angelovcom.net

Unnamed repository; edit this file 'description' to name the repository.
Log | Files | Refs

class-wp-image-editor-imagick.php (26873B)


      1 <?php
      2 /**
      3  * WordPress Imagick Image Editor
      4  *
      5  * @package WordPress
      6  * @subpackage Image_Editor
      7  */
      8 
      9 /**
     10  * WordPress Image Editor Class for Image Manipulation through Imagick PHP Module
     11  *
     12  * @since 3.5.0
     13  *
     14  * @see WP_Image_Editor
     15  */
     16 class WP_Image_Editor_Imagick extends WP_Image_Editor {
     17 	/**
     18 	 * Imagick object.
     19 	 *
     20 	 * @var Imagick
     21 	 */
     22 	protected $image;
     23 
     24 	public function __destruct() {
     25 		if ( $this->image instanceof Imagick ) {
     26 			// We don't need the original in memory anymore.
     27 			$this->image->clear();
     28 			$this->image->destroy();
     29 		}
     30 	}
     31 
     32 	/**
     33 	 * Checks to see if current environment supports Imagick.
     34 	 *
     35 	 * We require Imagick 2.2.0 or greater, based on whether the queryFormats()
     36 	 * method can be called statically.
     37 	 *
     38 	 * @since 3.5.0
     39 	 *
     40 	 * @param array $args
     41 	 * @return bool
     42 	 */
     43 	public static function test( $args = array() ) {
     44 
     45 		// First, test Imagick's extension and classes.
     46 		if ( ! extension_loaded( 'imagick' ) || ! class_exists( 'Imagick', false ) || ! class_exists( 'ImagickPixel', false ) ) {
     47 			return false;
     48 		}
     49 
     50 		if ( version_compare( phpversion( 'imagick' ), '2.2.0', '<' ) ) {
     51 			return false;
     52 		}
     53 
     54 		$required_methods = array(
     55 			'clear',
     56 			'destroy',
     57 			'valid',
     58 			'getimage',
     59 			'writeimage',
     60 			'getimageblob',
     61 			'getimagegeometry',
     62 			'getimageformat',
     63 			'setimageformat',
     64 			'setimagecompression',
     65 			'setimagecompressionquality',
     66 			'setimagepage',
     67 			'setoption',
     68 			'scaleimage',
     69 			'cropimage',
     70 			'rotateimage',
     71 			'flipimage',
     72 			'flopimage',
     73 			'readimage',
     74 			'readimageblob',
     75 		);
     76 
     77 		// Now, test for deep requirements within Imagick.
     78 		if ( ! defined( 'imagick::COMPRESSION_JPEG' ) ) {
     79 			return false;
     80 		}
     81 
     82 		$class_methods = array_map( 'strtolower', get_class_methods( 'Imagick' ) );
     83 		if ( array_diff( $required_methods, $class_methods ) ) {
     84 			return false;
     85 		}
     86 
     87 		return true;
     88 	}
     89 
     90 	/**
     91 	 * Checks to see if editor supports the mime-type specified.
     92 	 *
     93 	 * @since 3.5.0
     94 	 *
     95 	 * @param string $mime_type
     96 	 * @return bool
     97 	 */
     98 	public static function supports_mime_type( $mime_type ) {
     99 		$imagick_extension = strtoupper( self::get_extension( $mime_type ) );
    100 
    101 		if ( ! $imagick_extension ) {
    102 			return false;
    103 		}
    104 
    105 		// setIteratorIndex is optional unless mime is an animated format.
    106 		// Here, we just say no if you are missing it and aren't loading a jpeg.
    107 		if ( ! method_exists( 'Imagick', 'setIteratorIndex' ) && 'image/jpeg' !== $mime_type ) {
    108 				return false;
    109 		}
    110 
    111 		try {
    112 			// phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged
    113 			return ( (bool) @Imagick::queryFormats( $imagick_extension ) );
    114 		} catch ( Exception $e ) {
    115 			return false;
    116 		}
    117 	}
    118 
    119 	/**
    120 	 * Loads image from $this->file into new Imagick Object.
    121 	 *
    122 	 * @since 3.5.0
    123 	 *
    124 	 * @return true|WP_Error True if loaded; WP_Error on failure.
    125 	 */
    126 	public function load() {
    127 		if ( $this->image instanceof Imagick ) {
    128 			return true;
    129 		}
    130 
    131 		if ( ! is_file( $this->file ) && ! wp_is_stream( $this->file ) ) {
    132 			return new WP_Error( 'error_loading_image', __( 'File doesn&#8217;t exist?' ), $this->file );
    133 		}
    134 
    135 		/*
    136 		 * Even though Imagick uses less PHP memory than GD, set higher limit
    137 		 * for users that have low PHP.ini limits.
    138 		 */
    139 		wp_raise_memory_limit( 'image' );
    140 
    141 		try {
    142 			$this->image    = new Imagick();
    143 			$file_extension = strtolower( pathinfo( $this->file, PATHINFO_EXTENSION ) );
    144 
    145 			if ( 'pdf' === $file_extension ) {
    146 				$pdf_loaded = $this->pdf_load_source();
    147 
    148 				if ( is_wp_error( $pdf_loaded ) ) {
    149 					return $pdf_loaded;
    150 				}
    151 			} else {
    152 				if ( wp_is_stream( $this->file ) ) {
    153 					// Due to reports of issues with streams with `Imagick::readImageFile()`, uses `Imagick::readImageBlob()` instead.
    154 					$this->image->readImageBlob( file_get_contents( $this->file ), $this->file );
    155 				} else {
    156 					$this->image->readImage( $this->file );
    157 				}
    158 			}
    159 
    160 			if ( ! $this->image->valid() ) {
    161 				return new WP_Error( 'invalid_image', __( 'File is not an image.' ), $this->file );
    162 			}
    163 
    164 			// Select the first frame to handle animated images properly.
    165 			if ( is_callable( array( $this->image, 'setIteratorIndex' ) ) ) {
    166 				$this->image->setIteratorIndex( 0 );
    167 			}
    168 
    169 			$this->mime_type = $this->get_mime_type( $this->image->getImageFormat() );
    170 		} catch ( Exception $e ) {
    171 			return new WP_Error( 'invalid_image', $e->getMessage(), $this->file );
    172 		}
    173 
    174 		$updated_size = $this->update_size();
    175 
    176 		if ( is_wp_error( $updated_size ) ) {
    177 			return $updated_size;
    178 		}
    179 
    180 		return $this->set_quality();
    181 	}
    182 
    183 	/**
    184 	 * Sets Image Compression quality on a 1-100% scale.
    185 	 *
    186 	 * @since 3.5.0
    187 	 *
    188 	 * @param int $quality Compression Quality. Range: [1,100]
    189 	 * @return true|WP_Error True if set successfully; WP_Error on failure.
    190 	 */
    191 	public function set_quality( $quality = null ) {
    192 		$quality_result = parent::set_quality( $quality );
    193 		if ( is_wp_error( $quality_result ) ) {
    194 			return $quality_result;
    195 		} else {
    196 			$quality = $this->get_quality();
    197 		}
    198 
    199 		try {
    200 			switch ( $this->mime_type ) {
    201 				case 'image/jpeg':
    202 					$this->image->setImageCompressionQuality( $quality );
    203 					$this->image->setImageCompression( imagick::COMPRESSION_JPEG );
    204 					break;
    205 				case 'image/webp':
    206 					$webp_info = wp_get_webp_info( $this->file );
    207 
    208 					if ( 'lossless' === $webp_info['type'] ) {
    209 						// Use WebP lossless settings.
    210 						$this->image->setImageCompressionQuality( 100 );
    211 						$this->image->setOption( 'webp:lossless', 'true' );
    212 					} else {
    213 						$this->image->setImageCompressionQuality( $quality );
    214 					}
    215 					break;
    216 				default:
    217 					$this->image->setImageCompressionQuality( $quality );
    218 			}
    219 		} catch ( Exception $e ) {
    220 			return new WP_Error( 'image_quality_error', $e->getMessage() );
    221 		}
    222 		return true;
    223 	}
    224 
    225 
    226 	/**
    227 	 * Sets or updates current image size.
    228 	 *
    229 	 * @since 3.5.0
    230 	 *
    231 	 * @param int $width
    232 	 * @param int $height
    233 	 * @return true|WP_Error
    234 	 */
    235 	protected function update_size( $width = null, $height = null ) {
    236 		$size = null;
    237 		if ( ! $width || ! $height ) {
    238 			try {
    239 				$size = $this->image->getImageGeometry();
    240 			} catch ( Exception $e ) {
    241 				return new WP_Error( 'invalid_image', __( 'Could not read image size.' ), $this->file );
    242 			}
    243 		}
    244 
    245 		if ( ! $width ) {
    246 			$width = $size['width'];
    247 		}
    248 
    249 		if ( ! $height ) {
    250 			$height = $size['height'];
    251 		}
    252 
    253 		return parent::update_size( $width, $height );
    254 	}
    255 
    256 	/**
    257 	 * Resizes current image.
    258 	 *
    259 	 * At minimum, either a height or width must be provided.
    260 	 * If one of the two is set to null, the resize will
    261 	 * maintain aspect ratio according to the provided dimension.
    262 	 *
    263 	 * @since 3.5.0
    264 	 *
    265 	 * @param int|null $max_w Image width.
    266 	 * @param int|null $max_h Image height.
    267 	 * @param bool     $crop
    268 	 * @return true|WP_Error
    269 	 */
    270 	public function resize( $max_w, $max_h, $crop = false ) {
    271 		if ( ( $this->size['width'] == $max_w ) && ( $this->size['height'] == $max_h ) ) {
    272 			return true;
    273 		}
    274 
    275 		$dims = image_resize_dimensions( $this->size['width'], $this->size['height'], $max_w, $max_h, $crop );
    276 		if ( ! $dims ) {
    277 			return new WP_Error( 'error_getting_dimensions', __( 'Could not calculate resized image dimensions' ) );
    278 		}
    279 
    280 		list( $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h ) = $dims;
    281 
    282 		if ( $crop ) {
    283 			return $this->crop( $src_x, $src_y, $src_w, $src_h, $dst_w, $dst_h );
    284 		}
    285 
    286 		// Execute the resize.
    287 		$thumb_result = $this->thumbnail_image( $dst_w, $dst_h );
    288 		if ( is_wp_error( $thumb_result ) ) {
    289 			return $thumb_result;
    290 		}
    291 
    292 		return $this->update_size( $dst_w, $dst_h );
    293 	}
    294 
    295 	/**
    296 	 * Efficiently resize the current image
    297 	 *
    298 	 * This is a WordPress specific implementation of Imagick::thumbnailImage(),
    299 	 * which resizes an image to given dimensions and removes any associated profiles.
    300 	 *
    301 	 * @since 4.5.0
    302 	 *
    303 	 * @param int    $dst_w       The destination width.
    304 	 * @param int    $dst_h       The destination height.
    305 	 * @param string $filter_name Optional. The Imagick filter to use when resizing. Default 'FILTER_TRIANGLE'.
    306 	 * @param bool   $strip_meta  Optional. Strip all profiles, excluding color profiles, from the image. Default true.
    307 	 * @return void|WP_Error
    308 	 */
    309 	protected function thumbnail_image( $dst_w, $dst_h, $filter_name = 'FILTER_TRIANGLE', $strip_meta = true ) {
    310 		$allowed_filters = array(
    311 			'FILTER_POINT',
    312 			'FILTER_BOX',
    313 			'FILTER_TRIANGLE',
    314 			'FILTER_HERMITE',
    315 			'FILTER_HANNING',
    316 			'FILTER_HAMMING',
    317 			'FILTER_BLACKMAN',
    318 			'FILTER_GAUSSIAN',
    319 			'FILTER_QUADRATIC',
    320 			'FILTER_CUBIC',
    321 			'FILTER_CATROM',
    322 			'FILTER_MITCHELL',
    323 			'FILTER_LANCZOS',
    324 			'FILTER_BESSEL',
    325 			'FILTER_SINC',
    326 		);
    327 
    328 		/**
    329 		 * Set the filter value if '$filter_name' name is in the allowed list and the related
    330 		 * Imagick constant is defined or fall back to the default filter.
    331 		 */
    332 		if ( in_array( $filter_name, $allowed_filters, true ) && defined( 'Imagick::' . $filter_name ) ) {
    333 			$filter = constant( 'Imagick::' . $filter_name );
    334 		} else {
    335 			$filter = defined( 'Imagick::FILTER_TRIANGLE' ) ? Imagick::FILTER_TRIANGLE : false;
    336 		}
    337 
    338 		/**
    339 		 * Filters whether to strip metadata from images when they're resized.
    340 		 *
    341 		 * This filter only applies when resizing using the Imagick editor since GD
    342 		 * always strips profiles by default.
    343 		 *
    344 		 * @since 4.5.0
    345 		 *
    346 		 * @param bool $strip_meta Whether to strip image metadata during resizing. Default true.
    347 		 */
    348 		if ( apply_filters( 'image_strip_meta', $strip_meta ) ) {
    349 			$this->strip_meta(); // Fail silently if not supported.
    350 		}
    351 
    352 		try {
    353 			/*
    354 			 * To be more efficient, resample large images to 5x the destination size before resizing
    355 			 * whenever the output size is less that 1/3 of the original image size (1/3^2 ~= .111),
    356 			 * unless we would be resampling to a scale smaller than 128x128.
    357 			 */
    358 			if ( is_callable( array( $this->image, 'sampleImage' ) ) ) {
    359 				$resize_ratio  = ( $dst_w / $this->size['width'] ) * ( $dst_h / $this->size['height'] );
    360 				$sample_factor = 5;
    361 
    362 				if ( $resize_ratio < .111 && ( $dst_w * $sample_factor > 128 && $dst_h * $sample_factor > 128 ) ) {
    363 					$this->image->sampleImage( $dst_w * $sample_factor, $dst_h * $sample_factor );
    364 				}
    365 			}
    366 
    367 			/*
    368 			 * Use resizeImage() when it's available and a valid filter value is set.
    369 			 * Otherwise, fall back to the scaleImage() method for resizing, which
    370 			 * results in better image quality over resizeImage() with default filter
    371 			 * settings and retains backward compatibility with pre 4.5 functionality.
    372 			 */
    373 			if ( is_callable( array( $this->image, 'resizeImage' ) ) && $filter ) {
    374 				$this->image->setOption( 'filter:support', '2.0' );
    375 				$this->image->resizeImage( $dst_w, $dst_h, $filter, 1 );
    376 			} else {
    377 				$this->image->scaleImage( $dst_w, $dst_h );
    378 			}
    379 
    380 			// Set appropriate quality settings after resizing.
    381 			if ( 'image/jpeg' === $this->mime_type ) {
    382 				if ( is_callable( array( $this->image, 'unsharpMaskImage' ) ) ) {
    383 					$this->image->unsharpMaskImage( 0.25, 0.25, 8, 0.065 );
    384 				}
    385 
    386 				$this->image->setOption( 'jpeg:fancy-upsampling', 'off' );
    387 			}
    388 
    389 			if ( 'image/png' === $this->mime_type ) {
    390 				$this->image->setOption( 'png:compression-filter', '5' );
    391 				$this->image->setOption( 'png:compression-level', '9' );
    392 				$this->image->setOption( 'png:compression-strategy', '1' );
    393 				$this->image->setOption( 'png:exclude-chunk', 'all' );
    394 			}
    395 
    396 			/*
    397 			 * If alpha channel is not defined, set it opaque.
    398 			 *
    399 			 * Note that Imagick::getImageAlphaChannel() is only available if Imagick
    400 			 * has been compiled against ImageMagick version 6.4.0 or newer.
    401 			 */
    402 			if ( is_callable( array( $this->image, 'getImageAlphaChannel' ) )
    403 				&& is_callable( array( $this->image, 'setImageAlphaChannel' ) )
    404 				&& defined( 'Imagick::ALPHACHANNEL_UNDEFINED' )
    405 				&& defined( 'Imagick::ALPHACHANNEL_OPAQUE' )
    406 			) {
    407 				if ( $this->image->getImageAlphaChannel() === Imagick::ALPHACHANNEL_UNDEFINED ) {
    408 					$this->image->setImageAlphaChannel( Imagick::ALPHACHANNEL_OPAQUE );
    409 				}
    410 			}
    411 
    412 			// Limit the bit depth of resized images to 8 bits per channel.
    413 			if ( is_callable( array( $this->image, 'getImageDepth' ) ) && is_callable( array( $this->image, 'setImageDepth' ) ) ) {
    414 				if ( 8 < $this->image->getImageDepth() ) {
    415 					$this->image->setImageDepth( 8 );
    416 				}
    417 			}
    418 
    419 			if ( is_callable( array( $this->image, 'setInterlaceScheme' ) ) && defined( 'Imagick::INTERLACE_NO' ) ) {
    420 				$this->image->setInterlaceScheme( Imagick::INTERLACE_NO );
    421 			}
    422 		} catch ( Exception $e ) {
    423 			return new WP_Error( 'image_resize_error', $e->getMessage() );
    424 		}
    425 	}
    426 
    427 	/**
    428 	 * Create multiple smaller images from a single source.
    429 	 *
    430 	 * Attempts to create all sub-sizes and returns the meta data at the end. This
    431 	 * may result in the server running out of resources. When it fails there may be few
    432 	 * "orphaned" images left over as the meta data is never returned and saved.
    433 	 *
    434 	 * As of 5.3.0 the preferred way to do this is with `make_subsize()`. It creates
    435 	 * the new images one at a time and allows for the meta data to be saved after
    436 	 * each new image is created.
    437 	 *
    438 	 * @since 3.5.0
    439 	 *
    440 	 * @param array $sizes {
    441 	 *     An array of image size data arrays.
    442 	 *
    443 	 *     Either a height or width must be provided.
    444 	 *     If one of the two is set to null, the resize will
    445 	 *     maintain aspect ratio according to the provided dimension.
    446 	 *
    447 	 *     @type array $size {
    448 	 *         Array of height, width values, and whether to crop.
    449 	 *
    450 	 *         @type int  $width  Image width. Optional if `$height` is specified.
    451 	 *         @type int  $height Image height. Optional if `$width` is specified.
    452 	 *         @type bool $crop   Optional. Whether to crop the image. Default false.
    453 	 *     }
    454 	 * }
    455 	 * @return array An array of resized images' metadata by size.
    456 	 */
    457 	public function multi_resize( $sizes ) {
    458 		$metadata = array();
    459 
    460 		foreach ( $sizes as $size => $size_data ) {
    461 			$meta = $this->make_subsize( $size_data );
    462 
    463 			if ( ! is_wp_error( $meta ) ) {
    464 				$metadata[ $size ] = $meta;
    465 			}
    466 		}
    467 
    468 		return $metadata;
    469 	}
    470 
    471 	/**
    472 	 * Create an image sub-size and return the image meta data value for it.
    473 	 *
    474 	 * @since 5.3.0
    475 	 *
    476 	 * @param array $size_data {
    477 	 *     Array of size data.
    478 	 *
    479 	 *     @type int  $width  The maximum width in pixels.
    480 	 *     @type int  $height The maximum height in pixels.
    481 	 *     @type bool $crop   Whether to crop the image to exact dimensions.
    482 	 * }
    483 	 * @return array|WP_Error The image data array for inclusion in the `sizes` array in the image meta,
    484 	 *                        WP_Error object on error.
    485 	 */
    486 	public function make_subsize( $size_data ) {
    487 		if ( ! isset( $size_data['width'] ) && ! isset( $size_data['height'] ) ) {
    488 			return new WP_Error( 'image_subsize_create_error', __( 'Cannot resize the image. Both width and height are not set.' ) );
    489 		}
    490 
    491 		$orig_size  = $this->size;
    492 		$orig_image = $this->image->getImage();
    493 
    494 		if ( ! isset( $size_data['width'] ) ) {
    495 			$size_data['width'] = null;
    496 		}
    497 
    498 		if ( ! isset( $size_data['height'] ) ) {
    499 			$size_data['height'] = null;
    500 		}
    501 
    502 		if ( ! isset( $size_data['crop'] ) ) {
    503 			$size_data['crop'] = false;
    504 		}
    505 
    506 		$resized = $this->resize( $size_data['width'], $size_data['height'], $size_data['crop'] );
    507 
    508 		if ( is_wp_error( $resized ) ) {
    509 			$saved = $resized;
    510 		} else {
    511 			$saved = $this->_save( $this->image );
    512 
    513 			$this->image->clear();
    514 			$this->image->destroy();
    515 			$this->image = null;
    516 		}
    517 
    518 		$this->size  = $orig_size;
    519 		$this->image = $orig_image;
    520 
    521 		if ( ! is_wp_error( $saved ) ) {
    522 			unset( $saved['path'] );
    523 		}
    524 
    525 		return $saved;
    526 	}
    527 
    528 	/**
    529 	 * Crops Image.
    530 	 *
    531 	 * @since 3.5.0
    532 	 *
    533 	 * @param int  $src_x   The start x position to crop from.
    534 	 * @param int  $src_y   The start y position to crop from.
    535 	 * @param int  $src_w   The width to crop.
    536 	 * @param int  $src_h   The height to crop.
    537 	 * @param int  $dst_w   Optional. The destination width.
    538 	 * @param int  $dst_h   Optional. The destination height.
    539 	 * @param bool $src_abs Optional. If the source crop points are absolute.
    540 	 * @return true|WP_Error
    541 	 */
    542 	public function crop( $src_x, $src_y, $src_w, $src_h, $dst_w = null, $dst_h = null, $src_abs = false ) {
    543 		if ( $src_abs ) {
    544 			$src_w -= $src_x;
    545 			$src_h -= $src_y;
    546 		}
    547 
    548 		try {
    549 			$this->image->cropImage( $src_w, $src_h, $src_x, $src_y );
    550 			$this->image->setImagePage( $src_w, $src_h, 0, 0 );
    551 
    552 			if ( $dst_w || $dst_h ) {
    553 				// If destination width/height isn't specified,
    554 				// use same as width/height from source.
    555 				if ( ! $dst_w ) {
    556 					$dst_w = $src_w;
    557 				}
    558 				if ( ! $dst_h ) {
    559 					$dst_h = $src_h;
    560 				}
    561 
    562 				$thumb_result = $this->thumbnail_image( $dst_w, $dst_h );
    563 				if ( is_wp_error( $thumb_result ) ) {
    564 					return $thumb_result;
    565 				}
    566 
    567 				return $this->update_size();
    568 			}
    569 		} catch ( Exception $e ) {
    570 			return new WP_Error( 'image_crop_error', $e->getMessage() );
    571 		}
    572 
    573 		return $this->update_size();
    574 	}
    575 
    576 	/**
    577 	 * Rotates current image counter-clockwise by $angle.
    578 	 *
    579 	 * @since 3.5.0
    580 	 *
    581 	 * @param float $angle
    582 	 * @return true|WP_Error
    583 	 */
    584 	public function rotate( $angle ) {
    585 		/**
    586 		 * $angle is 360-$angle because Imagick rotates clockwise
    587 		 * (GD rotates counter-clockwise)
    588 		 */
    589 		try {
    590 			$this->image->rotateImage( new ImagickPixel( 'none' ), 360 - $angle );
    591 
    592 			// Normalise EXIF orientation data so that display is consistent across devices.
    593 			if ( is_callable( array( $this->image, 'setImageOrientation' ) ) && defined( 'Imagick::ORIENTATION_TOPLEFT' ) ) {
    594 				$this->image->setImageOrientation( Imagick::ORIENTATION_TOPLEFT );
    595 			}
    596 
    597 			// Since this changes the dimensions of the image, update the size.
    598 			$result = $this->update_size();
    599 			if ( is_wp_error( $result ) ) {
    600 				return $result;
    601 			}
    602 
    603 			$this->image->setImagePage( $this->size['width'], $this->size['height'], 0, 0 );
    604 		} catch ( Exception $e ) {
    605 			return new WP_Error( 'image_rotate_error', $e->getMessage() );
    606 		}
    607 
    608 		return true;
    609 	}
    610 
    611 	/**
    612 	 * Flips current image.
    613 	 *
    614 	 * @since 3.5.0
    615 	 *
    616 	 * @param bool $horz Flip along Horizontal Axis
    617 	 * @param bool $vert Flip along Vertical Axis
    618 	 * @return true|WP_Error
    619 	 */
    620 	public function flip( $horz, $vert ) {
    621 		try {
    622 			if ( $horz ) {
    623 				$this->image->flipImage();
    624 			}
    625 
    626 			if ( $vert ) {
    627 				$this->image->flopImage();
    628 			}
    629 
    630 			// Normalise EXIF orientation data so that display is consistent across devices.
    631 			if ( is_callable( array( $this->image, 'setImageOrientation' ) ) && defined( 'Imagick::ORIENTATION_TOPLEFT' ) ) {
    632 				$this->image->setImageOrientation( Imagick::ORIENTATION_TOPLEFT );
    633 			}
    634 		} catch ( Exception $e ) {
    635 			return new WP_Error( 'image_flip_error', $e->getMessage() );
    636 		}
    637 
    638 		return true;
    639 	}
    640 
    641 	/**
    642 	 * Check if a JPEG image has EXIF Orientation tag and rotate it if needed.
    643 	 *
    644 	 * As ImageMagick copies the EXIF data to the flipped/rotated image, proceed only
    645 	 * if EXIF Orientation can be reset afterwards.
    646 	 *
    647 	 * @since 5.3.0
    648 	 *
    649 	 * @return bool|WP_Error True if the image was rotated. False if no EXIF data or if the image doesn't need rotation.
    650 	 *                       WP_Error if error while rotating.
    651 	 */
    652 	public function maybe_exif_rotate() {
    653 		if ( is_callable( array( $this->image, 'setImageOrientation' ) ) && defined( 'Imagick::ORIENTATION_TOPLEFT' ) ) {
    654 			return parent::maybe_exif_rotate();
    655 		} else {
    656 			return new WP_Error( 'write_exif_error', __( 'The image cannot be rotated because the embedded meta data cannot be updated.' ) );
    657 		}
    658 	}
    659 
    660 	/**
    661 	 * Saves current image to file.
    662 	 *
    663 	 * @since 3.5.0
    664 	 *
    665 	 * @param string $destfilename
    666 	 * @param string $mime_type
    667 	 * @return array|WP_Error {'path'=>string, 'file'=>string, 'width'=>int, 'height'=>int, 'mime-type'=>string}
    668 	 */
    669 	public function save( $destfilename = null, $mime_type = null ) {
    670 		$saved = $this->_save( $this->image, $destfilename, $mime_type );
    671 
    672 		if ( ! is_wp_error( $saved ) ) {
    673 			$this->file      = $saved['path'];
    674 			$this->mime_type = $saved['mime-type'];
    675 
    676 			try {
    677 				$this->image->setImageFormat( strtoupper( $this->get_extension( $this->mime_type ) ) );
    678 			} catch ( Exception $e ) {
    679 				return new WP_Error( 'image_save_error', $e->getMessage(), $this->file );
    680 			}
    681 		}
    682 
    683 		return $saved;
    684 	}
    685 
    686 	/**
    687 	 * @param Imagick $image
    688 	 * @param string  $filename
    689 	 * @param string  $mime_type
    690 	 * @return array|WP_Error
    691 	 */
    692 	protected function _save( $image, $filename = null, $mime_type = null ) {
    693 		list( $filename, $extension, $mime_type ) = $this->get_output_format( $filename, $mime_type );
    694 
    695 		if ( ! $filename ) {
    696 			$filename = $this->generate_filename( null, null, $extension );
    697 		}
    698 
    699 		try {
    700 			// Store initial format.
    701 			$orig_format = $this->image->getImageFormat();
    702 
    703 			$this->image->setImageFormat( strtoupper( $this->get_extension( $mime_type ) ) );
    704 		} catch ( Exception $e ) {
    705 			return new WP_Error( 'image_save_error', $e->getMessage(), $filename );
    706 		}
    707 
    708 		$write_image_result = $this->write_image( $this->image, $filename );
    709 		if ( is_wp_error( $write_image_result ) ) {
    710 			return $write_image_result;
    711 		}
    712 
    713 		try {
    714 			// Reset original format.
    715 			$this->image->setImageFormat( $orig_format );
    716 		} catch ( Exception $e ) {
    717 			return new WP_Error( 'image_save_error', $e->getMessage(), $filename );
    718 		}
    719 
    720 		// Set correct file permissions.
    721 		$stat  = stat( dirname( $filename ) );
    722 		$perms = $stat['mode'] & 0000666; // Same permissions as parent folder, strip off the executable bits.
    723 		chmod( $filename, $perms );
    724 
    725 		return array(
    726 			'path'      => $filename,
    727 			/** This filter is documented in wp-includes/class-wp-image-editor-gd.php */
    728 			'file'      => wp_basename( apply_filters( 'image_make_intermediate_size', $filename ) ),
    729 			'width'     => $this->size['width'],
    730 			'height'    => $this->size['height'],
    731 			'mime-type' => $mime_type,
    732 		);
    733 	}
    734 
    735 	/**
    736 	 * Writes an image to a file or stream.
    737 	 *
    738 	 * @since 5.6.0
    739 	 *
    740 	 * @param Imagick $image
    741 	 * @param string  $filename The destination filename or stream URL.
    742 	 * @return true|WP_Error
    743 	 */
    744 	private function write_image( $image, $filename ) {
    745 		if ( wp_is_stream( $filename ) ) {
    746 			/*
    747 			 * Due to reports of issues with streams with `Imagick::writeImageFile()` and `Imagick::writeImage()`, copies the blob instead.
    748 			 * Checks for exact type due to: https://www.php.net/manual/en/function.file-put-contents.php
    749 			 */
    750 			if ( file_put_contents( $filename, $image->getImageBlob() ) === false ) {
    751 				return new WP_Error(
    752 					'image_save_error',
    753 					sprintf(
    754 						/* translators: %s: PHP function name. */
    755 						__( '%s failed while writing image to stream.' ),
    756 						'<code>file_put_contents()</code>'
    757 					),
    758 					$filename
    759 				);
    760 			} else {
    761 				return true;
    762 			}
    763 		} else {
    764 			$dirname = dirname( $filename );
    765 
    766 			if ( ! wp_mkdir_p( $dirname ) ) {
    767 				return new WP_Error(
    768 					'image_save_error',
    769 					sprintf(
    770 						/* translators: %s: Directory path. */
    771 						__( 'Unable to create directory %s. Is its parent directory writable by the server?' ),
    772 						esc_html( $dirname )
    773 					)
    774 				);
    775 			}
    776 
    777 			try {
    778 				return $image->writeImage( $filename );
    779 			} catch ( Exception $e ) {
    780 				return new WP_Error( 'image_save_error', $e->getMessage(), $filename );
    781 			}
    782 		}
    783 	}
    784 
    785 	/**
    786 	 * Streams current image to browser.
    787 	 *
    788 	 * @since 3.5.0
    789 	 *
    790 	 * @param string $mime_type The mime type of the image.
    791 	 * @return true|WP_Error True on success, WP_Error object on failure.
    792 	 */
    793 	public function stream( $mime_type = null ) {
    794 		list( $filename, $extension, $mime_type ) = $this->get_output_format( null, $mime_type );
    795 
    796 		try {
    797 			// Temporarily change format for stream.
    798 			$this->image->setImageFormat( strtoupper( $extension ) );
    799 
    800 			// Output stream of image content.
    801 			header( "Content-Type: $mime_type" );
    802 			print $this->image->getImageBlob();
    803 
    804 			// Reset image to original format.
    805 			$this->image->setImageFormat( $this->get_extension( $this->mime_type ) );
    806 		} catch ( Exception $e ) {
    807 			return new WP_Error( 'image_stream_error', $e->getMessage() );
    808 		}
    809 
    810 		return true;
    811 	}
    812 
    813 	/**
    814 	 * Strips all image meta except color profiles from an image.
    815 	 *
    816 	 * @since 4.5.0
    817 	 *
    818 	 * @return true|WP_Error True if stripping metadata was successful. WP_Error object on error.
    819 	 */
    820 	protected function strip_meta() {
    821 
    822 		if ( ! is_callable( array( $this->image, 'getImageProfiles' ) ) ) {
    823 			return new WP_Error(
    824 				'image_strip_meta_error',
    825 				sprintf(
    826 					/* translators: %s: ImageMagick method name. */
    827 					__( '%s is required to strip image meta.' ),
    828 					'<code>Imagick::getImageProfiles()</code>'
    829 				)
    830 			);
    831 		}
    832 
    833 		if ( ! is_callable( array( $this->image, 'removeImageProfile' ) ) ) {
    834 			return new WP_Error(
    835 				'image_strip_meta_error',
    836 				sprintf(
    837 					/* translators: %s: ImageMagick method name. */
    838 					__( '%s is required to strip image meta.' ),
    839 					'<code>Imagick::removeImageProfile()</code>'
    840 				)
    841 			);
    842 		}
    843 
    844 		/*
    845 		 * Protect a few profiles from being stripped for the following reasons:
    846 		 *
    847 		 * - icc:  Color profile information
    848 		 * - icm:  Color profile information
    849 		 * - iptc: Copyright data
    850 		 * - exif: Orientation data
    851 		 * - xmp:  Rights usage data
    852 		 */
    853 		$protected_profiles = array(
    854 			'icc',
    855 			'icm',
    856 			'iptc',
    857 			'exif',
    858 			'xmp',
    859 		);
    860 
    861 		try {
    862 			// Strip profiles.
    863 			foreach ( $this->image->getImageProfiles( '*', true ) as $key => $value ) {
    864 				if ( ! in_array( $key, $protected_profiles, true ) ) {
    865 					$this->image->removeImageProfile( $key );
    866 				}
    867 			}
    868 		} catch ( Exception $e ) {
    869 			return new WP_Error( 'image_strip_meta_error', $e->getMessage() );
    870 		}
    871 
    872 		return true;
    873 	}
    874 
    875 	/**
    876 	 * Sets up Imagick for PDF processing.
    877 	 * Increases rendering DPI and only loads first page.
    878 	 *
    879 	 * @since 4.7.0
    880 	 *
    881 	 * @return string|WP_Error File to load or WP_Error on failure.
    882 	 */
    883 	protected function pdf_setup() {
    884 		try {
    885 			// By default, PDFs are rendered in a very low resolution.
    886 			// We want the thumbnail to be readable, so increase the rendering DPI.
    887 			$this->image->setResolution( 128, 128 );
    888 
    889 			// Only load the first page.
    890 			return $this->file . '[0]';
    891 		} catch ( Exception $e ) {
    892 			return new WP_Error( 'pdf_setup_failed', $e->getMessage(), $this->file );
    893 		}
    894 	}
    895 
    896 	/**
    897 	 * Load the image produced by Ghostscript.
    898 	 *
    899 	 * Includes a workaround for a bug in Ghostscript 8.70 that prevents processing of some PDF files
    900 	 * when `use-cropbox` is set.
    901 	 *
    902 	 * @since 5.6.0
    903 	 *
    904 	 * @return true|WP_error
    905 	 */
    906 	protected function pdf_load_source() {
    907 		$filename = $this->pdf_setup();
    908 
    909 		if ( is_wp_error( $filename ) ) {
    910 			return $filename;
    911 		}
    912 
    913 		try {
    914 			// When generating thumbnails from cropped PDF pages, Imagemagick uses the uncropped
    915 			// area (resulting in unnecessary whitespace) unless the following option is set.
    916 			$this->image->setOption( 'pdf:use-cropbox', true );
    917 
    918 			// Reading image after Imagick instantiation because `setResolution`
    919 			// only applies correctly before the image is read.
    920 			$this->image->readImage( $filename );
    921 		} catch ( Exception $e ) {
    922 			// Attempt to run `gs` without the `use-cropbox` option. See #48853.
    923 			$this->image->setOption( 'pdf:use-cropbox', false );
    924 
    925 			$this->image->readImage( $filename );
    926 		}
    927 
    928 		return true;
    929 	}
    930 
    931 }