2012-04-13 10:20:54 +02:00
|
|
|
<?php
|
|
|
|
|
|
2012-04-13 14:34:53 +02:00
|
|
|
/*
|
|
|
|
|
* This file is part of PHP-FFmpeg.
|
|
|
|
|
*
|
|
|
|
|
* (c) Alchemy <info@alchemy.fr>
|
|
|
|
|
*
|
|
|
|
|
* For the full copyright and license information, please view the LICENSE
|
|
|
|
|
* file that was distributed with this source code.
|
|
|
|
|
*/
|
|
|
|
|
|
2012-04-13 10:20:54 +02:00
|
|
|
namespace FFMpeg;
|
|
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
use FFMpeg\Exception\InvalidArgumentException;
|
|
|
|
|
use FFMpeg\Exception\LogicException;
|
|
|
|
|
use FFMpeg\Exception\RuntimeException;
|
2012-05-25 20:54:54 +02:00
|
|
|
use FFMpeg\Format\Audio;
|
|
|
|
|
use FFMpeg\Format\Video;
|
2012-05-25 16:21:16 +02:00
|
|
|
use Symfony\Component\Process\Process;
|
2012-04-13 15:12:43 +02:00
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* FFMpeg driver
|
|
|
|
|
*
|
|
|
|
|
* @author Romain Neutron imprec@gmail.com
|
|
|
|
|
*/
|
2012-04-13 10:20:54 +02:00
|
|
|
class FFMpeg extends Binary
|
|
|
|
|
{
|
|
|
|
|
protected $pathfile;
|
|
|
|
|
|
2012-05-28 19:46:49 +02:00
|
|
|
/**
|
|
|
|
|
*
|
2012-05-30 12:23:55 +02:00
|
|
|
* @var FFProbe
|
2012-05-28 19:46:49 +02:00
|
|
|
*/
|
|
|
|
|
protected $prober;
|
|
|
|
|
|
2012-05-30 18:44:09 +02:00
|
|
|
/**
|
|
|
|
|
* Destructor
|
|
|
|
|
*/
|
2012-05-28 19:46:49 +02:00
|
|
|
public function __destruct()
|
|
|
|
|
{
|
|
|
|
|
$this->prober = null;
|
|
|
|
|
parent::__destruct();
|
|
|
|
|
}
|
2012-05-30 12:23:55 +02:00
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
/**
|
|
|
|
|
* Opens a file in order to be processed
|
|
|
|
|
*
|
2012-05-25 16:22:16 +02:00
|
|
|
* @param string $pathfile A pathfile
|
2012-04-17 16:33:36 +02:00
|
|
|
* @return \FFMpeg\FFMpeg
|
2012-05-25 16:21:16 +02:00
|
|
|
* @throws InvalidArgumentException
|
2012-04-13 15:12:43 +02:00
|
|
|
*/
|
2012-04-13 10:20:54 +02:00
|
|
|
public function open($pathfile)
|
|
|
|
|
{
|
2012-05-11 00:30:02 +02:00
|
|
|
if ( ! file_exists($pathfile)) {
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addError(sprintf('FFmpeg failed to open %s', $pathfile));
|
2012-04-13 15:12:43 +02:00
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
throw new InvalidArgumentException(sprintf('File %s does not exists', $pathfile));
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$this->logger->addInfo(sprintf('FFmpeg opens %s', $pathfile));
|
|
|
|
|
|
|
|
|
|
$this->pathfile = $pathfile;
|
2012-04-17 16:33:36 +02:00
|
|
|
|
|
|
|
|
return $this;
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-28 19:46:49 +02:00
|
|
|
/**
|
|
|
|
|
* Set a prober
|
2012-05-30 12:23:55 +02:00
|
|
|
*
|
2012-05-28 19:46:49 +02:00
|
|
|
* @return \FFMpeg\FFMpeg
|
|
|
|
|
*/
|
|
|
|
|
public function setProber(FFProbe $prober)
|
|
|
|
|
{
|
|
|
|
|
$this->prober = $prober;
|
2012-05-30 12:23:55 +02:00
|
|
|
|
2012-05-28 19:46:49 +02:00
|
|
|
return $this;
|
|
|
|
|
}
|
|
|
|
|
|
2012-04-17 16:33:36 +02:00
|
|
|
/**
|
|
|
|
|
* Close a file
|
|
|
|
|
*
|
|
|
|
|
* @return \FFMpeg\FFMpeg
|
|
|
|
|
*/
|
|
|
|
|
public function close()
|
|
|
|
|
{
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg closes %s', $this->pathfile));
|
|
|
|
|
|
2012-04-17 16:33:36 +02:00
|
|
|
$this->pathfile = null;
|
|
|
|
|
|
|
|
|
|
return $this;
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
/**
|
2012-05-30 18:44:09 +02:00
|
|
|
* Extract an image from a media file
|
2012-04-13 15:12:43 +02:00
|
|
|
*
|
2012-05-25 16:22:16 +02:00
|
|
|
* @param integer $time The time in second where to take the snapshot
|
|
|
|
|
* @param string $output The pathfile where to write
|
2012-04-17 16:33:36 +02:00
|
|
|
* @return \FFMpeg\FFMpeg
|
2012-05-25 16:21:16 +02:00
|
|
|
* @throws RuntimeException
|
|
|
|
|
* @throws LogicException
|
2012-04-13 15:12:43 +02:00
|
|
|
*/
|
2012-04-18 10:36:29 +02:00
|
|
|
public function extractImage($time, $output)
|
2012-04-13 10:20:54 +02:00
|
|
|
{
|
2012-05-11 00:30:02 +02:00
|
|
|
if ( ! $this->pathfile) {
|
2012-05-25 16:21:16 +02:00
|
|
|
throw new LogicException('No file open');
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$cmd = $this->binary
|
2012-05-11 00:30:02 +02:00
|
|
|
. ' -i ' . escapeshellarg($this->pathfile)
|
|
|
|
|
. ' -vframes 1 -ss ' . $time
|
|
|
|
|
. ' -f image2 ' . escapeshellarg($output);
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg executes command %s', $cmd));
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
$process = new Process($cmd);
|
|
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
try {
|
2012-04-13 15:12:43 +02:00
|
|
|
$process->run();
|
2012-05-11 00:30:02 +02:00
|
|
|
} catch (\RuntimeException $e) {
|
2012-06-01 19:03:00 +02:00
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
}
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
if ( ! $process->isSuccessful()) {
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addError(sprintf('FFmpeg command failed : %s', $process->getErrorOutput()));
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
$this->cleanupTemporaryFile($output);
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
throw new RuntimeException('Failed to extract image');
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg command successful'));
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-04-17 16:33:36 +02:00
|
|
|
return $this;
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
/**
|
|
|
|
|
* Encode the file to the specified format
|
|
|
|
|
*
|
2012-05-25 20:54:54 +02:00
|
|
|
* @param Audio $format The output format
|
2012-05-25 16:22:16 +02:00
|
|
|
* @param string $outputPathfile The pathfile where to write
|
|
|
|
|
* @param integer $threads The number of threads to use
|
2012-04-17 16:33:36 +02:00
|
|
|
* @return \FFMpeg\FFMpeg
|
2012-05-25 16:21:16 +02:00
|
|
|
* @throws RuntimeException
|
|
|
|
|
* @throws LogicException
|
2012-04-13 15:12:43 +02:00
|
|
|
*/
|
2012-05-25 20:54:54 +02:00
|
|
|
public function encode(Audio $format, $outputPathfile, $threads = 1)
|
2012-04-13 10:20:54 +02:00
|
|
|
{
|
2012-05-11 00:30:02 +02:00
|
|
|
if ( ! $this->pathfile) {
|
2012-05-25 16:21:16 +02:00
|
|
|
throw new LogicException('No file open');
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$threads = max(min($threads, 64), 1);
|
|
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
switch (true) {
|
2012-05-25 20:54:54 +02:00
|
|
|
case $format instanceof Video:
|
2012-04-17 16:33:36 +02:00
|
|
|
$this->encodeVideo($format, $outputPathfile, $threads);
|
2012-04-13 14:15:56 +02:00
|
|
|
break;
|
|
|
|
|
default:
|
2012-05-25 20:54:54 +02:00
|
|
|
case $format instanceof Audio:
|
2012-04-17 16:33:36 +02:00
|
|
|
$this->encodeAudio($format, $outputPathfile, $threads);
|
2012-04-13 14:15:56 +02:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2012-04-17 16:33:36 +02:00
|
|
|
return $this;
|
2012-04-13 14:15:56 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
/**
|
|
|
|
|
* Encode to audio
|
|
|
|
|
*
|
2012-05-25 20:54:54 +02:00
|
|
|
* @param Audio $format The output format
|
2012-05-25 16:22:16 +02:00
|
|
|
* @param string $outputPathfile The pathfile where to write
|
|
|
|
|
* @param integer $threads The number of threads to use
|
2012-04-17 16:33:36 +02:00
|
|
|
* @return \FFMpeg\FFMpeg
|
2012-05-25 16:21:16 +02:00
|
|
|
* @throws RuntimeException
|
2012-04-13 15:12:43 +02:00
|
|
|
*/
|
2012-05-25 20:54:54 +02:00
|
|
|
protected function encodeAudio(Audio $format, $outputPathfile, $threads)
|
2012-04-13 14:15:56 +02:00
|
|
|
{
|
|
|
|
|
$cmd = $this->binary
|
2012-05-11 00:30:02 +02:00
|
|
|
. ' -y -i '
|
|
|
|
|
. escapeshellarg($this->pathfile)
|
|
|
|
|
. ' ' . $format->getExtraParams()
|
|
|
|
|
. ' -threads ' . $threads
|
|
|
|
|
. ' -ab ' . $format->getKiloBitrate() . 'k '
|
|
|
|
|
. ' ' . escapeshellarg($outputPathfile);
|
2012-04-13 14:15:56 +02:00
|
|
|
|
2012-05-30 15:06:53 +02:00
|
|
|
if ($format instanceof Audio\Transcodable) {
|
|
|
|
|
$cmd .= ' -acodec ' . $format->getAudioCodec();
|
|
|
|
|
}
|
2012-05-31 15:54:28 +02:00
|
|
|
|
2012-05-30 15:06:53 +02:00
|
|
|
if ($format instanceof Audio\Resamplable) {
|
|
|
|
|
$cmd .= ' -ac 2 -ar ' . $format->getAudioSampleRate();
|
|
|
|
|
}
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
$process = new Process($cmd);
|
|
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg executes command %s', $cmd));
|
|
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
try {
|
2012-04-13 15:12:43 +02:00
|
|
|
$process->run();
|
2012-05-11 00:30:02 +02:00
|
|
|
} catch (\RuntimeException $e) {
|
2012-06-01 19:03:00 +02:00
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
}
|
2012-04-13 14:15:56 +02:00
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
if ( ! $process->isSuccessful()) {
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg command failed'));
|
|
|
|
|
throw new RuntimeException(sprintf('Encoding failed : %s', $process->getErrorOutput()));
|
2012-04-13 14:15:56 +02:00
|
|
|
}
|
|
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg command successful'));
|
|
|
|
|
|
2012-04-17 16:33:36 +02:00
|
|
|
return $this;
|
2012-04-13 14:15:56 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
/**
|
|
|
|
|
* Encode to video
|
|
|
|
|
*
|
2012-05-25 20:54:54 +02:00
|
|
|
* @param Video $format The output format
|
2012-05-25 16:22:16 +02:00
|
|
|
* @param string $outputPathfile The pathfile where to write
|
|
|
|
|
* @param integer $threads The number of threads to use
|
2012-04-18 10:36:29 +02:00
|
|
|
* @return \FFMpeg\FFMpeg
|
2012-05-25 16:21:16 +02:00
|
|
|
* @throws RuntimeException
|
2012-04-13 15:12:43 +02:00
|
|
|
*/
|
2012-05-25 20:54:54 +02:00
|
|
|
protected function encodeVideo(Video $format, $outputPathfile, $threads)
|
2012-04-13 14:15:56 +02:00
|
|
|
{
|
2012-04-13 10:20:54 +02:00
|
|
|
$cmd_part1 = $this->binary
|
2012-05-11 00:30:02 +02:00
|
|
|
. ' -y -i '
|
|
|
|
|
. escapeshellarg($this->pathfile) . ' '
|
|
|
|
|
. $format->getExtraParams() . ' ';
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-05-25 18:24:37 +02:00
|
|
|
$cmd_part2 = '';
|
|
|
|
|
|
2012-05-30 16:53:36 +02:00
|
|
|
if ($format instanceof Video\Resizable) {
|
2012-05-30 12:23:55 +02:00
|
|
|
if ( ! $this->prober) {
|
|
|
|
|
throw new LogicException('You must set a valid prober if you use RESIZEMODE_INSET');
|
|
|
|
|
}
|
2012-05-28 19:46:49 +02:00
|
|
|
|
2012-05-31 15:54:28 +02:00
|
|
|
$result = json_decode($this->prober->probeStreams($this->pathfile), true);
|
2012-05-28 19:46:49 +02:00
|
|
|
|
2012-05-30 15:06:53 +02:00
|
|
|
$originalWidth = $originalHeight = null;
|
2012-05-28 19:46:49 +02:00
|
|
|
|
2012-05-30 12:23:55 +02:00
|
|
|
foreach ($result as $stream) {
|
2012-05-31 15:54:28 +02:00
|
|
|
foreach ($stream as $name => $value) {
|
|
|
|
|
if ($name == 'width') {
|
2012-05-31 16:12:51 +02:00
|
|
|
$originalWidth = $value;
|
2012-05-30 12:23:55 +02:00
|
|
|
continue;
|
2012-05-28 19:46:49 +02:00
|
|
|
}
|
2012-06-01 19:03:00 +02:00
|
|
|
if ($name == 'height') {
|
2012-05-31 16:12:51 +02:00
|
|
|
$originalHeight = $value;
|
2012-05-30 12:23:55 +02:00
|
|
|
continue;
|
2012-05-29 10:32:37 +02:00
|
|
|
}
|
2012-05-30 12:23:55 +02:00
|
|
|
}
|
2012-05-28 19:46:49 +02:00
|
|
|
}
|
|
|
|
|
|
2012-06-01 19:03:00 +02:00
|
|
|
if ($originalHeight !== null && $originalWidth !== null) {
|
|
|
|
|
$this->logger->addInfo(sprintf('Read dimension for resizin succesful : %s x %s', $originalWidth, $originalHeight));
|
|
|
|
|
} else {
|
|
|
|
|
$this->logger->addInfo(sprintf('Read dimension for resizin failed !'));
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-30 15:06:53 +02:00
|
|
|
if ($originalHeight !== null && $originalWidth !== null) {
|
2012-05-30 16:53:36 +02:00
|
|
|
$dimensions = $format->getComputedDimensions($originalWidth, $originalHeight);
|
2012-05-30 15:06:53 +02:00
|
|
|
|
2012-05-30 16:53:36 +02:00
|
|
|
$width = $this->getMultiple($dimensions->getWidth(), 16);
|
|
|
|
|
$height = $this->getMultiple($dimensions->getHeight(), 16);
|
2012-05-30 12:23:55 +02:00
|
|
|
|
2012-05-30 15:06:53 +02:00
|
|
|
$cmd_part2 .= ' -s ' . $width . 'x' . $height;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($format instanceof Video\Resamplable) {
|
|
|
|
|
$cmd_part2 .= ' -r ' . $format->getFrameRate();
|
|
|
|
|
}
|
2012-05-30 12:23:55 +02:00
|
|
|
|
2012-05-30 15:06:53 +02:00
|
|
|
if ($format instanceof Video\Transcodable) {
|
|
|
|
|
$cmd_part2 .= ' -vcodec ' . $format->getVideoCodec();
|
2012-05-25 18:24:37 +02:00
|
|
|
}
|
|
|
|
|
|
2012-05-30 15:06:53 +02:00
|
|
|
$cmd_part2 .= ' -b ' . $format->getKiloBitrate() . 'k -g 25 -bf 3'
|
2012-05-11 00:30:02 +02:00
|
|
|
. ' -threads ' . $threads
|
|
|
|
|
. ' -refs 6 -b_strategy 1 -coder 1 -qmin 10 -qmax 51 '
|
|
|
|
|
. ' -sc_threshold 40 -flags +loop -cmp +chroma'
|
|
|
|
|
. ' -me_range 16 -subq 7 -i_qfactor 0.71 -qcomp 0.6 -qdiff 4 '
|
|
|
|
|
. ' -trellis 1 -qscale 1 '
|
2012-05-30 15:06:53 +02:00
|
|
|
. ' -ab 92k ';
|
|
|
|
|
|
|
|
|
|
if ($format instanceof Audio\Transcodable) {
|
|
|
|
|
$cmd_part2 .= '-acodec ' . $format->getAudioCodec();
|
|
|
|
|
}
|
2012-04-13 10:20:54 +02:00
|
|
|
|
|
|
|
|
$tmpFile = new \SplFileInfo(tempnam(sys_get_temp_dir(), 'temp') . '.' . pathinfo($outputPathfile, PATHINFO_EXTENSION));
|
|
|
|
|
|
|
|
|
|
$passes = array();
|
|
|
|
|
|
|
|
|
|
$passes[] = $cmd_part1 . ' -pass 1 ' . $cmd_part2
|
2012-05-11 00:30:02 +02:00
|
|
|
. ' -an ' . escapeshellarg($tmpFile->getPathname());
|
2012-04-13 10:20:54 +02:00
|
|
|
|
|
|
|
|
$passes[] = $cmd_part1 . ' -pass 2 ' . $cmd_part2
|
2012-05-11 00:30:02 +02:00
|
|
|
. ' -ac 2 -ar 44100 ' . escapeshellarg($outputPathfile);
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
$process = null;
|
|
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
foreach ($passes as $pass) {
|
2012-05-25 16:21:16 +02:00
|
|
|
|
|
|
|
|
$this->logger->addInfo(sprintf('FFmpeg executes command %s', $pass));
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
$process = new Process($pass);
|
2012-04-13 10:20:54 +02:00
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
try {
|
2012-05-11 00:34:19 +02:00
|
|
|
$process->run();
|
2012-05-11 00:30:02 +02:00
|
|
|
} catch (\RuntimeException $e) {
|
2012-04-13 10:20:54 +02:00
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
$this->cleanupTemporaryFile($tmpFile->getPathname());
|
|
|
|
|
$this->cleanupTemporaryFile(getcwd() . '/ffmpeg2pass-0.log');
|
2012-05-30 18:46:11 +02:00
|
|
|
$this->cleanupTemporaryFile(getcwd() . '/av2pass-0.log');
|
2012-04-13 10:20:54 +02:00
|
|
|
$this->cleanupTemporaryFile(getcwd() . '/ffmpeg2pass-0.log.mbtree');
|
|
|
|
|
|
2012-05-11 00:30:02 +02:00
|
|
|
if ( ! $process->isSuccessful()) {
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg command failed'));
|
|
|
|
|
throw new RuntimeException(sprintf('Encoding failed : %s', $process->getErrorOutput()));
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
2012-05-25 16:21:16 +02:00
|
|
|
$this->logger->addInfo(sprintf('FFmpeg command successful'));
|
|
|
|
|
|
2012-04-17 16:33:36 +02:00
|
|
|
return $this;
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
/**
|
|
|
|
|
* Removes unnecessary file
|
|
|
|
|
*
|
|
|
|
|
* @param string $pathfile
|
|
|
|
|
*/
|
2012-04-13 10:20:54 +02:00
|
|
|
protected function cleanupTemporaryFile($pathfile)
|
|
|
|
|
{
|
2012-05-11 00:30:02 +02:00
|
|
|
if (file_exists($pathfile) && is_writable($pathfile)) {
|
2012-04-13 10:20:54 +02:00
|
|
|
unlink($pathfile);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2012-05-28 19:46:49 +02:00
|
|
|
/**
|
|
|
|
|
* Returns the nearest multiple for a value
|
|
|
|
|
*
|
|
|
|
|
* @param integer $value
|
|
|
|
|
* @param integer $multiple
|
|
|
|
|
* @return integer
|
|
|
|
|
*/
|
|
|
|
|
protected function getMultiple($value, $multiple)
|
|
|
|
|
{
|
|
|
|
|
$modulo = $value % $multiple;
|
|
|
|
|
|
|
|
|
|
$ret = (int) $multiple;
|
|
|
|
|
|
|
|
|
|
$halfDistance = $multiple / 2;
|
|
|
|
|
if ($modulo <= $halfDistance)
|
|
|
|
|
$bound = 'bottom';
|
|
|
|
|
else
|
|
|
|
|
$bound = 'top';
|
|
|
|
|
|
|
|
|
|
switch ($bound) {
|
|
|
|
|
default:
|
|
|
|
|
case 'top':
|
|
|
|
|
$ret = $value + $multiple - $modulo;
|
|
|
|
|
break;
|
|
|
|
|
case 'bottom':
|
|
|
|
|
$ret = $value - $modulo;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if ($ret < $multiple) {
|
|
|
|
|
$ret = (int) $multiple;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return (int) $ret;
|
|
|
|
|
}
|
|
|
|
|
|
2012-04-13 15:12:43 +02:00
|
|
|
/**
|
2012-05-25 16:21:16 +02:00
|
|
|
* {@inheritdoc}
|
2012-04-13 15:12:43 +02:00
|
|
|
*
|
|
|
|
|
* @return string
|
|
|
|
|
*/
|
2012-04-13 10:20:54 +02:00
|
|
|
protected static function getBinaryName()
|
|
|
|
|
{
|
2012-05-11 00:30:02 +02:00
|
|
|
return array('avconv', 'ffmpeg');
|
2012-04-13 10:20:54 +02:00
|
|
|
}
|
|
|
|
|
}
|