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

533 lines
18 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
//
2017-06-16 23:10:25 +08:00
var popupOptions = "resizable,scrollbars,status=no,toolbar=yes";
2013-03-17 07:45:21 +08:00
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
}
}
2013-03-17 07:45:21 +08:00
// Deprecated
2017-06-06 03:22:32 +08:00
function newWindow( url, name, width, height ) {
window.open( url, name, popupOptions+",width="+width+",height="+height );
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function getPopupSize( tag, width, height ) {
if ( typeof popupSizes == 'undefined' ) {
2019-07-04 21:04:43 +08:00
Error("Can't find any window sizes");
return {'width': 0, 'height': 0};
}
2019-07-04 21:04:43 +08:00
var popupSize = Object.clone(popupSizes[tag]);
2017-06-06 03:22:32 +08:00
if ( !popupSize ) {
2019-07-04 21:04:43 +08:00
Error("Can't find window size for tag '"+tag+"'");
return {'width': 0, 'height': 0};
2017-06-06 03:22:32 +08:00
}
if ( popupSize.width && popupSize.height ) {
if ( width || height ) {
2019-07-04 21:04:43 +08:00
Warning("Ignoring passed dimensions "+width+"x"+height+" when getting popup size for tag '"+tag+"'");
}
2019-07-04 21:04:43 +08:00
return popupSize;
2017-06-06 03:22:32 +08:00
}
if ( popupSize.addWidth ) {
popupSize.width = popupSize.addWidth;
if ( !width ) {
2019-07-04 21:04:43 +08:00
Error("Got addWidth but no passed width when getting popup size for tag '"+tag+"'");
} else {
2017-06-06 03:22:32 +08:00
popupSize.width += parseInt(width);
}
2017-06-06 03:22:32 +08:00
} else if ( width ) {
popupSize.width = width;
2019-07-04 21:04:43 +08:00
Error("Got passed width but no addWidth when getting popup size for tag '"+tag+"'");
2017-06-06 03:22:32 +08:00
}
if ( popupSize.minWidth && popupSize.width < popupSize.minWidth ) {
2019-07-04 21:04:43 +08:00
Warning("Adjusting to minimum width when getting popup size for tag '"+tag+"'");
2017-06-06 03:22:32 +08:00
popupSize.width = popupSize.minWidth;
}
if ( popupSize.addHeight ) {
popupSize.height = popupSize.addHeight;
if ( !height ) {
2019-07-04 21:04:43 +08:00
Error("Got addHeight but no passed height when getting popup size for tag '"+tag+"'");
} else {
2017-06-06 03:22:32 +08:00
popupSize.height += parseInt(height);
}
2017-06-06 03:22:32 +08:00
} else if ( height ) {
popupSize.height = height;
2019-07-04 21:04:43 +08:00
Error("Got passed height but no addHeight when getting popup size for tag '"+tag+"'");
2017-06-06 03:22:32 +08:00
}
2017-07-13 22:25:14 +08:00
if ( popupSize.minHeight && ( popupSize.height < popupSize.minHeight ) ) {
2019-07-04 21:04:43 +08:00
Warning("Adjusting to minimum height ("+popupSize.minHeight+") when getting popup size for tag '"+tag+"' because calculated height is " + popupSize.height);
2017-06-06 03:22:32 +08:00
popupSize.height = popupSize.minHeight;
}
2019-07-04 21:04:43 +08:00
return popupSize;
2013-03-17 07:45:21 +08:00
}
function zmWindow(sub_url) {
var zmWin = window.open( 'https://www.zoneminder.com'+sub_url, 'ZoneMinder' );
2017-06-06 03:22:32 +08:00
if ( ! zmWin ) {
// if popup blocking is enabled, the popup won't be defined.
console.log("Please disable popup blocking.");
} else {
zmWin.focus();
}
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function createPopup( url, name, tag, width, height ) {
var popupSize = getPopupSize( tag, width, height );
var popupDimensions = "";
if ( popupSize.width > 0 ) {
2017-06-06 03:22:32 +08:00
popupDimensions += ",width="+popupSize.width;
}
if ( popupSize.height > 0 ) {
2017-06-06 03:22:32 +08:00
popupDimensions += ",height="+popupSize.height;
}
var popup = window.open( url+"&popup=1", name, popupOptions+popupDimensions );
2017-06-06 03:22:32 +08:00
if ( ! popup ) {
// if popup blocking is enabled, the popup won't be defined.
console.log("Please disable popup blocking.");
} else {
popup.focus();
}
2013-03-17 07:45:21 +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();
}
});
});
document.querySelectorAll(".popup-link").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");
}
var name = el.getAttribute("data-window-name");
var tag = el.getAttribute("data-window-tag");
var width = el.getAttribute("data-window-width");
var height = el.getAttribute("data-window-height");
evt.preventDefault();
createPopup(url, name, tag, width, height);
});
});
document.querySelectorAll(".tabList a").forEach(function addOnClick(el) {
el.addEventListener("click", submitTab);
});
// 'data-on-click-this' calls the global function in the attribute value with the element when a click happens.
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.
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() {
window[fnName]();
};
});
2019-02-06 05:45:05 +08:00
// 'data-on-click-true' calls the global function in the attribute value with no arguments when a click happens.
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);
};
});
// 'data-on-change-this' calls the global function in the attribute value with the element when a change happens.
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.
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.
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.
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.onchange = window[fnName].bind(el, el);
});
});
2017-06-06 03:22:32 +08:00
function createEventPopup( eventId, eventFilter, width, height ) {
var url = '?view=event&eid='+eventId;
if ( eventFilter ) {
2017-06-06 03:22:32 +08:00
url += eventFilter;
}
2017-06-06 03:22:32 +08:00
var name = 'zmEvent';
var popupSize = getPopupSize( 'event', width, height );
var popup = window.open( url, name, popupOptions+",width="+popupSize.width+",height="+popupSize.height );
if ( ! popup ) {
// if popup blocking is enabled, the popup won't be defined.
console.log("Please disable popup blocking.");
} else {
popup.focus();
}
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function createFramesPopup( eventId, width, height ) {
var url = '?view=frames&eid='+eventId;
var name = 'zmFrames';
var popupSize = getPopupSize( 'frames', width, height );
var popup = window.open( url, name, popupOptions+",width="+popupSize.width+",height="+popupSize.height );
if ( ! popup ) {
// if popup blocking is enabled, the popup won't be defined.
console.log("Please disable popup blocking.");
} else {
popup.focus();
}
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function createFramePopup( eventId, frameId, width, height ) {
var url = '?view=frame&eid='+eventId+'&fid='+frameId;
var name = 'zmFrame';
var popupSize = getPopupSize( 'frame', width, height );
var popup = window.open( url, name, popupOptions+",width="+popupSize.width+",height="+popupSize.height );
if ( ! popup ) {
// if popup blocking is enabled, the popup won't be defined.
console.log("Please disable popup blocking.");
} else {
popup.focus();
}
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() {
2019-02-06 05:45:05 +08:00
if ( $j('.navbar').length ) {
2018-02-15 02:16:14 +08:00
setInterval(getNavBar, navBarRefresh);
}
});
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);
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) {
if ( data.auth ) {
2018-02-15 02:16:14 +08:00
if ( data.auth != auth_hash ) {
// Update authentication token.
auth_hash = data.auth;
}
}
$j('#reload').replaceWith(data.message);
}
}
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");
2017-06-06 03:22:32 +08:00
var form = $('contentForm');
form.action.value = "";
form.tab.value = tab;
form.submit();
evt.preventDefault();
2013-03-17 07:45:21 +08:00
}
function submitThisForm() {
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
}
if ( refreshParent ) {
2017-06-06 03:22:32 +08:00
refreshParentWindow();
2013-03-17 07:45:21 +08:00
}
if ( focusWindow ) {
2017-06-06 03:22:32 +08:00
windowToFront();
2013-03-17 07:45:21 +08:00
}
if ( closePopup ) {
closeWindow();
}
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 = {"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', "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};
}