2020-01-26 13:19:35 +00:00
< ? php declare ( strict_types = 1 );
2019-06-12 17:54:06 -05:00
2019-10-10 10:32:01 -05:00
require_once " config.php " ;
2019-06-12 17:54:06 -05:00
/*
* This is used by the image transcoding code when there is an error while transcoding
*/
2019-06-14 13:47:50 +01:00
class ImageTranscodeException extends SCoreException
{
}
2019-06-12 17:54:06 -05:00
class TranscodeImage extends Extension
{
2020-02-04 00:46:36 +00:00
/** @var TranscodeImageTheme */
protected $theme ;
2019-06-27 12:26:09 -05:00
const ACTION_BULK_TRANSCODE = " bulk_transcode " ;
2020-06-14 11:05:55 -05:00
const INPUT_MIMES = [
" BMP " => MimeType :: BMP ,
" GIF " => MimeType :: GIF ,
" ICO " => MimeType :: ICO ,
" JPG " => MimeType :: JPEG ,
" PNG " => MimeType :: PNG ,
2020-10-26 07:25:47 -05:00
" PPM " => MimeType :: PPM ,
2020-06-14 11:05:55 -05:00
" PSD " => MimeType :: PSD ,
" TIFF " => MimeType :: TIFF ,
" WEBP " => MimeType :: WEBP
2019-06-12 17:54:06 -05:00
];
2020-06-14 11:05:55 -05:00
const OUTPUT_MIMES = [
2019-06-12 17:54:06 -05:00
" " => " " ,
2020-06-14 11:05:55 -05:00
" JPEG (lossy) " => MimeType :: JPEG ,
" PNG (lossless) " => MimeType :: PNG ,
" WEBP (lossy) " => MimeType :: WEBP ,
" WEBP (lossless) " => MimeType :: WEBP_LOSSLESS ,
2019-06-12 17:54:06 -05:00
];
/**
2019-06-13 11:45:34 -05:00
* Needs to be after upload , but before the processing extensions
2019-06-12 17:54:06 -05:00
*/
public function get_priority () : int
{
return 45 ;
}
public function onInitExt ( InitExtEvent $event )
{
global $config ;
2019-06-18 13:45:59 -05:00
$config -> set_default_bool ( TranscodeConfig :: ENABLED , true );
2020-06-16 19:06:15 -05:00
$config -> set_default_bool ( TranscodeConfig :: GET_ENABLED , false );
2019-06-18 13:45:59 -05:00
$config -> set_default_bool ( TranscodeConfig :: UPLOAD , false );
2019-06-24 10:05:16 -05:00
$config -> set_default_string ( TranscodeConfig :: ENGINE , MediaEngine :: GD );
2019-06-18 13:45:59 -05:00
$config -> set_default_int ( TranscodeConfig :: QUALITY , 80 );
2020-06-14 11:36:52 -05:00
$config -> set_default_string ( TranscodeConfig :: ALPHA_COLOR , Media :: DEFAULT_ALPHA_CONVERSION_COLOR );
2019-06-12 17:54:06 -05:00
2020-06-14 11:05:55 -05:00
foreach ( array_values ( self :: INPUT_MIMES ) as $mime ) {
$config -> set_default_string ( self :: get_mapping_name ( $mime ), " " );
2019-06-12 17:54:06 -05:00
}
}
2020-06-14 11:05:55 -05:00
private static function get_mapping_name ( string $mime ) : string
{
$mime = str_replace ( " . " , " _ " , $mime );
$mime = str_replace ( " / " , " _ " , $mime );
return TranscodeConfig :: UPLOAD_PREFIX . $mime ;
}
private static function get_mapping ( String $mime ) : ? string
{
global $config ;
return $config -> get_string ( self :: get_mapping_name ( $mime ));
}
private static function set_mapping ( String $from_mime , ? String $to_mime ) : void
{
global $config ;
$config -> set_string ( self :: get_mapping_name ( $from_mime ), $to_mime );
}
public static function get_enabled_mimes () : array
{
$output = [];
foreach ( array_values ( self :: INPUT_MIMES ) as $mime ) {
$value = self :: get_mapping ( $mime );
if ( ! empty ( $value )) {
$output [] = $mime ;
}
}
return $output ;
}
public function onDatabaseUpgrade ( DatabaseUpgradeEvent $event )
{
global $config ;
if ( $this -> get_version ( TranscodeConfig :: VERSION ) < 1 ) {
$old_extensions = [];
foreach ( array_values ( self :: INPUT_MIMES ) as $mime ) {
$old_extensions = array_merge ( $old_extensions , FileExtension :: get_all_for_mime ( $mime ));
}
foreach ( $old_extensions as $old_extension ) {
$oldValue = $this -> get_mapping ( $old_extension );
if ( ! empty ( $oldValue )) {
$from_mime = MimeType :: get_for_extension ( $old_extension );
if ( empty ( $from_mime )) {
continue ;
}
$to_mime = MimeType :: get_for_extension ( $oldValue );
if ( empty ( $to_mime )) {
continue ;
}
$this -> set_mapping ( $from_mime , $to_mime );
$this -> set_mapping ( $old_extension , null );
}
}
$this -> set_version ( TranscodeConfig :: VERSION , 1 );
}
}
2019-06-12 17:54:06 -05:00
public function onImageAdminBlockBuilding ( ImageAdminBlockBuildingEvent $event )
{
global $user , $config ;
2019-09-29 19:00:51 +01:00
if ( $user -> can ( Permissions :: EDIT_FILES )) {
2019-06-18 13:45:59 -05:00
$engine = $config -> get_string ( TranscodeConfig :: ENGINE );
2020-06-14 11:05:55 -05:00
if ( $this -> can_convert_mime ( $engine , $event -> image -> get_mime ())) {
$options = $this -> get_supported_output_mimes ( $engine , $event -> image -> get_mime ());
2019-06-12 17:54:06 -05:00
$event -> add_part ( $this -> theme -> get_transcode_html ( $event -> image , $options ));
}
}
}
2019-08-07 14:53:59 -05:00
2019-06-12 17:54:06 -05:00
public function onSetupBuilding ( SetupBuildingEvent $event )
{
global $config ;
2019-06-18 13:45:59 -05:00
$engine = $config -> get_string ( TranscodeConfig :: ENGINE );
2019-06-12 17:54:06 -05:00
$sb = new SetupBlock ( " Image Transcode " );
2019-06-26 23:00:49 -05:00
$sb -> start_table ();
2020-06-14 11:36:52 -05:00
$sb -> add_bool_option ( TranscodeConfig :: ENABLED , " Allow transcoding images " , true );
2020-06-16 19:06:15 -05:00
$sb -> add_bool_option ( TranscodeConfig :: GET_ENABLED , " Enable GET args " , true );
2020-06-14 11:36:52 -05:00
$sb -> add_bool_option ( TranscodeConfig :: UPLOAD , " Transcode on upload " , true );
$sb -> add_choice_option ( TranscodeConfig :: ENGINE , MediaEngine :: IMAGE_ENGINES , " Engine " , true );
2020-06-14 11:05:55 -05:00
foreach ( self :: INPUT_MIMES as $display => $mime ) {
if ( MediaEngine :: is_input_supported ( $engine , $mime )) {
$outputs = $this -> get_supported_output_mimes ( $engine , $mime );
$sb -> add_choice_option ( self :: get_mapping_name ( $mime ), $outputs , " $display " , true );
2019-06-14 13:47:50 +01:00
}
2019-06-12 17:54:06 -05:00
}
2020-06-11 16:58:19 -05:00
$sb -> add_int_option ( TranscodeConfig :: QUALITY , " Lossy Format Quality " , true );
2020-06-14 11:36:52 -05:00
$sb -> add_color_option ( TranscodeConfig :: ALPHA_COLOR , " Alpha Conversion Color " , true );
2019-06-26 23:00:49 -05:00
$sb -> end_table ();
2019-06-12 17:54:06 -05:00
$event -> panel -> add_block ( $sb );
}
public function onDataUpload ( DataUploadEvent $event )
{
2019-10-02 11:23:57 +01:00
global $config ;
2019-06-12 17:54:06 -05:00
2019-06-18 13:45:59 -05:00
if ( $config -> get_bool ( TranscodeConfig :: UPLOAD ) == true ) {
2020-06-14 11:05:55 -05:00
$mime = strtolower ( $event -> mime );
if ( $mime === MimeType :: GIF && MimeType :: is_animated_gif ( $event -> tmpname )) {
2019-06-12 17:54:06 -05:00
return ;
}
2020-06-14 11:05:55 -05:00
if ( in_array ( $mime , array_values ( self :: INPUT_MIMES ))) {
$target_mime = self :: get_mapping ( $mime );
if ( empty ( $target_mime )) {
2019-06-12 17:54:06 -05:00
return ;
}
try {
2020-06-14 11:05:55 -05:00
$new_image = $this -> transcode_image ( $event -> tmpname , $mime , $target_mime );
$event -> set_mime ( $target_mime );
2019-06-12 17:54:06 -05:00
$event -> set_tmpname ( $new_image );
2019-06-14 13:47:50 +01:00
} catch ( Exception $e ) {
log_error ( " transcode " , " Error while performing upload transcode: " . $e -> getMessage ());
// We don't want to interfere with the upload process,
2019-06-12 17:54:06 -05:00
// so if something goes wrong the untranscoded image jsut continues
}
}
2019-06-14 13:47:50 +01:00
}
2019-06-12 17:54:06 -05:00
}
public function onPageRequest ( PageRequestEvent $event )
{
global $page , $user ;
2019-09-29 19:00:51 +01:00
if ( $event -> page_matches ( " transcode " ) && $user -> can ( Permissions :: EDIT_FILES )) {
2019-11-04 00:42:06 +00:00
if ( $event -> count_args () >= 1 ) {
2019-11-04 00:40:10 +00:00
$image_id = int_escape ( $event -> get_arg ( 0 ));
2019-11-04 00:42:06 +00:00
} elseif ( isset ( $_POST [ 'image_id' ])) {
2019-11-04 00:40:10 +00:00
$image_id = int_escape ( $_POST [ 'image_id' ]);
2019-11-04 00:42:06 +00:00
} else {
2019-06-14 13:47:50 +01:00
throw new ImageTranscodeException ( " Can not resize Image: No valid Image ID given. " );
}
$image_obj = Image :: by_id ( $image_id );
if ( is_null ( $image_obj )) {
$this -> theme -> display_error ( 404 , " Image not found " , " No image in the database has the ID # $image_id " );
} else {
2020-06-14 11:05:55 -05:00
if ( isset ( $_POST [ 'transcode_mime' ])) {
2019-06-14 13:47:50 +01:00
try {
2020-06-14 11:05:55 -05:00
$this -> transcode_and_replace_image ( $image_obj , $_POST [ 'transcode_mime' ]);
2019-06-18 20:58:28 -05:00
$page -> set_mode ( PageMode :: REDIRECT );
2019-06-12 17:54:06 -05:00
$page -> set_redirect ( make_link ( " post/view/ " . $image_id ));
2019-06-14 13:47:50 +01:00
} catch ( ImageTranscodeException $e ) {
$this -> theme -> display_transcode_error ( $page , " Error Transcoding " , $e -> getMessage ());
}
}
}
}
2019-06-12 17:54:06 -05:00
}
2020-06-16 19:06:15 -05:00
public function onImageDownloading ( ImageDownloadingEvent $event )
{
global $config , $user ;
if ( $config -> get_bool ( TranscodeConfig :: GET_ENABLED ) &&
isset ( $_GET [ 'transcode' ]) &&
$user -> can ( Permissions :: EDIT_FILES ) &&
$this -> can_convert_mime ( $config -> get_string ( TranscodeConfig :: ENGINE ), $event -> image -> get_mime ())) {
$target_mime = $_GET [ 'transcode' ];
if ( ! MimeType :: is_mime ( $target_mime )) {
$target_mime = MimeType :: get_for_extension ( $target_mime );
}
if ( empty ( $target_mime )) {
throw new ImageTranscodeException ( " Unable to determine output MIME for " . $_GET [ 'transcode' ]);
}
MediaEngine :: is_output_supported ( $config -> get_string ( TranscodeConfig :: ENGINE ), $target_mime );
$source_mime = $event -> image -> get_mime ();
if ( $source_mime != $target_mime ) {
$tmp_filename = $this -> transcode_image ( $event -> path , $source_mime , $target_mime );
if ( $event -> file_modified === true && $event -> path != $event -> image -> get_image_filename ()) {
// This means that we're dealing with a temp file that will need cleaned up
unlink ( $event -> path );
}
$event -> path = $tmp_filename ;
$event -> mime = $target_mime ;
$event -> file_modified = true ;
}
}
}
2019-08-07 14:53:59 -05:00
2019-06-12 17:54:06 -05:00
public function onBulkActionBlockBuilding ( BulkActionBlockBuildingEvent $event )
{
global $user , $config ;
2019-06-18 13:45:59 -05:00
$engine = $config -> get_string ( TranscodeConfig :: ENGINE );
2019-06-12 17:54:06 -05:00
2019-09-29 19:00:51 +01:00
if ( $user -> can ( Permissions :: EDIT_FILES )) {
2020-08-28 09:11:14 -05:00
$event -> add_action ( self :: ACTION_BULK_TRANSCODE , " Transcode Image " , null , " " , $this -> theme -> get_transcode_picker_html ( $this -> get_supported_output_mimes ( $engine )));
2019-06-12 17:54:06 -05:00
}
}
public function onBulkAction ( BulkActionEvent $event )
{
2019-12-15 19:47:18 +00:00
global $user , $database , $page ;
2019-06-12 17:54:06 -05:00
2019-06-14 13:47:50 +01:00
switch ( $event -> action ) {
2019-06-18 13:45:59 -05:00
case self :: ACTION_BULK_TRANSCODE :
2020-06-14 11:05:55 -05:00
if ( ! isset ( $_POST [ 'transcode_mime' ])) {
2019-06-12 17:54:06 -05:00
return ;
}
2019-09-29 19:00:51 +01:00
if ( $user -> can ( Permissions :: EDIT_FILES )) {
2020-06-14 11:05:55 -05:00
$mime = $_POST [ 'transcode_mime' ];
2019-06-12 17:54:06 -05:00
$total = 0 ;
2020-02-24 14:49:40 +00:00
$size_difference = 0 ;
2019-07-05 10:24:46 -05:00
foreach ( $event -> items as $image ) {
2019-06-12 17:54:06 -05:00
try {
2020-06-12 13:39:57 -05:00
$database -> begin_transaction ();
2019-07-05 10:24:46 -05:00
2020-02-24 14:49:40 +00:00
$before_size = $image -> filesize ;
2020-06-14 11:05:55 -05:00
$new_image = $this -> transcode_and_replace_image ( $image , $mime );
2019-06-18 13:45:59 -05:00
// If a subsequent transcode fails, the database needs to have everything about the previous
// transcodes recorded already, otherwise the image entries will be stuck pointing to
// missing image files
2019-06-12 17:54:06 -05:00
$database -> commit ();
$total ++ ;
2020-02-24 14:49:40 +00:00
$size_difference += ( $before_size - $new_image -> filesize );
2019-06-14 13:47:50 +01:00
} catch ( Exception $e ) {
2020-06-14 11:05:55 -05:00
log_error ( " transcode " , " Error while bulk transcode on item { $image -> id } to $mime : " . $e -> getMessage ());
2019-06-12 17:54:06 -05:00
try {
$database -> rollback ();
2019-06-14 13:47:50 +01:00
} catch ( Exception $e ) {
2019-10-02 10:10:47 +01:00
// is this safe? o.o
2019-06-14 13:47:50 +01:00
}
2019-06-12 17:54:06 -05:00
}
}
2020-02-24 14:49:40 +00:00
if ( $size_difference > 0 ) {
2020-02-24 15:30:23 +00:00
$page -> flash ( " Transcoded $total items, reduced size by " . human_filesize ( $size_difference ));
2020-02-24 14:49:40 +00:00
} elseif ( $size_difference < 0 ) {
2020-02-24 15:30:23 +00:00
$page -> flash ( " Transcoded $total items, increased size by " . human_filesize ( - 1 * $size_difference ));
2020-02-24 14:49:40 +00:00
} else {
$page -> flash ( " Transcoded $total items, no size difference " );
}
2019-06-12 17:54:06 -05:00
}
break ;
}
}
2020-06-14 11:05:55 -05:00
private function can_convert_mime ( $engine , $mime ) : bool
2019-06-12 17:54:06 -05:00
{
2020-06-14 11:05:55 -05:00
return MediaEngine :: is_input_supported ( $engine , $mime );
2019-06-12 17:54:06 -05:00
}
2019-06-18 13:45:59 -05:00
2020-06-14 11:05:55 -05:00
private function get_supported_output_mimes ( $engine , ? String $omit_mime = null ) : array
2019-06-12 17:54:06 -05:00
{
$output = [];
2019-06-25 15:17:13 -05:00
2020-06-14 11:05:55 -05:00
foreach ( self :: OUTPUT_MIMES as $key => $value ) {
2019-06-14 13:47:50 +01:00
if ( $value == " " ) {
2019-06-12 17:54:06 -05:00
$output [ $key ] = $value ;
continue ;
}
2020-06-14 11:05:55 -05:00
if ( MediaEngine :: is_output_supported ( $engine , $value )
&& ( empty ( $omit_mime ) || $omit_mime != $value )) {
2019-06-12 17:54:06 -05:00
$output [ $key ] = $value ;
2019-06-14 13:47:50 +01:00
}
2019-06-12 17:54:06 -05:00
}
return $output ;
}
2019-08-07 14:53:59 -05:00
2019-06-18 13:45:59 -05:00
2019-06-12 17:54:06 -05:00
2020-06-14 11:05:55 -05:00
private function transcode_and_replace_image ( Image $image_obj , String $target_mime ) : Image
2019-06-12 17:54:06 -05:00
{
2019-06-15 11:18:52 -05:00
$original_file = warehouse_path ( Image :: IMAGE_DIR , $image_obj -> hash );
2019-06-12 17:54:06 -05:00
2020-06-14 11:05:55 -05:00
$tmp_filename = $this -> transcode_image ( $original_file , $image_obj -> get_mime (), $target_mime );
2019-08-07 14:53:59 -05:00
2019-06-12 17:54:06 -05:00
$new_image = new Image ();
$new_image -> hash = md5_file ( $tmp_filename );
$new_image -> filesize = filesize ( $tmp_filename );
$new_image -> filename = $image_obj -> filename ;
$new_image -> width = $image_obj -> width ;
$new_image -> height = $image_obj -> height ;
/* Move the new image into the main storage location */
2019-06-15 11:18:52 -05:00
$target = warehouse_path ( Image :: IMAGE_DIR , $new_image -> hash );
2019-06-12 17:54:06 -05:00
if ( !@ copy ( $tmp_filename , $target )) {
throw new ImageTranscodeException ( " Failed to copy new image file from temporary location ( { $tmp_filename } ) to archive ( $target ) " );
}
2019-08-07 14:53:59 -05:00
2019-06-12 17:54:06 -05:00
/* Remove temporary file */
@ unlink ( $tmp_filename );
send_event ( new ImageReplaceEvent ( $image_obj -> id , $new_image ));
2020-02-24 14:49:40 +00:00
return $new_image ;
2019-06-14 13:47:50 +01:00
}
2019-06-12 17:54:06 -05:00
2020-06-14 11:05:55 -05:00
private function transcode_image ( String $source_name , String $source_mime , string $target_mime ) : string
2019-06-12 17:54:06 -05:00
{
global $config ;
2020-06-14 11:05:55 -05:00
if ( $source_mime == $target_mime ) {
throw new ImageTranscodeException ( " Source and target MIMEs are the same: " . $source_mime );
2019-06-12 17:54:06 -05:00
}
2020-06-14 11:36:52 -05:00
$engine = $config -> get_string ( TranscodeConfig :: ENGINE );
2019-06-12 17:54:06 -05:00
2020-06-14 11:05:55 -05:00
if ( ! $this -> can_convert_mime ( $engine , $source_mime )) {
throw new ImageTranscodeException ( " Engine $engine does not support input MIME $source_mime " );
2019-06-12 17:54:06 -05:00
}
2020-06-14 11:05:55 -05:00
if ( ! MediaEngine :: is_output_supported ( $engine , $target_mime )) {
throw new ImageTranscodeException ( " Engine $engine does not support output MIME $target_mime " );
2019-06-12 17:54:06 -05:00
}
2019-06-14 13:47:50 +01:00
switch ( $engine ) {
2019-06-12 17:54:06 -05:00
case " gd " :
2020-06-14 11:05:55 -05:00
return $this -> transcode_image_gd ( $source_name , $source_mime , $target_mime );
2019-06-12 17:54:06 -05:00
case " convert " :
2020-06-14 11:05:55 -05:00
return $this -> transcode_image_convert ( $source_name , $source_mime , $target_mime );
2019-10-02 11:23:57 +01:00
default :
throw new ImageTranscodeException ( " No engine specified " );
2019-06-12 17:54:06 -05:00
}
}
2020-06-14 11:05:55 -05:00
private function transcode_image_gd ( String $source_name , String $source_mime , string $target_mime ) : string
2019-06-12 17:54:06 -05:00
{
global $config ;
2019-08-07 14:53:59 -05:00
2020-08-28 09:11:14 -05:00
$q = $config -> get_int ( TranscodeConfig :: QUALITY );
2019-06-12 17:54:06 -05:00
2020-06-16 18:18:56 -05:00
$tmp_name = tempnam ( sys_get_temp_dir (), " shimmie_transcode " );
2019-06-12 17:54:06 -05:00
$image = imagecreatefromstring ( file_get_contents ( $source_name ));
try {
$result = false ;
2020-06-14 11:05:55 -05:00
switch ( $target_mime ) {
case MimeType :: WEBP :
2019-06-12 17:54:06 -05:00
$result = imagewebp ( $image , $tmp_name , $q );
break ;
2020-06-14 11:05:55 -05:00
case MimeType :: PNG :
2019-06-12 17:54:06 -05:00
$result = imagepng ( $image , $tmp_name , 9 );
break ;
2020-06-14 11:05:55 -05:00
case MimeType :: JPEG :
2019-06-12 17:54:06 -05:00
// In case of alpha channels
$width = imagesx ( $image );
$height = imagesy ( $image );
$new_image = imagecreatetruecolor ( $width , $height );
2019-06-14 13:47:50 +01:00
if ( $new_image === false ) {
2019-06-12 17:54:06 -05:00
throw new ImageTranscodeException ( " Could not create image with dimensions $width x $height " );
}
2019-06-14 13:47:50 +01:00
try {
2020-06-14 11:36:52 -05:00
$background_color = Media :: hex_color_allocate ( $new_image , $config -> get_string ( TranscodeConfig :: ALPHA_COLOR ));
if ( $background_color === false ) {
2019-06-12 17:54:06 -05:00
throw new ImageTranscodeException ( " Could not allocate background color " );
}
2020-06-14 11:36:52 -05:00
if ( imagefilledrectangle ( $new_image , 0 , 0 , $width , $height , $background_color ) === false ) {
2019-06-12 17:54:06 -05:00
throw new ImageTranscodeException ( " Could not fill background color " );
}
2019-06-14 13:47:50 +01:00
if ( imagecopy ( $new_image , $image , 0 , 0 , 0 , 0 , $width , $height ) === false ) {
2019-06-12 17:54:06 -05:00
throw new ImageTranscodeException ( " Could not copy source image to new image " );
}
$result = imagejpeg ( $new_image , $tmp_name , $q );
} finally {
imagedestroy ( $new_image );
}
break ;
}
} finally {
imagedestroy ( $image );
}
2019-11-04 01:04:08 +00:00
if ( $result === false ) {
2020-06-14 11:05:55 -05:00
throw new ImageTranscodeException ( " Error while transcoding " . $source_name . " to " . $target_mime );
2019-11-04 01:04:08 +00:00
}
return $tmp_name ;
2019-06-12 17:54:06 -05:00
}
2020-06-14 11:05:55 -05:00
private function transcode_image_convert ( String $source_name , String $source_mime , string $target_mime ) : string
2019-06-12 17:54:06 -05:00
{
global $config ;
2019-08-07 14:53:59 -05:00
2020-08-28 09:11:14 -05:00
$q = $config -> get_int ( TranscodeConfig :: QUALITY );
2019-06-24 10:05:16 -05:00
$convert = $config -> get_string ( MediaConfig :: CONVERT_PATH );
2019-06-12 17:54:06 -05:00
2020-08-28 09:11:14 -05:00
if ( empty ( $convert )) {
2019-06-12 17:54:06 -05:00
throw new ImageTranscodeException ( " ImageMagick path not configured " );
}
2020-06-14 11:05:55 -05:00
$ext = Media :: determine_ext ( $target_mime );
2019-06-12 17:54:06 -05:00
2020-06-14 11:36:52 -05:00
$args = " -background " ;
2020-06-14 11:05:55 -05:00
if ( Media :: supports_alpha ( $target_mime )) {
$args .= " none " ;
} else {
2020-06-14 11:36:52 -05:00
$args .= " \" " . $config -> get_string ( TranscodeConfig :: ALPHA_COLOR ) . " \" " ;
2020-06-14 11:05:55 -05:00
}
2020-06-14 11:36:52 -05:00
$args .= " -flatten " ;
2020-06-14 11:05:55 -05:00
switch ( $target_mime ) {
case MimeType :: PNG :
$args .= ' -define png:compression-level=9' ;
2019-06-12 17:54:06 -05:00
break ;
2020-06-14 11:05:55 -05:00
case MimeType :: WEBP_LOSSLESS :
$args .= ' -define webp:lossless=true -quality 100 ' ;
2019-06-12 17:54:06 -05:00
break ;
default :
2020-06-14 11:05:55 -05:00
$args .= ' -quality ' . $q ;
2019-06-12 17:54:06 -05:00
break ;
}
2020-06-16 18:18:56 -05:00
$tmp_name = tempnam ( sys_get_temp_dir (), " shimmie_transcode " );
2019-06-12 17:54:06 -05:00
2020-06-14 11:05:55 -05:00
$source_type = FileExtension :: get_for_mime ( $source_mime );
$format = '"%s" %s:"%s" %s %s:"%s" 2>&1' ;
$cmd = sprintf ( $format , $convert , $source_type , $source_name , $args , $ext , $tmp_name );
2019-06-14 12:59:12 -05:00
2019-06-12 17:54:06 -05:00
$cmd = str_replace ( " \" convert \" " , " convert " , $cmd ); // quotes are only needed if the path to convert contains a space; some other times, quotes break things, see github bug #27
exec ( $cmd , $output , $ret );
log_debug ( 'transcode' , " Transcoding with command ` $cmd `, returns $ret " );
2019-06-14 13:47:50 +01:00
if ( $ret !== 0 ) {
2019-06-14 12:59:12 -05:00
throw new ImageTranscodeException ( " Transcoding failed with command " . $cmd . " , returning " . implode ( " \r \n " , $output ));
2019-06-12 17:54:06 -05:00
}
return $tmp_name ;
}
}