+
+
+
+
diff --git a/public/css/icinga/selection-toolbar.less b/public/css/icinga/selection-toolbar.less
new file mode 100644
index 000000000..91d83fbc4
--- /dev/null
+++ b/public/css/icinga/selection-toolbar.less
@@ -0,0 +1,8 @@
+div.selection-toolbar {
+ float: right;
+ padding-right: 20px;
+}
+
+div.selection-toolbar a {
+ color: #049baf
+}
\ No newline at end of file
diff --git a/public/js/icinga/events.js b/public/js/icinga/events.js
index 614b85c38..d78229a16 100644
--- a/public/js/icinga/events.js
+++ b/public/js/icinga/events.js
@@ -13,6 +13,12 @@
Icinga.Events.prototype = {
+ keyboard: {
+ ctrlKey: false,
+ altKey: false,
+ shiftKey: false
+ },
+
/**
* Icinga will call our initialize() function once it's ready
*/
@@ -20,6 +26,7 @@
this.applyGlobalDefaults();
this.applyHandlers($('#layout'));
this.icinga.ui.prepareContainers();
+ this.icinga.ui.prepareMultiselectTables($(document));
},
// TODO: What's this?
@@ -61,11 +68,27 @@
$('input.autofocus', el).focus();
- $('.inlinepie', el).sparkline('html', {
- type: 'pie',
- sliceColors: ['#44bb77', '#ffaa44', '#ff5566', '#dcd'],
- width: '2em',
- height: '2em',
+ $('div.inlinepie', el).each(function() {
+ var $img = $(this).find('img');
+ var title = $img.attr('title'),
+ values = $img.data('icinga-values'),
+ colors = $img.data('icinga-colors'),
+ width = $img.css('width'),
+ height = $img.css('height');
+ if (colors) {
+ colors = colors.split(',');
+ }
+ $img.replaceWith(values);
+ $(this).sparkline(
+ 'html',
+ {
+ type: 'pie',
+ sliceColors: colors || ['#44bb77', '#ffaa44', '#ff5566', '#dcd'],
+ width: width,
+ height: height,
+ tooltipChartTitle: title
+ }
+ );
});
},
@@ -89,8 +112,9 @@
// We want to catch each link click
$(document).on('click', 'a', { self: this }, this.linkClicked);
- // We treat tr's with a href attribute like links
- $(document).on('click', 'tr[href]', { self: this }, this.linkClicked);
+ // Select a table row
+ $(document).on('click', 'table.action tr[href]', { self: this }, this.rowSelected);
+ $(document).on('click', 'table.action tr a', { self: this }, this.rowSelected);
$(document).on('click', 'button', { self: this }, this.submitForm);
@@ -110,6 +134,8 @@
$(document).on('mouseleave', '#sidebar', this.leaveSidebar);
$(document).on('click', '.tree .handle', { self: this }, this.treeNodeToggle);
+ // Toggle all triStateButtons
+ $(document).on('click', 'div.tristate .tristate-dummy', { self: this }, this.clickTriState);
// TBD: a global autocompletion handler
// $(document).on('keyup', 'form.auto input', this.formChangeDelayed);
@@ -225,6 +251,38 @@
return event.data.self.submitForm(event, true);
},
+ clickTriState: function (event) {
+ var $tristate = $(this);
+ var triState = parseInt($tristate.data('icinga-tristate'), 10);
+
+ // load current values
+ var old = $tristate.data('icinga-old').toString();
+ var value = $tristate.parent().find('input:radio:checked').first().prop('checked', false).val();
+
+ // calculate the new value
+ if (triState) {
+ // 1 => 0
+ // 0 => unchanged
+ // unchanged => 1
+ value = value === '1' ? '0' : (value === '0' ? 'unchanged' : '1');
+ } else {
+ // 1 => 0
+ // 0 => 1
+ value = value === '1' ? '0' : '1';
+ }
+
+ // update form value
+ $tristate.parent().find('input:radio[value="' + value + '"]').prop('checked', true);
+ // update dummy
+
+ if (value !== old) {
+ $tristate.parent().find('b.tristate-changed').css('visibility', 'visible');
+ } else {
+ $tristate.parent().find('b.tristate-changed').css('visibility', 'hidden');
+ }
+ self.icinga.ui.setTriState(value.toString(), $tristate);
+ },
+
/**
*
*/
@@ -277,6 +335,92 @@
return false;
},
+ handleExternalTarget: function($node) {
+ var linkTarget = $node.attr('target');
+
+ // TODO: Let remote links pass through. Right now they only work
+ // combined with target="_blank" or target="_self"
+ // window.open is used as return true; didn't work reliable
+ if (linkTarget === '_blank' || linkTarget === '_self') {
+ window.open(href, linkTarget);
+ return true;
+ }
+ return false;
+ },
+
+ /**
+ * Handle table selection.
+ */
+ rowSelected: function(event) {
+ var self = event.data.self;
+ var icinga = self.icinga;
+ var $tr = $(this);
+ var $table = $tr.closest('table.multiselect');
+ var data = self.icinga.ui.getSelectionKeys($table);
+ var multisel = $table.hasClass('multiselect');
+ var url = $table.data('icinga-multiselect-url');
+
+ // When the selection points to a link, select the closest row
+ if ($tr.prop('tagName').toLowerCase() === 'a') {
+ $tr = $tr.closest('tr').first();
+ }
+
+ event.stopPropagation();
+ event.preventDefault();
+
+ if (icinga.events.handleExternalTarget($tr)) {
+ // link handled externally
+ return false;
+ }
+ if (multisel && !data) {
+ icinga.logger.error('A table with multiselection must define the attribute "data-icinga-multiselect-data"');
+ return;
+ }
+ if (multisel && !url) {
+ icinga.logger.error('A table with multiselection must define the attribute "data-icinga-multiselect-url"');
+ return;
+ }
+
+ // update selection
+ if ((event.ctrlKey || event.metaKey) && multisel) {
+ icinga.ui.toogleTableRowSelection($tr);
+ // multi selection
+ } else if (event.shiftKey && multisel) {
+ // range selection
+ icinga.ui.addTableRowRangeSelection($tr);
+ } else {
+ // single selection
+ icinga.ui.setTableRowSelection($tr);
+ }
+ // focus only the current table.
+ icinga.ui.focusTable($table[0]);
+
+ // update url
+ var $target = self.getLinkTargetFor($tr);
+ if (multisel) {
+ var $trs = $table.find('tr[href].active');
+ if ($trs.length > 1) {
+ var queries = [];
+ var selectionData = icinga.ui.getSelectionSetData($trs, data);
+ var query = icinga.ui.selectionDataToQuery(selectionData, data, icinga);
+ icinga.loader.loadUrl(url + '?' + query, $target);
+ icinga.ui.storeSelectionData(selectionData);
+ } else if ($trs.length === 1) {
+ // display a single row
+ icinga.loader.loadUrl($tr.attr('href'), $target);
+ icinga.ui.storeSelectionData($tr.attr('href'));
+ } else {
+ // display nothing
+ icinga.loader.loadUrl('#');
+ icinga.ui.storeSelectionData(null);
+ }
+ } else {
+ icinga.loader.loadUrl($tr.attr('href'), $target);
+ }
+ return false;
+ },
+
+
/**
* Someone clicked a link or tr[href]
*/
@@ -299,6 +443,11 @@
return false;
}
+ // ignore links inside of tables.
+ if ($a.closest('table tr').length > 0) {
+ return;
+ }
+
// Handle all other links as XHR requests
event.stopPropagation();
event.preventDefault();
@@ -409,7 +558,8 @@
$(window).off('beforeunload', this.onUnload);
$(document).off('scroll', '.container', this.onContainerScroll);
$(document).off('click', 'a', this.linkClicked);
- $(document).off('click', 'tr[href]', this.linkClicked);
+ $(document).off('click', 'table.action tr[href]', this.rowSelected);
+ $(document).off('click', 'table.action tr a', this.rowSelected);
$(document).off('submit', 'form', this.submitForm);
$(document).off('click', 'button', this.submitForm);
$(document).off('change', 'form select.autosubmit', this.submitForm);
@@ -417,6 +567,7 @@
$(document).off('mouseleave', '.historycolorgrid td', this.historycolorgidUnhover);
$(document).off('mouseenter', 'li.dropdown', this.dropdownHover);
$(document).off('mouseleave', 'li.dropdown', this.dropdownLeave);
+ $(document).off('click', 'div.tristate .tristate-dummy', this.clickTriState);
},
destroy: function() {
diff --git a/public/js/icinga/loader.js b/public/js/icinga/loader.js
index 04035e1e6..f525c2437 100644
--- a/public/js/icinga/loader.js
+++ b/public/js/icinga/loader.js
@@ -247,7 +247,9 @@
if (! req.autorefresh) {
// TODO: Hook for response/url?
var $forms = $('[action="' + this.icinga.utils.parseUrl(url).path + '"]');
+
var $matches = $.merge($('[href="' + url + '"]'), $forms);
+
$matches.each(function (idx, el) {
if ($(el).closest('#menu').length) {
$('#menu .active').removeClass('active');
@@ -273,6 +275,7 @@
});
} else {
// TODO: next container url
+ // Get first container url?
active = $('[href].active', req.$target).attr('href');
}
@@ -387,19 +390,12 @@
this.icinga.history.pushCurrentState();
}
}
+ this.icinga.ui.initializeTriStates($resp);
-
- /*
- * Replace SVG piecharts with jQuery-Sparkline
+ /**
+ * Make multiselection-tables not selectable.
*/
- $('.inlinepie', $resp).each(function(){
- var title = $(this).attr('title'),
- style = $(this).attr('style'),
- values = $(this).data('icinga-values');
- var html = '
' + values + '
';
- $(this).replaceWith(html);
- });
-
+ this.icinga.ui.prepareMultiselectTables($resp);
/* Should we try to fiddle with responses containing full HTML? */
/*
@@ -436,7 +432,39 @@
}
if (active) {
- $('[href="' + active + '"]', req.$target).addClass('active');
+ var focusedUrl = this.icinga.ui.getFocusedContainerDataUrl();
+ var oldSelectionData = this.icinga.ui.loadSelectionData();
+ if (typeof oldSelectionData === 'string') {
+ $('[href="' + oldSelectionData + '"]', req.$target).addClass('active');
+
+ } else if (oldSelectionData !== null) {
+ var $container;
+ if (!focusedUrl) {
+ $container = $('document').first();
+ } else {
+ $container = $('.container[data-icinga-url="' + focusedUrl + '"]');;
+ }
+
+ var $table = $container.find('table.action').first();
+ var keys = self.icinga.ui.getSelectionKeys($table);
+
+ // build map of selected queries
+ var oldSelectionQueries = {};
+ $.each(oldSelectionData, function(i, query){
+ oldSelectionQueries[self.icinga.ui.selectionDataToQueryComp(query)] = true;
+ });
+
+ // set all new selections to active
+ $table.find('tr[href]').filter(function(){
+ var $tr = $(this);
+ var rowData = self.icinga.ui.getSelectionData($tr, keys, self.icinga);
+ var newSelectionQuery = self.icinga.ui.selectionDataToQueryComp(rowData);
+ if (oldSelectionQueries[newSelectionQuery]) {
+ return true;
+ }
+ return false;
+ }).addClass('active');
+ }
}
req.$target.trigger('rendered');
},
diff --git a/public/js/icinga/ui.js b/public/js/icinga/ui.js
index 27ebae7a4..e71310d62 100644
--- a/public/js/icinga/ui.js
+++ b/public/js/icinga/ui.js
@@ -7,6 +7,13 @@
'use strict';
+ // Stores the icinga-data-url of the last focused table.
+ var focusedTableDataUrl = null;
+
+ // The stored selection data, useful for preserving selections over
+ // multiple reload-cycles.
+ var selectionData = null;
+
Icinga.UI = function (icinga) {
this.icinga = icinga;
@@ -198,12 +205,10 @@
return true;
}
}
-
this.icinga.logger.error(
'Someone messed up our responsiveness hacks, html font-family is',
layout
);
-
return false;
},
@@ -271,6 +276,220 @@
*/
},
+ /**
+ * Prepare all multiselectable tables for multi-selection by
+ * removing the regular text selection.
+ */
+ prepareMultiselectTables: function () {
+ var $rows = $('table.multiselect tr[href]');
+ $rows.find('td').attr('unselectable', 'on')
+ .css('user-select', 'none')
+ .css('-webkit-user-select', 'none')
+ .css('-moz-user-select', 'none')
+ .css('-ms-user-select', 'none');
+ },
+
+ /**
+ * Add the given table-row to the selection of the closest
+ * table and deselect all other rows of the closest table.
+ *
+ * @param $tr {jQuery} The selected table row.
+ * @returns {boolean} If the selection was changed.
+ */
+ setTableRowSelection: function ($tr) {
+ var $table = $tr.closest('table.multiselect');
+ $table.find('tr[href].active').removeClass('active');
+ $tr.addClass('active');
+ return true;
+ },
+
+ /**
+ * Toggle the given table row to "on" when not selected, or to "off" when
+ * currently selected.
+ *
+ * @param $tr {jQuery} The table row.
+ * @returns {boolean} If the selection was changed.
+ */
+ toogleTableRowSelection: function ($tr) {
+ // multi selection
+ if ($tr.hasClass('active')) {
+ $tr.removeClass('active');
+ } else {
+ $tr.addClass('active');
+ }
+ return true;
+ },
+
+ /**
+ * Add a new selection range to the closest table, using the selected row as
+ * range target.
+ *
+ * @param $tr {jQuery} The target of the selected range.
+ * @returns {boolean} If the selection was changed.
+ */
+ addTableRowRangeSelection: function ($tr) {
+ var $table = $tr.closest('table.multiselect');
+ var $rows = $table.find('tr[href]'),
+ from, to;
+ var selected = $tr.first().get(0);
+ $rows.each(function(i, el) {
+ if ($(el).hasClass('active') || el === selected) {
+ if (!from) {
+ from = el;
+ }
+ to = el;
+ }
+ });
+ var inRange = false;
+ $rows.each(function(i, el){
+ if (el === from) {
+ inRange = true;
+ }
+ if (inRange) {
+ $(el).addClass('active');
+ }
+ if (el === to) {
+ inRange = false;
+ }
+ });
+ return false;
+ },
+
+
+ /**
+ * Read the data from a whole set of selections.
+ *
+ * @param $selections {jQuery} All selected rows in a jQuery-selector.
+ * @param keys {Array} An array containing all valid keys.
+ * @returns {Array} An array containing an object with the data for each selection.
+ */
+ getSelectionSetData: function($selections, keys) {
+ var selections = [];
+ var icinga = this.icinga;
+
+ // read all current selections
+ $selections.each(function(ind, selected) {
+ selections.push(icinga.ui.getSelectionData($(selected), keys, icinga));
+ });
+ return selections;
+ },
+
+ getSelectionKeys: function($selection)
+ {
+ var d = $selection.data('icinga-multiselect-data') && $selection.data('icinga-multiselect-data').split(',');
+ return d || [];
+ },
+
+ /**
+ * Read the data from the given selected object.
+ *
+ * @param $selection {jQuery} The selected object.
+ * @param keys {Array} An array containing all valid keys.
+ * @param icinga {Icinga} The main icinga object.
+ * @returns {Object} An object containing all key-value pairs associated with this selection.
+ */
+ getSelectionData: function($selection, keys, icinga)
+ {
+ var url = $selection.attr('href');
+ var params = this.icinga.utils.parseUrl(url).params;
+ var tuple = {};
+ for (var i = 0; i < keys.length; i++) {
+ var key = keys[i];
+ if (params[key]) {
+ tuple[key] = params[key];
+ }
+ }
+ return tuple;
+ },
+
+ /**
+ * Convert a set of selection data to a single query.
+ *
+ * @param selectionData {Array} The selection data generated from getSelectionData
+ * @returns {String} The formatted and uri-encoded query-string.
+ */
+ selectionDataToQuery: function (selectionData) {
+ var queries = [];
+
+ // create new url
+ if (selectionData.length < 2) {
+ // single-selection
+ $.each(selectionData[0], function(key, value){
+ queries.push(key + '=' + encodeURIComponent(value));
+ });
+ } else {
+ // multi-selection
+ $.each(selectionData, function(i, el){
+ $.each(el, function(key, value) {
+ queries.push(key + '[' + i + ']=' + encodeURIComponent(value));
+ });
+ });
+ }
+ return queries.join('&');
+ },
+
+ /**
+ * Create a single query-argument (not compatible to selectionDataToQuery)
+ *
+ * @param data
+ * @returns {string}
+ */
+ selectionDataToQueryComp: function(data) {
+ var queries = [];
+ $.each(data, function(key, value){
+ queries.push(key + '=' + encodeURIComponent(value));
+ });
+ return queries.join('&');
+ },
+
+ /**
+ * Store a set of selection-data to preserve it accross page-reloads
+ *
+ * @param data {Array|String|Null} The selection-data be an Array of Objects,
+ * containing the selection data (when multiple rows where selected), a
+ * String containing a single url (when only a single row was selected) or
+ * Null when nothing was selected.
+ */
+ storeSelectionData: function(data) {
+ selectionData = data;
+ },
+
+ /**
+ * Load the last stored set of selection-data
+ *
+ * @returns {Array|String|Null} May be an Array of Objects, containing the selection data
+ * (when multiple rows where selected), a String containing a single url
+ * (when only a single row was selected) or Null when nothing was selected.
+ */
+ loadSelectionData: function() {
+ return selectionData;
+ },
+
+ /**
+ * Focus the given table by deselecting all selections on all other tables.
+ *
+ * Focusing a table is important for environments with multiple tables like
+ * the dashboard. It should only be possible to select rows at one table at a time,
+ * when a user selects a row on a table all rows that are not child of the given table
+ * will be removed from the selection.
+ *
+ * @param table {htmlElement} The table to focus.
+ */
+ focusTable: function (table) {
+ $('table').filter(function(){ return this !== table; }).find('tr[href]').removeClass('active');
+ var n = $(table).closest('div.container').attr('data-icinga-url');
+ focusedTableDataUrl = n;
+ },
+
+ /**
+ * Return the URL of the last focused table container.
+ *
+ * @returns {String} The data-icinga-url of the last focused table, which should be unique in each site.
+ */
+ getFocusedContainerDataUrl: function() {
+ return focusedTableDataUrl;
+ },
+
refreshDebug: function () {
var size = this.getDefaultFontSize().toString();
@@ -369,6 +588,62 @@
return $calc.width() / 1000;
},
+ /**
+ * Initialize all TriStateCheckboxes in the given html
+ */
+ initializeTriStates: function ($html) {
+ var self = this;
+ $('div.tristate', $html).each(function(index, item) {
+ var $target = $(item);
+
+ // hide input boxess and remove text nodes
+ $target.find("input").hide();
+ $target.contents().filter(function() { return this.nodeType === 3; }).remove();
+
+ // has three states?
+ var triState = $target.find('input[value="unchanged"]').size() > 0 ? 1 : 0;
+
+ // fetch current value from radiobuttons
+ var value = $target.find('input:checked').first().val();
+
+ $target.append(
+ ' (changed) '
+ );
+ if (triState) {
+ // TODO: find a better way to activate indeterminate checkboxes after load.
+ $target.append(
+ ''
+ );
+ }
+ });
+ },
+
+ /**
+ * Set the value of the given TriStateCheckbox
+ *
+ * @param value {String} The value to set, can be '1', '0' and 'unchanged'
+ * @param $checkbox {jQuery} The checkbox
+ */
+ setTriState: function(value, $checkbox)
+ {
+ switch (value) {
+ case ('1'):
+ $checkbox.prop('checked', true).prop('indeterminate', false);
+ break;
+ case ('0'):
+ $checkbox.prop('checked', false).prop('indeterminate', false);
+ break;
+ case ('unchanged'):
+ $checkbox.prop('checked', false).prop('indeterminate', true);
+ break;
+ }
+ },
+
initializeControls: function (parent) {
var self = this;
diff --git a/public/js/icinga/utils.js b/public/js/icinga/utils.js
index 7a756092e..bf081d016 100644
--- a/public/js/icinga/utils.js
+++ b/public/js/icinga/utils.js
@@ -104,7 +104,7 @@
path : a.pathname.replace(/^([^\/])/,'/$1'),
relative: (a.href.match(/tps?:\/\/[^\/]+(.+)/) || [,''])[1],
segments: a.pathname.replace(/^\//,'').split('/'),
- params : this.parseParams(a),
+ params : this.parseParams(a)
};
a = null;