666 lines
17 KiB
PHP
666 lines
17 KiB
PHP
<?php defined('SYSPATH') OR die('No direct script access.');
|
|
/**
|
|
* Support for image manipulation using [GD](http://php.net/GD).
|
|
*
|
|
* @package Kohana/Image
|
|
* @category Drivers
|
|
* @author Kohana Team
|
|
* @copyright (c) 2008-2009 Kohana Team
|
|
* @license http://kohanaphp.com/license.html
|
|
*/
|
|
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 = array();
|
|
|
|
/**
|
|
* 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 = array(
|
|
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',
|
|
array('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;
|
|
}
|
|
|
|
if ( ! isset($create) OR ! function_exists($create))
|
|
{
|
|
throw new Kohana_Exception('Installed GD does not support :type images',
|
|
array(':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 (is_resource($this->_image))
|
|
{
|
|
// Free all resources
|
|
imagedestroy($this->_image);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Loads an image into GD.
|
|
*
|
|
* @return void
|
|
*/
|
|
protected function _load_image()
|
|
{
|
|
if ( ! is_resource($this->_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);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* 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',
|
|
array(':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',
|
|
array(':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 = array
|
|
(
|
|
array(-1, -1, -1),
|
|
array(-1, $amount, -1),
|
|
array(-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',
|
|
array(':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',
|
|
array(':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 = 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 = 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 '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;
|
|
default:
|
|
throw new Kohana_Exception('Installed GD does not support :type images',
|
|
array(':type' => $extension));
|
|
break;
|
|
}
|
|
|
|
return array($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;
|
|
}
|
|
|
|
} // End Image_GD
|