123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689 |
- <?php
- /**
- * Support for image manipulation using [GD](http://php.net/GD).
- *
- * @package Kohana/Image
- * @category Drivers
- * @author Kohana Team
- * @copyright (c) Kohana Team
- * @license https://koseven.ga/LICENSE.md
- */
- class Kohana_Image_GD extends Image {
- // Which GD functions are available?
- const IMAGEROTATE = 'imagerotate';
- const IMAGECONVOLUTION = 'imageconvolution';
- const IMAGEFILTER = 'imagefilter';
- const IMAGELAYEREFFECT = 'imagelayereffect';
- protected static $_available_functions = [];
- /**
- * Checks if GD is enabled and verify that key methods exist, some of which require GD to
- * be bundled with PHP. Exceptions will be thrown from those methods when GD is not
- * bundled.
- *
- * @return boolean
- */
- public static function check()
- {
- if ( ! function_exists('gd_info'))
- {
- throw new Kohana_Exception('GD is either not installed or not enabled, check your configuration');
- }
- $functions = [
- Image_GD::IMAGEROTATE,
- Image_GD::IMAGECONVOLUTION,
- Image_GD::IMAGEFILTER,
- Image_GD::IMAGELAYEREFFECT
- ];
- foreach ($functions as $function)
- {
- Image_GD::$_available_functions[$function] = function_exists($function);
- }
- if (defined('GD_VERSION'))
- {
- // Get the version via a constant, available in PHP 5.2.4+
- $version = GD_VERSION;
- }
- else
- {
- // Get the version information
- $info = gd_info();
- // Extract the version number
- preg_match('/\d+\.\d+(?:\.\d+)?/', $info['GD Version'], $matches);
- // Get the major version
- $version = $matches[0];
- }
- if ( ! version_compare($version, '2.0.1', '>='))
- {
- throw new Kohana_Exception('Image_GD requires GD version :required or greater, you have :version',
- ['required' => '2.0.1', ':version' => $version]);
- }
- return Image_GD::$_checked = TRUE;
- }
- // Temporary image resource
- protected $_image;
- // Function name to open Image
- protected $_create_function;
- /**
- * Runs [Image_GD::check] and loads the image.
- *
- * @param string $file image file path
- * @return void
- * @throws Kohana_Exception
- */
- public function __construct($file)
- {
- if ( ! Image_GD::$_checked)
- {
- // Run the install check
- Image_GD::check();
- }
- parent::__construct($file);
- // Set the image creation function name
- switch ($this->type)
- {
- case IMAGETYPE_JPEG:
- $create = 'imagecreatefromjpeg';
- break;
- case IMAGETYPE_GIF:
- $create = 'imagecreatefromgif';
- break;
- case IMAGETYPE_PNG:
- $create = 'imagecreatefrompng';
- break;
- case self::IMAGETYPE_WEBP:
- $create = 'imagecreatefromwebp';
- break;
- }
- if ( ! isset($create) OR ! function_exists($create))
- {
- throw new Kohana_Exception('Installed GD does not support :type images',
- [':type' => image_type_to_extension($this->type, FALSE)]);
- }
- // Save function for future use
- $this->_create_function = $create;
- // Save filename for lazy loading
- $this->_image = $this->file;
- }
- /**
- * Destroys the loaded image to free up resources.
- *
- * @return void
- */
- public function __destruct()
- {
- if ( $this->_has_image() )
- {
- // Free all resources
- imagedestroy($this->_image);
- }
- }
- /**
- * Loads an image into GD.
- *
- * @return void
- */
- protected function _load_image()
- {
- if ( ! $this->_has_image() )
- {
- // Gets create function
- $create = $this->_create_function;
- // Open the temporary image
- $this->_image = $create($this->file);
- // Preserve transparency when saving
- imagesavealpha($this->_image, TRUE);
- }
- }
- /**
- * checks if an image has been set
- *
- * @return boolean
- */
- protected function _has_image()
- {
- if ( version_compare(PHP_VERSION, '8', '>=') )
- {
- return is_object($this->_image);
- }
- return is_resource($this->_image);
- }
- /**
- * Execute a resize.
- *
- * @param integer $width new width
- * @param integer $height new height
- * @return void
- */
- protected function _do_resize($width, $height)
- {
- // Presize width and height
- $pre_width = $this->width;
- $pre_height = $this->height;
- // Loads image if not yet loaded
- $this->_load_image();
- // Test if we can do a resize without resampling to speed up the final resize
- if ($width > ($this->width / 2) AND $height > ($this->height / 2))
- {
- // The maximum reduction is 10% greater than the final size
- $reduction_width = round($width * 1.1);
- $reduction_height = round($height * 1.1);
- while ($pre_width / 2 > $reduction_width AND $pre_height / 2 > $reduction_height)
- {
- // Reduce the size using an O(2n) algorithm, until it reaches the maximum reduction
- $pre_width /= 2;
- $pre_height /= 2;
- }
- // Create the temporary image to copy to
- $image = $this->_create($pre_width, $pre_height);
- if (imagecopyresized($image, $this->_image, 0, 0, 0, 0, $pre_width, $pre_height, $this->width, $this->height))
- {
- // Swap the new image for the old one
- imagedestroy($this->_image);
- $this->_image = $image;
- }
- }
- // Create the temporary image to copy to
- $image = $this->_create($width, $height);
- // Execute the resize
- if (imagecopyresampled($image, $this->_image, 0, 0, 0, 0, $width, $height, $pre_width, $pre_height))
- {
- // Swap the new image for the old one
- imagedestroy($this->_image);
- $this->_image = $image;
- // Reset the width and height
- $this->width = imagesx($image);
- $this->height = imagesy($image);
- }
- }
- /**
- * Execute a crop.
- *
- * @param integer $width new width
- * @param integer $height new height
- * @param integer $offset_x offset from the left
- * @param integer $offset_y offset from the top
- * @return void
- */
- protected function _do_crop($width, $height, $offset_x, $offset_y)
- {
- // Create the temporary image to copy to
- $image = $this->_create($width, $height);
- // Loads image if not yet loaded
- $this->_load_image();
- // Execute the crop
- if (imagecopyresampled($image, $this->_image, 0, 0, $offset_x, $offset_y, $width, $height, $width, $height))
- {
- // Swap the new image for the old one
- imagedestroy($this->_image);
- $this->_image = $image;
- // Reset the width and height
- $this->width = imagesx($image);
- $this->height = imagesy($image);
- }
- }
- /**
- * Execute a rotation.
- *
- * @param integer $degrees degrees to rotate
- * @return void
- */
- protected function _do_rotate($degrees)
- {
- if (empty(Image_GD::$_available_functions[Image_GD::IMAGEROTATE]))
- {
- throw new Kohana_Exception('This method requires :function, which is only available in the bundled version of GD',
- [':function' => 'imagerotate']);
- }
- // Loads image if not yet loaded
- $this->_load_image();
- // Transparent black will be used as the background for the uncovered region
- $transparent = imagecolorallocatealpha($this->_image, 0, 0, 0, 127);
- // Rotate, setting the transparent color
- $image = imagerotate($this->_image, 360 - $degrees, $transparent, 1);
- // Save the alpha of the rotated image
- imagesavealpha($image, TRUE);
- // Get the width and height of the rotated image
- $width = imagesx($image);
- $height = imagesy($image);
- if (imagecopymerge($this->_image, $image, 0, 0, 0, 0, $width, $height, 100))
- {
- // Swap the new image for the old one
- imagedestroy($this->_image);
- $this->_image = $image;
- // Reset the width and height
- $this->width = $width;
- $this->height = $height;
- }
- }
- /**
- * Execute a flip.
- *
- * @param integer $direction direction to flip
- * @return void
- */
- protected function _do_flip($direction)
- {
- // Create the flipped image
- $flipped = $this->_create($this->width, $this->height);
- // Loads image if not yet loaded
- $this->_load_image();
- if ($direction === Image::HORIZONTAL)
- {
- for ($x = 0; $x < $this->width; $x++)
- {
- // Flip each row from top to bottom
- imagecopy($flipped, $this->_image, $x, 0, $this->width - $x - 1, 0, 1, $this->height);
- }
- }
- else
- {
- for ($y = 0; $y < $this->height; $y++)
- {
- // Flip each column from left to right
- imagecopy($flipped, $this->_image, 0, $y, 0, $this->height - $y - 1, $this->width, 1);
- }
- }
- // Swap the new image for the old one
- imagedestroy($this->_image);
- $this->_image = $flipped;
- // Reset the width and height
- $this->width = imagesx($flipped);
- $this->height = imagesy($flipped);
- }
- /**
- * Execute a sharpen.
- *
- * @param integer $amount amount to sharpen
- * @return void
- */
- protected function _do_sharpen($amount)
- {
- if (empty(Image_GD::$_available_functions[Image_GD::IMAGECONVOLUTION]))
- {
- throw new Kohana_Exception('This method requires :function, which is only available in the bundled version of GD',
- [':function' => 'imageconvolution']);
- }
- // Loads image if not yet loaded
- $this->_load_image();
- // Amount should be in the range of 18-10
- $amount = round(abs(-18 + ($amount * 0.08)), 2);
- // Gaussian blur matrix
- $matrix = [
- [-1, -1, -1],
- [-1, $amount, -1],
- [-1, -1, -1],
- ];
- // Perform the sharpen
- if (imageconvolution($this->_image, $matrix, $amount - 8, 0))
- {
- // Reset the width and height
- $this->width = imagesx($this->_image);
- $this->height = imagesy($this->_image);
- }
- }
- /**
- * Execute a reflection.
- *
- * @param integer $height reflection height
- * @param integer $opacity reflection opacity
- * @param boolean $fade_in TRUE to fade out, FALSE to fade in
- * @return void
- */
- protected function _do_reflection($height, $opacity, $fade_in)
- {
- if (empty(Image_GD::$_available_functions[Image_GD::IMAGEFILTER]))
- {
- throw new Kohana_Exception('This method requires :function, which is only available in the bundled version of GD',
- [':function' => 'imagefilter']);
- }
- // Loads image if not yet loaded
- $this->_load_image();
- // Convert an opacity range of 0-100 to 127-0
- $opacity = round(abs(($opacity * 127 / 100) - 127));
- if ($opacity < 127)
- {
- // Calculate the opacity stepping
- $stepping = (127 - $opacity) / $height;
- }
- else
- {
- // Avoid a "divide by zero" error
- $stepping = 127 / $height;
- }
- // Create the reflection image
- $reflection = $this->_create($this->width, $this->height + $height);
- // Copy the image to the reflection
- imagecopy($reflection, $this->_image, 0, 0, 0, 0, $this->width, $this->height);
- for ($offset = 0; $height >= $offset; $offset++)
- {
- // Read the next line down
- $src_y = $this->height - $offset - 1;
- // Place the line at the bottom of the reflection
- $dst_y = $this->height + $offset;
- if ($fade_in === TRUE)
- {
- // Start with the most transparent line first
- $dst_opacity = round($opacity + ($stepping * ($height - $offset)));
- }
- else
- {
- // Start with the most opaque line first
- $dst_opacity = round($opacity + ($stepping * $offset));
- }
- // Create a single line of the image
- $line = $this->_create($this->width, 1);
- // Copy a single line from the current image into the line
- imagecopy($line, $this->_image, 0, 0, 0, $src_y, $this->width, 1);
- // Colorize the line to add the correct alpha level
- imagefilter($line, IMG_FILTER_COLORIZE, 0, 0, 0, $dst_opacity);
- // Copy a the line into the reflection
- imagecopy($reflection, $line, 0, $dst_y, 0, 0, $this->width, 1);
- }
- // Swap the new image for the old one
- imagedestroy($this->_image);
- $this->_image = $reflection;
- // Reset the width and height
- $this->width = imagesx($reflection);
- $this->height = imagesy($reflection);
- }
- /**
- * Execute a watermarking.
- *
- * @param Image $image watermarking Image
- * @param integer $offset_x offset from the left
- * @param integer $offset_y offset from the top
- * @param integer $opacity opacity of watermark
- * @return void
- */
- protected function _do_watermark(Image $watermark, $offset_x, $offset_y, $opacity)
- {
- if (empty(Image_GD::$_available_functions[Image_GD::IMAGELAYEREFFECT]))
- {
- throw new Kohana_Exception('This method requires :function, which is only available in the bundled version of GD',
- [':function' => 'imagelayereffect']);
- }
- // Loads image if not yet loaded
- $this->_load_image();
- // Create the watermark image resource
- $overlay = imagecreatefromstring($watermark->render());
- imagesavealpha($overlay, TRUE);
- // Get the width and height of the watermark
- $width = imagesx($overlay);
- $height = imagesy($overlay);
- if ($opacity < 100)
- {
- // Convert an opacity range of 0-100 to 127-0
- $opacity = round(abs(($opacity * 127 / 100) - 127));
- // Allocate transparent gray
- $color = imagecolorallocatealpha($overlay, 127, 127, 127, $opacity);
- // The transparent image will overlay the watermark
- imagelayereffect($overlay, IMG_EFFECT_OVERLAY);
- // Fill the background with the transparent color
- imagefilledrectangle($overlay, 0, 0, $width, $height, $color);
- }
- // Alpha blending must be enabled on the background!
- imagealphablending($this->_image, TRUE);
- if (imagecopy($this->_image, $overlay, $offset_x, $offset_y, 0, 0, $width, $height))
- {
- // Destroy the overlay image
- imagedestroy($overlay);
- }
- }
- /**
- * Execute a background.
- *
- * @param integer $r red
- * @param integer $g green
- * @param integer $b blue
- * @param integer $opacity opacity
- * @return void
- */
- protected function _do_background($r, $g, $b, $opacity)
- {
- // Loads image if not yet loaded
- $this->_load_image();
- // Convert an opacity range of 0-100 to 127-0
- $opacity = round(abs(($opacity * 127 / 100) - 127));
- // Create a new background
- $background = $this->_create($this->width, $this->height);
- // Allocate the color
- $color = imagecolorallocatealpha($background, $r, $g, $b, $opacity);
- // Fill the image with white
- imagefilledrectangle($background, 0, 0, $this->width, $this->height, $color);
- // Alpha blending must be enabled on the background!
- imagealphablending($background, TRUE);
- // Copy the image onto a white background to remove all transparency
- if (imagecopy($background, $this->_image, 0, 0, 0, 0, $this->width, $this->height))
- {
- // Swap the new image for the old one
- imagedestroy($this->_image);
- $this->_image = $background;
- }
- }
- /**
- * Execute a save.
- *
- * @param string $file new image filename
- * @param integer $quality quality
- * @return boolean
- */
- protected function _do_save($file, $quality)
- {
- // Loads image if not yet loaded
- $this->_load_image();
- // Get the extension of the file
- $extension = pathinfo($file, PATHINFO_EXTENSION);
- // Get the save function and IMAGETYPE
- list($save, $type) = $this->_save_function($extension, $quality);
- // Save the image to a file
- $status = isset($quality) ? $save($this->_image, $file, $quality) : $save($this->_image, $file);
- if ($status === TRUE AND $type !== $this->type)
- {
- // Reset the image type and mime type
- $this->type = $type;
- $this->mime = $this->image_type_to_mime_type($type);
- }
- return TRUE;
- }
- /**
- * Execute a render.
- *
- * @param string $type image type: png, jpg, gif, etc
- * @param integer $quality quality
- * @return string
- */
- protected function _do_render($type, $quality)
- {
- // Loads image if not yet loaded
- $this->_load_image();
- // Get the save function and IMAGETYPE
- list($save, $type) = $this->_save_function($type, $quality);
- // Capture the output
- ob_start();
- // Render the image
- $status = isset($quality) ? $save($this->_image, NULL, $quality) : $save($this->_image, NULL);
- if ($status === TRUE AND $type !== $this->type)
- {
- // Reset the image type and mime type
- $this->type = $type;
- $this->mime = $this->image_type_to_mime_type($type);
- }
- return ob_get_clean();
- }
- /**
- * Get the GD saving function and image type for this extension.
- * Also normalizes the quality setting
- *
- * @param string $extension image type: png, jpg, etc
- * @param integer $quality image quality
- * @return array save function, IMAGETYPE_* constant
- * @throws Kohana_Exception
- */
- protected function _save_function($extension, & $quality)
- {
- if ( ! $extension)
- {
- // Use the current image type
- $extension = image_type_to_extension($this->type, FALSE);
- }
- switch (strtolower($extension))
- {
- case 'jpg':
- case 'jpe':
- case 'jpeg':
- // Save a JPG file
- $save = 'imagejpeg';
- $type = IMAGETYPE_JPEG;
- break;
- case 'gif':
- // Save a GIF file
- $save = 'imagegif';
- $type = IMAGETYPE_GIF;
- // GIFs do not a quality setting
- $quality = NULL;
- break;
- case 'png':
- // Save a PNG file
- $save = 'imagepng';
- $type = IMAGETYPE_PNG;
- // Use a compression level of 9 (does not affect quality!)
- $quality = 9;
- break;
- case 'webp':
- // Save a WEBP file
- $save = 'imagewebp';
- $type = self::IMAGETYPE_WEBP;
- $quality = 80;
- break;
- default:
- throw new Kohana_Exception('Installed GD does not support :type images',
- [':type' => $extension]);
- break;
- }
- return [$save, $type];
- }
- /**
- * Create an empty image with the given width and height.
- *
- * @param integer $width image width
- * @param integer $height image height
- * @return resource
- */
- protected function _create($width, $height)
- {
- // Create an empty image
- $image = imagecreatetruecolor($width, $height);
- // Do not apply alpha blending
- imagealphablending($image, FALSE);
- // Save alpha levels
- imagesavealpha($image, TRUE);
- return $image;
- }
- }
|