[ Index ]

PHP Cross Reference of WordPress Trunk (Updated Daily)

Search

title

Body

[close]

/wp-includes/ -> class-wp-image-editor-gd.php (source)

   1  <?php
   2  /**
   3   * WordPress GD Image Editor
   4   *
   5   * @package WordPress
   6   * @subpackage Image_Editor
   7   */
   8  
   9  /**
  10   * WordPress Image Editor Class for Image Manipulation through GD
  11   *
  12   * @since 3.5.0
  13   *
  14   * @see WP_Image_Editor
  15   */
  16  class WP_Image_Editor_GD extends WP_Image_Editor {
  17      /**
  18       * GD Resource.
  19       *
  20       * @var resource|GdImage
  21       */
  22      protected $image;
  23  
  24  	public function __destruct() {
  25          if ( $this->image ) {
  26              // We don't need the original in memory anymore.
  27              imagedestroy( $this->image );
  28          }
  29      }
  30  
  31      /**
  32       * Checks to see if current environment supports GD.
  33       *
  34       * @since 3.5.0
  35       *
  36       * @param array $args
  37       * @return bool
  38       */
  39  	public static function test( $args = array() ) {
  40          if ( ! extension_loaded( 'gd' ) || ! function_exists( 'gd_info' ) ) {
  41              return false;
  42          }
  43  
  44          // On some setups GD library does not provide imagerotate() - Ticket #11536.
  45          if ( isset( $args['methods'] ) &&
  46              in_array( 'rotate', $args['methods'], true ) &&
  47              ! function_exists( 'imagerotate' ) ) {
  48  
  49                  return false;
  50          }
  51  
  52          return true;
  53      }
  54  
  55      /**
  56       * Checks to see if editor supports the mime-type specified.
  57       *
  58       * @since 3.5.0
  59       *
  60       * @param string $mime_type
  61       * @return bool
  62       */
  63  	public static function supports_mime_type( $mime_type ) {
  64          $image_types = imagetypes();
  65          switch ( $mime_type ) {
  66              case 'image/jpeg':
  67                  return ( $image_types & IMG_JPG ) !== 0;
  68              case 'image/png':
  69                  return ( $image_types & IMG_PNG ) !== 0;
  70              case 'image/gif':
  71                  return ( $image_types & IMG_GIF ) !== 0;
  72              case 'image/webp':
  73                  return ( $image_types & IMG_WEBP ) !== 0;
  74              case 'image/avif':
  75                  return ( $image_types & IMG_AVIF ) !== 0 && function_exists( 'imageavif' );
  76          }
  77  
  78          return false;
  79      }
  80  
  81      /**
  82       * Loads image from $this->file into new GD Resource.
  83       *
  84       * @since 3.5.0
  85       *
  86       * @return true|WP_Error True if loaded successfully; WP_Error on failure.
  87       */
  88  	public function load() {
  89          if ( $this->image ) {
  90              return true;
  91          }
  92  
  93          if ( ! is_file( $this->file ) && ! preg_match( '|^https?://|', $this->file ) ) {
  94              return new WP_Error( 'error_loading_image', __( 'File does not exist?' ), $this->file );
  95          }
  96  
  97          // Set artificially high because GD uses uncompressed images in memory.
  98          wp_raise_memory_limit( 'image' );
  99  
 100          $file_contents = @file_get_contents( $this->file );
 101  
 102          if ( ! $file_contents ) {
 103              return new WP_Error( 'error_loading_image', __( 'File does not exist?' ), $this->file );
 104          }
 105  
 106          // Handle WebP and AVIF mime types explicitly, falling back to imagecreatefromstring.
 107          if (
 108              function_exists( 'imagecreatefromwebp' ) && ( 'image/webp' === wp_get_image_mime( $this->file ) )
 109          ) {
 110              $this->image = @imagecreatefromwebp( $this->file );
 111          } elseif (
 112              function_exists( 'imagecreatefromavif' ) && ( 'image/avif' === wp_get_image_mime( $this->file ) )
 113          ) {
 114              $this->image = @imagecreatefromavif( $this->file );
 115          } else {
 116              $this->image = @imagecreatefromstring( $file_contents );
 117          }
 118  
 119          if ( ! is_gd_image( $this->image ) ) {
 120              return new WP_Error( 'invalid_image', __( 'File is not an image.' ), $this->file );
 121          }
 122  
 123          $size = wp_getimagesize( $this->file );
 124  
 125          if ( ! $size ) {
 126              return new WP_Error( 'invalid_image', __( 'Could not read image size.' ), $this->file );
 127          }
 128  
 129          if ( function_exists( 'imagealphablending' ) && function_exists( 'imagesavealpha' ) ) {
 130              imagealphablending( $this->image, false );
 131              imagesavealpha( $this->image, true );
 132          }
 133  
 134          $this->update_size( $size[0], $size[1] );
 135          $this->mime_type = $size['mime'];
 136  
 137          return $this->set_quality();
 138      }
 139  
 140      /**
 141       * Sets or updates current image size.
 142       *
 143       * @since 3.5.0
 144       *
 145       * @param int $width
 146       * @param int $height
 147       * @return true
 148       */
 149  	protected function update_size( $width = false, $height = false ) {
 150          if ( ! $width ) {
 151              $width = imagesx( $this->image );
 152          }
 153  
 154          if ( ! $height ) {
 155              $height = imagesy( $this->image );
 156          }
 157  
 158          return parent::update_size( $width, $height );
 159      }
 160  
 161      /**
 162       * Resizes current image.
 163       *
 164       * Wraps `::_resize()` which returns a GD resource or GdImage instance.
 165       *
 166       * At minimum, either a height or width must be provided. If one of the two is set
 167       * to null, the resize will maintain aspect ratio according to the provided dimension.
 168       *
 169       * @since 3.5.0
 170       *
 171       * @param int|null   $max_w Image width.
 172       * @param int|null   $max_h Image height.
 173       * @param bool|array $crop  {
 174       *     Optional. Image cropping behavior. If false, the image will be scaled (default).
 175       *     If true, image will be cropped to the specified dimensions using center positions.
 176       *     If an array, the image will be cropped using the array to specify the crop location:
 177       *
 178       *     @type string $0 The x crop position. Accepts 'left', 'center', or 'right'.
 179       *     @type string $1 The y crop position. Accepts 'top', 'center', or 'bottom'.
 180       * }
 181       * @return true|WP_Error
 182       */
 183  	public function resize( $max_w, $max_h, $crop = false ) {
 184          if ( ( $this->size['width'] === $max_w ) && ( $this->size['height'] === $max_h ) ) {
 185              return true;
 186          }
 187  
 188          $resized = $this->_resize( $max_w, $max_h, $crop );
 189  
 190          if ( is_gd_image( $resized ) ) {
 191              imagedestroy( $this->image );
 192              $this->image = $resized;
 193              return true;
 194  
 195          } elseif ( is_wp_error( $resized ) ) {
 196              return $resized;
 197          }
 198  
 199          return new WP_Error( 'image_resize_error', __( 'Image resize failed.' ), $this->file );
 200      }
 201  
 202      /**
 203       * @param int        $max_w
 204       * @param int        $max_h
 205       * @param bool|array $crop  {
 206       *     Optional. Image cropping behavior. If false, the image will be scaled (default).
 207       *     If true, image will be cropped to the specified dimensions using center positions.
 208       *     If an array, the image will be cropped using the array to specify the crop location:
 209       *
 210       *     @type string $0 The x crop position. Accepts 'left', 'center', or 'right'.
 211       *     @type string $1 The y crop position. Accepts 'top', 'center', or 'bottom'.
 212       * }
 213       * @return resource|GdImage|WP_Error
 214       */
 215  	protected function _resize( $max_w, $max_h, $crop = false ) {
 216          $dims = image_resize_dimensions( $this->size['width'], $this->size['height'], $max_w, $max_h, $crop );
 217  
 218          if ( ! $dims ) {
 219              return new WP_Error( 'error_getting_dimensions', __( 'Could not calculate resized image dimensions' ), $this->file );
 220          }
 221  
 222          list( $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h ) = $dims;
 223  
 224          $this->set_quality(
 225              null,
 226              array(
 227                  'width'  => $dst_w,
 228                  'height' => $dst_h,
 229              )
 230          );
 231  
 232          $resized = wp_imagecreatetruecolor( $dst_w, $dst_h );
 233          imagecopyresampled( $resized, $this->image, $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h );
 234  
 235          if ( is_gd_image( $resized ) ) {
 236              $this->update_size( $dst_w, $dst_h );
 237              return $resized;
 238          }
 239  
 240          return new WP_Error( 'image_resize_error', __( 'Image resize failed.' ), $this->file );
 241      }
 242  
 243      /**
 244       * Create multiple smaller images from a single source.
 245       *
 246       * Attempts to create all sub-sizes and returns the meta data at the end. This
 247       * may result in the server running out of resources. When it fails there may be few
 248       * "orphaned" images left over as the meta data is never returned and saved.
 249       *
 250       * As of 5.3.0 the preferred way to do this is with `make_subsize()`. It creates
 251       * the new images one at a time and allows for the meta data to be saved after
 252       * each new image is created.
 253       *
 254       * @since 3.5.0
 255       *
 256       * @param array $sizes {
 257       *     An array of image size data arrays.
 258       *
 259       *     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 source image.
 262       *
 263       *     @type array ...$0 {
 264       *         Array of height, width values, and whether to crop.
 265       *
 266       *         @type int        $width  Image width. Optional if `$height` is specified.
 267       *         @type int        $height Image height. Optional if `$width` is specified.
 268       *         @type bool|array $crop   Optional. Whether to crop the image. Default false.
 269       *     }
 270       * }
 271       * @return array An array of resized images' metadata by size.
 272       */
 273  	public function multi_resize( $sizes ) {
 274          $metadata = array();
 275  
 276          foreach ( $sizes as $size => $size_data ) {
 277              $meta = $this->make_subsize( $size_data );
 278  
 279              if ( ! is_wp_error( $meta ) ) {
 280                  $metadata[ $size ] = $meta;
 281              }
 282          }
 283  
 284          return $metadata;
 285      }
 286  
 287      /**
 288       * Create an image sub-size and return the image meta data value for it.
 289       *
 290       * @since 5.3.0
 291       *
 292       * @param array $size_data {
 293       *     Array of size data.
 294       *
 295       *     @type int        $width  The maximum width in pixels.
 296       *     @type int        $height The maximum height in pixels.
 297       *     @type bool|array $crop   Whether to crop the image to exact dimensions.
 298       * }
 299       * @return array|WP_Error The image data array for inclusion in the `sizes` array in the image meta,
 300       *                        WP_Error object on error.
 301       */
 302  	public function make_subsize( $size_data ) {
 303          if ( ! isset( $size_data['width'] ) && ! isset( $size_data['height'] ) ) {
 304              return new WP_Error( 'image_subsize_create_error', __( 'Cannot resize the image. Both width and height are not set.' ) );
 305          }
 306  
 307          $orig_size = $this->size;
 308  
 309          if ( ! isset( $size_data['width'] ) ) {
 310              $size_data['width'] = null;
 311          }
 312  
 313          if ( ! isset( $size_data['height'] ) ) {
 314              $size_data['height'] = null;
 315          }
 316  
 317          if ( ! isset( $size_data['crop'] ) ) {
 318              $size_data['crop'] = false;
 319          }
 320  
 321          $resized = $this->_resize( $size_data['width'], $size_data['height'], $size_data['crop'] );
 322  
 323          if ( is_wp_error( $resized ) ) {
 324              $saved = $resized;
 325          } else {
 326              $saved = $this->_save( $resized );
 327              imagedestroy( $resized );
 328          }
 329  
 330          $this->size = $orig_size;
 331  
 332          if ( ! is_wp_error( $saved ) ) {
 333              unset( $saved['path'] );
 334          }
 335  
 336          return $saved;
 337      }
 338  
 339      /**
 340       * Crops Image.
 341       *
 342       * @since 3.5.0
 343       *
 344       * @param int  $src_x   The start x position to crop from.
 345       * @param int  $src_y   The start y position to crop from.
 346       * @param int  $src_w   The width to crop.
 347       * @param int  $src_h   The height to crop.
 348       * @param int  $dst_w   Optional. The destination width.
 349       * @param int  $dst_h   Optional. The destination height.
 350       * @param bool $src_abs Optional. If the source crop points are absolute.
 351       * @return true|WP_Error
 352       */
 353  	public function crop( $src_x, $src_y, $src_w, $src_h, $dst_w = null, $dst_h = null, $src_abs = false ) {
 354          /*
 355           * If destination width/height isn't specified,
 356           * use same as width/height from source.
 357           */
 358          if ( ! $dst_w ) {
 359              $dst_w = $src_w;
 360          }
 361          if ( ! $dst_h ) {
 362              $dst_h = $src_h;
 363          }
 364  
 365          foreach ( array( $src_w, $src_h, $dst_w, $dst_h ) as $value ) {
 366              if ( ! is_numeric( $value ) || (int) $value <= 0 ) {
 367                  return new WP_Error( 'image_crop_error', __( 'Image crop failed.' ), $this->file );
 368              }
 369          }
 370  
 371          $dst = wp_imagecreatetruecolor( (int) $dst_w, (int) $dst_h );
 372  
 373          if ( $src_abs ) {
 374              $src_w -= $src_x;
 375              $src_h -= $src_y;
 376          }
 377  
 378          if ( function_exists( 'imageantialias' ) ) {
 379              imageantialias( $dst, true );
 380          }
 381  
 382          imagecopyresampled( $dst, $this->image, 0, 0, (int) $src_x, (int) $src_y, (int) $dst_w, (int) $dst_h, (int) $src_w, (int) $src_h );
 383  
 384          if ( is_gd_image( $dst ) ) {
 385              imagedestroy( $this->image );
 386              $this->image = $dst;
 387              $this->update_size();
 388              return true;
 389          }
 390  
 391          return new WP_Error( 'image_crop_error', __( 'Image crop failed.' ), $this->file );
 392      }
 393  
 394      /**
 395       * Rotates current image counter-clockwise by $angle.
 396       * Ported from image-edit.php
 397       *
 398       * @since 3.5.0
 399       *
 400       * @param float $angle
 401       * @return true|WP_Error
 402       */
 403  	public function rotate( $angle ) {
 404          if ( function_exists( 'imagerotate' ) ) {
 405              $transparency = imagecolorallocatealpha( $this->image, 255, 255, 255, 127 );
 406              $rotated      = imagerotate( $this->image, $angle, $transparency );
 407  
 408              if ( is_gd_image( $rotated ) ) {
 409                  imagealphablending( $rotated, true );
 410                  imagesavealpha( $rotated, true );
 411                  imagedestroy( $this->image );
 412                  $this->image = $rotated;
 413                  $this->update_size();
 414                  return true;
 415              }
 416          }
 417  
 418          return new WP_Error( 'image_rotate_error', __( 'Image rotate failed.' ), $this->file );
 419      }
 420  
 421      /**
 422       * Flips current image.
 423       *
 424       * @since 3.5.0
 425       *
 426       * @param bool $horz Flip along Horizontal Axis.
 427       * @param bool $vert Flip along Vertical Axis.
 428       * @return true|WP_Error
 429       */
 430  	public function flip( $horz, $vert ) {
 431          $w   = $this->size['width'];
 432          $h   = $this->size['height'];
 433          $dst = wp_imagecreatetruecolor( $w, $h );
 434  
 435          if ( is_gd_image( $dst ) ) {
 436              $sx = $vert ? ( $w - 1 ) : 0;
 437              $sy = $horz ? ( $h - 1 ) : 0;
 438              $sw = $vert ? -$w : $w;
 439              $sh = $horz ? -$h : $h;
 440  
 441              if ( imagecopyresampled( $dst, $this->image, 0, 0, $sx, $sy, $w, $h, $sw, $sh ) ) {
 442                  imagedestroy( $this->image );
 443                  $this->image = $dst;
 444                  return true;
 445              }
 446          }
 447  
 448          return new WP_Error( 'image_flip_error', __( 'Image flip failed.' ), $this->file );
 449      }
 450  
 451      /**
 452       * Saves current in-memory image to file.
 453       *
 454       * @since 3.5.0
 455       * @since 5.9.0 Renamed `$filename` to `$destfilename` to match parent class
 456       *              for PHP 8 named parameter support.
 457       * @since 6.0.0 The `$filesize` value was added to the returned array.
 458       *
 459       * @param string|null $destfilename Optional. Destination filename. Default null.
 460       * @param string|null $mime_type    Optional. The mime-type. Default null.
 461       * @return array|WP_Error {
 462       *     Array on success or WP_Error if the file failed to save.
 463       *
 464       *     @type string $path      Path to the image file.
 465       *     @type string $file      Name of the image file.
 466       *     @type int    $width     Image width.
 467       *     @type int    $height    Image height.
 468       *     @type string $mime-type The mime type of the image.
 469       *     @type int    $filesize  File size of the image.
 470       * }
 471       */
 472  	public function save( $destfilename = null, $mime_type = null ) {
 473          $saved = $this->_save( $this->image, $destfilename, $mime_type );
 474  
 475          if ( ! is_wp_error( $saved ) ) {
 476              $this->file      = $saved['path'];
 477              $this->mime_type = $saved['mime-type'];
 478          }
 479  
 480          return $saved;
 481      }
 482  
 483      /**
 484       * @since 3.5.0
 485       * @since 6.0.0 The `$filesize` value was added to the returned array.
 486       *
 487       * @param resource|GdImage $image
 488       * @param string|null      $filename
 489       * @param string|null      $mime_type
 490       * @return array|WP_Error {
 491       *     Array on success or WP_Error if the file failed to save.
 492       *
 493       *     @type string $path      Path to the image file.
 494       *     @type string $file      Name of the image file.
 495       *     @type int    $width     Image width.
 496       *     @type int    $height    Image height.
 497       *     @type string $mime-type The mime type of the image.
 498       *     @type int    $filesize  File size of the image.
 499       * }
 500       */
 501  	protected function _save( $image, $filename = null, $mime_type = null ) {
 502          list( $filename, $extension, $mime_type ) = $this->get_output_format( $filename, $mime_type );
 503  
 504          if ( ! $filename ) {
 505              $filename = $this->generate_filename( null, null, $extension );
 506          }
 507  
 508          if ( function_exists( 'imageinterlace' ) ) {
 509              /**
 510               * Filters whether to output progressive images (if available).
 511               *
 512               * @since 6.5.0
 513               *
 514               * @param bool   $interlace Whether to use progressive images for output if available. Default false.
 515               * @param string $mime_type The mime type being saved.
 516               */
 517              imageinterlace( $image, apply_filters( 'image_save_progressive', false, $mime_type ) );
 518          }
 519  
 520          if ( 'image/gif' === $mime_type ) {
 521              if ( ! $this->make_image( $filename, 'imagegif', array( $image, $filename ) ) ) {
 522                  return new WP_Error( 'image_save_error', __( 'Image Editor Save Failed' ) );
 523              }
 524          } elseif ( 'image/png' === $mime_type ) {
 525              // Convert from full colors to index colors, like original PNG.
 526              if ( function_exists( 'imageistruecolor' ) && ! imageistruecolor( $image ) ) {
 527                  imagetruecolortopalette( $image, false, imagecolorstotal( $image ) );
 528              }
 529  
 530              if ( ! $this->make_image( $filename, 'imagepng', array( $image, $filename ) ) ) {
 531                  return new WP_Error( 'image_save_error', __( 'Image Editor Save Failed' ) );
 532              }
 533          } elseif ( 'image/jpeg' === $mime_type ) {
 534              if ( ! $this->make_image( $filename, 'imagejpeg', array( $image, $filename, $this->get_quality() ) ) ) {
 535                  return new WP_Error( 'image_save_error', __( 'Image Editor Save Failed' ) );
 536              }
 537          } elseif ( 'image/webp' === $mime_type ) {
 538              if ( ! function_exists( 'imagewebp' )
 539                  || ! $this->make_image( $filename, 'imagewebp', array( $image, $filename, $this->get_quality() ) )
 540              ) {
 541                  return new WP_Error( 'image_save_error', __( 'Image Editor Save Failed' ) );
 542              }
 543          } elseif ( 'image/avif' === $mime_type ) {
 544              if ( ! function_exists( 'imageavif' )
 545                  || ! $this->make_image( $filename, 'imageavif', array( $image, $filename, $this->get_quality() ) )
 546              ) {
 547                  return new WP_Error( 'image_save_error', __( 'Image Editor Save Failed' ) );
 548              }
 549          } else {
 550              return new WP_Error( 'image_save_error', __( 'Image Editor Save Failed' ) );
 551          }
 552  
 553          // Set correct file permissions.
 554          $stat  = stat( dirname( $filename ) );
 555          $perms = $stat['mode'] & 0000666; // Same permissions as parent folder, strip off the executable bits.
 556          chmod( $filename, $perms );
 557  
 558          return array(
 559              'path'      => $filename,
 560              /**
 561               * Filters the name of the saved image file.
 562               *
 563               * @since 2.6.0
 564               *
 565               * @param string $filename Name of the file.
 566               */
 567              'file'      => wp_basename( apply_filters( 'image_make_intermediate_size', $filename ) ),
 568              'width'     => $this->size['width'],
 569              'height'    => $this->size['height'],
 570              'mime-type' => $mime_type,
 571              'filesize'  => wp_filesize( $filename ),
 572          );
 573      }
 574  
 575      /**
 576       * Sets Image Compression quality on a 1-100% scale. Handles WebP lossless images.
 577       *
 578       * @since 6.7.0
 579       * @since 6.8.0 The `$dims` parameter was added.
 580       *
 581       * @param int   $quality Compression Quality. Range: [1,100]
 582       * @param array $dims    Optional. Image dimensions array with 'width' and 'height' keys.
 583       * @return true|WP_Error True if set successfully; WP_Error on failure.
 584       */
 585  	public function set_quality( $quality = null, $dims = array() ) {
 586          $quality_result = parent::set_quality( $quality, $dims );
 587          if ( is_wp_error( $quality_result ) ) {
 588              return $quality_result;
 589          } else {
 590              $quality = $this->get_quality();
 591          }
 592  
 593          // Handle setting the quality for WebP lossless images, see https://php.watch/versions/8.1/gd-webp-lossless.
 594          try {
 595              if ( 'image/webp' === $this->mime_type && defined( 'IMG_WEBP_LOSSLESS' ) ) {
 596                  $webp_info = wp_get_webp_info( $this->file );
 597                  if ( ! empty( $webp_info['type'] ) && 'lossless' === $webp_info['type'] ) {
 598                      $quality = IMG_WEBP_LOSSLESS;
 599                      parent::set_quality( $quality, $dims );
 600                  }
 601              }
 602          } catch ( Exception $e ) {
 603              return new WP_Error( 'image_quality_error', $e->getMessage() );
 604          }
 605          $this->quality = $quality;
 606          return true;
 607      }
 608  
 609      /**
 610       * Returns stream of current image.
 611       *
 612       * @since 3.5.0
 613       *
 614       * @param string $mime_type The mime type of the image.
 615       * @return bool True on success, false on failure.
 616       */
 617  	public function stream( $mime_type = null ) {
 618          list( $filename, $extension, $mime_type ) = $this->get_output_format( null, $mime_type );
 619  
 620          switch ( $mime_type ) {
 621              case 'image/png':
 622                  header( 'Content-Type: image/png' );
 623                  return imagepng( $this->image );
 624              case 'image/gif':
 625                  header( 'Content-Type: image/gif' );
 626                  return imagegif( $this->image );
 627              case 'image/webp':
 628                  if ( function_exists( 'imagewebp' ) ) {
 629                      header( 'Content-Type: image/webp' );
 630                      return imagewebp( $this->image, null, $this->get_quality() );
 631                  } else {
 632                      // Fall back to JPEG.
 633                      header( 'Content-Type: image/jpeg' );
 634                      return imagejpeg( $this->image, null, $this->get_quality() );
 635                  }
 636              case 'image/avif':
 637                  if ( function_exists( 'imageavif' ) ) {
 638                      header( 'Content-Type: image/avif' );
 639                      return imageavif( $this->image, null, $this->get_quality() );
 640                  }
 641                  // Fall back to JPEG.
 642              default:
 643                  header( 'Content-Type: image/jpeg' );
 644                  return imagejpeg( $this->image, null, $this->get_quality() );
 645          }
 646      }
 647  
 648      /**
 649       * Either calls editor's save function or handles file as a stream.
 650       *
 651       * @since 3.5.0
 652       *
 653       * @param string   $filename
 654       * @param callable $callback
 655       * @param array    $arguments
 656       * @return bool
 657       */
 658  	protected function make_image( $filename, $callback, $arguments ) {
 659          if ( wp_is_stream( $filename ) ) {
 660              $arguments[1] = null;
 661          }
 662  
 663          return parent::make_image( $filename, $callback, $arguments );
 664      }
 665  }


Generated : Tue Jan 21 08:20:01 2025 Cross-referenced by PHPXref