icingaweb2/public/js/icinga/events.js

420 lines
16 KiB
JavaScript

/*! Icinga Web 2 | (c) 2014 Icinga Development Team | GPLv2+ */
/**
* Icinga.Events
*
* Event handlers
*/
(function (Icinga, $) {
'use strict';
Icinga.Events = function (icinga) {
this.icinga = icinga;
this.searchValue = '';
this.searchTimer = null;
};
Icinga.Events.prototype = {
/**
* Icinga will call our initialize() function once it's ready
*/
initialize: function () {
this.applyGlobalDefaults();
},
/**
* Global default event handlers
*/
applyGlobalDefaults: function () {
$(document).on('visibilitychange', { self: this }, this.onVisibilityChange);
$.each(this.icinga.behaviors, function (name, behavior) {
behavior.bind($(document));
});
var _this = this;
$('.container').each(function () {
// Initialize module javascript (Applies only to module.js code)
var moduleName = $(this).data('icingaModule');
if (moduleName) {
if (_this.icinga.hasModule(moduleName) && ! _this.icinga.isLoadedModule(moduleName)) {
_this.icinga.loadModule(moduleName);
}
}
});
// We catch resize events
$(window).on('resize', { self: this.icinga.ui }, this.icinga.ui.onWindowResize);
// Trigger 'rendered' event also on page loads
$(document).on('icinga-init', { self: this }, this.onInit);
// Destroy Icinga, clean up and interrupt pending requests on unload
$( window ).on('unload', { self: this }, this.onUnload);
$( window ).on('beforeunload', { self: this }, this.onUnload);
// Remove notifications on click
$(document).on('click', '#notifications li', function () { $(this).remove(); });
// We want to catch each link click
$(document).on('click', 'a', { self: this }, this.linkClicked);
$(document).on('click', 'tr[href]', { self: this }, this.linkClicked);
$(document).on('click', 'input[type="submit"], button[type="submit"]', this.rememberSubmitButton);
// We catch all form submit events
$(document).on('submit', 'form', { self: this }, this.submitForm);
// We support an 'autosubmit' class on dropdown form elements
$(document).on('change', 'form select.autosubmit', { self: this }, this.autoSubmitForm);
$(document).on('change', 'form input.autosubmit', { self: this }, this.autoSubmitForm);
// Automatically check a radio button once a specific input is focused
$(document).on('focus', 'form select[data-related-radiobtn]', { self: this }, this.autoCheckRadioButton);
$(document).on('focus', 'form input[data-related-radiobtn]', { self: this }, this.autoCheckRadioButton);
$(document).on('rendered', '#menu', { self: this }, this.onRenderedMenu);
$(document).on('keyup', '#search', { self: this }, this.autoSubmitSearch);
$(document).on('click', '.tree .handle', { self: this }, this.treeNodeToggle);
$(document).on('click', '#search + .search-reset', this.clearSearch);
$(document).on('rendered', '.container', { self: this }, this.loadDashlets);
// TBD: a global autocompletion handler
// $(document).on('keyup', 'form.auto input', this.formChangeDelayed);
// $(document).on('change', 'form.auto input', this.formChanged);
// $(document).on('change', 'form.auto select', this.submitForm);
},
treeNodeToggle: function () {
var $parent = $(this).closest('li');
if ($parent.hasClass('collapsed')) {
$('li', $parent).addClass('collapsed');
$parent.removeClass('collapsed');
} else {
$parent.addClass('collapsed');
}
},
onInit: function (event) {
$('body').removeClass('loading');
// Trigger the initial `rendered` events
$('.container').trigger('rendered');
// Additionally trigger a `rendered` event on the layout, some behaviors may
// want to differentiate whether a container or the entire layout is rendered
$('#layout').trigger('rendered');
},
onUnload: function (event) {
var icinga = event.data.self.icinga;
icinga.logger.info('Unloading Icinga');
icinga.destroy();
},
onVisibilityChange: function (event) {
var icinga = event.data.self.icinga;
if (document.visibilityState === undefined || document.visibilityState === 'visible') {
icinga.loader.autorefreshSuspended = false;
icinga.logger.debug('Page visible, enabling auto-refresh');
} else {
icinga.loader.autorefreshSuspended = true;
icinga.logger.debug('Page invisible, disabling auto-refresh');
}
},
autoCheckRadioButton: function (event) {
var $input = $(event.currentTarget);
var $radio = $('#' + $input.attr('data-related-radiobtn'));
if ($radio.length) {
$radio.prop('checked', true);
}
return true;
},
onRenderedMenu: function(event) {
var _this = event.data.self;
var $target = $(event.target);
var $searchField = $target.find('input.search');
// Remember initial search field value if any
if ($searchField.length && $searchField.val().length) {
_this.searchValue = $searchField.val();
}
},
autoSubmitSearch: function(event) {
var _this = event.data.self;
var $searchField = $(event.target);
if ($searchField.val() === _this.searchValue) {
return;
}
_this.searchValue = $searchField.val();
if (_this.searchTimer !== null) {
clearTimeout(_this.searchTimer);
_this.searchTimer = null;
}
var _event = $.extend({}, event); // event seems gc'd once the timeout is over
_this.searchTimer = setTimeout(function () {
_this.submitForm(_event, $searchField);
_this.searchTimer = null;
}, 500);
},
loadDashlets: function(event) {
var _this = event.data.self;
var $target = $(event.target);
if ($target.children('.dashboard').length) {
$target.find('.dashboard > .container').each(function () {
var $dashlet = $(this);
var url = $dashlet.data('icingaUrl');
if (typeof url !== 'undefined') {
_this.icinga.loader.loadUrl(url, $dashlet).autorefresh = true;
}
});
}
},
rememberSubmitButton: function(e) {
var $button = $(this);
var $form = $button.closest('form');
$form.data('submitButton', $button);
},
autoSubmitForm: function (event) {
return event.data.self.submitForm(event, $(event.currentTarget));
},
/**
*
*/
submitForm: function (event, $autoSubmittedBy) {
var _this = event.data.self;
// .closest is not required unless subelements to trigger this
var $form = $(event.currentTarget).closest('form');
var $button;
var $rememberedSubmittButton = $form.data('submitButton');
if (typeof $rememberedSubmittButton != 'undefined') {
if ($form.has($rememberedSubmittButton)) {
$button = $rememberedSubmittButton;
}
$form.removeData('submitButton');
}
if ($form.closest('[data-no-icinga-ajax]').length > 0) {
return true;
}
if (typeof $button === 'undefined') {
var $el;
if (typeof event.originalEvent !== 'undefined'
&& typeof event.originalEvent.explicitOriginalTarget === 'object') { // Firefox
$el = $(event.originalEvent.explicitOriginalTarget);
_this.icinga.logger.debug('events/submitForm: Button is event.originalEvent.explicitOriginalTarget');
} else {
$el = $(event.currentTarget);
_this.icinga.logger.debug('events/submitForm: Button is event.currentTarget');
}
if ($el && ($el.is('input[type=submit]') || $el.is('button[type=submit]'))) {
$button = $el;
} else {
_this.icinga.logger.debug(
'events/submitForm: Can not determine submit button, using the first one in form'
);
}
}
_this.icinga.loader.submitForm($form, $autoSubmittedBy, $button);
event.stopPropagation();
event.preventDefault();
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($node.attr('href'), linkTarget);
return true;
}
return false;
},
/**
* Someone clicked a link or tr[href]
*/
linkClicked: function (event) {
var _this = event.data.self;
var icinga = _this.icinga;
var $a = $(this);
var $eventTarget = $(event.target);
var href = $a.attr('href');
var linkTarget = $a.attr('target');
var $target;
var formerUrl;
if (! href) {
return;
}
if (href.match(/^(?:(?:mailto|javascript|data):|[a-z]+:\/\/)/)) {
event.stopPropagation();
return true;
}
if ($a.closest('[data-no-icinga-ajax]').length > 0) {
return true;
}
// Check for ctrl or cmd click to open new tab unless clicking on a multiselect row
if ((event.ctrlKey || event.metaKey) && href !== '#' && $a.is('a')) {
window.open(href, linkTarget);
return false;
}
// Special checks for link clicks in action tables
if (! $a.is('tr[href]') && $a.closest('table.action').length > 0) {
// ignore clicks to ANY link with special key pressed
if ($a.closest('table.multiselect').length > 0 && (event.ctrlKey || event.metaKey || event.shiftKey)) {
return true;
}
// ignore inner links matching the row URL
if ($a.attr('href') === $a.closest('tr[href]').attr('href')) {
return true;
}
}
// window.open is used as return true; didn't work reliable
if (linkTarget === '_blank' || linkTarget === '_self') {
window.open(href, linkTarget);
return false;
}
if (! $eventTarget.is($a)) {
if ($eventTarget.is('input') || $eventTarget.is('button')) {
// Ignore form elements in action rows
return;
} else {
var $button = $('input[type=submit]:focus').add('button[type=submit]:focus');
if ($button.length > 0 && $.contains($button[0], $eventTarget[0])) {
// Ignore any descendant of form elements
return;
}
}
}
// ignore multiselect table row clicks
if ($a.is('tr') && $a.closest('table.multiselect').length > 0) {
return;
}
// Handle all other links as XHR requests
event.stopPropagation();
event.preventDefault();
// This is an anchor only
if (href.substr(0, 1) === '#' && href.length > 1
&& href.substr(1, 1) !== '!'
) {
icinga.ui.focusElement(href.substr(1), $a.closest('.container'));
return;
}
// activate spinner indicator
if ($a.hasClass('spinner')) {
$a.addClass('active');
}
// If link has hash tag...
if (href.match(/#/)) {
if (href === '#') {
if ($a.hasClass('close-container-control')) {
if (! icinga.ui.isOneColLayout()) {
var $cont = $a.closest('.container').first();
if ($cont.attr('id') === 'col1') {
icinga.ui.moveToLeft();
icinga.ui.layout1col();
} else {
icinga.ui.layout1col();
}
icinga.history.pushCurrentState();
}
}
return false;
}
$target = icinga.loader.getLinkTargetFor($a);
formerUrl = $target.data('icingaUrl');
if (typeof formerUrl !== 'undefined' && formerUrl.split(/#/)[0] === href.split(/#/)[0]) {
icinga.ui.focusElement(href.split(/#/)[1], $target);
$target.data('icingaUrl', href);
if (formerUrl !== href) {
icinga.history.pushCurrentState();
}
return false;
}
} else {
$target = icinga.loader.getLinkTargetFor($a);
}
// Load link URL
icinga.loader.loadUrl(href, $target);
if ($a.closest('#menu').length > 0) {
// Menu links should remove all but the first layout column
icinga.ui.layout1col();
}
return false;
},
clearSearch: function (event) {
$(event.target).parent().find('#search').attr('value', '');
},
unbindGlobalHandlers: function () {
$.each(this.icinga.behaviors, function (name, behavior) {
behavior.unbind($(document));
});
$(window).off('resize', this.onWindowResize);
$(window).off('load', this.onLoad);
$(window).off('unload', this.onUnload);
$(window).off('beforeunload', this.onUnload);
$(document).off('scroll', '.container', this.onContainerScroll);
$(document).off('click', 'a', this.linkClicked);
$(document).off('submit', 'form', this.submitForm);
$(document).off('change', 'form select.autosubmit', this.submitForm);
$(document).off('change', 'form input.autosubmit', this.submitForm);
$(document).off('focus', 'form select[data-related-radiobtn]', this.autoCheckRadioButton);
$(document).off('focus', 'form input[data-related-radiobtn]', this.autoCheckRadioButton);
$(document).off('visibilitychange', this.onVisibilityChange);
},
destroy: function() {
// This is gonna be hard, clean up the mess
this.unbindGlobalHandlers();
this.icinga = null;
}
};
}(Icinga, jQuery));