* * For the full copyright and license information, please view the LICENSE * file that was distributed with this source code. */ namespace FFMpeg\Media; use Alchemy\BinaryDriver\Exception\ExecutionFailureException; use FFMpeg\Coordinate\TimeCode; use FFMpeg\Coordinate\Dimension; use FFMpeg\Filters\Audio\SimpleFilter; use FFMpeg\Exception\InvalidArgumentException; use FFMpeg\Exception\RuntimeException; use FFMpeg\Filters\Video\VideoFilters; use FFMpeg\Filters\FilterInterface; use FFMpeg\Format\FormatInterface; use FFMpeg\Format\ProgressableInterface; use FFMpeg\Format\AudioInterface; use FFMpeg\Format\VideoInterface; use Neutron\TemporaryFilesystem\Manager as FsManager; class Video extends Audio { /** * FileSystem Manager instance * @var Manager */ protected $fs; /** * FileSystem Manager ID * @var int */ protected $fsId; /** * @inheritDoc * @return VideoFilters */ public function filters() { return new VideoFilters($this); } /** * @inheritDoc * @return Video */ public function addFilter(FilterInterface $filter) { $this->filters->add($filter); return $this; } /** * Exports the video in the desired format, applies registered filters. * * @param FormatInterface $format * @param string $outputPathfile * @return Video * @throws RuntimeException */ public function save(FormatInterface $format, string $outputPathfile) { $passes = $this->buildCommand($format, $outputPathfile); $failure = null; $totalPasses = $format->getPasses(); foreach ($passes as $pass => $passCommands) { try { /** add listeners here */ $listeners = null; if ($format instanceof ProgressableInterface) { $listeners = $format->createProgressListener($this, $this->ffprobe, $pass + 1, $totalPasses); } $this->driver->command($passCommands, false, $listeners); } catch (ExecutionFailureException $e) { $failure = $e; break; } } $this->fs->clean($this->fsId); if (null !== $failure) { throw new RuntimeException('Encoding failed', $failure->getCode(), $failure); } return $this; } /** * NOTE: This method is different to the Audio's one, because Video is using passes. * @inheritDoc */ public function getFinalCommand(FormatInterface $format, string $outputPathfile) { $finalCommands = array(); foreach($this->buildCommand($format, $outputPathfile) as $pass => $passCommands) { $finalCommands[] = implode(' ', $passCommands); } $this->fs->clean($this->fsId); return $finalCommands; } /** * **NOTE:** This creates passes instead of a single command! * * @inheritDoc * @return string[][] */ protected function buildCommand(FormatInterface $format, string $outputPathfile) { $commands = array('-y', '-i', $this->pathfile); $filters = clone $this->filters; $filters->add(new SimpleFilter($format->getExtraParams(), 10)); if ($this->driver->getConfiguration()->has('ffmpeg.threads')) { $filters->add(new SimpleFilter(array('-threads', $this->driver->getConfiguration()->get('ffmpeg.threads')))); } if ($format instanceof VideoInterface) { if (null !== $format->getVideoCodec()) { $filters->add(new SimpleFilter(array('-vcodec', $format->getVideoCodec()))); } } if ($format instanceof AudioInterface) { if (null !== $format->getAudioCodec()) { $filters->add(new SimpleFilter(array('-acodec', $format->getAudioCodec()))); } } foreach ($filters as $filter) { $commands = array_merge($commands, $filter->apply($this, $format)); } if ($format instanceof VideoInterface) { $commands[] = '-b:v'; $commands[] = $format->getKiloBitrate() . 'k'; $commands[] = '-refs'; $commands[] = '6'; $commands[] = '-coder'; $commands[] = '1'; $commands[] = '-sc_threshold'; $commands[] = '40'; $commands[] = '-flags'; $commands[] = '+loop'; $commands[] = '-me_range'; $commands[] = '16'; $commands[] = '-subq'; $commands[] = '7'; $commands[] = '-i_qfactor'; $commands[] = '0.71'; $commands[] = '-qcomp'; $commands[] = '0.6'; $commands[] = '-qdiff'; $commands[] = '4'; $commands[] = '-trellis'; $commands[] = '1'; } if ($format instanceof AudioInterface) { if (null !== $format->getAudioKiloBitrate()) { $commands[] = '-b:a'; $commands[] = $format->getAudioKiloBitrate() . 'k'; } if (null !== $format->getAudioChannels()) { $commands[] = '-ac'; $commands[] = $format->getAudioChannels(); } } // If the user passed some additional parameters if ($format instanceof VideoInterface) { if (null !== $format->getAdditionalParameters()) { foreach ($format->getAdditionalParameters() as $additionalParameter) { $commands[] = $additionalParameter; } } } // Merge Filters into one command $videoFilterVars = $videoFilterProcesses = array(); for($i=0;$i $process) { $command = '[' . $lastInput .']'; $command .= $process; $lastInput = 'p' . $i; if($i === (count($videoFilterProcesses) - 1)) { $command .= '[out]'; } else { $command .= '[' . $lastInput . ']'; } $videoFilterCommands[] = $command; } $videoFilterCommand = implode(';', $videoFilterCommands); if($videoFilterCommand) { $commands[] = '-vf'; $commands[] = $videoFilterCommand; } $this->fs = FsManager::create(); $this->fsId = uniqid('ffmpeg-passes'); $passPrefix = $this->fs->createTemporaryDirectory(0777, 50, $this->fsId) . '/' . uniqid('pass-'); $passes = array(); $totalPasses = $format->getPasses(); if(!$totalPasses) { throw new InvalidArgumentException('Pass number should be a positive value.'); } for($i = 1; $i <= $totalPasses; $i++) { $pass = $commands; if ($totalPasses > 1) { $pass[] = '-pass'; $pass[] = $i; $pass[] = '-passlogfile'; $pass[] = $passPrefix; } $pass[] = $outputPathfile; $passes[] = $pass; } return $passes; } /** * Gets the frame at timecode. * * @param TimeCode $at * @return Frame */ public function frame(TimeCode $at) { return new Frame($this, $this->driver, $this->ffprobe, $at); } /** * Extracts a gif from a sequence of the video. * * @param TimeCode $at * @param Dimension $dimension * @param integer $duration * @return Gif */ public function gif(TimeCode $at, Dimension $dimension, $duration = null) { return new Gif($this, $this->driver, $this->ffprobe, $at, $dimension, $duration); } /** * Concatenates a list of videos into one unique video. * * @param array $sources * @return Concat */ public function concat($sources) { return new Concat($sources, $this->driver, $this->ffprobe); } }