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
2016-12-26 23:23:16 +08:00
// 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
//
2020-09-24 21:25:39 +08:00
// 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'
} ;
2014-04-07 18:07:29 +08:00
function checkSize ( ) {
2017-11-10 04:03:23 +08:00
if ( 0 ) {
2019-01-19 23:32:40 +08:00
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
}
2014-04-07 18:07:29 +08:00
}
2019-02-09 13:54:23 +08:00
// Polyfill for NodeList.prototype.forEach on IE.
if ( window . NodeList && ! NodeList . prototype . forEach ) {
NodeList . prototype . forEach = Array . prototype . forEach ;
}
2019-01-22 21:00:40 +08:00
window . addEventListener ( "DOMContentLoaded" , function onSkinDCL ( ) {
document . querySelectorAll ( "form.validateFormOnSubmit" ) . forEach ( function ( el ) {
el . addEventListener ( "submit" , function onSubmit ( evt ) {
if ( ! validateForm ( this ) ) {
evt . preventDefault ( ) ;
}
} ) ;
2019-01-19 22:41:53 +08:00
} ) ;
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 ) {
2019-01-22 21:00:40 +08:00
el . addEventListener ( "click" , submitTab ) ;
} ) ;
2019-01-19 22:42:12 +08:00
2020-10-14 21:51:44 +08:00
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 ( ) {
2019-01-16 22:59:58 +08:00
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" ) ;
2019-07-23 22:03:28 +08:00
if ( ! window [ fnName ] ) {
2019-12-03 01:17:18 +08:00
console . error ( "Nothing found to bind to " + fnName + " on element " + el . name ) ;
2019-07-23 22:03:28 +08:00
return ;
}
2019-01-16 22:59:58 +08:00
el . onclick = window [ fnName ] . bind ( el , el ) ;
} ) ;
2020-10-14 21:51:44 +08:00
}
2019-01-16 22:59:58 +08:00
2020-10-14 21:51:44 +08:00
// '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 ) {
2019-01-16 22:59:58 +08:00
var fnName = el . getAttribute ( "data-on-click" ) ;
2019-07-23 22:03:28 +08:00
if ( ! window [ fnName ] ) {
2019-12-03 01:17:18 +08:00
console . error ( "Nothing found to bind to " + fnName + " on element " + el . name ) ;
2019-07-23 22:03:28 +08:00
return ;
2020-01-01 09:24:51 +08:00
}
2020-01-01 08:11:14 +08:00
2020-04-30 04:32:55 +08:00
el . onclick = function ( ev ) {
window [ fnName ] ( ev ) ;
2019-01-16 22:59:58 +08:00
} ;
} ) ;
2020-10-14 21:51:44 +08:00
}
2019-01-16 22:59:58 +08:00
2020-10-14 21:51:44 +08:00
// '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" ) ;
2019-07-23 22:03:28 +08:00
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 ) ;
} ;
} ) ;
2020-10-14 21:51:44 +08:00
}
2019-02-06 05:45:05 +08:00
2020-10-14 21:51:44 +08:00
// 'data-on-change-this' calls the global function in the attribute value with the element when a change happens.
function dataOnChangeThis ( ) {
2019-01-18 22:51:06 +08:00
document . querySelectorAll ( "select[data-on-change-this], input[data-on-change-this]" ) . forEach ( function attachOnChangeThis ( el ) {
var fnName = el . getAttribute ( "data-on-change-this" ) ;
2019-07-23 22:03:28 +08:00
if ( ! window [ fnName ] ) {
console . error ( "Nothing found to bind to " + fnName ) ;
return ;
}
2019-01-18 22:51:06 +08:00
el . onchange = window [ fnName ] . bind ( el , el ) ;
} ) ;
2020-10-14 21:51:44 +08:00
}
2019-01-18 22:51:06 +08:00
2020-10-14 21:51:44 +08:00
// 'data-on-change' adds an event listener for the global function in the attribute value when a change happens.
function dataOnChange ( ) {
2019-01-16 22:59:58 +08:00
document . querySelectorAll ( "select[data-on-change], input[data-on-change]" ) . forEach ( function attachOnChange ( el ) {
var fnName = el . getAttribute ( "data-on-change" ) ;
2019-07-23 22:03:28 +08:00
if ( ! window [ fnName ] ) {
console . error ( "Nothing found to bind to " + fnName ) ;
return ;
}
2019-01-16 22:59:58 +08:00
el . onchange = window [ fnName ] ;
} ) ;
2020-10-14 21:51:44 +08:00
}
2019-11-02 01:38:09 +08:00
2020-10-14 21:51:44 +08:00
// 'data-on-input' adds an event listener for the global function in the attribute value when an input happens.
function dataOnInput ( ) {
2019-11-02 01:38:09 +08:00
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 ] ;
} ) ;
2020-10-14 21:51:44 +08:00
}
2019-11-02 01:38:09 +08:00
2020-10-14 21:51:44 +08:00
// 'data-on-input-this' calls the global function in the attribute value with the element when an input happens.
function dataOnInputThis ( ) {
2019-11-02 01:38:09 +08:00
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 ;
}
2020-07-30 04:40:57 +08:00
el . oninput = window [ fnName ] . bind ( el , el ) ;
2019-11-02 01:38:09 +08:00
} ) ;
2020-10-14 21:51:44 +08:00
}
2019-01-15 22:01:58 +08:00
2020-10-13 02:26:20 +08:00
function openEvent ( eventId , eventFilter ) {
2017-06-06 03:22:32 +08:00
var url = '?view=event&eid=' + eventId ;
2019-01-19 23:32:40 +08:00
if ( eventFilter ) {
2017-06-06 03:22:32 +08:00
url += eventFilter ;
2019-01-19 23:32:40 +08:00
}
2020-10-13 02:26:20 +08:00
window . location . assign ( url ) ;
2013-03-17 07:45:21 +08:00
}
2020-10-13 02:26:20 +08:00
function openFrames ( eventId ) {
2017-06-06 03:22:32 +08:00
var url = '?view=frames&eid=' + eventId ;
2020-10-13 02:26:20 +08:00
window . location . assign ( url ) ;
2013-03-17 07:45:21 +08:00
}
2020-10-13 02:26:20 +08:00
function openFrame ( eventId , frameId , width , height ) {
2017-06-06 03:22:32 +08:00
var url = '?view=frame&eid=' + eventId + '&fid=' + frameId ;
2020-10-13 02:26:20 +08:00
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
}
2019-09-29 00:57:43 +08:00
function backWindow ( ) {
window . history . back ( ) ;
}
2013-03-17 07:45:21 +08:00
2017-03-28 01:13:08 +08:00
function refreshParentWindow ( ) {
2017-06-06 03:22:32 +08:00
if ( refreshParent ) {
if ( window . opener ) {
2019-01-19 23:32:40 +08:00
if ( refreshParent == true ) {
2017-06-06 03:22:32 +08:00
window . opener . location . reload ( true ) ;
2019-01-19 23:32:40 +08:00
} else {
2017-06-06 03:22:32 +08:00
window . opener . location . href = refreshParent ;
2019-01-19 23:32:40 +08:00
}
2017-06-06 03:22:32 +08:00
}
2017-03-28 01:13:08 +08:00
}
2013-03-17 07:45:21 +08:00
}
2018-06-12 20:58:19 +08:00
if ( currentView != 'none' && currentView != 'login' ) {
2019-01-19 23:32:40 +08:00
$j . ajaxSetup ( { timeout : AJAX _TIMEOUT } ) ; //sets timeout for all getJSON.
2017-11-30 12:04:38 +08:00
2017-12-05 06:04:53 +08:00
$j ( document ) . ready ( function ( ) {
2020-09-20 01:36:04 +08:00
// Load the Logout and State modals into the dom
2020-09-19 23:44:05 +08:00
getLogoutModal ( ) ;
2020-09-20 01:36:04 +08:00
if ( canEditSystem ) $j ( '#stateModalBtn' ) . click ( getStateModal ) ;
2020-09-19 23:44:05 +08:00
2020-08-09 01:06:57 +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 ) ;
}
2020-08-09 01:06:57 +08:00
// 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 ( ) ;
2020-08-09 01:06:57 +08:00
// 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' ) ;
2020-10-07 23:26:18 +08:00
Cookie . write ( 'zmHeaderFlip' , 'down' , { duration : 10 * 365 , samesite : 'strict' } ) ;
2020-08-09 01:06:57 +08:00
} else {
flip . html ( 'keyboard_arrow_up' ) ;
2020-10-07 23:26:18 +08:00
Cookie . write ( 'zmHeaderFlip' , 'up' , { duration : 10 * 365 , samesite : 'strict' } ) ;
2020-08-09 01:06:57 +08:00
}
} ) ;
// 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' ) ;
2020-10-07 23:26:18 +08:00
Cookie . write ( 'zmFilterBarFlip' , 'down' , { duration : 10 * 365 , samesite : 'strict' } ) ;
2020-08-09 01:06:57 +08:00
} else {
fbflip . html ( 'keyboard_arrow_up' ) ;
2020-10-07 23:26:18 +08:00
Cookie . write ( 'zmFilterBarFlip' , 'up' , { duration : 10 * 365 , samesite : 'strict' } ) ;
2020-08-09 01:06:57 +08:00
$j ( '.chosen' ) . chosen ( "destroy" ) ;
$j ( '.chosen' ) . chosen ( ) ;
}
} ) ;
2020-08-27 22:31:06 +08:00
// 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' ) ;
2020-10-07 23:26:18 +08:00
Cookie . write ( 'zmMonitorFilterBarFlip' , 'up' , { duration : 10 * 365 , samesite : 'strict' } ) ;
2020-08-27 22:31:06 +08:00
} else {
mfbflip . html ( 'keyboard_arrow_up' ) ;
2020-10-07 23:26:18 +08:00
Cookie . write ( 'zmMonitorFilterBarFlip' , 'down' , { duration : 10 * 365 , samesite : 'strict' } ) ;
2020-08-27 22:31:06 +08:00
$j ( '.chosen' ) . chosen ( "destroy" ) ;
$j ( '.chosen' ) . chosen ( ) ;
}
} ) ;
2020-08-09 01:06:57 +08:00
// 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 )
2020-09-20 21:41:16 +08:00
. fail ( logAjaxFail ) ;
2020-09-11 05:56:58 +08:00
} ) ;
2017-12-05 06:04:53 +08:00
} ) ;
2017-11-30 12:04:38 +08:00
2020-09-11 05:56:58 +08:00
// Manage the modal html we received after user clicks help link
function optionhelpModal ( data ) {
if ( $j ( '#optionhelp' ) . length ) {
$j ( '#optionhelp' ) . replaceWith ( data . html ) ;
} else {
$j ( "body" ) . append ( data . html ) ;
}
$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 ( ) {
2018-05-31 22:25:53 +08:00
$j . getJSON ( thisUrl + '?view=request&request=status&entity=navBar' )
2019-01-19 23:32:40 +08:00
. done ( setNavBar )
2019-02-06 05:45:05 +08:00
. fail ( function ( jqxhr , textStatus , error ) {
console . log ( "Request Failed: " + textStatus + ", " + error ) ;
2020-09-20 21:41:16 +08:00
console . log ( "Response Text: " + jqxhr . responseText . replace ( /(<([^>]+)>)/gi , '' ) ) ;
2019-01-19 23:32:40 +08:00
if ( textStatus != "timeout" ) {
2018-10-11 23:30:30 +08:00
// 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 ) ;
2019-01-19 23:32:40 +08:00
}
} ) ;
2017-12-05 06:04:53 +08:00
}
2017-11-30 12:04:38 +08:00
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 ;
}
2018-02-14 23:44:48 +08:00
if ( data . auth ) {
2018-02-15 02:16:14 +08:00
if ( data . auth != auth _hash ) {
// Update authentication token.
auth _hash = data . auth ;
}
2018-02-14 23:44:48 +08:00
}
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 ] ) ;
2020-08-09 01:06:57 +08:00
if ( key == 'getBandwidthHTML' ) bwClickFunction ( ) ;
2020-08-03 02:33:25 +08:00
}
2017-12-05 06:04:53 +08:00
}
2017-11-30 12:04:38 +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 ;
2019-01-19 23:32:40 +08:00
if ( timeSecs < 10 ) {
2017-06-06 03:22:32 +08:00
timeSecs = '0' + timeSecs . toString ( ) . substr ( 0 , 4 ) ;
2019-01-19 23:32:40 +08:00
} else {
2017-06-06 03:22:32 +08:00
timeSecs = timeSecs . toString ( ) . substr ( 0 , 5 ) ;
2019-01-19 23:32:40 +08:00
}
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 ;
2019-01-19 23:32:40 +08:00
if ( timeMins < 10 ) {
2017-06-06 03:22:32 +08:00
timeMins = '0' + timeMins . toString ( ) . substr ( 0 , 4 ) ;
2019-01-19 23:32:40 +08:00
} else {
2017-06-06 03:22:32 +08:00
timeMins = timeMins . toString ( ) . substr ( 0 , 5 ) ;
2019-01-19 23:32:40 +08:00
}
if ( timeSecs < 10 ) {
2017-06-06 03:22:32 +08:00
timeSecs = '0' + timeSecs . toString ( ) . substr ( 0 , 4 ) ;
2019-01-19 23:32:40 +08:00
} else {
2017-06-06 03:22:32 +08:00
timeSecs = timeSecs . toString ( ) . substr ( 0 , 5 ) ;
2019-01-19 23:32:40 +08:00
}
2017-06-06 03:22:32 +08:00
timeString = timeHours + ":" + timeMins + ":" + timeSecs ;
}
2019-01-19 23:32:40 +08:00
return ( timeString ) ;
2013-03-17 07:45:21 +08:00
}
2019-01-19 22:42:12 +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 ( ) ;
2019-01-19 22:42:12 +08:00
evt . preventDefault ( ) ;
2013-03-17 07:45:21 +08:00
}
2019-01-16 22:59:58 +08:00
function submitThisForm ( ) {
2020-07-28 00:40:25 +08:00
if ( ! this . form ) {
console . log ( "No this.form. element with onchange is not in a form" ) ;
return ;
}
2019-01-16 22:59:58 +08:00
this . form . submit ( ) ;
}
2019-01-22 00:11:40 +08:00
/ * *
* @ 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 ;
2019-01-19 23:32:40 +08:00
for ( var i = 0 ; i < form . elements . length ; i ++ ) {
if ( form . elements [ i ] . name . indexOf ( name ) == 0 ) {
form . elements [ i ] . checked = checked ;
}
}
2019-01-22 00:11:40 +08:00
setButtonStates ( headerCheckbox ) ;
2017-10-06 04:11:21 +08:00
}
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
}
2017-03-28 01:13:08 +08:00
function confirmDelete ( message ) {
2019-01-19 23:32:40 +08:00
return ( confirm ( message ? message : 'Are you sure you wish to delete?' ) ) ;
2013-03-17 07:45:21 +08:00
}
2019-01-22 00:14:32 +08:00
window . addEventListener ( 'DOMContentLoaded' , checkSize ) ;
2014-10-21 03:32:30 +08:00
2019-01-18 23:00:55 +08:00
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)
2020-04-17 21:44:07 +08:00
var replacements = {
2020-04-17 23:04:07 +08:00
'a' : 'ddd' ,
2020-04-17 21:44:07 +08:00
'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 + ']' ,
'%' : '%' } ;
2019-01-18 23:00:55 +08:00
var momentLabelFormat = Object . keys ( replacements ) . reduce ( function ( momentFormat , key ) {
var value = replacements [ key ] ;
2020-04-17 21:44:07 +08:00
return momentFormat . replace ( '%' + key , value ) ;
2019-01-18 23:00:55 +08:00
} , LabelFormat ) ;
return momentLabelFormat ;
2015-11-19 13:21:56 +08:00
}
2019-01-18 23:00:55 +08:00
function addVideoTimingTrack ( video , LabelFormat , monitorName , duration , startTime ) {
2017-10-22 10:08:18 +08:00
//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.
2018-12-04 04:17:16 +08:00
var cues = vid . textTracks ( ) [ 0 ] . cues ( ) ;
var labelFormat = convertLabelFormat ( LabelFormat , monitorName ) ;
2017-10-22 10:08:18 +08:00
startTime = moment ( startTime ) ;
2020-04-17 21:44:07 +08:00
for ( var i = 0 ; i <= duration ; i ++ ) {
2019-01-22 01:01:46 +08:00
cues [ i ] = { id : i , index : i , startTime : i , endTime : i + 1 , text : startTime . format ( labelFormat ) } ;
2017-10-22 10:08:18 +08:00
startTime . add ( 1 , 's' ) ;
}
}
/ *
2019-01-19 23:32:40 +08:00
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 ) ;
2015-11-19 13:21:56 +08:00
}
2017-10-22 10:08:18 +08:00
* /
2017-10-01 02:19:32 +08:00
2017-11-21 02:36:45 +08:00
var resizeTimer ;
function endOfResize ( e ) {
clearTimeout ( resizeTimer ) ;
resizeTimer = setTimeout ( changeScale , 250 ) ;
}
2019-01-18 23:00:55 +08:00
function scaleToFit ( baseWidth , baseHeight , scaleEl , bottomEl ) {
2019-01-19 23:32:40 +08:00
$j ( window ) . on ( 'resize' , endOfResize ) ; //set delayed scaling when Scale to Fit is selected
2018-12-04 04:17:16 +08:00
var ratio = baseWidth / baseHeight ;
var container = $j ( '#content' ) ;
var viewPort = $j ( window ) ;
2019-01-19 23:32:40 +08:00
// jquery does not provide a bottom offet, and offset dows not include margins. outerHeight true minus false gives total vertical margins.
2018-12-04 04:17:16 +08:00
var bottomLoc = bottomEl . offset ( ) . top + ( bottomEl . outerHeight ( true ) - bottomEl . outerHeight ( ) ) + bottomEl . outerHeight ( true ) ;
2019-01-18 23:00:55 +08:00
var newHeight = viewPort . height ( ) - ( bottomLoc - scaleEl . outerHeight ( true ) ) ;
2018-12-04 04:17:16 +08:00
var newWidth = ratio * newHeight ;
2017-11-21 02:36:45 +08:00
if ( newWidth > container . innerWidth ( ) ) {
newWidth = container . innerWidth ( ) ;
newHeight = newWidth / ratio ;
}
2018-12-04 04:17:16 +08:00
var autoScale = Math . round ( newWidth / baseWidth * SCALE _BASE ) ;
2019-01-19 23:32:40 +08:00
var scales = $j ( '#scale option' ) . map ( function ( ) {
return parseInt ( $j ( this ) . val ( ) ) ;
} ) . get ( ) ;
2017-11-21 02:36:45 +08:00
scales . shift ( ) ;
2018-12-04 04:17:16 +08:00
var closest ;
2019-01-18 23:00:55 +08:00
$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-02-26 02:17:02 +08:00
2020-03-03 00:52:32 +08:00
function setButtonState ( element _id , butClass ) {
var element = $ ( element _id ) ;
2020-02-26 02:17:02 +08:00
if ( element ) {
element . className = butClass ;
if ( butClass == 'unavail' || ( butClass == 'active' && ( element . id == 'pauseBtn' || element . id == 'playBtn' ) ) ) {
element . disabled = true ;
} else {
element . disabled = false ;
}
} else {
2020-03-03 00:52:32 +08:00
console . log ( 'Element was null or not found in setButtonState. id:' + element _id ) ;
2020-02-26 02:17:02 +08:00
}
}
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 ( ) ;
}
2020-10-07 23:26:18 +08:00
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;' ;
}
2020-08-09 01:06:57 +08:00
function bwClickFunction ( ) {
$j ( "#dropdown_bandwidth a" ) . 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' ) ;
2020-09-12 03:30:36 +08:00
$j . getJSON ( thisUrl + '?view=version&action=version&option=' + option )
. done ( window . location . reload ( true ) ) //Do a full refresh to update ZM_DYN_LAST_VERSION
2020-09-20 21:41:16 +08:00
. 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 ) {
if ( $j ( '#ENoPerm' ) . length ) {
$j ( '#ENoPerm' ) . replaceWith ( data . html ) ;
} else {
$j ( "body" ) . append ( data . html ) ;
}
$j ( '#ENoPerm' ) . modal ( 'show' ) ;
// Manage the CLOSE optionhelp modal button
document . getElementById ( "enpCloseBtn" ) . addEventListener ( "click" , function onENPCloseClick ( evt ) {
$j ( '#ENoPerm' ) . modal ( 'hide' ) ;
} ) ;
} )
2020-09-20 21:41:16 +08:00
. 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 ) {
if ( $j ( '#modalLogout' ) . length ) {
$j ( '#modalLogout' ) . replaceWith ( data . html ) ;
} else {
$j ( "body" ) . append ( data . html ) ;
}
} )
2020-09-20 21:41:16 +08:00
. fail ( logAjaxFail ) ;
2020-09-19 23:44:05 +08:00
}
2020-09-20 01:36:04 +08:00
function getStateModal ( ) {
$j . getJSON ( thisUrl + '?request=modal&modal=state' )
. done ( function ( data ) {
if ( $j ( '#modalState' ) . length ) {
$j ( '#modalState' ) . replaceWith ( data . html ) ;
} else {
$j ( "body" ) . append ( data . html ) ;
}
$j ( '#modalState' ) . modal ( 'show' ) ;
manageStateModalBtns ( ) ;
} )
2020-09-20 21:41:16 +08:00
. 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 ( ) ;
} ) ;
}
2020-09-20 21:41:16 +08:00
function logAjaxFail ( jqxhr , textStatus , error ) {
2020-10-02 23:28:50 +08:00
var responseText = jqxhr . responseText . replace ( /(<([^>]+)>)/gi , '' ) . trim ( ) ; // strip any html or whitespace from the response
2020-09-20 21:41:16 +08:00
console . log ( "Request Failed: " + textStatus + ", " + error ) ;
2020-10-02 23:28:50 +08:00
if ( responseText ) console . log ( "Response Text: " + responseText ) ;
2020-09-20 21:41:16 +08:00
}
2020-09-22 01:00:55 +08:00
// 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 ;
}
if ( $j ( '#' + id ) . length ) {
$j ( '#' + id ) . replaceWith ( data . html ) ;
} else {
$j ( 'body' ) . append ( 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
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 ) ;
} ) ;
}
2020-09-23 04:21:59 +08:00
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 ) {
console . log ( data ) ;
2020-09-30 22:25:52 +08:00
2020-09-30 22:12:54 +08:00
var generated = ( data . result == 'Ok' ) ? 1 : 0 ;
var exportFile = '?view=archive&type=' + data . exportFormat + '&connkey=' + data . connkey ;
$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 ) ;
startDownload . pass ( exportFile ) . delay ( 1500 ) ;
} else {
$j ( '#exportProgress' ) . addClass ( 'text-danger' ) ;
$j ( '#exportProgress' ) . text ( exportFailedString ) ;
}
}
2020-09-23 04:21:59 +08:00
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 ) {
if ( $j ( '#shutdownModal' ) . length ) {
$j ( '#shutdownModal' ) . replaceWith ( data . html ) ;
} else {
$j ( "body" ) . append ( 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
}