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 ) {
2017-06-06 03:22:32 +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 ) {
var windowId = 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' ) {
2017-10-03 00:30:30 +08:00
Error ( "Can't find any window sizes" ) ;
return ( { 'width' : 0 , 'height' : 0 } ) ;
}
2017-06-06 03:22:32 +08:00
var popupSize = Object . clone ( popupSizes [ tag ] ) ;
if ( ! popupSize ) {
Error ( "Can't find window size for tag '" + tag + "'" ) ;
return ( { 'width' : 0 , 'height' : 0 } ) ;
}
if ( popupSize . width && popupSize . height ) {
if ( width || height )
Warning ( "Ignoring passed dimensions " + width + "x" + height + " when getting popup size for tag '" + tag + "'" ) ;
2013-03-17 07:45:21 +08:00
return ( popupSize ) ;
2017-06-06 03:22:32 +08:00
}
if ( popupSize . addWidth ) {
popupSize . width = popupSize . addWidth ;
if ( ! width )
Error ( "Got addWidth but no passed width when getting popup size for tag '" + tag + "'" ) ;
else
popupSize . width += parseInt ( width ) ;
} else if ( width ) {
popupSize . width = width ;
Error ( "Got passed width but no addWidth when getting popup size for tag '" + tag + "'" ) ;
}
if ( popupSize . minWidth && popupSize . width < popupSize . minWidth ) {
Warning ( "Adjusting to minimum width when getting popup size for tag '" + tag + "'" ) ;
popupSize . width = popupSize . minWidth ;
}
if ( popupSize . addHeight ) {
popupSize . height = popupSize . addHeight ;
if ( ! height )
Error ( "Got addHeight but no passed height when getting popup size for tag '" + tag + "'" ) ;
else
popupSize . height += parseInt ( height ) ;
} else if ( height ) {
popupSize . height = height ;
Error ( "Got passed height but no addHeight when getting popup size for tag '" + tag + "'" ) ;
}
2017-07-13 22:25:14 +08:00
if ( popupSize . minHeight && ( popupSize . height < popupSize . minHeight ) ) {
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 ;
}
return ( popupSize ) ;
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function zmWindow ( ) {
var zmWin = window . open ( 'http://www.zoneminder.com' , 'ZoneMinder' ) ;
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 )
popupDimensions += ",width=" + popupSize . width ;
if ( popupSize . height > 0 )
popupDimensions += ",height=" + popupSize . height ;
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-01-15 22:01:58 +08:00
$j ( document ) . ready ( function ( ) {
$j ( ".popup-link" ) . 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" ) ;
createPopup ( url , name , tag , width , height ) ;
evt . preventDefault ( ) ;
} ) ;
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" ) ;
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.
document . querySelectorAll ( "a[data-on-click], button[data-on-click], input[data-on-click]" ) . forEach ( function attachOnClick ( el ) {
var fnName = el . getAttribute ( "data-on-click" ) ;
el . onclick = function ( ) {
window [ fnName ] ( ) ;
} ;
} ) ;
// '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" ) ;
el . onchange = window [ fnName ] ;
} ) ;
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 ;
if ( eventFilter )
url += eventFilter ;
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
}
2017-03-28 01:13:08 +08:00
function refreshParentWindow ( ) {
2017-06-06 03:22:32 +08:00
if ( refreshParent ) {
if ( window . opener ) {
if ( refreshParent == true )
window . opener . location . reload ( true ) ;
else
window . opener . location . href = refreshParent ;
}
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' ) {
2017-12-05 06:04:53 +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 ( ) {
2018-02-15 02:16:14 +08:00
if ( $j ( '.navbar' ) . length ) {
setInterval ( getNavBar , navBarRefresh ) ;
}
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' )
. done ( setNavBar )
. fail ( function ( jqxhr , textStatus , error ) {
2018-10-11 23:30:30 +08:00
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.
window . location . reload ( true ) ;
}
2018-05-31 22:25:53 +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
}
2017-12-05 06:04:53 +08:00
$j ( '#reload' ) . replaceWith ( data . message ) ;
}
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 ;
if ( timeSecs < 10 )
timeSecs = '0' + timeSecs . toString ( ) . substr ( 0 , 4 ) ;
2013-03-17 07:45:21 +08:00
else
2017-06-06 03:22:32 +08:00
timeSecs = timeSecs . toString ( ) . substr ( 0 , 5 ) ;
timeString = timeMins + ":" + timeSecs ;
} else {
var timeHours = parseInt ( seconds / 3600 ) ;
var timeMins = ( seconds % 3600 ) / 60 ;
var timeSecs = seconds % 60 ;
if ( timeMins < 10 )
timeMins = '0' + timeMins . toString ( ) . substr ( 0 , 4 ) ;
else
timeMins = timeMins . toString ( ) . substr ( 0 , 5 ) ;
if ( timeSecs < 10 )
timeSecs = '0' + timeSecs . toString ( ) . substr ( 0 , 4 ) ;
else
timeSecs = timeSecs . toString ( ) . substr ( 0 , 5 ) ;
timeString = timeHours + ":" + timeMins + ":" + timeSecs ;
}
return ( timeString ) ;
2013-03-17 07:45:21 +08:00
}
2017-06-06 03:22:32 +08:00
function submitTab ( tab ) {
var form = $ ( 'contentForm' ) ;
form . action . value = "" ;
form . tab . value = tab ;
form . submit ( ) ;
2013-03-17 07:45:21 +08:00
}
2019-01-16 22:59:58 +08:00
function submitThisForm ( ) {
this . form . submit ( ) ;
}
2017-10-06 04:11:21 +08:00
function toggleCheckbox ( element , name ) {
var form = element . form ;
var checked = element . checked ;
for ( var i = 0 ; i < form . elements . length ; i ++ )
if ( form . elements [ i ] . name . indexOf ( name ) == 0 )
form . elements [ i ] . checked = checked ;
}
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 ) {
2017-06-06 03:22:32 +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
2017-07-06 22:48:06 +08:00
window . addEvent ( 'domready' , checkSize ) ;
2014-10-21 03:32:30 +08:00
2015-11-19 13:21:56 +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)
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 ) {
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 ) ;
2018-12-04 04:17:16 +08:00
for ( var i = 0 ; i <= duration ; i ++ ) {
2017-10-22 10:08:18 +08:00
cues [ i ] = { id : i , index : i , startTime : i , Ca : i + 1 , text : startTime . format ( labelFormat ) } ;
startTime . add ( 1 , 's' ) ;
}
}
/ *
2015-11-19 13:21:56 +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 ) ;
}
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 ) ;
}
2017-11-22 11:46:45 +08:00
function scaleToFit ( baseWidth , baseHeight , scaleEl , bottomEl ) {
2017-11-21 02:36:45 +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 ) ;
2017-11-21 02:36:45 +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 ) ;
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 ;
}
2018-12-04 04:17:16 +08:00
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 ( ) ;
2018-12-04 04:17:16 +08:00
var closest ;
2017-11-21 02:36:45 +08:00
$j ( scales ) . each ( function ( ) { //Set zms scale to nearest regular scale. Zoom does not like arbitrary scale values.
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 } ;
}