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
//
2017-06-16 23:10:25 +08:00
var popupOptions = "resizable,scrollbars,status=no,toolbar=yes" ;
2013-03-17 07:45:21 +08:00
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
}
2013-03-17 07:45:21 +08:00
// Deprecated
2017-06-06 03:22:32 +08:00
function newWindow ( url , name , width , height ) {
2019-01-19 23:32:40 +08:00
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 ) {
2017-10-03 00:43:52 +08:00
if ( typeof popupSizes == 'undefined' ) {
2019-07-04 21:04:43 +08:00
Error ( "Can't find any window sizes" ) ;
return { 'width' : 0 , 'height' : 0 } ;
2017-10-03 00:30:30 +08:00
}
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 ) {
2019-01-19 23:32:40 +08:00
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-01-19 23:32:40 +08:00
}
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 ;
2019-01-19 23:32:40 +08:00
if ( ! width ) {
2019-07-04 21:04:43 +08:00
Error ( "Got addWidth but no passed width when getting popup size for tag '" + tag + "'" ) ;
2019-01-19 23:32:40 +08:00
} else {
2017-06-06 03:22:32 +08:00
popupSize . width += parseInt ( width ) ;
2019-01-19 23:32:40 +08:00
}
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 ;
2019-01-19 23:32:40 +08:00
if ( ! height ) {
2019-07-04 21:04:43 +08:00
Error ( "Got addHeight but no passed height when getting popup size for tag '" + tag + "'" ) ;
2019-01-19 23:32:40 +08:00
} else {
2017-06-06 03:22:32 +08:00
popupSize . height += parseInt ( height ) ;
2019-01-19 23:32:40 +08:00
}
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
}
2019-11-05 23:07:42 +08:00
function zmWindow ( sub _url ) {
2020-02-22 23:50:00 +08:00
var zmWin = window . open ( 'https://www.zoneminder.com' + ( sub _url ? 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 = "" ;
2019-01-19 23:32:40 +08:00
if ( popupSize . width > 0 ) {
2017-06-06 03:22:32 +08:00
popupDimensions += ",width=" + popupSize . width ;
2019-01-19 23:32:40 +08:00
}
if ( popupSize . height > 0 ) {
2017-06-06 03:22:32 +08:00
popupDimensions += ",height=" + popupSize . height ;
2019-01-19 23:32:40 +08:00
}
2017-12-05 22:39:18 +08:00
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
}
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
} ) ;
2019-01-22 21:00:40 +08:00
document . querySelectorAll ( ".popup-link" ) . forEach ( function ( el ) {
el . addEventListener ( "click" , function onClick ( evt ) {
var el = this ;
var url ;
2019-07-23 22:03:28 +08:00
if ( el . hasAttribute ( "href" ) ) {
2019-01-22 21:00:40 +08:00
// <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 ( ) ;
2019-02-11 05:10:36 +08:00
createPopup ( url , name , tag , width , height ) ;
2019-01-22 21:00:40 +08:00
} ) ;
2019-01-15 22:01:58 +08:00
} ) ;
2019-01-16 22:59:58 +08:00
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
2019-01-16 22:59:58 +08:00
// '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" ) ;
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 ) ;
} ) ;
// '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 ) {
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
} ;
} ) ;
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" ) ;
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 ) ;
} ;
} ) ;
2019-01-18 22:51:06 +08:00
// '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" ) ;
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 ) ;
} ) ;
2019-01-16 22:59:58 +08:00
// '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" ) ;
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 ] ;
} ) ;
2019-11-02 01:38:09 +08:00
// '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 ;
}
2020-07-30 04:40:57 +08:00
el . oninput = window [ fnName ] . bind ( el , el ) ;
2019-11-02 01:38:09 +08:00
} ) ;
2019-01-15 22:01:58 +08:00
} ) ;
2017-06-06 03:22:32 +08:00
function createEventPopup ( eventId , eventFilter , width , height ) {
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
}
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
}
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-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-08-09 01:26:42 +08:00
Cookie . write ( 'zmHeaderFlip' , 'down' , { duration : 10 * 365 } ) ;
2020-08-09 01:06:57 +08:00
} else {
flip . html ( 'keyboard_arrow_up' ) ;
Cookie . write ( 'zmHeaderFlip' , 'up' , { duration : 10 * 365 } ) ;
}
} ) ;
// 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-08-09 01:26:42 +08:00
Cookie . write ( 'zmFilterBarFlip' , 'down' , { duration : 10 * 365 } ) ;
2020-08-09 01:06:57 +08:00
} else {
fbflip . html ( 'keyboard_arrow_up' ) ;
Cookie . write ( 'zmFilterBarFlip' , 'up' , { duration : 10 * 365 } ) ;
$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-08-27 22:54:54 +08:00
Cookie . write ( 'zmMonitorFilterBarFlip' , 'up' , { duration : 10 * 365 } ) ;
2020-08-27 22:31:06 +08:00
} else {
mfbflip . html ( 'keyboard_arrow_up' ) ;
2020-08-27 22:54:54 +08:00
Cookie . write ( 'zmMonitorFilterBarFlip' , 'down' , { duration : 10 * 365 } ) ;
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 ( ) ;
}
} ) ;
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 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 ) ;
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 ) {
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
}
2017-03-28 01:13:08 +08:00
if ( refreshParent ) {
2017-06-06 03:22:32 +08:00
refreshParentWindow ( ) ;
2013-03-17 07:45:21 +08:00
}
2017-03-28 01:13:08 +08:00
if ( focusWindow ) {
2017-06-06 03:22:32 +08:00
windowToFront ( ) ;
2013-03-17 07:45:21 +08:00
}
2017-07-06 22:48:06 +08:00
if ( closePopup ) {
closeWindow ( ) ;
}
2017-03-28 01:13:08 +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 ( ) ;
}
document . cookie = name + "=" + ( value || "" ) + expires + "; path=/" ;
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' ) ;
$j . getJSON ( thisUrl + '?view=version&action=version&option=' + option ) ;
2020-08-10 06:43:39 +08:00
window . location . reload ( true ) ; //Do a full refresh to update ZM_DYN_LAST_VERSION
2020-08-10 03:18:44 +08:00
} ) ;
}