From 2aeed1774c6d3f01c51d8ee30ddb976e6b0c942c Mon Sep 17 00:00:00 2001 From: Sullivan SENECHAL Date: Tue, 18 Nov 2014 14:34:08 +0100 Subject: [PATCH 1/3] Begin of Imagick implementation --- Adapter/Imagick.php | 780 ++++++++++++++++++++++++-------------------- Image.php | 32 +- 2 files changed, 438 insertions(+), 374 deletions(-) diff --git a/Adapter/Imagick.php b/Adapter/Imagick.php index 9da3969..cf4a26b 100644 --- a/Adapter/Imagick.php +++ b/Adapter/Imagick.php @@ -4,386 +4,448 @@ use Gregwar\Image\Image; -class Imagick extends Common{ - public function __construct(){ - throw new \Exception('Imagick is not supported right now'); - } - - /** - * Gets the name of the adapter - * - * @return string - */ - public function getName(){ - return 'ImageMagick'; - } - - /** - * Image width - * - * @return int - */ - public function width(){ - // TODO: Implement width() method. - } - - /** - * Image height - * - * @return int - */ - public function height(){ - // TODO: Implement height() method. - } - - /** - * Save the image as a gif - * - * @return $this - */ - public function saveGif($file){ - // TODO: Implement saveGif() method. - } - - /** - * Save the image as a png - * - * @return $this - */ - public function savePng($file){ - // TODO: Implement savePng() method. - } - - /** - * Save the image as a jpeg - * - * @return $this - */ - public function saveJpeg($file, $quality){ - // TODO: Implement saveJpeg() method. - } - - /** - * Crops the image - * - * @param int $x the top-left x position of the crop box - * @param int $y the top-left y position of the crop box - * @param int $width the width of the crop box - * @param int $height the height of the crop box - * - * @return $this - */ - public function crop($x, $y, $width, $height){ - // TODO: Implement crop() method. - } - - /** - * Fills the image background to $bg if the image is transparent - * - * @param int $background background color - * - * @return $this - */ - public function fillBackground($background = 0xffffff){ - // TODO: Implement fillBackground() method. - } - - /** - * Negates the image - * - * @return $this - */ - public function negate(){ - // TODO: Implement negate() method. - } - - /** - * Changes the brightness of the image - * - * @param int $brightness the brightness - * - * @return $this - */ - public function brightness($brightness){ - // TODO: Implement brightness() method. - } - - /** - * Contrasts the image - * - * @param int $contrast the contrast [-100, 100] - * - * @return $this - */ - public function contrast($contrast){ - // TODO: Implement contrast() method. - } - - /** - * Apply a grayscale level effect on the image - * - * @return $this - */ - public function grayscale(){ - // TODO: Implement grayscale() method. - } - - /** - * Emboss the image - * - * @return $this - */ - public function emboss(){ - // TODO: Implement emboss() method. - } - - /** - * Smooth the image - * - * @param int $p value between [-10,10] - * - * @return $this - */ - public function smooth($p){ - // TODO: Implement smooth() method. - } - - /** - * Sharps the image - * - * @return $this - */ - public function sharp(){ - // TODO: Implement sharp() method. - } +class Imagick extends Common +{ + /** + * @var \Imagick + */ + protected $resource; - /** - * Edges the image - * - * @return $this - */ - public function edge(){ - // TODO: Implement edge() method. - } + /** + * @inheritdoc + */ + public function __construct() + { + parent::__construct(); - /** - * Colorize the image - * - * @param int $red value in range [-255, 255] - * @param int $green value in range [-255, 255] - * @param int $blue value in range [-255, 255] - * - * @return $this - */ - public function colorize($red, $green, $blue){ - // TODO: Implement colorize() method. + if (!(extension_loaded('imagick') && class_exists('Imagick'))) { + throw new \RuntimeException('You need to install Imagick PHP Extension to use this library'); + } } - /** - * apply sepia to the image - * - * @return $this - */ - public function sepia(){ - // TODO: Implement sepia() method. + /** + * @inheritdoc + */ + public function getName() + { + return 'ImageMagick'; } - /** - * Merge with another image - * - * @param Image $other - * @param int $x - * @param int $y - * @param int $width - * @param int $height - * - * @return $this - */ - public function merge(Image $other, $x = 0, $y = 0, $width = null, $height = null){ - // TODO: Implement merge() method. - } + /** + * @inheritdoc + */ + public function width() + { + if (null === $this->resource) { + $this->init(); + } - /** - * Rotate the image - * - * @param float $angle - * @param int $background - * - * @return $this - */ - public function rotate($angle, $background = 0xffffff){ - // TODO: Implement rotate() method. - } + return $this->resource->getImageWidth(); + } - /** - * Fills the image - * - * @param int $color - * @param int $x - * @param int $y - * - * @return $this - */ - public function fill($color = 0xffffff, $x = 0, $y = 0){ - // TODO: Implement fill() method. - } + /** + * @inheritdoc + */ + public function height() + { + if (null === $this->resource) { + $this->init(); + } - /** - * write text to the image - * - * @param string $font - * @param string $text - * @param int $x - * @param int $y - * @param int $size - * @param int $angle - * @param int $color - * @param string $align - */ - public function write($font, $text, $x = 0, $y = 0, $size = 12, $angle = 0, $color = 0x000000, $align = 'left'){ - // TODO: Implement write() method. - } + return $this->resource->getImageHeight(); + } - /** - * Draws a rectangle - * - * @param int $x1 - * @param int $y1 - * @param int $x2 - * @param int $y2 - * @param int $color - * @param bool $filled - * - * @return $this - */ - public function rectangle($x1, $y1, $x2, $y2, $color, $filled = false){ - // TODO: Implement rectangle() method. - } + /** + * @inheritdoc + */ + public function saveGif($file) + { + $this->resource->setImageFormat('gif'); + $this->resource->writeImage($file); - /** - * Draws a rounded rectangle - * - * @param int $x1 - * @param int $y1 - * @param int $x2 - * @param int $y2 - * @param int $radius - * @param int $color - * @param bool $filled - * - * @return $this - */ - public function roundedRectangle($x1, $y1, $x2, $y2, $radius, $color, $filled = false){ - // TODO: Implement roundedRectangle() method. - } + return $this; + } - /** - * Draws a line - * - * @param int $x1 - * @param int $y1 - * @param int $x2 - * @param int $y2 - * @param int $color - * - * @return $this - */ - public function line($x1, $y1, $x2, $y2, $color = 0x000000){ - // TODO: Implement line() method. - } + /** + * @inheritdoc + */ + public function savePng($file) + { + $this->resource->setImageFormat('png'); + $this->resource->writeImage($file); - /** - * Draws an ellipse - * - * @param int $cx - * @param int $cy - * @param int $width - * @param int $height - * @param int $color - * @param bool $filled - * - * @return $this - */ - public function ellipse($cx, $cy, $width, $height, $color = 0x000000, $filled = false){ - // TODO: Implement ellipse() method. - } + return $this; + } - /** - * Draws a circle - * - * @param int $cx - * @param int $cy - * @param int $r - * @param int $color - * @param bool $filled - * - * @return $this - */ - public function circle($cx, $cy, $r, $color = 0x000000, $filled = false){ - // TODO: Implement circle() method. - } + /** + * @inheritdoc + */ + public function saveJpeg($file, $quality) + { + $this->resource->setImageFormat('jpeg'); + $this->resource->setCompressionQuality($quality); + $this->resource->writeImage($file); - /** - * Draws a polygon - * - * @param array $points - * @param int $color - * @param bool $filled - * - * @return $this - */ - public function polygon(array $points, $color, $filled = false){ - // TODO: Implement polygon() method. - } + return $this; + } - /** - * @inheritdoc + /** + * @inheritdoc */ - public function flip($flipVertical, $flipHorizontal) { - // TODO: Implement flip method - } - - /** - * Opens the image - */ - protected function openGif($file){ - // TODO: Implement openGif() method. - } + public function crop($x, $y, $width, $height) + { + $this->resource->cropImage($width, $height, $x, $y); + + return $this; + } + + /** + * Fills the image background to $bg if the image is transparent + * + * @param int $background background color + * + * @return $this + */ + public function fillBackground($background = 0xffffff) + { + // TODO: Implement fillBackground() method. + } + + /** + * Negates the image + * + * @return $this + */ + public function negate() + { + // TODO: Implement negate() method. + } + + /** + * Changes the brightness of the image + * + * @param int $brightness the brightness + * + * @return $this + */ + public function brightness($brightness) + { + // TODO: Implement brightness() method. + } + + /** + * Contrasts the image + * + * @param int $contrast the contrast [-100, 100] + * + * @return $this + */ + public function contrast($contrast) + { + // TODO: Implement contrast() method. + } + + /** + * Apply a grayscale level effect on the image + * + * @return $this + */ + public function grayscale() + { + // TODO: Implement grayscale() method. + } + + /** + * Emboss the image + * + * @return $this + */ + public function emboss() + { + // TODO: Implement emboss() method. + } + + /** + * Smooth the image + * + * @param int $p value between [-10,10] + * + * @return $this + */ + public function smooth($p) + { + // TODO: Implement smooth() method. + } + + /** + * Sharps the image + * + * @return $this + */ + public function sharp() + { + // TODO: Implement sharp() method. + } + + /** + * Edges the image + * + * @return $this + */ + public function edge() + { + // TODO: Implement edge() method. + } + + /** + * Colorize the image + * + * @param int $red value in range [-255, 255] + * @param int $green value in range [-255, 255] + * @param int $blue value in range [-255, 255] + * + * @return $this + */ + public function colorize($red, $green, $blue) + { + // TODO: Implement colorize() method. + } + + /** + * apply sepia to the image + * + * @return $this + */ + public function sepia() + { + // TODO: Implement sepia() method. + } + + /** + * Merge with another image + * + * @param Image $other + * @param int $x + * @param int $y + * @param int $width + * @param int $height + * + * @return $this + */ + public function merge(Image $other, $x = 0, $y = 0, $width = null, $height = null) + { + // TODO: Implement merge() method. + } + + /** + * Rotate the image + * + * @param float $angle + * @param int $background + * + * @return $this + */ + public function rotate($angle, $background = 0xffffff) + { + // TODO: Implement rotate() method. + } + + /** + * Fills the image + * + * @param int $color + * @param int $x + * @param int $y + * + * @return $this + */ + public function fill($color = 0xffffff, $x = 0, $y = 0) + { + // TODO: Implement fill() method. + } + + /** + * write text to the image + * + * @param string $font + * @param string $text + * @param int $x + * @param int $y + * @param int $size + * @param int $angle + * @param int $color + * @param string $align + */ + public function write($font, $text, $x = 0, $y = 0, $size = 12, $angle = 0, $color = 0x000000, $align = 'left') + { + // TODO: Implement write() method. + } + + /** + * Draws a rectangle + * + * @param int $x1 + * @param int $y1 + * @param int $x2 + * @param int $y2 + * @param int $color + * @param bool $filled + * + * @return $this + */ + public function rectangle($x1, $y1, $x2, $y2, $color, $filled = false) + { + // TODO: Implement rectangle() method. + } + + /** + * Draws a rounded rectangle + * + * @param int $x1 + * @param int $y1 + * @param int $x2 + * @param int $y2 + * @param int $radius + * @param int $color + * @param bool $filled + * + * @return $this + */ + public function roundedRectangle($x1, $y1, $x2, $y2, $radius, $color, $filled = false) + { + // TODO: Implement roundedRectangle() method. + } + + /** + * Draws a line + * + * @param int $x1 + * @param int $y1 + * @param int $x2 + * @param int $y2 + * @param int $color + * + * @return $this + */ + public function line($x1, $y1, $x2, $y2, $color = 0x000000) + { + // TODO: Implement line() method. + } + + /** + * Draws an ellipse + * + * @param int $cx + * @param int $cy + * @param int $width + * @param int $height + * @param int $color + * @param bool $filled + * + * @return $this + */ + public function ellipse($cx, $cy, $width, $height, $color = 0x000000, $filled = false) + { + // TODO: Implement ellipse() method. + } + + /** + * Draws a circle + * + * @param int $cx + * @param int $cy + * @param int $r + * @param int $color + * @param bool $filled + * + * @return $this + */ + public function circle($cx, $cy, $r, $color = 0x000000, $filled = false) + { + // TODO: Implement circle() method. + } + + /** + * Draws a polygon + * + * @param array $points + * @param int $color + * @param bool $filled + * + * @return $this + */ + public function polygon(array $points, $color, $filled = false) + { + // TODO: Implement polygon() method. + } + + /** + * Flips the image + * + * @param int $flipVertical + * @param int $flipHorizontal + * + * @return $this + */ + public function flip($flipVertical, $flipHorizontal) + { + // TODO: Implement flip() method. + } - protected function openJpeg($file){ - // TODO: Implement openJpeg() method. - } + /** + * @inheritdoc + */ + protected function openGif($file) + { + $this->resource = new \Imagick($file); + } - protected function openPng($file){ - // TODO: Implement openPng() method. - } + /** + * @inheritdoc + */ + protected function openJpeg($file) + { + $this->resource = new \Imagick($file); + } - /** - * Creates an image - */ - protected function createImage($width, $height){ - // TODO: Implement createImage() method. - } + /** + * @inheritdoc + */ + protected function openPng($file) + { + $this->resource = new \Imagick($file); + } - /** - * Creating an image using $data - */ - protected function createImageFromData($data){ - // TODO: Implement createImageFromData() method. - } + /** + * Creates an image + */ + protected function createImage($width, $height) + { + // TODO: Implement createImage() method. + } - /** - * Resizes the image to an image having size of $target_width, $target_height, using - * $new_width and $new_height and padding with $bg color - */ - protected function doResize($bg, $target_width, $target_height, $new_width, $new_height){ - // TODO: Implement doResize() method. - } + /** + * Creating an image using $data + */ + protected function createImageFromData($data) + { + // TODO: Implement createImageFromData() method. + } + + /** + * Resizes the image to an image having size of $target_width, $target_height, using + * $new_width and $new_height and padding with $bg color + */ + protected function doResize($bg, $target_width, $target_height, $new_width, $new_height) + { + // TODO: Implement doResize() method. + } - /** - * Gets the color of the $x, $y pixel - */ - protected function getColor($x, $y){ - // TODO: Implement getColor() method. - } + /** + * Gets the color of the $x, $y pixel + */ + protected function getColor($x, $y) + { + // TODO: Implement getColor() method. + } } diff --git a/Image.php b/Image.php index 4a51fc0..9fb9f4f 100644 --- a/Image.php +++ b/Image.php @@ -87,7 +87,7 @@ class Image 'png' => 'png', 'gif' => 'gif', ); - + /** * Fallback image */ @@ -252,8 +252,8 @@ public function getCacheFallback() public function getAdapter() { if (null === $this->adapter) { - // Defaults to GD - $this->setAdapter('gd'); + // Defaults to Imagick + $this->setAdapter('imagick'); } return $this->adapter; @@ -268,16 +268,18 @@ public function setAdapter($adapter) $adapter = strtolower($adapter); switch ($adapter) { - case 'gd': - $this->adapter = new Adapter\GD(); - break; - case 'imagemagick': - case 'imagick': - $this->adapter = new Adapter\Imagick(); - break; - default: - throw new \Exception('Unknown adapter: '.$adapter); - break; + case 'gd': + $this->adapter = new Adapter\GD(); + break; + case 'imagemagick': + $this->adapter = new Adapter\Imagick(); + break; + case 'imagick': + $this->adapter = new Adapter\Imagick(); + break; + default: + throw new \Exception('Unknown adapter: '.$adapter); + break; } } else { throw new \Exception('Unable to load the given adapter (not string or Adapter)'); @@ -454,8 +456,8 @@ public function cacheFile($type = 'jpg', $quality = 80, $actual = false) // If the files does not exists, save it $image = $this; - // Target file should be younger than all the current image - // dependencies + // Target file should be younger than all the current image + // dependencies $conditions = array( 'younger-than' => $this->getDependencies() ); From 1ceb956a08ed793a0f15e32935d6fe86eafecda2 Mon Sep 17 00:00:00 2001 From: Sullivan SENECHAL Date: Tue, 18 Nov 2014 15:46:38 +0100 Subject: [PATCH 2/3] Clear image on descruct and fill another adapter methods --- Adapter/GD.php | 4 ++-- Adapter/Imagick.php | 55 ++++++++++++++++++++++++--------------------- 2 files changed, 32 insertions(+), 27 deletions(-) diff --git a/Adapter/GD.php b/Adapter/GD.php index b63c734..65ef014 100644 --- a/Adapter/GD.php +++ b/Adapter/GD.php @@ -380,7 +380,7 @@ public function flip($flipVertical, $flipHorizontal) { } else if (!$flipVertical && $flipHorizontal) { $flipMode = \IMG_FLIP_HORIZONTAL; } - + imageflip($this->resource, $flipMode); } else { $width = $this->width(); @@ -410,7 +410,7 @@ public function flip($flipVertical, $flipHorizontal) { $this->resource = $imgdest; } } - + return $this; } diff --git a/Adapter/Imagick.php b/Adapter/Imagick.php index cf4a26b..dd60628 100644 --- a/Adapter/Imagick.php +++ b/Adapter/Imagick.php @@ -23,6 +23,14 @@ public function __construct() } } + public function __destruct() + { + if ($this->resource !== null) { + $this->resource->clear(); + $this->resource->destroy(); + } + } + /** * @inheritdoc */ @@ -100,59 +108,56 @@ public function crop($x, $y, $width, $height) } /** - * Fills the image background to $bg if the image is transparent - * - * @param int $background background color - * - * @return $this + * @inheritdoc */ public function fillBackground($background = 0xffffff) { - // TODO: Implement fillBackground() method. + // TODO: Check if int works or if hexadecimal notation is needed. + $this->resource->setImageBackgroundColor($background); + + return $this; } /** - * Negates the image - * - * @return $this + * @inheritdoc */ public function negate() { - // TODO: Implement negate() method. + // TODO: Check if $gray should be false or true according to GD method + $this->resource->negateImage(true); + + return $this; } /** - * Changes the brightness of the image - * - * @param int $brightness the brightness - * - * @return $this + * @inheritdoc */ public function brightness($brightness) { - // TODO: Implement brightness() method. + // TODO: Test according to this note: http://php.net/manual/en/imagick.modulateimage.php#87330 + $this->resource->modulateImage($brightness, 100, 100); + + return $this; } /** - * Contrasts the image - * - * @param int $contrast the contrast [-100, 100] - * - * @return $this + * @inheritdoc */ public function contrast($contrast) { // TODO: Implement contrast() method. + // Note: Imagick want a boolean. How to convert it? } /** - * Apply a grayscale level effect on the image - * - * @return $this + * @inheritdoc */ public function grayscale() { - // TODO: Implement grayscale() method. + // TODO: Don't know if it is the good solution. Can we add colors after? Test it. + $this->resource->setImageColorspace(\Imagick::COLORSPACE_GRAY); + + return $this; } /** From b9c3fac443e75d618210c92c13c8509cc2755577 Mon Sep 17 00:00:00 2001 From: Sullivan SENECHAL Date: Thu, 20 Nov 2014 22:44:55 +0100 Subject: [PATCH 3/3] Add some filters --- Adapter/Imagick.php | 83 ++++++++++++++++++++++++--------------------- 1 file changed, 44 insertions(+), 39 deletions(-) diff --git a/Adapter/Imagick.php b/Adapter/Imagick.php index dd60628..7b8ac86 100644 --- a/Adapter/Imagick.php +++ b/Adapter/Imagick.php @@ -161,84 +161,89 @@ public function grayscale() } /** - * Emboss the image - * - * @return $this + * @inheritdoc */ public function emboss() { - // TODO: Implement emboss() method. + // TODO: Manage parameters? http://php.net/manual/en/imagick.embossImage.php + $this->resource->embossImage(0, 1); + + return $this; } /** - * Smooth the image - * - * @param int $p value between [-10,10] - * - * @return $this + * @inheritdoc */ public function smooth($p) { - // TODO: Implement smooth() method. + $this->resource->reduceNoiseImage($p); + + return $this; } /** - * Sharps the image - * - * @return $this + * @inheritdoc */ public function sharp() { - // TODO: Implement sharp() method. + // TODO: Manage parameters? http://php.net/manual/en/imagick.sharpenimage.php + $this->resource->sharpenImage(0, 1); + + return $this; } /** - * Edges the image - * - * @return $this + * @inheritdoc */ public function edge() { - // TODO: Implement edge() method. + // TODO: Manage parameters? http://php.net/manual/en/imagick.edgeimage.php + $this->resource->edgeImage(0); + + return $this; } /** - * Colorize the image - * - * @param int $red value in range [-255, 255] - * @param int $green value in range [-255, 255] - * @param int $blue value in range [-255, 255] - * - * @return $this + * @inheritdoc */ public function colorize($red, $green, $blue) { - // TODO: Implement colorize() method. + $color = sprintf('rgb(%d, %d, %d)', $red, $green, $blue); + // TODO: Manage opacity? http://php.net/manual/en/imagick.colorizeimage.php + $this->resource->colorizeImage($color, 1.0); + + return $this; } /** - * apply sepia to the image - * - * @return $this + * @inheritdoc */ public function sepia() { - // TODO: Implement sepia() method. + // TODO: Manage threshold parameter? http://php.net/manual/en/imagick.sepiatoneimage.php + $this->resource->sepiaToneImage(80); + + return $this; } /** - * Merge with another image - * - * @param Image $other - * @param int $x - * @param int $y - * @param int $width - * @param int $height - * - * @return $this + * @inheritdoc */ public function merge(Image $other, $x = 0, $y = 0, $width = null, $height = null) { + // TODO: implement it on Common class. + $other = clone $other; + $other->init(); + $other->applyOperations(); + + if (null == $width) { + $width = $other->width(); + } + if (null == $height) { + $height = $other->height(); + } + // ENDTOTO + // TODO: Implement merge() method. }