Added video generation frame rate and scaling.

git-svn-id: http://svn.zoneminder.com/svn/zm/trunk@651 e3e1d417-86f3-4887-817a-d78f3d33393f
This commit is contained in:
stan 2003-10-10 15:46:36 +00:00
parent 945e8e4145
commit 22490ebfac
4 changed files with 160 additions and 38 deletions

View File

@ -77,18 +77,24 @@ $ENV{SHELL} = '/bin/sh' if exists $ENV{SHELL};
delete @ENV{qw(IFS CDPATH ENV BASH_ENV)}; delete @ENV{qw(IFS CDPATH ENV BASH_ENV)};
my $event_id; my $event_id;
my $rate = 1;
my $scale = 1;
my $overwrite = 0;
sub Usage sub Usage
{ {
print( " print( "
Usage: zmvideo.pl -e <event_id>,--event=<event_id> Usage: zmvideo.pl -e <event_id>,--event=<event_id> [-r <rate>,--rate=<rate>] [-s <scale>,--scale=<scale>] [-o,--overwrite]
Parameters are :- Parameters are :-
-e<event_id>, --event=<event_id> - What event to start create the video for -e<event_id>, --event=<event_id> - What event to create the video for
-r<rate>, --rate=<rate> - Relative rate to use, 1 = realtime, 2 = double speed , -2 = half speed etc
-s<scale>, --scale=<scale> - Scale to use, 1 = normal, 2 = double size, -2 = half size etc
-o, --overwrite - Whether to overwrite an existing file, off by default.
"); ");
exit( -1 ); exit( -1 );
} }
if ( !GetOptions( 'event=i'=>\$event_id ) ) if ( !GetOptions( 'event=i'=>\$event_id, 'rate=i'=>\$rate, 'scale=i'=>\$scale, 'overwrite'=>\$overwrite ) )
{ {
Usage(); Usage();
} }
@ -105,6 +111,30 @@ if ( ZM_OPT_MPEG eq "no" )
exit(-1); exit(-1);
} }
if ( ZM_OPT_MPEG eq "mpeg_encode" && $rate != 1 )
{
print( STDERR "Variable rate not supported with mpeg_encode\n" );
exit(-1);
}
if ( $rate < -4 || $rate > 10 )
{
print( STDERR "Rate is out of range, -4 >= rate <= 10\n" );
Usage();
}
if ( !$scale || $scale < -4 || $scale > 4 )
{
print( STDERR "Scale is out of range, -4 >= scale <= 4\n" );
Usage();
}
my ( $detaint_rate ) = $rate =~ /^(-?\d+(?:\.\d+)?)$/;
my ( $detaint_scale ) = $scale =~ /^(-?\d+(?:\.\d+)?)$/;
$rate = $detaint_rate;
$scale = $detaint_scale;
my $log_file = LOG_FILE; my $log_file = LOG_FILE;
open( LOG, ">>$log_file" ) or die( "Can't open log file: $!" ); open( LOG, ">>$log_file" ) or die( "Can't open log file: $!" );
#open( STDOUT, ">&LOG" ) || die( "Can't dup stdout: $!" ); #open( STDOUT, ">&LOG" ) || die( "Can't dup stdout: $!" );
@ -116,7 +146,7 @@ select( LOG ); $| = 1;
my $dbh = DBI->connect( "DBI:mysql:database=".ZM_DB_NAME.";host=".ZM_DB_SERVER, ZM_DB_USERB, ZM_DB_PASSB ); my $dbh = DBI->connect( "DBI:mysql:database=".ZM_DB_NAME.";host=".ZM_DB_SERVER, ZM_DB_USERB, ZM_DB_PASSB );
my @filters; my @filters;
my $sql = "select max(F.Delta)-min(F.Delta) as FullLength,E.*,M.Name as MonitorName, M.Palette from Frames as F inner join Events as E on F.EventId = E.Id inner join Monitors as M on E.MonitorId = M.Id where EventId = '$event_id' group by F.EventId"; my $sql = "select max(F.Delta)-min(F.Delta) as FullLength, E.*, M.Name as MonitorName, M.Width as MonitorWidth, M.Height as MonitorHeight, M.Palette from Frames as F inner join Events as E on F.EventId = E.Id inner join Monitors as M on E.MonitorId = M.Id where EventId = '$event_id' group by F.EventId";
my $sth = $dbh->prepare_cached( $sql ) or die( "Can't prepare '$sql': ".$dbh->errstr() ); my $sth = $dbh->prepare_cached( $sql ) or die( "Can't prepare '$sql': ".$dbh->errstr() );
my $res = $sth->execute() or die( "Can't execute: ".$sth->errstr() ); my $res = $sth->execute() or die( "Can't execute: ".$sth->errstr() );
my $event = $sth->fetchrow_hashref(); my $event = $sth->fetchrow_hashref();
@ -125,7 +155,7 @@ chdir( ZM_PATH_WEB.'/'.ZM_DIR_EVENTS.'/'.$event->{MonitorName}.'/'.$event->{Id}
( my $video_name = $event->{Name} ) =~ s/\s/_/g; ( my $video_name = $event->{Name} ) =~ s/\s/_/g;
my $video_file = "$video_name.mpg"; my $video_file = "$video_name.mpg";
if ( !-s $video_file ) if ( $overwrite || !-s $video_file )
{ {
print( LOG "Creating video file $video_file for event $event->{Id}\n" ); print( LOG "Creating video file $video_file for event $event->{Id}\n" );
@ -153,6 +183,28 @@ if ( !-s $video_file )
print( PARAMS "REFERENCE_FRAME ORIGINAL\n" ); print( PARAMS "REFERENCE_FRAME ORIGINAL\n" );
print( PARAMS "FRAME_RATE 24\n" ); print( PARAMS "FRAME_RATE 24\n" );
my $scale_conversion = "";
if ( $scale != 1 )
{
if ( $scale > 1 )
{
$scale_conversion = ZM_PATH_NETPBM."/pnmscale $scale";
}
else
{
$scale_conversion = ZM_PATH_NETPBM."/pnmscale ".(1/$scale);
}
if ( $event->{Palette} == 1 && !ZM_COLOUR_JPEG_FILES )
{
print( PARAMS "INPUT_CONVERT ".ZM_PATH_NETPBM."/jpegtopnm * | ".$scale_conversion." | ".ZM_PATH_NETPBM."/pgmtoppm white | ".ZM_PATH_NETPBM."/ppmtojpeg\n" );
}
else
{
print( PARAMS "INPUT_CONVERT ".ZM_PATH_NETPBM."/jpegtopnm * | ".$scale_conversion." | ".ZM_PATH_NETPBM."/ppmtojpeg\n" );
}
}
else
{
if ( $event->{Palette} == 1 && !ZM_COLOUR_JPEG_FILES ) if ( $event->{Palette} == 1 && !ZM_COLOUR_JPEG_FILES )
{ {
print( PARAMS "INPUT_CONVERT ".ZM_PATH_NETPBM."/jpegtopnm * | ".ZM_PATH_NETPBM."/pgmtoppm white | ".ZM_PATH_NETPBM."/ppmtojpeg\n" ); print( PARAMS "INPUT_CONVERT ".ZM_PATH_NETPBM."/jpegtopnm * | ".ZM_PATH_NETPBM."/pgmtoppm white | ".ZM_PATH_NETPBM."/ppmtojpeg\n" );
@ -161,6 +213,7 @@ if ( !-s $video_file )
{ {
print( PARAMS "INPUT_CONVERT *\n" ); print( PARAMS "INPUT_CONVERT *\n" );
} }
}
print( PARAMS "INPUT_DIR .\n" ); print( PARAMS "INPUT_DIR .\n" );
print( PARAMS "INPUT\n" ); print( PARAMS "INPUT\n" );
@ -172,13 +225,50 @@ if ( !-s $video_file )
close( PARAMS ); close( PARAMS );
my $command = ZM_PATH_MPEG_ENCODE." $param_file >mpeg.log"; my $command = ZM_PATH_MPEG_ENCODE." $param_file >mpeg.log";
print( LOG $command."\n" );
my $output = qx($command); my $output = qx($command);
print( LOG $output."\n" );
} }
elsif ( ZM_OPT_MPEG eq "ffmpeg" ) elsif ( ZM_OPT_MPEG eq "ffmpeg" )
{ {
my $frame_rate = sprintf( "%.2f", $event->{Frames}/$event->{FullLength} ); my $frame_rate = sprintf( "%.2f", $event->{Frames}/$event->{FullLength} );
my $command = ZM_PATH_FFMPEG." -y ".ZM_FFMPEG_OPTIONS." -r $frame_rate -i capture-%03d.jpg $video_file > mpeg.log"; if ( $rate )
{
if ( $rate != 1 )
{
if ( $rate > 1 )
{
$frame_rate *= $rate;
}
else
{
$frame_rate /= $rate;
}
}
}
else
{
$frame_rate = 25.0;
}
my $width = $event->{MonitorWidth};
my $height = $event->{MonitorHeight};
if ( $scale != 1 )
{
if ( $scale > 1 )
{
$width = int($width*$scale);
$height = int($height*$scale);
}
else
{
$width = int($width/$scale);
$height = int($height/$scale);
}
}
my $command = ZM_PATH_FFMPEG." -y ".ZM_FFMPEG_OPTIONS." -r $frame_rate -s ${width}x${height} -i capture-%03d.jpg $video_file > mpeg.log";
print( LOG $command."\n" );
my $output = qx($command); my $output = qx($command);
print( LOG $output."\n" );
} }
else else
{ {

View File

@ -52,6 +52,14 @@ $rates = array(
"-4" => "1/4x", "-4" => "1/4x",
); );
$scales = array(
"4" => "4x",
"2" => "2x",
"1" => "Real",
"-2" => "1/2x",
"-4" => "1/4x",
);
require_once( 'zm_db.php' ); require_once( 'zm_db.php' );
loadConfig(); loadConfig();

View File

@ -307,9 +307,11 @@ function zmaCheck( $monitor )
return( daemonCheck( "zma", "-m $monitor" ) ); return( daemonCheck( "zma", "-m $monitor" ) );
} }
function createVideo( $event ) function createVideo( $event, $rate, $scale, $overwrite=0 )
{ {
$command = ZM_PATH_BIN."/zmvideo.pl -e $event[Id]"; $command = ZM_PATH_BIN."/zmvideo.pl -e $event[Id] -r $rate -s $scale";
if ( $overwite )
$command .= " -o";
$result = exec( $command, $output, $status ); $result = exec( $command, $output, $status );
return( $status?"":rtrim($result) ); return( $status?"":rtrim($result) );
} }

View File

@ -9,9 +9,12 @@
die( mysql_error() ); die( mysql_error() );
$event = mysql_fetch_assoc( $result ); $event = mysql_fetch_assoc( $result );
ob_start(); if ( !isset( $scale ) )
$scale = 1;
if ( !isset( $rate ) )
$rate = 1;
// Note this all has a bunch of extra padding as IE won't flush less than 1024 chars ob_start();
?> ?>
<html> <html>
<head> <head>
@ -19,12 +22,28 @@
<link rel="stylesheet" href="zm_styles.css" type="text/css"> <link rel="stylesheet" href="zm_styles.css" type="text/css">
</head> </head>
<body> <body>
<form name="video_form" method="get" action="<?= $PHP_SELF ?>">
<input type="hidden" name="view" value="<?= $view ?>">
<input type="hidden" name="action" value="<?= $action ?>">
<input type="hidden" name="eid" value="<?= $eid ?>">
<input type="hidden" name="generate" value="1">
<table align="center" border="0" cellspacing="0" cellpadding="2" width="250">
<tr><td colspan="2" class="head" align="center">Video Generation Parameters</td></tr>
<tr><td colspan="2">&nbsp;</td></tr>
<tr><td class="text" align="right">Frame Rate</td><td><?php buildSelect( "rate", $rates ) ?></td></tr>
<tr><td class="text" align="right">Video Size</td><td><?php buildSelect( "scale", $scales ) ?></td></tr>
<tr><td class="text" align="right">Overwite Existing</td><td><input type="checkbox" class="form-noborder" name="overwrite" value="1"<?php if ( $overwrite ) { ?> checked<?php } ?>></td></tr>
<tr><td colspan="2">&nbsp;</td></tr>
<tr><td colspan="2" align="center"><input type="submit" class="form" value="Generate Video"></td></tr>
</table>
</form>
<?php
if ( $generate )
{
?>
<table border="0" cellspacing="0" cellpadding="4" width="100%"> <table border="0" cellspacing="0" cellpadding="4" width="100%">
<tr><td>&nbsp;</td></tr> <tr><td>&nbsp;</td></tr>
<tr><td>&nbsp;</td></tr> <tr><td>&nbsp;</td></tr>
<tr><td>&nbsp;</td></tr>
<tr><td>&nbsp;</td></tr>
<tr><td>&nbsp;</td></tr>
<tr> <tr>
<td align="center" class="head">Generating Video</td> <td align="center" class="head">Generating Video</td>
</tr> </tr>
@ -32,6 +51,7 @@
<tr><td>&nbsp;</td></tr> <tr><td>&nbsp;</td></tr>
</table> </table>
</body> </body>
</html>
<?php <?php
$buffer_string = "<!-- This is some long buffer text to ensure that IE flushes correctly -->"; $buffer_string = "<!-- This is some long buffer text to ensure that IE flushes correctly -->";
for ( $i = 0; $i < 4096/strlen($buffer_string); $i++ ) for ( $i = 0; $i < 4096/strlen($buffer_string); $i++ )
@ -39,10 +59,9 @@
echo $buffer_string."\n"; echo $buffer_string."\n";
} }
?> ?>
</html>
<?php <?php
ob_end_flush(); ob_end_flush();
if ( $video_file = createVideo( $event ) ) if ( $video_file = createVideo( $event, $rate, $scale, $overwrite ) )
{ {
$event_dir = ZM_DIR_EVENTS."/$event[MonitorName]/".sprintf( "%d", $eid ); $event_dir = ZM_DIR_EVENTS."/$event[MonitorName]/".sprintf( "%d", $eid );
$video_path = $event_dir.'/'.$video_file; $video_path = $event_dir.'/'.$video_file;
@ -50,8 +69,6 @@
?> ?>
<html> <html>
<head> <head>
<title>ZM - Video - <?= $event[Name] ?></title>
<link rel="stylesheet" href="zm_styles.css" type="text/css">
<script language="JavaScript"> <script language="JavaScript">
location.replace('<?= $video_path ?>'); location.replace('<?= $video_path ?>');
</script> </script>
@ -61,16 +78,21 @@ location.replace('<?= $video_path ?>');
} }
else else
{ {
ob_end_flush();
?> ?>
<html> <html>
<head> <head>
<title>ZM - Video - <?= $event[Name] ?></title>
<link rel="stylesheet" href="zm_styles.css" type="text/css"> <link rel="stylesheet" href="zm_styles.css" type="text/css">
</head> </head>
<body> <body>
<p class="head" align="center"><font color="red"><br><br><br>Video Generation Failed!<br><br><br></font></p> <p class="head" align="center"><font color="red"><br><br>Video Generation Failed!<br><br></font></p>
</body>
</html>
<?php <?php
} }
}
else
{
ob_end_flush();
}
?> ?>
</body>
</html>