2020-08-14 02:46:54 +09:00
|
|
|
<?php
|
2021-04-26 06:24:43 +09:00
|
|
|
// {{{ License
|
2020-08-14 02:46:54 +09:00
|
|
|
// This file is part of GNU social - https://www.gnu.org/software/social
|
|
|
|
//
|
|
|
|
// GNU social is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// GNU social 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 Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with GNU social. If not, see <http://www.gnu.org/licenses/>.
|
2021-04-26 06:24:43 +09:00
|
|
|
// }}}
|
2020-08-14 02:46:54 +09:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Animated GIF resize support via PHP-FFMpeg
|
|
|
|
*
|
|
|
|
* @package GNUsocial
|
2021-04-15 00:44:45 +09:00
|
|
|
*
|
2020-08-14 02:46:54 +09:00
|
|
|
* @author Bruno Casteleiro <up201505347@fc.up.pt>
|
|
|
|
* @copyright 2020 Free Software Foundation, Inc http://www.fsf.org
|
|
|
|
* @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
|
2021-04-15 00:44:45 +09:00
|
|
|
*
|
|
|
|
* @see http://www.gnu.org/software/social/
|
2020-08-14 02:46:54 +09:00
|
|
|
*/
|
|
|
|
|
2021-04-18 13:47:16 +09:00
|
|
|
namespace Plugin\VideoEncoder;
|
2021-04-15 00:44:45 +09:00
|
|
|
|
2021-07-23 04:49:12 +09:00
|
|
|
use App\Core\Event;
|
2021-04-18 13:47:16 +09:00
|
|
|
use App\Core\Modules\Plugin;
|
2021-07-23 04:49:12 +09:00
|
|
|
use App\Util\Formatting;
|
2020-08-14 02:46:54 +09:00
|
|
|
|
2021-04-18 13:47:16 +09:00
|
|
|
class VideoEncoder extends Plugin
|
2020-08-14 02:46:54 +09:00
|
|
|
{
|
|
|
|
const PLUGIN_VERSION = '0.1.0';
|
|
|
|
|
2021-04-15 00:44:45 +09:00
|
|
|
/**
|
|
|
|
* Handle resizing GIF files
|
|
|
|
*/
|
2020-09-04 19:15:23 +09:00
|
|
|
public function onStartResizeImageFile(
|
2021-04-18 13:47:16 +09:00
|
|
|
ImageValidate $imagefile,
|
2020-09-04 19:15:23 +09:00
|
|
|
string $outpath,
|
2021-04-15 00:44:45 +09:00
|
|
|
array $box
|
2020-09-04 19:15:23 +09:00
|
|
|
): bool {
|
2020-08-14 02:46:54 +09:00
|
|
|
switch ($imagefile->mimetype) {
|
|
|
|
case 'image/gif':
|
|
|
|
// resize only if an animated GIF
|
|
|
|
if ($imagefile->animated) {
|
|
|
|
return !$this->resizeImageFileAnimatedGif($imagefile, $outpath, $box);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-07-23 05:17:23 +09:00
|
|
|
/**
|
|
|
|
* @param array $event_map
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function onResizerAvailable(array &$event_map): bool
|
|
|
|
{
|
|
|
|
//$event_map['video'] = 'ResizeVideoPath';
|
|
|
|
return Event::next;
|
|
|
|
}
|
|
|
|
|
2021-07-23 04:49:12 +09:00
|
|
|
/**
|
|
|
|
* Generates the view for attachments of type Video
|
|
|
|
*
|
|
|
|
* @param array $vars
|
|
|
|
* @param array $res
|
2021-07-23 05:17:23 +09:00
|
|
|
*
|
2021-07-23 04:49:12 +09:00
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function onViewAttachmentVideo(array $vars, array &$res): bool
|
|
|
|
{
|
2021-08-12 11:43:11 +09:00
|
|
|
$res[] = Formatting::twigRenderFile('videoEncoder/videoEncoderView.html.twig',
|
|
|
|
['attachment' => $vars['attachment'],
|
|
|
|
'thumbnail_parameters' => $vars['thumbnail_parameters'],
|
|
|
|
'note' => $vars['note'],
|
|
|
|
]);
|
2021-07-23 04:49:12 +09:00
|
|
|
return Event::stop;
|
|
|
|
}
|
|
|
|
|
2020-08-14 02:46:54 +09:00
|
|
|
/**
|
|
|
|
* High quality GIF conversion.
|
|
|
|
*
|
|
|
|
* @see http://blog.pkh.me/p/21-high-quality-gif-with-ffmpeg.html
|
|
|
|
* @see https://github.com/PHP-FFMpeg/PHP-FFMpeg/pull/592
|
|
|
|
*/
|
2021-04-18 13:47:16 +09:00
|
|
|
public function resizeImageFileAnimatedGif(ImageValidate $imagefile, string $outpath, array $box): bool
|
2020-08-14 02:46:54 +09:00
|
|
|
{
|
|
|
|
// Create FFMpeg instance
|
|
|
|
// Need to explictly tell the drivers location or it won't find them
|
|
|
|
$ffmpeg = FFMpeg\FFMpeg::create([
|
2021-04-15 00:44:45 +09:00
|
|
|
'ffmpeg.binaries' => exec('which ffmpeg'),
|
|
|
|
'ffprobe.binaries' => exec('which ffprobe'),
|
2020-08-14 02:46:54 +09:00
|
|
|
]);
|
|
|
|
|
|
|
|
// FFmpeg can't edit existing files in place,
|
|
|
|
// generate temporary output file to avoid that
|
2021-05-03 00:02:26 +09:00
|
|
|
$tempfile = new TemporaryFile(['prefix' => 'video']);
|
2020-08-14 02:46:54 +09:00
|
|
|
|
|
|
|
// Generate palette file. FFmpeg explictly needs to be told the
|
|
|
|
// extension for PNG files outputs
|
|
|
|
$palette = $this->tempnam_sfx(sys_get_temp_dir(), '.png');
|
|
|
|
|
|
|
|
// Build filters
|
|
|
|
$filters = 'fps=30';
|
|
|
|
$filters .= ",crop={$box['w']}:{$box['h']}:{$box['x']}:{$box['y']}";
|
|
|
|
$filters .= ",scale={$box['width']}:{$box['height']}:flags=lanczos";
|
|
|
|
|
|
|
|
// Assemble commands for palette generation
|
|
|
|
$commands[] = $commands_2[] = '-f';
|
|
|
|
$commands[] = $commands_2[] = 'gif';
|
|
|
|
$commands[] = $commands_2[] = '-i';
|
|
|
|
$commands[] = $commands_2[] = $imagefile->filepath;
|
|
|
|
$commands[] = '-vf';
|
|
|
|
$commands[] = $filters . ',palettegen';
|
|
|
|
$commands[] = '-y';
|
|
|
|
$commands[] = $palette;
|
|
|
|
|
|
|
|
// Assemble commands for GIF generation
|
|
|
|
$commands_2[] = '-i';
|
|
|
|
$commands_2[] = $palette;
|
|
|
|
$commands_2[] = '-lavfi';
|
|
|
|
$commands_2[] = $filters . ' [x]; [x][1:v] paletteuse';
|
|
|
|
$commands_2[] = '-f';
|
|
|
|
$commands_2[] = 'gif';
|
|
|
|
$commands_2[] = '-y';
|
2020-09-04 19:15:23 +09:00
|
|
|
$commands_2[] = $tempfile->getRealPath();
|
2020-08-14 02:46:54 +09:00
|
|
|
|
|
|
|
$success = true;
|
|
|
|
|
|
|
|
// Generate the palette image
|
|
|
|
try {
|
|
|
|
$ffmpeg->getFFMpegDriver()->command($commands);
|
|
|
|
} catch (Exception $e) {
|
|
|
|
$this->log(LOG_ERR, 'Unable to generate the palette image');
|
|
|
|
$success = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Generate GIF
|
|
|
|
try {
|
|
|
|
if ($success) {
|
|
|
|
$ffmpeg->getFFMpegDriver()->command($commands_2);
|
|
|
|
}
|
|
|
|
} catch (Exception $e) {
|
|
|
|
$this->log(LOG_ERR, 'Unable to generate the GIF image');
|
|
|
|
$success = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ($success) {
|
2020-09-15 02:37:48 +09:00
|
|
|
try {
|
|
|
|
$tempfile->commit($outpath);
|
|
|
|
} catch (TemporaryFileException $e) {
|
|
|
|
$this->log(LOG_ERR, 'Unable to save the GIF image');
|
|
|
|
$success = false;
|
|
|
|
}
|
2020-08-14 02:46:54 +09:00
|
|
|
}
|
|
|
|
|
|
|
|
@unlink($palette);
|
|
|
|
|
|
|
|
return $success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Suffix version of tempnam.
|
|
|
|
* Courtesy of tomas at slax dot org:
|
2021-04-15 00:44:45 +09:00
|
|
|
*
|
2020-08-14 02:46:54 +09:00
|
|
|
* @see https://www.php.net/manual/en/function.tempnam.php#98232
|
|
|
|
*/
|
|
|
|
private function tempnam_sfx(string $dir, string $suffix): string
|
|
|
|
{
|
|
|
|
do {
|
2021-04-15 00:44:45 +09:00
|
|
|
$file = $dir . '/' . mt_rand() . $suffix;
|
|
|
|
$fp = @fopen($file, 'x');
|
2020-08-14 02:46:54 +09:00
|
|
|
} while (!$fp);
|
|
|
|
|
|
|
|
fclose($fp);
|
|
|
|
return $file;
|
|
|
|
}
|
|
|
|
|
|
|
|
public function onPluginVersion(array &$versions): bool
|
|
|
|
{
|
|
|
|
$versions[] = ['name' => 'FFmpeg',
|
2021-04-15 00:44:45 +09:00
|
|
|
'version' => self::PLUGIN_VERSION,
|
|
|
|
'author' => 'Bruno Casteleiro',
|
|
|
|
'homepage' => 'https://notabug.org/diogo/gnu-social/src/nightly/plugins/FFmpeg',
|
|
|
|
'rawdescription' => // TRANS: Plugin description.
|
|
|
|
_m('Use PHP-FFMpeg for resizing animated GIFs'), ];
|
2020-08-14 02:46:54 +09:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|