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
|
|
|
|
//
|
|
|
|
|
|
|
|
var popupOptions = "resizable,scrollbars,status=no";
|
|
|
|
|
2014-04-07 18:07:29 +08:00
|
|
|
function checkSize() {
|
|
|
|
if (window.outerHeight) {
|
|
|
|
var w = window.outerWidth;
|
|
|
|
var prevW = w;
|
|
|
|
var h = window.outerHeight;
|
|
|
|
var prevH = h;
|
|
|
|
if (h > screen.availHeight)
|
2017-03-18 15:00:51 +08:00
|
|
|
h = screen.availHeight;
|
2014-04-07 18:07:29 +08:00
|
|
|
if (w > screen.availWidth)
|
2017-03-18 15:00:51 +08:00
|
|
|
w = screen.availWidth;
|
2014-04-07 18:07:29 +08:00
|
|
|
if (w != prevW || h != prevH)
|
2017-03-18 15:00:51 +08:00
|
|
|
window.resizeTo(w, h);
|
2014-04-07 18:07:29 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-17 07:45:21 +08:00
|
|
|
// Deprecated
|
|
|
|
function newWindow( url, name, width, height )
|
|
|
|
{
|
|
|
|
var windowId = window.open( url, name, popupOptions+",width="+width+",height="+height );
|
|
|
|
}
|
|
|
|
|
|
|
|
function getPopupSize( tag, width, height )
|
|
|
|
{
|
|
|
|
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 )
|
2017-03-18 15:00:51 +08:00
|
|
|
Warning( "Ignoring passed dimensions "+width+"x"+height+" when getting popup size for tag '"+tag+"'" );
|
2013-03-17 07:45:21 +08:00
|
|
|
return( popupSize );
|
|
|
|
}
|
|
|
|
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+"'" );
|
|
|
|
}
|
|
|
|
if ( popupSize.minHeight && popupSize.height < popupSize.minHeight )
|
|
|
|
{
|
|
|
|
Warning( "Adjusting to minimum height when getting popup size for tag '"+tag+"'" );
|
|
|
|
popupSize.height = popupSize.minHeight;
|
|
|
|
}
|
|
|
|
Debug( popupSize );
|
|
|
|
return( popupSize );
|
|
|
|
}
|
|
|
|
|
|
|
|
function zmWindow()
|
|
|
|
{
|
|
|
|
var zmWin = window.open( 'http://www.zoneminder.com', 'ZoneMinder' );
|
2016-10-11 23:48:29 +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
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
var popup = window.open( url, name, popupOptions+popupDimensions );
|
2016-10-11 23:48:29 +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
|
|
|
}
|
|
|
|
|
|
|
|
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 );
|
2016-10-11 23:48:29 +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
|
|
|
}
|
|
|
|
|
|
|
|
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 );
|
2016-10-11 23:48:29 +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
|
|
|
}
|
|
|
|
|
|
|
|
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 );
|
2016-10-11 23:48:29 +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
|
|
|
}
|
|
|
|
|
|
|
|
function windowToFront()
|
|
|
|
{
|
|
|
|
top.window.focus();
|
|
|
|
}
|
|
|
|
|
|
|
|
function closeWindow()
|
|
|
|
{
|
|
|
|
top.window.close();
|
|
|
|
}
|
|
|
|
|
|
|
|
function refreshWindow()
|
|
|
|
{
|
|
|
|
window.location.reload( true );
|
|
|
|
}
|
|
|
|
|
|
|
|
function refreshParentWindow()
|
|
|
|
{
|
|
|
|
if ( window.opener )
|
|
|
|
window.opener.location.reload( true );
|
|
|
|
}
|
|
|
|
|
|
|
|
//Shows a message if there is an error in the streamObj or the stream doesn't exist. Returns true if error, false otherwise.
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
|
|
|
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 );
|
|
|
|
else
|
|
|
|
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 );
|
|
|
|
}
|
|
|
|
|
|
|
|
function submitTab( tab )
|
|
|
|
{
|
|
|
|
var form = $('contentForm');
|
|
|
|
form.action.value = "";
|
|
|
|
form.tab.value = tab;
|
|
|
|
form.submit();
|
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
form.deleteBtn.disabled = !checked;
|
|
|
|
}
|
|
|
|
|
|
|
|
function confirmDelete( message )
|
|
|
|
{
|
|
|
|
return( confirm( message?message:'Are you sure you wish to delete?' ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( refreshParent )
|
|
|
|
{
|
|
|
|
refreshParentWindow();
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( focusWindow )
|
|
|
|
{
|
|
|
|
windowToFront();
|
|
|
|
}
|
2014-10-21 03:32:30 +08:00
|
|
|
window.addEvent( 'domready', checkSize);
|
|
|
|
|
Feature h264 videostorage (#1882)
* Moved writing of configure options from Controller to Model. Fixes #191.
* Initial commit for saving events as videos :)
* Add zm_video.cpp to autotools
* Add zm_video.h to autotools
* Search for MP4V2 header file 3 times: mp4v2/mp4v2.h, mp4v2.h, mp4.h
* Fix serve memory leak
* Few minor code improvements
* Added the ability to override preset, tune, profile and few other improvements
* Correctly write SPS & PPS from x264 encoder headers
* Remove unnessecary SPS & PPS writing code
* Imported missing files from master to feature-h264-videostorage
* Audio support including fixes for dts/pts, split on keyframe and update to mkv extension to prevent ffmpeg problems writing rtsp audio to mp4 containter (header problem)
* Updates to make gcc happy
* Add html5 video control to timeline and event to support mkv playback
* Add zm_videostore.cpp to CMakeLists.txt
* Remove Modern Branch for now
* Fix minor bug
* Option handled added in master, removing duplicate declaration
* Add CaptureandRecord from zm_camera.h
* Putting placeholder in for CaptureAndRecord function
* Removed duplicate code and brackets
* add digest auth file for cmake
Conflicts:
src/CMakeLists.txt
* Add web dir back into Makefile.am
Revert "Removed web from SUBDIRS in Makefile.am"
This reverts commit d9bbcdf3a94cba4d8955fcd03bd965ed2772f34d.
* Add CaptureAndRecord to vlc, still need to make it record
* Resolve SegFault on videostore
* Swap to mp4 container
* mp4 changes
* spaces to tabs, hide video stuff if video writer is turned off
* Make timeline open event.mp4 instead of mkv
* Missed mkv in timeline.js
* Fix some issues from the merge conflict
* Resolve post merge build issues with braces
* Fix whitespace
* Update Jpeg and Video options for passthrough options
* Whitespace fix zm_camera.h
* Fix array mkssing comma
* Add support for Jpeg save options for h264 branch snapshot. Might remove altogether if snapshots not needed
* Update VideoStoreData memory size comment
* Change from config.use_mkv_storage to per monitor option VideoWriter from video branch
* Fix bracket issues post merge
* Clean up comments and add av_free_packet
* Convert from event_directory to event file as per Video branch
* Testing videojs for video playback
* Fixed a missing bracket post merge and also SQL_values now used for EventID and Monitors
* bring recent improvements in ffmpeg capture function into captureandrecord
* Remove pict from writeAudioFramePacket as not used
* Add translate options for h264 Storage options in Monitor and update en_gb file
* Cherry-Pick from iconnor - make it compile on ubuntu 15.04. Which is libav 56.1.0
Conflicts:
src/zm_ffmpeg.cpp
src/zm_remote_camera_rtsp.cpp
Conflicts:
distros/ubuntu1204/changelog
* Clean up videostore code and remove lots of unused code
* proof of concept for dynamic/automatic video rotation using video-js plugin zoomrotate
Conflicts:
web/skins/classic/views/event.php
* removed redundant field in sql query
Conflicts:
web/skins/classic/views/event.php
* local storage of video js plugin
* Beautify!
Make the code somewhat readable.
* added missing videojs.zoomrotate.js file
added missing videojs.zoomrotate.js file
* Typo
added missing "
* Added missing brackets
* fix to display thumbnails when only storing snapshot.jpg
* added control for video playback rate
Conflicts:
web/skins/classic/views/event.php
* dynamically create jpegs from video file for viewing in browser
* fix timeline view for SaveJPEGs monitors (without enabled VideoWriter)
* only expose monitor info which are being used in client
* fix segmentation fault in zma with ubuntu 14.04 and ffmpeg 2.5.8 (gcc 4.8)
when libx264 is not installed
* better way of detecting showing image or video in timeline and event view
instead of Monitor.VideoWriter, Event.DefaultVideo is used, so even if
VideoWriter/SaveJPEG option is changed, a valid image or video will always be
displayed for historical events in both timeline and event view
this also fixes loading videos in timeline view
* Fixes problem of crashing zmc when bad packet arrives causing av_interleaved_write_frame() to return non-zero (-22). Prefilters common packet issues. Add metadata title to generated video file
* Remove syslog.h
* fixed SaveJPEGs are not working
which is caused in errors introduced when merging with master
* Update README.md
* Fix build warnings specific to h264 branch, unused FrameImg, unused ret and int64_t snprintf issues
* Fix PRId64 issue in travis, builds locally fine, but I can see a gcc version issue here
* Fix PRId64 issue in travis, another try
* Try "STDC_FORMAT_MACROS" to see if that helps Travis on gcc 4.6.3
* Revert space removal around PRId64
* video branch ffmpeg 2.9 fixes
ffmpeg 2.9 patched removed SSE2 CPU
* Add FFMPEGInit back
* use webvvt to overlay timestamp (honoring Monitor.LabelFormat) to videos in timeline and event
also fixed bug which prevented seeking in timeline video preview
* ffmpeg 3.0 API build failure fixes
* Update README.md
* merge all the commits from the messed up iconnor_video branch
* fix whitespace
* revert
* whitespace fixes
* spelling fix
* put back some text
* add these back
* fix spelling mistake
* Steal some packet dumping routines from ffmpeg. Convert them to use our logging routines
* add a test and error message if the codec is not h264
* these have been removed in master
* add a view to check auth and just send the video
* add some comments, and dump filename and AVFormatContext on failure to write header
* add the toggle for RecordAudio so that the checkbox works to turn off Audio
* Must init videoStore in constuctor
* more debug and comments, return checking
* Fix dropped part of sql query.
* fix extra else and some whitespace
* Fix missing } from merge that was preventing building.
* fix tabs
* get rid of use of separator, just use \n
* Restore lost fixes for deprecation
* Why are these failing
* Respect record_audio flag when setting up video file so dont try and initiliase mp4 with unsupported audio
* Forgot that I was trying to solve case of stream is true and record_audio
is false.
* Pass swscale_ctx back in to getCachedContext or it will create new
context every frame and leak memory like a mofo.
* Add libx264-dev and libmp4v2-dev to build requires to save hassle of
ensuring they are installed before build.
* Merge my Rotation/Orientation work and fixes for bad h264 streams
* need arpa/inet for reverse lookups
* pull in the new byte range code for viewing videos
* Move our recording flag deeper into closeevent
* add braces and only call closeEvent if there is an event
* deprecate the z_frame_rate stuff which is deprecated in ffmpeg
* remark out some debugging
* fix for video on stream 1
* fix audio_stream to audio_st
* Ignore bad decodes
* fix problems with content-length causing viewing to not work in chrome/android
* change logic of sending file contents to handle an off by one and be more readable
* Some fixes pointed out by Maxim Romanov. Also simply the loading of events to not join the Monitors table
* fix to sql for timeline
* added RecordAudio to sql in README
* Use sub queries instead of joins to fix errors when using new mysql defaults.
* fix sql queries
* Dockerfile to build feature-h264-videostorage
* Must cast codec
* add php-acpu as a dependency
* require php5-acpu
* fix typo
* remove extra /
* Add a line for out-of-tree builds to do api/lib/Cake/bootstrap.php
* delete merge conflict files
* delete merge conflict files
2017-05-16 10:02:48 +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){
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|