zoneminder/web/skins/classic/js/skin.js

922 lines
28 KiB
JavaScript
Raw Normal View History

2013-03-17 07:45:21 +08:00
//
// ZoneMinder base static javascript file, $Date$, $Revision$
// Copyright (C) 2001-2008 Philip Coombes
//
// This program is free software; you can redistribute it and/or
// modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 2
// of the License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program; if not, write to the Free Software
// Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
2013-03-17 07:45:21 +08:00
//
//
// This file should only contain static JavaScript and no php.
// Use skin.js.php for JavaScript that need pre-processing
//
// Globally define the icons used in the bootstrap-table top-right toolbar
var icons = {
paginationSwitchDown: 'fa-caret-square-o-down',
paginationSwitchUp: 'fa-caret-square-o-up',
export: 'fa-download',
refresh: 'fa-retweet',
autoRefresh: 'fa-clock-o',
advancedSearchIcon: 'fa-chevron-down',
toggleOff: 'fa-toggle-off',
toggleOn: 'fa-toggle-on',
columns: 'fa-th-list',
fullscreen: 'fa-arrows-alt',
detailOpen: 'fa-plus',
detailClose: 'fa-minus'
};
function checkSize() {
2017-11-10 04:03:23 +08:00
if ( 0 ) {
if (window.outerHeight) {
var w = window.outerWidth;
var prevW = w;
var h = window.outerHeight;
var prevH = h;
if (h > screen.availHeight) {
h = screen.availHeight;
}
if (w > screen.availWidth) {
w = screen.availWidth;
}
if (w != prevW || h != prevH) {
window.resizeTo(w, h);
}
}
2017-11-10 04:03:23 +08:00
}
}
// Polyfill for NodeList.prototype.forEach on IE.
if (window.NodeList && !NodeList.prototype.forEach) {
NodeList.prototype.forEach = Array.prototype.forEach;
}
window.addEventListener("DOMContentLoaded", function onSkinDCL() {
document.querySelectorAll("form.validateFormOnSubmit").forEach(function(el) {
el.addEventListener("submit", function onSubmit(evt) {
if (!validateForm(this)) {
evt.preventDefault();
}
});
});
2020-09-26 01:34:04 +08:00
document.querySelectorAll(".zmlink").forEach(function(el) {
el.addEventListener("click", function onClick(evt) {
var el = this;
var url;
if ( el.hasAttribute("href") ) {
// <a>
url = el.getAttribute("href");
} else {
// buttons
url = el.getAttribute("data-url");
}
evt.preventDefault();
window.location.assign(url);
});
});
2020-08-29 02:09:49 +08:00
document.querySelectorAll(".pillList a").forEach(function addOnClick(el) {
el.addEventListener("click", submitTab);
});
dataOnClickThis();
dataOnClick();
dataOnClickTrue();
dataOnChangeThis();
dataOnChange();
dataOnInput();
dataOnInputThis();
});
// 'data-on-click-this' calls the global function in the attribute value with the element when a click happens.
function dataOnClickThis() {
document.querySelectorAll("a[data-on-click-this], button[data-on-click-this], input[data-on-click-this]").forEach(function attachOnClick(el) {
var fnName = el.getAttribute("data-on-click-this");
if ( !window[fnName] ) {
console.error("Nothing found to bind to " + fnName + " on element " + el.name);
return;
}
el.onclick = window[fnName].bind(el, el);
});
}
// 'data-on-click' calls the global function in the attribute value with no arguments when a click happens.
function dataOnClick() {
2020-01-01 08:11:14 +08:00
document.querySelectorAll("i[data-on-click], a[data-on-click], button[data-on-click], input[data-on-click]").forEach(function attachOnClick(el) {
var fnName = el.getAttribute("data-on-click");
if ( !window[fnName] ) {
console.error("Nothing found to bind to " + fnName + " on element " + el.name);
return;
2020-01-01 09:24:51 +08:00
}
2020-01-01 08:11:14 +08:00
el.onclick = function(ev) {
window[fnName](ev);
};
});
}
// 'data-on-click-true' calls the global function in the attribute value with no arguments when a click happens.
function dataOnClickTrue() {
2019-02-06 05:45:05 +08:00
document.querySelectorAll("a[data-on-click-true], button[data-on-click-true], input[data-on-click-true]").forEach(function attachOnClick(el) {
var fnName = el.getAttribute("data-on-click-true");
if ( !window[fnName] ) {
console.error("Nothing found to bind to " + fnName);
return;
}
2019-02-06 05:45:05 +08:00
el.onclick = function() {
window[fnName](true);
};
});
}
2019-02-06 05:45:05 +08:00
// 'data-on-change-this' calls the global function in the attribute value with the element when a change happens.
function dataOnChangeThis() {
document.querySelectorAll("select[data-on-change-this], input[data-on-change-this]").forEach(function attachOnChangeThis(el) {
var fnName = el.getAttribute("data-on-change-this");
if ( !window[fnName] ) {
console.error("Nothing found to bind to " + fnName);
return;
}
el.onchange = window[fnName].bind(el, el);
});
}
// 'data-on-change' adds an event listener for the global function in the attribute value when a change happens.
function dataOnChange() {
document.querySelectorAll("select[data-on-change], input[data-on-change]").forEach(function attachOnChange(el) {
var fnName = el.getAttribute("data-on-change");
if ( !window[fnName] ) {
console.error("Nothing found to bind to " + fnName);
return;
}
el.onchange = window[fnName];
});
}
// 'data-on-input' adds an event listener for the global function in the attribute value when an input happens.
function dataOnInput() {
document.querySelectorAll("input[data-on-input]").forEach(function(el) {
var fnName = el.getAttribute("data-on-input");
if ( !window[fnName] ) {
console.error("Nothing found to bind to " + fnName);
return;
}
el.oninput = window[fnName];
});
}
// 'data-on-input-this' calls the global function in the attribute value with the element when an input happens.
function dataOnInputThis() {
document.querySelectorAll("input[data-on-input-this]").forEach(function(el) {
var fnName = el.getAttribute("data-on-input-this");
if ( !window[fnName] ) {
console.error("Nothing found to bind to " + fnName);
return;
}
el.oninput = window[fnName].bind(el, el);
});
}
function openEvent( eventId, eventFilter ) {
2017-06-06 03:22:32 +08:00
var url = '?view=event&eid='+eventId;
if ( eventFilter ) {
2017-06-06 03:22:32 +08:00
url += eventFilter;
}
window.location.assign(url);
2013-03-17 07:45:21 +08:00
}
function openFrames( eventId ) {
2017-06-06 03:22:32 +08:00
var url = '?view=frames&eid='+eventId;
window.location.assign(url);
2013-03-17 07:45:21 +08:00
}
function openFrame( eventId, frameId, width, height ) {
2017-06-06 03:22:32 +08:00
var url = '?view=frame&eid='+eventId+'&fid='+frameId;
window.location.assign(url);
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function windowToFront() {
top.window.focus();
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function closeWindow() {
top.window.close();
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function refreshWindow() {
window.location.reload( true );
2013-03-17 07:45:21 +08:00
}
function backWindow() {
window.history.back();
}
2013-03-17 07:45:21 +08:00
function refreshParentWindow() {
2017-06-06 03:22:32 +08:00
if ( refreshParent ) {
if ( window.opener ) {
if ( refreshParent == true ) {
2017-06-06 03:22:32 +08:00
window.opener.location.reload( true );
} else {
2017-06-06 03:22:32 +08:00
window.opener.location.href = refreshParent;
}
2017-06-06 03:22:32 +08:00
}
}
2013-03-17 07:45:21 +08:00
}
2018-06-12 20:58:19 +08:00
if ( currentView != 'none' && currentView != 'login' ) {
$j.ajaxSetup({timeout: AJAX_TIMEOUT}); //sets timeout for all getJSON.
$j(document).ready(function() {
2020-09-20 01:36:04 +08:00
// Load the Logout and State modals into the dom
$j('#logoutButton').click(clickLogout);
2020-12-09 04:25:48 +08:00
if ( canEdit.System ) $j('#stateModalBtn').click(getStateModal);
2020-09-19 23:44:05 +08:00
// Trigger autorefresh of the widget bar stats on the navbar
2019-02-06 05:45:05 +08:00
if ( $j('.navbar').length ) {
2018-02-15 02:16:14 +08:00
setInterval(getNavBar, navBarRefresh);
}
// Workaround Bootstrap-Mootools conflict
var bootstrapLoaded = (typeof $j().carousel == 'function');
var mootoolsLoaded = (typeof MooTools != 'undefined');
if (bootstrapLoaded && mootoolsLoaded) {
Element.implement({
hide: function() {
return this;
},
show: function(v) {
return this;
},
slide: function(v) {
return this;
}
});
}
// Update zmBandwidth cookie when the user makes a selection from the dropdown
bwClickFunction();
2020-08-10 03:18:44 +08:00
// Update update reminders when the user makes a selection from the dropdown
reminderClickFunction();
// Manage the widget bar minimize chevron
$j("#flip").click(function() {
$j("#panel").slideToggle("slow");
var flip = $j("#flip");
if ( flip.html() == 'keyboard_arrow_up' ) {
flip.html('keyboard_arrow_down');
setCookie('zmHeaderFlip', 'down', 3600);
} else {
flip.html('keyboard_arrow_up');
setCookie('zmHeaderFlip', 'up', 3600);
}
});
// Manage the web console filter bar minimize chevron
$j("#fbflip").click(function() {
$j("#fbpanel").slideToggle("slow");
var fbflip = $j("#fbflip");
if ( fbflip.html() == 'keyboard_arrow_up' ) {
fbflip.html('keyboard_arrow_down');
setCookie('zmFilterBarFlip', 'down', 3600);
} else {
fbflip.html('keyboard_arrow_up');
setCookie('zmFilterBarFlip', 'up', 3600);
$j('.chosen').chosen("destroy");
$j('.chosen').chosen();
}
});
// Manage the web console filter bar minimize chevron
$j("#mfbflip").click(function() {
$j("#mfbpanel").slideToggle("slow");
var mfbflip = $j("#mfbflip");
if ( mfbflip.html() == 'keyboard_arrow_up' ) {
mfbflip.html('keyboard_arrow_down');
setCookie('zmMonitorFilterBarFlip', 'up', 3600);
} else {
mfbflip.html('keyboard_arrow_up');
setCookie('zmMonitorFilterBarFlip', 'down', 3600);
$j('.chosen').chosen("destroy");
$j('.chosen').chosen();
}
});
// Autoclose the hamburger button if the end user clicks outside the button
$j(document).click(function(event) {
var target = $j(event.target);
var _mobileMenuOpen = $j("#main-header-nav").hasClass("show");
if (_mobileMenuOpen === true && !target.hasClass("navbar-toggler")) {
$j("button.navbar-toggler").click();
}
});
2020-09-11 05:56:58 +08:00
// Manage the optionhelp links
$j(".optionhelp").click(function(evt) {
$j.getJSON(thisUrl + '?request=modal&modal=optionhelp&ohndx=' + evt.target.id)
2020-09-11 06:11:31 +08:00
.done(optionhelpModal)
.fail(logAjaxFail);
2020-09-11 05:56:58 +08:00
});
});
2020-10-15 04:58:39 +08:00
// After retieving modal html via Ajax, this will insert it into the DOM
function insertModalHtml(name, html) {
var modal = $j('#' + name);
if ( modal.length ) {
modal.replaceWith(html);
2020-09-11 05:56:58 +08:00
} else {
2020-10-15 04:58:39 +08:00
$j("body").append(html);
2020-09-11 05:56:58 +08:00
}
2020-10-15 04:58:39 +08:00
}
// Manage the modal html we received after user clicks help link
function optionhelpModal(data) {
insertModalHtml('optionhelp', data.html);
2020-09-11 05:56:58 +08:00
$j('#optionhelp').modal('show');
// Manage the CLOSE optionhelp modal button
document.getElementById("ohCloseBtn").addEventListener("click", function onOhCloseClick(evt) {
$j('#optionhelp').modal('hide');
});
}
2018-02-15 02:16:14 +08:00
function getNavBar() {
$j.getJSON(thisUrl + '?view=request&request=status&entity=navBar')
.done(setNavBar)
2019-02-06 05:45:05 +08:00
.fail(function(jqxhr, textStatus, error) {
console.log("Request Failed: " + textStatus + ", " + error);
2020-10-27 01:06:41 +08:00
if ( ! jqxhr.responseText ) {
console.log("No responseText in jqxhr");
console.log(jqxhr);
return;
}
console.log("Response Text: " + jqxhr.responseText.replace(/(<([^>]+)>)/gi, ''));
if ( textStatus != "timeout" ) {
// The idea is that this should only fail due to auth, so reload the page
// which should go to login if it can't stay logged in.
2019-02-06 05:45:05 +08:00
window.location.reload(true);
}
});
}
2018-02-15 02:16:14 +08:00
function setNavBar(data) {
2020-09-18 01:51:31 +08:00
if ( !data ) {
console.error("No data in setNavBar");
return;
}
if ( data.auth ) {
2018-02-15 02:16:14 +08:00
if ( data.auth != auth_hash ) {
console.log("Update auth_hash to "+data.auth);
2018-02-15 02:16:14 +08:00
// Update authentication token.
auth_hash = data.auth;
}
}
if ( data.auth_relay ) {
auth_relay = data.auth_relay;
}
2020-08-03 02:33:25 +08:00
// iterate through all the keys then update each element id with the same name
for (var key of Object.keys(data)) {
if ( key == "auth" ) continue;
2020-08-03 23:43:24 +08:00
if ( $j('#'+key).hasClass("show") ) continue; // don't update if the user has the dropdown open
2020-08-03 02:33:25 +08:00
if ( $j('#'+key).length ) $j('#'+key).replaceWith(data[key]);
if ( key == 'getBandwidthHTML' ) bwClickFunction();
2020-08-03 02:33:25 +08:00
}
}
}
2013-03-17 07:45:21 +08:00
//Shows a message if there is an error in the streamObj or the stream doesn't exist. Returns true if error, false otherwise.
2017-06-06 03:22:32 +08:00
function checkStreamForErrors( funcName, streamObj ) {
if ( !streamObj ) {
Error( funcName+": stream object was null" );
return true;
}
if ( streamObj.result == "Error" ) {
Error( funcName+" stream error: "+streamObj.message );
return true;
}
return false;
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function secsToTime( seconds ) {
var timeString = "--";
if ( seconds < 60 ) {
timeString = seconds.toString();
} else if ( seconds < 60*60 ) {
var timeMins = parseInt(seconds/60);
var timeSecs = seconds%60;
if ( timeSecs < 10 ) {
2017-06-06 03:22:32 +08:00
timeSecs = '0'+timeSecs.toString().substr( 0, 4 );
} else {
2017-06-06 03:22:32 +08:00
timeSecs = timeSecs.toString().substr( 0, 5 );
}
2017-06-06 03:22:32 +08:00
timeString = timeMins+":"+timeSecs;
} else {
var timeHours = parseInt(seconds/3600);
var timeMins = (seconds%3600)/60;
var timeSecs = seconds%60;
if ( timeMins < 10 ) {
2017-06-06 03:22:32 +08:00
timeMins = '0'+timeMins.toString().substr( 0, 4 );
} else {
2017-06-06 03:22:32 +08:00
timeMins = timeMins.toString().substr( 0, 5 );
}
if ( timeSecs < 10 ) {
2017-06-06 03:22:32 +08:00
timeSecs = '0'+timeSecs.toString().substr( 0, 4 );
} else {
2017-06-06 03:22:32 +08:00
timeSecs = timeSecs.toString().substr( 0, 5 );
}
2017-06-06 03:22:32 +08:00
timeString = timeHours+":"+timeMins+":"+timeSecs;
}
return ( timeString );
2013-03-17 07:45:21 +08:00
}
function submitTab(evt) {
var tab = this.getAttribute("data-tab-name");
2020-12-24 02:29:39 +08:00
var form = $j('#contentForm');
form.attr('action', '');
form.attr('tab', tab);
2017-06-06 03:22:32 +08:00
form.submit();
evt.preventDefault();
2013-03-17 07:45:21 +08:00
}
function submitThisForm() {
if ( ! this.form ) {
console.log("No this.form. element with onchange is not in a form");
return;
}
this.form.submit();
}
/**
* @param {Element} headerCheckbox The select all/none checkbox that was just toggled.
* @param {DOMString} name The name of the checkboxes to toggle.
*/
function updateFormCheckboxesByName( headerCheckbox ) {
var name = headerCheckbox.getAttribute("data-checkbox-name");
var form = headerCheckbox.form;
var checked = headerCheckbox.checked;
for (var i = 0; i < form.elements.length; i++) {
if (form.elements[i].name.indexOf(name) == 0) {
form.elements[i].checked = checked;
}
}
setButtonStates(headerCheckbox);
}
2017-06-06 03:22:32 +08:00
function configureDeleteButton( element ) {
var form = element.form;
var checked = element.checked;
if ( !checked ) {
for ( var i = 0; i < form.elements.length; i++ ) {
if ( form.elements[i].name == element.name ) {
if ( form.elements[i].checked ) {
checked = true;
break;
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
}
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
}
form.deleteBtn.disabled = !checked;
2013-03-17 07:45:21 +08:00
}
function confirmDelete( message ) {
return ( confirm( message?message:'Are you sure you wish to delete?' ) );
2013-03-17 07:45:21 +08:00
}
window.addEventListener( 'DOMContentLoaded', checkSize );
function convertLabelFormat(LabelFormat, monitorName) {
//convert label format from strftime to moment's format (modified from
//https://raw.githubusercontent.com/benjaminoakes/moment-strftime/master/lib/moment-strftime.js
//added %f and %N below (TODO: add %Q)
var replacements = {
2020-04-17 23:04:07 +08:00
'a': 'ddd',
'A': 'dddd',
'b': 'MMM',
'B': 'MMMM',
'd': 'DD',
'e': 'D',
'F': 'YYYY-MM-DD',
'H': 'HH',
'I': 'hh',
'j': 'DDDD',
'k': 'H',
'l': 'h',
'm': 'MM',
'M': 'mm',
'p': 'A',
'r': 'hh:mm:ss A',
'S': 'ss',
'u': 'E',
'w': 'd',
'W': 'WW',
'y': 'YY',
'Y': 'YYYY',
'z': 'ZZ',
'Z': 'z',
'f': 'SS',
'N': '['+monitorName+']',
'%': '%'};
var momentLabelFormat = Object.keys(replacements).reduce(function(momentFormat, key) {
var value = replacements[key];
return momentFormat.replace('%' + key, value);
}, LabelFormat);
return momentLabelFormat;
}
function addVideoTimingTrack(video, LabelFormat, monitorName, duration, startTime) {
//This is a hacky way to handle changing the texttrack. If we ever upgrade vjs in a revamp replace this. Old method preserved because it's the right way.
var cues = vid.textTracks()[0].cues();
var labelFormat = convertLabelFormat(LabelFormat, monitorName);
startTime = moment(startTime);
for ( var i = 0; i <= duration; i++ ) {
cues[i] = {id: i, index: i, startTime: i, endTime: i+1, text: startTime.format(labelFormat)};
startTime.add(1, 's');
}
}
/*
var labelFormat = convertLabelFormat(LabelFormat, monitorName);
var webvttformat = 'HH:mm:ss.SSS', webvttdata="WEBVTT\n\n";
startTime = moment(startTime);
var seconds = moment({s:0}), endduration = moment({s:duration});
while(seconds.isBefore(endduration)){
webvttdata += seconds.format(webvttformat) + " --> ";
seconds.add(1,'s');
webvttdata += seconds.format(webvttformat) + "\n";
webvttdata += startTime.format(labelFormat) + "\n\n";
startTime.add(1, 's');
}
var track = document.createElement('track');
track.kind = "captions";
track.srclang = "en";
track.label = "English";
track['default'] = true;
track.src = 'data:plain/text;charset=utf-8,'+encodeURIComponent(webvttdata);
video.appendChild(track);
}
*/
2017-11-21 02:36:45 +08:00
var resizeTimer;
function endOfResize(e) {
clearTimeout(resizeTimer);
resizeTimer = setTimeout(changeScale, 250);
}
function scaleToFit(baseWidth, baseHeight, scaleEl, bottomEl) {
$j(window).on('resize', endOfResize); //set delayed scaling when Scale to Fit is selected
var ratio = baseWidth / baseHeight;
var container = $j('#content');
var viewPort = $j(window);
// jquery does not provide a bottom offet, and offset dows not include margins. outerHeight true minus false gives total vertical margins.
var bottomLoc = bottomEl.offset().top + (bottomEl.outerHeight(true) - bottomEl.outerHeight()) + bottomEl.outerHeight(true);
var newHeight = viewPort.height() - (bottomLoc - scaleEl.outerHeight(true));
var newWidth = ratio * newHeight;
2017-11-21 02:36:45 +08:00
if (newWidth > container.innerWidth()) {
newWidth = container.innerWidth();
newHeight = newWidth / ratio;
}
var autoScale = Math.round(newWidth / baseWidth * SCALE_BASE);
var scales = $j('#scale option').map(function() {
return parseInt($j(this).val());
}).get();
2017-11-21 02:36:45 +08:00
scales.shift();
var closest;
$j(scales).each(function() { //Set zms scale to nearest regular scale. Zoom does not like arbitrary scale values.
2017-11-21 02:36:45 +08:00
if (closest == null || Math.abs(this - autoScale) < Math.abs(closest - autoScale)) {
closest = this.valueOf();
}
});
autoScale = closest;
return {width: Math.floor(newWidth), height: Math.floor(newHeight), autoScale: autoScale};
}
2020-12-24 02:20:31 +08:00
function setButtonState(element_id, btnClass) {
var element = document.getElementById(element_id);
if ( element ) {
2020-12-24 02:20:31 +08:00
element.className = btnClass;
if (btnClass == 'unavail' || (btnClass == 'active' && (element.id == 'pauseBtn' || element.id == 'playBtn'))) {
element.disabled = true;
} else {
element.disabled = false;
}
} else {
console.log('Element was null or not found in setButtonState. id:'+element_id);
}
}
2020-08-08 23:07:26 +08:00
2020-08-09 00:19:29 +08:00
function setCookie(name, value, days) {
var expires = "";
if (days) {
var date = new Date();
date.setTime(date.getTime() + (days*24*60*60*1000));
expires = "; expires=" + date.toUTCString();
}
document.cookie = name + "=" + (value || "") + expires + "; path=/; samesite=strict";
2020-08-08 23:07:26 +08:00
}
2020-08-17 05:02:43 +08:00
function getCookie(name) {
2020-08-17 05:30:35 +08:00
var nameEQ = name + "=";
var ca = document.cookie.split(';');
for (var i=0; i < ca.length; i++) {
var c = ca[i];
while (c.charAt(0)==' ') c = c.substring(1, c.length);
if (c.indexOf(nameEQ) == 0) return c.substring(nameEQ.length, c.length);
}
return null;
2020-08-17 05:02:43 +08:00
}
2020-08-08 23:07:26 +08:00
function delCookie(name) {
document.cookie = name + '=; expires=Thu, 01 Jan 1970 00:00:01 GMT;';
}
function bwClickFunction() {
$j('.bwselect').click(function() {
var bwval = $j(this).data('pdsa-dropdown-val');
setCookie("zmBandwidth", bwval, 3600);
getNavBar();
});
}
2020-08-10 03:18:44 +08:00
function reminderClickFunction() {
$j("#dropdown_reminder a").click(function() {
var option = $j(this).data('pdsa-dropdown-val');
$j.getJSON(thisUrl + '?view=version&action=version&option=' + option)
.done(window.location.reload(true)) //Do a full refresh to update ZM_DYN_LAST_VERSION
.fail(logAjaxFail);
2020-08-10 03:18:44 +08:00
});
}
2020-09-14 00:04:04 +08:00
// Load then show the "You No Permission" error modal
function enoperm() {
$j.getJSON(thisUrl + '?request=modal&modal=enoperm')
.done(function(data) {
2020-10-15 04:58:39 +08:00
insertModalHtml('ENoPerm', data.html);
2020-09-14 00:04:04 +08:00
$j('#ENoPerm').modal('show');
// Manage the CLOSE optionhelp modal button
document.getElementById("enpCloseBtn").addEventListener("click", function onENPCloseClick(evt) {
$j('#ENoPerm').modal('hide');
});
})
.fail(logAjaxFail);
2020-09-14 00:04:04 +08:00
}
2020-09-19 23:44:05 +08:00
function getLogoutModal() {
$j.getJSON(thisUrl + '?request=modal&modal=logout')
.done(function(data) {
2020-10-15 04:58:39 +08:00
insertModalHtml('modalLogout', data.html);
manageModalBtns('modalLogout');
clickLogout();
2020-09-19 23:44:05 +08:00
})
.fail(logAjaxFail);
2020-09-19 23:44:05 +08:00
}
function clickLogout() {
if ( ! $j('#modalLogout').length ) {
getLogoutModal();
return;
}
$j('#modalLogout').modal('show');
}
2020-09-20 01:36:04 +08:00
function getStateModal() {
$j.getJSON(thisUrl + '?request=modal&modal=state')
.done(function(data) {
2020-10-15 04:58:39 +08:00
insertModalHtml('modalState', data.html);
2020-09-20 01:36:04 +08:00
$j('#modalState').modal('show');
manageStateModalBtns();
})
.fail(logAjaxFail);
2020-09-20 01:36:04 +08:00
}
function manageStateModalBtns() {
// Enable or disable the Delete button depending on the selected run state
$j("#runState").change(function() {
runstate = $j(this).val();
if ( (runstate == 'stop') || (runstate == 'restart') || (runstate == 'start') || (runstate == 'default') ) {
$j("#btnDelete").prop("disabled", true);
} else {
$j("#btnDelete").prop("disabled", false);
}
});
// Enable or disable the Save button when entering a new state
$j("#newState").keyup(function() {
length = $j(this).val().length;
if ( length < 1 ) {
$j("#btnSave").prop("disabled", true);
} else {
$j("#btnSave").prop("disabled", false);
}
});
// Delete a state
$j("#btnDelete").click(function() {
stateStuff('delete', $j("#runState").val());
});
// Save a new state
$j("#btnSave").click(function() {
stateStuff('save', undefined, $j("#newState").val());
});
// Change state
$j("#btnApply").click(function() {
stateStuff('state', $j("#runState").val());
});
}
function stateStuff(action, runState, newState) {
// the state action will redirect to console
var formData = {
'view': 'state',
'action': action,
'apply': 1,
'runState': runState,
'newState': newState
};
$j("#pleasewait").toggleClass("hidden");
$j.ajax({
type: 'POST',
url: thisUrl,
data: formData,
dataType: 'html',
timeout: 0
}).done(function(data) {
location.reload();
});
}
function logAjaxFail(jqxhr, textStatus, error) {
console.log("Request Failed: " + textStatus + ", " + error);
2020-10-23 23:12:57 +08:00
if ( ! jqxhr.responseText ) {
console.log("Ajax request failed. No responseText. jqxhr follows:");
console.log(jqxhr);
return;
}
var responseText = jqxhr.responseText.replace(/(<([^>]+)>)/gi, '').trim(); // strip any html or whitespace from the response
if ( responseText ) console.log("Response Text: " + responseText);
}
// Load the Modal HTML via Ajax call
function getModal(id) {
$j.getJSON(thisUrl + '?request=modal&modal='+id)
.done(function(data) {
if ( !data ) {
console.error("Get modal returned no data");
return;
}
2020-10-15 04:58:39 +08:00
insertModalHtml(id, data.html);
manageModalBtns(id);
modal = $j('#'+id+'Modal');
if ( ! modal.length ) {
console.log('No modal found');
}
$j('#'+id+'Modal').modal('show');
})
.fail(logAjaxFail);
}
function manageModalBtns(id) {
// Manage the CANCEL modal button, note data-dismiss="modal" would work better
var cancelBtn = document.getElementById(id+"CancelBtn");
if ( cancelBtn ) {
document.getElementById(id+"CancelBtn").addEventListener('click', function onCancelClick(evt) {
$j('#'+id).modal('hide');
});
}
// 'data-on-click-this' calls the global function in the attribute value with the element when a click happens.
document.querySelectorAll('#'+id+'Modal button[data-on-click]').forEach(function attachOnClick(el) {
var fnName = el.getAttribute('data-on-click');
if ( !window[fnName] ) {
console.error('Nothing found to bind to ' + fnName + ' on element ' + el.name);
return;
} else {
console.log("Setting onclick for " + el.name);
}
el.onclick = window[fnName].bind(el, el);
});
}
function human_filesize(size, precision = 2) {
var units = Array('B', 'kB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB');
var step = 1024;
var i = 0;
while ((size / step) > 0.9) {
size = size / step;
i++;
}
return (Math.round(size*(10^precision))/(10^precision))+units[i];
}
2020-09-30 22:12:54 +08:00
function startDownload( exportFile ) {
console.log("Starting download from " + exportFile);
window.location.replace( exportFile );
}
function exportResponse(data, responseText) {
2020-12-03 02:02:24 +08:00
console.log('exportResponse data: ' + JSON.stringify(data));
2020-09-30 22:25:52 +08:00
2020-09-30 22:12:54 +08:00
var generated = (data.result=='Ok') ? 1 : 0;
2020-12-03 02:02:24 +08:00
//var exportFile = '?view=archive&type='+data.exportFormat+'&connkey='+data.connkey;
var exportFile = data.exportFile;
2020-09-30 22:12:54 +08:00
$j('#exportProgress').removeClass( 'text-warning' );
if ( generated ) {
$j('#downloadLink').text('Download');
$j('#downloadLink').attr("href", thisUrl + exportFile);
$j('#exportProgress').addClass( 'text-success' );
$j('#exportProgress').text(exportSucceededString);
2021-01-09 02:25:17 +08:00
setTimeout(startDownload, 1500, exportFile);
2020-09-30 22:12:54 +08:00
} else {
$j('#exportProgress').addClass( 'text-danger' );
$j('#exportProgress').text(exportFailedString);
}
}
2020-09-30 22:12:54 +08:00
function exportEvent() {
var form = $j('#downloadForm').serialize();
$j.getJSON(thisUrl + '?view=request&request=event&action=download', form)
.done(exportResponse)
2020-09-30 22:25:52 +08:00
.fail(logAjaxFail);
2020-09-30 22:12:54 +08:00
$j('#exportProgress').removeClass( 'invisible' );
}
2020-10-13 01:16:52 +08:00
2020-10-14 23:03:33 +08:00
// Loads the shutdown modal
2020-10-13 01:16:52 +08:00
function getShutdownModal() {
$j.getJSON(thisUrl + '?request=modal&modal=shutdown')
.done(function(data) {
2020-10-15 04:58:39 +08:00
insertModalHtml('shutdownModal', data.html);
2020-10-14 23:03:33 +08:00
dataOnClickThis();
2020-10-13 01:16:52 +08:00
$j('#shutdownModal').modal('show');
})
.fail(logAjaxFail);
}
2020-10-14 23:03:33 +08:00
function manageShutdownBtns(element) {
var cmd = element.getAttribute('data-command');
var when = $j('#when1min').is(':checked') ? '1min' : 'now';
var respText = $j('#respText');
2020-10-15 00:20:37 +08:00
$j.getJSON(thisUrl + '?request=shutdown&when=' + when + '&command=' + cmd)
.done(function(data) {
respText.removeClass('invisible');
if ( data.rc ) {
respText.html('<h2>Error</h2>' + data.output);
} else {
$j('#cancelBtn').prop('disabled', false);
if ( cmd == 'cancel' ) {
respText.html('<h2>Success</h2>Event has been cancelled');
2020-10-14 23:03:33 +08:00
} else {
2020-10-15 00:20:37 +08:00
respText.html('<h2>Success</h2>You may cancel this shutdown by clicking ' + cancelString);
2020-10-14 23:03:33 +08:00
}
2020-10-15 00:20:37 +08:00
}
})
.fail(logAjaxFail);
2020-10-14 23:03:33 +08:00
}
function thumbnail_onmouseover(event) {
2020-12-05 08:30:05 +08:00
timeout = setTimeout(function() {
var img = event.target;
2020-12-05 04:47:13 +08:00
var imgClass = ( currentView == 'console' ) ? 'zoom-console' : 'zoom';
var imgAttr = ( currentView == 'frames' ) ? 'full_img_src' : 'stream_src';
img.src = '';
img.src = img.getAttribute(imgAttr);
img.addClass(imgClass);
}, 350);
}
function thumbnail_onmouseout(event) {
clearTimeout(timeout);
var img = event.target;
2020-12-05 04:47:13 +08:00
var imgClass = ( currentView == 'console' ) ? 'zoom-console' : 'zoom';
var imgAttr = ( currentView == 'frames' ) ? 'img_src' : 'still_src';
img.src = '';
img.src = img.getAttribute(imgAttr);
img.removeClass(imgClass);
}
function initThumbAnimation() {
if ( ANIMATE_THUMBS ) {
$j('.colThumbnail img').each(function() {
this.addEventListener('mouseover', thumbnail_onmouseover, false);
this.addEventListener('mouseout', thumbnail_onmouseout, false);
});
}
}