/filemanager/include/php_image_magician.php
PHP | 3320 lines | 1624 code | 614 blank | 1082 comment | 289 complexity | 873b98d1cd68064b004ce98c73b40ab4 MD5 | raw file
Possible License(s): LGPL-2.1, LGPL-3.0
Large files files are truncated, but you can click here to view the full file
- <?php
- # ========================================================================#
- #
- # This work is licensed under the Creative Commons Attribution 3.0 Unported
- # License. To view a copy of this license,
- # visit http://creativecommons.org/licenses/by/3.0/ or send a letter to
- # Creative Commons, 444 Castro Street, Suite 900, Mountain View, California,
- # 94041, USA.
- #
- # All rights reserved.
- #
- # Author: Jarrod Oberto
- # Version: 1.5.1
- # Date: 10-05-11
- # Purpose: Provide tools for image manipulation using GD
- # Param In: See functions.
- # Param Out: Produces a resized image
- # Requires : Requires PHP GD library.
- # Usage Example:
- # include("lib/php_image_magician.php");
- # $magicianObj = new resize('images/car.jpg');
- # $magicianObj -> resizeImage(150, 100, 0);
- # $magicianObj -> saveImage('images/car_small.jpg', 100);
- #
- # - See end of doc for more examples -
- #
- # Supported file types include: jpg, png, gif, bmp, psd (read)
- #
- #
- #
- # The following functions are taken from phpThumb() [available from
- # http://phpthumb.sourceforge.net], and are used with written permission
- # from James Heinrich.
- # - GD2BMPstring
- # - GetPixelColor
- # - LittleEndian2String
- #
- # The following functions are from Marc Hibbins and are used with written
- # permission (are also under the Attribution-ShareAlike
- # [http://creativecommons.org/licenses/by-sa/3.0/] license.
- # -
- #
- # PhpPsdReader is used with written permission from Tim de Koning.
- # [http://www.kingsquare.nl/phppsdreader]
- #
- #
- #
- # Modificatoin history
- # Date Initials Ver Description
- # 10-05-11 J.C.O 0.0 Initial build
- # 01-06-11 J.C.O 0.1.1 * Added reflections
- # * Added Rounded corners
- # * You can now use PNG interlacing
- # * Added shadow
- # * Added caption box
- # * Added vintage filter
- # * Added dynamic image resizing (resize on the fly)
- # * minor bug fixes
- # 05-06-11 J.C.O 0.1.1.1 * Fixed undefined variables
- # 17-06-11 J.C.O 0.1.2 * Added image_batch_class.php class
- # * Minor bug fixes
- # 26-07-11 J.C.O 0.1.4 * Added support for external images
- # * Can now set the crop poisition
- # 03-08-11 J.C.O 0.1.5 * Added reset() method to reset resource to
- # original input file.
- # * Added method addTextToCaptionBox() to
- # simplify adding text to a caption box.
- # * Added experimental writeIPTC. (not finished)
- # * Added experimental readIPTC. (not finished)
- # 11-08-11 J.C.O * Added initial border presets.
- # 30-08-11 J.C.O * Added 'auto' crop option to crop portrait
- # images near the top.
- # 08-09-11 J.C.O * Added cropImage() method to allow standalone
- # cropping.
- # 17-09-11 J.C.O * Added setCropFromTop() set method - set the
- # percentage to crop from the top when using
- # crop 'auto' option.
- # * Added setTransparency() set method - allows you
- # to turn transparency off (like when saving
- # as a jpg).
- # * Added setFillColor() set method - set the
- # background color to use instead of transparency.
- # 05-11-11 J.C.O 0.1.5.1 * Fixed interlacing option
- # 0-07-12 J.C.O 1.0
- #
- # Known issues & Limitations:
- # -------------------------------
- # Not so much an issue, the image is destroyed on the deconstruct rather than
- # when we have finished with it. The reason for this is that we don't know
- # when we're finished with it as you can both save the image and display
- # it directly to the screen (imagedestroy($this->imageResized))
- #
- # Opening BMP files is slow. A test with 884 bmp files processed in a loop
- # takes forever - over 5 min. This test inlcuded opening the file, then
- # getting and displaying its width and height.
- #
- # $forceStretch:
- # -------------------------------
- # On by default.
- # $forceStretch can be disabled by calling method setForceStretch with false
- # parameter. If disabled, if an images original size is smaller than the size
- # specified by the user, the original size will be used. This is useful when
- # dealing with small images.
- #
- # If enabled, images smaller than the size specified will be stretched to
- # that size.
- #
- # Tips:
- # -------------------------------
- # * If you're resizing a transparent png and saving it as a jpg, set
- # $keepTransparency to false with: $magicianObj->setTransparency(false);
- #
- # FEATURES:
- # * EASY TO USE
- # * BMP SUPPORT (read & write)
- # * PSD (photoshop) support (read)
- # * RESIZE IMAGES
- # - Preserve transparency (png, gif)
- # - Apply sharpening (jpg) (requires PHP >= 5.1.0)
- # - Set image quality (jpg, png)
- # - Resize modes:
- # - exact size
- # - resize by width (auto height)
- # - resize by height (auto width)
- # - auto (automatically determine the best of the above modes to use)
- # - crop - resize as best as it can then crop the rest
- # - Force stretching of smaller images (upscale)
- # * APPLY FILTERS
- # - Convert to grey scale
- # - Convert to black and white
- # - Convert to sepia
- # - Convert to negative
- # * ROTATE IMAGES
- # - Rotate using predefined "left", "right", or "180"; or any custom degree amount
- # * EXTRACT EXIF DATA (requires exif module)
- # - make
- # - model
- # - date
- # - exposure
- # - aperture
- # - f-stop
- # - iso
- # - focal length
- # - exposure program
- # - metering mode
- # - flash status
- # - creator
- # - copyright
- # * ADD WATERMARK
- # - Specify exact x, y placement
- # - Or, specify using one of the 9 pre-defined placements such as "tl"
- # (for top left), "m" (for middle), "br" (for bottom right)
- # - also specify padding from edge amount (optional).
- # - Set opacity of watermark (png).
- # * ADD BORDER
- # * USE HEX WHEN SPECIFYING COLORS (eg: #ffffff)
- # * SAVE IMAGE OR OUTPUT TO SCREEN
- #
- #
- # ========================================================================#
- class imageLib
- {
- private $fileName;
- private $image;
- protected $imageResized;
- private $widthOriginal; # Always be the original width
- private $heightOriginal;
- private $width; # Current width (width after resize)
- private $height;
- private $imageSize;
- private $fileExtension;
- private $debug = true;
- private $errorArray = array();
- private $forceStretch = true;
- private $aggresiveSharpening = false;
- private $transparentArray = array('.png', '.gif');
- private $keepTransparency = true;
- private $fillColorArray = array('r'=>255, 'g'=>255, 'b'=>255);
- private $sharpenArray = array('jpg');
- private $psdReaderPath;
- private $filterOverlayPath;
- private $isInterlace;
- private $captionBoxPositionArray = array();
- private $fontDir = 'fonts';
- private $cropFromTopPercent = 10;
- ## --------------------------------------------------------
- function __construct($fileName)
- # Author: Jarrod Oberto
- # Date: 27-02-08
- # Purpose: Constructor
- # Param in: $fileName: File name and path.
- # Param out: n/a
- # Reference:
- # Notes:
- #
- {
- if (!$this->testGDInstalled()) { if ($this->debug) { throw new Exception('The GD Library is not installed.'); }else{ throw new Exception(); }};
- $this->initialise();
- // *** Save the image file name. Only store this incase you want to display it
- $this->fileName = $fileName;
- $this->fileExtension = fix_strtolower(strrchr($fileName, '.'));
- // *** Open up the file
- $this->image = $this->openImage($fileName);
- // *** Assign here so we don't modify the original
- $this->imageResized = $this->image;
- // *** If file is an image
- if ($this->testIsImage($this->image))
- {
- // *** Get width and height
- $this->width = imagesx($this->image);
- $this->widthOriginal = imagesx($this->image);
- $this->height = imagesy($this->image);
- $this->heightOriginal = imagesy($this->image);
- /* Added 15-09-08
- * Get the filesize using this build in method.
- * Stores an array of size
- *
- * $this->imageSize[1] = width
- * $this->imageSize[2] = height
- * $this->imageSize[3] = width x height
- *
- */
- $this->imageSize = getimagesize($this->fileName);
- } else {
- $this->errorArray[] = 'File is not an image';
- }
- }
- ## --------------------------------------------------------
- private function initialise () {
- $this->psdReaderPath = dirname(__FILE__) . '/classPhpPsdReader.php';
- $this->filterOverlayPath = dirname(__FILE__) . '/filters';
- // *** Set if image should be interlaced or not.
- $this->isInterlace = false;
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Resize
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function resizeImage($newWidth, $newHeight, $option = 0, $sharpen = false, $autoRotate = false)
- # Author: Jarrod Oberto
- # Date: 27-02-08
- # Purpose: Resizes the image
- # Param in: $newWidth:
- # $newHeight:
- # $option: 0 / exact = defined size;
- # 1 / portrait = keep aspect set height;
- # 2 / landscape = keep aspect set width;
- # 3 / auto = auto;
- # 4 / crop= resize and crop;
- #
- # $option can also be an array containing options for
- # cropping. E.G., array('crop', 'r')
- #
- # This array only applies to 'crop' and the 'r' refers to
- # "crop right". Other value include; tl, t, tr, l, m (default),
- # r, bl, b, br, or you can specify your own co-ords (which
- # isn't recommended.
- #
- # $sharpen: true: sharpen (jpg only);
- # false: don't sharpen
- # Param out: n/a
- # Reference:
- # Notes: To clarify the $option input:
- # 0 = The exact height and width dimensions you set.
- # 1 = Whatever height is passed in will be the height that
- # is set. The width will be calculated and set automatically
- # to a the value that keeps the original aspect ratio.
- # 2 = The same but based on the width. We try make the image the
- # biggest size we can while stil fitting inside the box size
- # 3 = Depending whether the image is landscape or portrait, this
- # will automatically determine whether to resize via
- # dimension 1,2 or 0
- # 4 = Will resize and then crop the image for best fit
- #
- # forceStretch can be applied to options 1,2,3 and 4
- #
- {
- // *** We can pass in an array of options to change the crop position
- $cropPos = 'm';
- if (is_array($option) && fix_strtolower($option[0]) == 'crop') {
- $cropPos = $option[1]; # get the crop option
- } else if (strpos($option, '-') !== false) {
- // *** Or pass in a hyphen seperated option
- $optionPiecesArray = explode('-', $option);
- $cropPos = end($optionPiecesArray);
- }
- // *** Check the option is valid
- $option = $this->prepOption($option);
- // *** Make sure the file passed in is valid
- if (!$this->image) { if ($this->debug) { throw new Exception('file ' . $this->getFileName() .' is missing or invalid'); }else{ throw new Exception(); }};
- // *** Get optimal width and height - based on $option
- $dimensionsArray = $this->getDimensions($newWidth, $newHeight, $option);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- // *** Resample - create image canvas of x, y size
- $this->imageResized = imagecreatetruecolor($optimalWidth, $optimalHeight);
- $this->keepTransparancy($optimalWidth, $optimalHeight, $this->imageResized);
- imagecopyresampled($this->imageResized, $this->image, 0, 0, 0, 0, $optimalWidth, $optimalHeight, $this->width, $this->height);
- // *** If '4', then crop too
- if ($option == 4 || $option == 'crop') {
- if (($optimalWidth >= $newWidth && $optimalHeight >= $newHeight)) {
- $this->crop($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos);
- }
- }
- // *** If Rotate.
- if ($autoRotate) {
- $exifData = $this->getExif(false);
- if (count($exifData) > 0) {
- switch($exifData['orientation']) {
- case 8:
- $this->imageResized = imagerotate($this->imageResized,90,0);
- break;
- case 3:
- $this->imageResized = imagerotate($this->imageResized,180,0);
- break;
- case 6:
- $this->imageResized = imagerotate($this->imageResized,-90,0);
- break;
- }
- }
- }
- // *** Sharpen image (if jpg and the user wishes to do so)
- if ($sharpen && in_array($this->fileExtension, $this->sharpenArray)) {
- // *** Sharpen
- $this->sharpen();
- }
- }
- ## --------------------------------------------------------
- public function cropImage($newWidth, $newHeight, $cropPos = 'm')
- # Author: Jarrod Oberto
- # Date: 08-09-11
- # Purpose: Crops the image
- # Param in: $newWidth: crop with
- # $newHeight: crop height
- # $cropPos: Can be any of the following:
- # tl, t, tr, l, m, r, bl, b, br, auto
- # Or:
- # a custom position such as '30x50'
- # Param out: n/a
- # Reference:
- # Notes:
- #
- {
- // *** Make sure the file passed in is valid
- if (!$this->image) { if ($this->debug) { throw new Exception('file ' . $this->getFileName() .' is missing or invalid'); }else{ throw new Exception(); }};
- $this->imageResized = $this->image;
- $this->crop($this->width, $this->height, $newWidth, $newHeight, $cropPos);
- }
- ## --------------------------------------------------------
- private function keepTransparancy($width, $height, $im)
- # Author: Jarrod Oberto
- # Date: 08-04-11
- # Purpose: Keep transparency for png and gif image
- # Param in:
- # Param out: n/a
- # Reference:
- # Notes:
- #
- {
- // *** If PNG, perform some transparency retention actions (gif untested)
- if (in_array($this->fileExtension, $this->transparentArray) && $this->keepTransparency) {
- imagealphablending($im, false);
- imagesavealpha($im, true);
- $transparent = imagecolorallocatealpha($im, 255, 255, 255, 127);
- imagefilledrectangle($im, 0, 0, $width, $height, $transparent);
- } else {
- $color = imagecolorallocate($im, $this->fillColorArray['r'], $this->fillColorArray['g'], $this->fillColorArray['b']);
- imagefilledrectangle($im, 0, 0, $width, $height, $color);
- }
- }
- ## --------------------------------------------------------
- private function crop($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos)
- # Author: Jarrod Oberto
- # Date: 15-09-08
- # Purpose: Crops the image
- # Param in: $newWidth:
- # $newHeight:
- # Param out: n/a
- # Reference:
- # Notes:
- #
- {
- // *** Get cropping co-ordinates
- $cropArray = $this->getCropPlacing($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos);
- $cropStartX = $cropArray['x'];
- $cropStartY = $cropArray['y'];
- // *** Crop this bad boy
- $crop = imagecreatetruecolor($newWidth , $newHeight);
- $this->keepTransparancy($optimalWidth, $optimalHeight, $crop);
- imagecopyresampled($crop, $this->imageResized, 0, 0, $cropStartX, $cropStartY, $newWidth, $newHeight , $newWidth, $newHeight);
- $this->imageResized = $crop;
- // *** Set new width and height to our variables
- $this->width = $newWidth;
- $this->height = $newHeight;
- }
- ## --------------------------------------------------------
- private function getCropPlacing($optimalWidth, $optimalHeight, $newWidth, $newHeight, $pos='m')
- #
- # Author: Jarrod Oberto
- # Date: July 11
- # Purpose: Set the cropping area.
- # Params in:
- # Params out: (array) the crop x and y co-ordinates.
- # Notes: When specifying the exact pixel crop position (eg 10x15), be
- # very careful as it's easy to crop out of the image leaving
- # black borders.
- #
- {
- $pos = fix_strtolower($pos);
- // *** If co-ords have been entered
- if (strstr($pos, 'x')) {
- $pos = str_replace(' ', '', $pos);
- $xyArray = explode('x', $pos);
- list($cropStartX, $cropStartY) = $xyArray;
- } else {
- switch ($pos) {
- case 'tl':
- $cropStartX = 0;
- $cropStartY = 0;
- break;
- case 't':
- $cropStartX = ( $optimalWidth / 2) - ( $newWidth /2 );
- $cropStartY = 0;
- break;
- case 'tr':
- $cropStartX = $optimalWidth - $newWidth;
- $cropStartY = 0;
- break;
- case 'l':
- $cropStartX = 0;
- $cropStartY = ( $optimalHeight/ 2) - ( $newHeight/2 );
- break;
- case 'm':
- $cropStartX = ( $optimalWidth / 2) - ( $newWidth /2 );
- $cropStartY = ( $optimalHeight/ 2) - ( $newHeight/2 );
- break;
- case 'r':
- $cropStartX = $optimalWidth - $newWidth;
- $cropStartY = ( $optimalHeight/ 2) - ( $newHeight/2 );
- break;
- case 'bl':
- $cropStartX = 0;
- $cropStartY = $optimalHeight - $newHeight;
- break;
- case 'b':
- $cropStartX = ( $optimalWidth / 2) - ( $newWidth /2 );
- $cropStartY = $optimalHeight - $newHeight;
- break;
- case 'br':
- $cropStartX = $optimalWidth - $newWidth;
- $cropStartY = $optimalHeight - $newHeight;
- break;
- case 'auto':
- // *** If image is a portrait crop from top, not center. v1.5
- if ($optimalHeight > $optimalWidth) {
- $cropStartX = ( $optimalWidth / 2) - ( $newWidth /2 );
- $cropStartY = ($this->cropFromTopPercent /100) * $optimalHeight;
- } else {
- // *** Else crop from the center
- $cropStartX = ( $optimalWidth / 2) - ( $newWidth /2 );
- $cropStartY = ( $optimalHeight/ 2) - ( $newHeight/2 );
- }
- break;
- default:
- // *** Default to center
- $cropStartX = ( $optimalWidth / 2) - ( $newWidth /2 );
- $cropStartY = ( $optimalHeight/ 2) - ( $newHeight/2 );
- break;
- }
- }
- return array('x' => $cropStartX, 'y' => $cropStartY);
- }
- ## --------------------------------------------------------
- private function getDimensions($newWidth, $newHeight, $option)
- # Author: Jarrod Oberto
- # Date: 17-11-09
- # Purpose: Get new image dimensions based on user specificaions
- # Param in: $newWidth:
- # $newHeight:
- # Param out: Array of new width and height values
- # Reference:
- # Notes: If $option = 3 then this function is call recursivly
- #
- # To clarify the $option input:
- # 0 = The exact height and width dimensions you set.
- # 1 = Whatever height is passed in will be the height that
- # is set. The width will be calculated and set automatically
- # to a the value that keeps the original aspect ratio.
- # 2 = The same but based on the width.
- # 3 = Depending whether the image is landscape or portrait, this
- # will automatically determine whether to resize via
- # dimension 1,2 or 0.
- # 4 = Resize the image as much as possible, then crop the
- # remainder.
- {
- switch (strval($option))
- {
- case '0':
- case 'exact':
- $optimalWidth = $newWidth;
- $optimalHeight= $newHeight;
- break;
- case '1':
- case 'portrait':
- $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- break;
- case '2':
- case 'landscape':
- $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- break;
- case '3':
- case 'auto':
- $dimensionsArray = $this->getSizeByAuto($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- break;
- case '4':
- case 'crop':
- $dimensionsArray = $this->getOptimalCrop($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- break;
- }
- return array('optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight);
- }
- ## --------------------------------------------------------
- private function getSizeByFixedHeight($newWidth, $newHeight)
- {
- // *** If forcing is off...
- if (!$this->forceStretch) {
- // *** ...check if actual height is less than target height
- if ($this->height < $newHeight) {
- return array('optimalWidth' => $this->width, 'optimalHeight' => $this->height);
- }
- }
- $ratio = $this->width / $this->height;
- $newWidth = $newHeight * $ratio;
- //return $newWidth;
- return array('optimalWidth' => $newWidth, 'optimalHeight' => $newHeight);
- }
- ## --------------------------------------------------------
- private function getSizeByFixedWidth($newWidth, $newHeight)
- {
- // *** If forcing is off...
- if (!$this->forceStretch) {
- // *** ...check if actual width is less than target width
- if ($this->width < $newWidth) {
- return array('optimalWidth' => $this->width, 'optimalHeight' => $this->height);
- }
- }
- $ratio = $this->height / $this->width;
- $newHeight = $newWidth * $ratio;
- //return $newHeight;
- return array('optimalWidth' => $newWidth, 'optimalHeight' => $newHeight);
- }
- ## --------------------------------------------------------
- private function getSizeByAuto($newWidth, $newHeight)
- # Author: Jarrod Oberto
- # Date: 19-08-08
- # Purpose: Depending on the height, choose to resize by 0, 1, or 2
- # Param in: The new height and new width
- # Notes:
- #
- {
- // *** If forcing is off...
- if (!$this->forceStretch) {
- // *** ...check if actual size is less than target size
- if ($this->width < $newWidth && $this->height < $newHeight) {
- return array('optimalWidth' => $this->width, 'optimalHeight' => $this->height);
- }
- }
- if ($this->height < $this->width)
- // *** Image to be resized is wider (landscape)
- {
- //$optimalWidth = $newWidth;
- //$optimalHeight= $this->getSizeByFixedWidth($newWidth);
- $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- }
- elseif ($this->height > $this->width)
- // *** Image to be resized is taller (portrait)
- {
- //$optimalWidth = $this->getSizeByFixedHeight($newHeight);
- //$optimalHeight= $newHeight;
- $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- }
- else
- // *** Image to be resizerd is a square
- {
- if ($newHeight < $newWidth) {
- //$optimalWidth = $newWidth;
- //$optimalHeight= $this->getSizeByFixedWidth($newWidth);
- $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- } else if ($newHeight > $newWidth) {
- //$optimalWidth = $this->getSizeByFixedHeight($newHeight);
- //$optimalHeight= $newHeight;
- $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
- $optimalWidth = $dimensionsArray['optimalWidth'];
- $optimalHeight = $dimensionsArray['optimalHeight'];
- } else {
- // *** Sqaure being resized to a square
- $optimalWidth = $newWidth;
- $optimalHeight= $newHeight;
- }
- }
- return array('optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight);
- }
- ## --------------------------------------------------------
- private function getOptimalCrop($newWidth, $newHeight)
- # Author: Jarrod Oberto
- # Date: 17-11-09
- # Purpose: Get optimal crop dimensions
- # Param in: width and height as requested by user (fig 3)
- # Param out: Array of optimal width and height (fig 2)
- # Reference:
- # Notes: The optimal width and height return are not the same as the
- # same as the width and height passed in. For example:
- #
- #
- # |-----------------| |------------| |-------|
- # | | => |**| |**| => | |
- # | | |**| |**| | |
- # | | |------------| |-------|
- # |-----------------|
- # original optimal crop
- # size size size
- # Fig 1 2 3
- #
- # 300 x 250 150 x 125 150 x 100
- #
- # The optimal size is the smallest size (that is closest to the crop size)
- # while retaining proportion/ratio.
- #
- # The crop size is the optimal size that has been cropped on one axis to
- # make the image the exact size specified by the user.
- #
- # * represent cropped area
- #
- {
- // *** If forcing is off...
- if (!$this->forceStretch) {
- // *** ...check if actual size is less than target size
- if ($this->width < $newWidth && $this->height < $newHeight) {
- return array('optimalWidth' => $this->width, 'optimalHeight' => $this->height);
- }
- }
- $heightRatio = $this->height / $newHeight;
- $widthRatio = $this->width / $newWidth;
- if ($heightRatio < $widthRatio) {
- $optimalRatio = $heightRatio;
- } else {
- $optimalRatio = $widthRatio;
- }
- $optimalHeight = round( $this->height / $optimalRatio );
- $optimalWidth = round( $this->width / $optimalRatio );
- return array('optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight);
- }
- ## --------------------------------------------------------
- private function sharpen()
- # Author: Jarrod Oberto
- # Date: 08 04 2011
- # Purpose: Sharpen image
- # Param in: n/a
- # Param out: n/a
- # Reference:
- # Notes:
- # Credit: Incorporates Joe Lencioni (August 6, 2008) code
- {
- if (version_compare(PHP_VERSION, '5.1.0') >= 0) {
- // ***
- if ($this->aggresiveSharpening) { # A more aggressive sharpening solution
- $sharpenMatrix = array( array( -1, -1, -1 ),
- array( -1, 16, -1 ),
- array( -1, -1, -1 ) );
- $divisor = 8;
- $offset = 0;
- imageconvolution($this->imageResized, $sharpenMatrix, $divisor, $offset);
- }
- else # More subtle and personally more desirable
- {
- $sharpness = $this->findSharp($this->widthOriginal, $this->width);
- $sharpenMatrix = array(
- array(-1, -2, -1),
- array(-2, $sharpness + 12, -2), //Lessen the effect of a filter by increasing the value in the center cell
- array(-1, -2, -1)
- );
- $divisor = $sharpness; // adjusts brightness
- $offset = 0;
- imageconvolution($this->imageResized, $sharpenMatrix, $divisor, $offset);
- }
- }
- else
- {
- if ($this->debug) { throw new Exception('Sharpening required PHP 5.1.0 or greater.'); }
- }
- }
- ## --------------------------------------------------------
- private function sharpen2($level)
- {
- $sharpenMatrix = array(
- array($level, $level, $level),
- array($level, (8*$level)+1, $level), //Lessen the effect of a filter by increasing the value in the center cell
- array($level, $level, $level)
- );
- }
- ## --------------------------------------------------------
- private function findSharp($orig, $final)
- # Author: Ryan Rud (http://adryrun.com)
- # Purpose: Find optimal sharpness
- # Param in: n/a
- # Param out: n/a
- # Reference:
- # Notes:
- #
- {
- $final = $final * (750.0 / $orig);
- $a = 52;
- $b = -0.27810650887573124;
- $c = .00047337278106508946;
- $result = $a + $b * $final + $c * $final * $final;
- return max(round($result), 0);
- }
- ## --------------------------------------------------------
- private function prepOption($option)
- # Author: Jarrod Oberto
- # Purpose: Prep option like change the passed in option to lowercase
- # Param in: (str/int) $option: eg. 'exact', 'crop'. 0, 4
- # Param out: lowercase string
- # Reference:
- # Notes:
- #
- {
- if (is_array($option)) {
- if (fix_strtolower($option[0]) == 'crop' && count($option) == 2) {
- return 'crop';
- } else {
- throw new Exception('Crop resize option array is badly formatted.');
- }
- } else if (strpos($option, 'crop') !== false) {
- return 'crop';
- }
- if (is_string($option)) {
- return fix_strtolower($option);
- }
- return $option;
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Presets
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- #
- # Preset are pre-defined templates you can apply to your image.
- #
- # These are inteded to be applied to thumbnail images.
- #
- public function borderPreset($preset)
- {
- switch ($preset)
- {
- case 'simple':
- $this->addBorder(7, '#fff');
- $this->addBorder(6, '#f2f1f0');
- $this->addBorder(2, '#fff');
- $this->addBorder(1, '#ccc');
- break;
- default:
- break;
- }
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Draw border
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function addBorder($thickness = 1, $rgbArray = array(255, 255, 255))
- # Author: Jarrod Oberto
- # Date: 05-05-11
- # Purpose: Add a border to the image
- # Param in:
- # Param out:
- # Reference:
- # Notes: This border is added to the INSIDE of the image
- #
- {
- if ($this->imageResized) {
- $rgbArray = $this->formatColor($rgbArray);
- $r = $rgbArray['r'];
- $g = $rgbArray['g'];
- $b = $rgbArray['b'];
- $x1 = 0;
- $y1 = 0;
- $x2 = ImageSX($this->imageResized) - 1;
- $y2 = ImageSY($this->imageResized) - 1;
- $rgbArray = ImageColorAllocate($this->imageResized, $r, $g, $b);
- for($i = 0; $i < $thickness; $i++) {
- ImageRectangle($this->imageResized, $x1++, $y1++, $x2--, $y2--, $rgbArray);
- }
- }
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Gray Scale
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function greyScale()
- # Author: Jarrod Oberto
- # Date: 07-05-2011
- # Purpose: Make image greyscale
- # Param in: n/a
- # Param out:
- # Reference:
- # Notes:
- #
- {
- if ($this->imageResized) {
- imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
- }
- }
- ## --------------------------------------------------------
- public function greyScaleEnhanced()
- # Author: Jarrod Oberto
- # Date: 07-05-2011
- # Purpose: Make image greyscale
- # Param in: n/a
- # Param out:
- # Reference:
- # Notes:
- #
- {
- if ($this->imageResized) {
- imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
- imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -15);
- imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, 2);
- $this->sharpen($this->width);
- }
- }
- ## --------------------------------------------------------
- public function greyScaleDramatic()
- # Alias of gd_filter_monopin
- {
- $this->gd_filter_monopin();
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Black 'n White
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function blackAndWhite()
- # Author: Jarrod Oberto
- # Date: 07-05-2011
- # Purpose: Make image black and white
- # Param in: n/a
- # Param out:
- # Reference:
- # Notes:
- #
- {
- if ($this->imageResized) {
- imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
- imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -1000);
- }
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Negative
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function negative()
- # Author: Jarrod Oberto
- # Date: 07-05-2011
- # Purpose: Make image negative
- # Param in: n/a
- # Param out:
- # Reference:
- # Notes:
- #
- {
- if ($this->imageResized) {
- imagefilter($this->imageResized, IMG_FILTER_NEGATE);
- }
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Sepia
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function sepia()
- # Author: Jarrod Oberto
- # Date: 07-05-2011
- # Purpose: Make image sepia
- # Param in: n/a
- # Param out:
- # Reference:
- # Notes:
- #
- {
- if ($this->imageResized) {
- imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
- imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, -10);
- imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -20);
- imagefilter($this->imageResized, IMG_FILTER_COLORIZE, 60, 30, -15);
- }
- }
- ## --------------------------------------------------------
- public function sepia2()
- {
- if ($this->imageResized) {
- $total = imagecolorstotal( $this->imageResized );
- for ( $i = 0; $i < $total; $i++ ) {
- $index = imagecolorsforindex( $this->imageResized, $i );
- $red = ( $index["red"] * 0.393 + $index["green"] * 0.769 + $index["blue"] * 0.189 ) / 1.351;
- $green = ( $index["red"] * 0.349 + $index["green"] * 0.686 + $index["blue"] * 0.168 ) / 1.203;
- $blue = ( $index["red"] * 0.272 + $index["green"] * 0.534 + $index["blue"] * 0.131 ) / 2.140;
- imagecolorset( $this->imageResized, $i, $red, $green, $blue );
- }
- }
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Vintage
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function vintage()
- # Alias of gd_filter_monopin
- {
- $this->gd_filter_vintage();
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Presets By Marc Hibbins
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- /** Apply 'Monopin' preset */
- public function gd_filter_monopin()
- {
- if ($this->imageResized) {
- imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
- imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, -15);
- imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -15);
- $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'vignette', 100);
- }
- }
- ## --------------------------------------------------------
- public function gd_filter_vintage()
- {
- if ($this->imageResized) {
- $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'vignette', 45);
- imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, 20);
- imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -35);
- imagefilter($this->imageResized, IMG_FILTER_COLORIZE, 60, -10, 35);
- imagefilter($this->imageResized, IMG_FILTER_SMOOTH, 7);
- $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'scratch', 10);
- }
- }
- ## --------------------------------------------------------
- /** Apply a PNG overlay */
- private function gd_apply_overlay($im, $type, $amount)
- #
- # Original Author: Marc Hibbins
- # License: Attribution-ShareAlike 3.0
- # Purpose:
- # Params in:
- # Params out:
- # Notes:
- #
- {
- $width = imagesx($im);
- $height = imagesy($im);
- $filter = imagecreatetruecolor($width, $height);
- imagealphablending($filter, false);
- imagesavealpha($filter, true);
- $transparent = imagecolorallocatealpha($filter, 255, 255, 255, 127);
- imagefilledrectangle($filter, 0, 0, $width, $height, $transparent);
- // *** Resize overlay
- $overlay = $this->filterOverlayPath . '/' . $type . '.png';
- $png = imagecreatefrompng($overlay);
- imagecopyresampled($filter, $png, 0, 0, 0, 0, $width, $height, imagesx($png), imagesy($png));
- $comp = imagecreatetruecolor($width, $height);
- imagecopy($comp, $im, 0, 0, 0, 0, $width, $height);
- imagecopy($comp, $filter, 0, 0, 0, 0, $width, $height);
- imagecopymerge($im, $comp, 0, 0, 0, 0, $width, $height, $amount);
- imagedestroy($comp);
- return $im;
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Colorise
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function image_colorize($rgb) {
- imageTrueColorToPalette($this->imageResized,true,256);
- $numColors = imageColorsTotal($this->imageResized);
- for ($x = 0; $x < $numColors; $x++) {
- list($r,$g,$b) = array_values(imageColorsForIndex($this->imageResized,$x));
- // calculate grayscale in percent
- $grayscale = ($r + $g + $b) / 3 / 0xff;
- imageColorSet($this->imageResized,$x,
- $grayscale * $rgb[0],
- $grayscale * $rgb[1],
- $grayscale * $rgb[2]
- );
- }
- return true;
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Reflection
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function addReflection($reflectionHeight = 50, $startingTransparency = 30, $inside = false, $bgColor = '#fff', $stretch=false, $divider = 0)
- {
- // *** Convert color
- $rgbArray = $this->formatColor($bgColor);
- $r = $rgbArray['r'];
- $g = $rgbArray['g'];
- $b = $rgbArray['b'];
- $im = $this->imageResized;
- $li = imagecreatetruecolor($this->width, 1);
- $bgc = imagecolorallocate($li, $r, $g, $b);
- imagefilledrectangle($li, 0, 0, $this->width, 1, $bgc);
- $bg = imagecreatetruecolor($this->width, $reflectionHeight);
- $wh = imagecolorallocate($im, 255, 255, 255);
- $im = imagerotate($im, -180, $wh);
- imagecopyresampled($bg, $im, 0, 0, 0, 0, $this->width, $this->height, $this->width, $this->height);
- $im = $bg;
- $bg = imagecreatetruecolor($this->width, $reflectionHeight);
- for ($x = 0; $x < $this->width; $x++) {
- imagecopy($bg, $im, $x, 0, $this->width-$x -1, 0, 1, $reflectionHeight);
- }
- $im = $bg;
- $transaprencyAmount = $this->invertTransparency($startingTransparency, 100);
- // *** Fade
- if ($stretch) {
- $step = 100/($reflectionHeight + $startingTransparency);
- } else{
- $step = 100/$reflectionHeight;
- }
- for($i=0; $i<=$reflectionHeight; $i++){
- if($startingTransparency>100) $startingTransparency = 100;
- if($startingTransparency< 1) $startingTransparency = 1;
- imagecopymerge($bg, $li, 0, $i, 0, 0, $this->width, 1, $startingTransparency);
- $startingTransparency+=$step;
- }
- // *** Apply fade
- imagecopymerge($im, $li, 0, 0, 0, 0, $this->width, $divider, 100); // Divider
- // *** width, height of reflection.
- $x = imagesx($im);
- $y = imagesy($im);
- // *** Determines if the reflection should be displayed inside or outside the image
- if ($inside) {
- // Create new blank image with sizes.
- $final = imagecreatetruecolor($this->width, $this->height);
- imagecopymerge ($final, $this->imageResized, 0, 0, 0, $reflectionHeight, $this->width, $this->height - $reflectionHeight, 100);
- imagecopymerge ($final, $im, 0, $this->height - $reflectionHeight, 0, 0, $x, $y, 100);
- } else {
- // Create new blank image with sizes.
- $final = imagecreatetruecolor($this->width, $this->height + $y);
- imagecopymerge ($final, $this->imageResized, 0, 0, 0, 0, $this->width, $this->height, 100);
- imagecopymerge ($final, $im, 0, $this->height, 0, 0, $x, $y, 100);
- }
- $this->imageResized = $final;
- imagedestroy($li);
- imagedestroy($im);
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Rotate
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function rotate($value = 90, $bgColor = 'transparent')
- # Author: Jarrod Oberto
- # Date: 07-05-2011
- # Purpose: Rotate image
- # Param in: (mixed) $degrees: (int) number of degress to rotate image
- # (str) param "left": rotate left
- # (str) param "right": rotate right
- # (str) param "upside": upside-down image
- # Param out:
- # Reference:
- # Notes: The default direction of imageRotate() is counter clockwise.
- #
- {
- if ($this->imageResized) {
- if (is_integer($value)) {
- $degrees = $value;
- }
- // *** Convert color
- $rgbArray = $this->formatColor($bgColor);
- $r = $rgbArray['r'];
- $g = $rgbArray['g'];
- $b = $rgbArray['b'];
- if (isset($rgbArray['a'])) {$a = $rgbArray['a']; }
- if (is_string($value)) {
- $value = fix_strtolower($value);
- switch ($value) {
- case 'left':
- $degrees = 90;
- break;
- case 'right':
- $degrees = 270;
- break;
- case 'upside':
- $degrees = 180;
- break;
- default:
- break;
- }
- }
- // *** The default direction of imageRotate() is counter clockwise
- // * This makes it clockwise
- $degrees = 360 - $degrees;
- // *** Create background color
- $bg = ImageColorAllocateAlpha($this->imageResized, $r, $g, $b, $a);
- // *** Fill with background
- ImageFill($this->imageResized, 0, 0 , $bg);
- // *** Rotate
- $this->imageResized = imagerotate($this->imageResized, $degrees, $bg); // Rotate 45 degrees and allocated the transparent colour as the one to make transparent (obviously)
- // Ensure alpha transparency
- ImageSaveAlpha($this->imageResized,true);
- }
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Round corners
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function roundCorners($radius = 5, $bgColor = 'transparent')
- # Author: Jarrod Oberto
- # Date: 19-05-2011
- # Purpose: Create rounded corners on your image
- # Param in: (int) radius = the amount of curvature
- # (mixed) $bgColor = the corner background color
- # Param out: n/a
- # Reference:
- # Notes:
- #
- {
- // *** Check if the user wants transparency
- $isTransparent = false;
- if (!is_array($bgColor)) {
- if (fix_strtolower($bgColor) == 'transparent') {
- $isTransparent = true;
- }
- }
- // *** If we use transparency, we need to color our curved mask with a unique color
- if ($isTransparent) {
- $bgColor = $this->findUnusedGreen();
- }
- // *** Convert color
- $rgbArray = $this->formatColor($bgColor);
- $r = $rgbArray['r'];
- $g = $rgbArray['g'];
- $b = $rgbArray['b'];
- if (isset($rgbArray['a'])) {$a = $rgbArray['a']; }
- // *** Create top-left corner mask (square)
- $cornerImg = imagecreatetruecolor($radius, $radius);
- //$cornerImg = imagecreate($radius, $radius);
- //imagealphablending($cornerImg, true);
- //imagesavealpha($cornerImg, true);
- //imagealphablending($this->imageResized, false);
- //imagesavealpha($this->imageResized, true);
- // *** Give it a color
- $maskColor = imagecolorallocate($cornerImg, 0, 0, 0);
- // *** Replace the mask color (black) to transparent
- imagecolortransparent($cornerImg, $maskColor);
- // *** Create the image background color
- $imagebgColor = imagecolorallocate($cornerImg, $r, $g, $b);
- // *** Fill the corner area to the user defined color
- imagefill($cornerImg, 0, 0, $imagebgColor);
- imagefilledellipse($cornerImg, $radius, $radius, $radius * 2, $radius * 2, $maskColor );
- // *** Map to top left corner
- imagecopymerge($this->imageResized, $cornerImg, 0, 0, 0, 0, $radius, $radius, 100); #tl
- // *** Map rounded corner to other corners by rotating and applying the mask
- $cornerImg = imagerotate($cornerImg, 90, 0);
- imagecopymerge($this->imageResized, $cornerImg, 0, $this->height - $radius, 0, 0, $radius, $radius, 100); #bl
- $cornerImg = imagerotate($cornerImg, 90, 0);
- imagecopymerge($this->imageResized, $cornerImg, $this->width - $radius, $this->height - $radius, 0, 0, $radius, $radius, 100); #br
- $cornerImg = imagerotate($cornerImg, 90, 0);
- imagecopymerge($this->imageResized, $cornerImg, $this->width - $radius, 0, 0, 0, $radius, $radius, 100); #tr
- // *** If corners are to be transparent, we fill our chromakey color as transparent.
- if ($isTransparent) {
- //imagecolortransparent($this->imageResized, $imagebgColor);
- $this->imageResized = $this->transparentImage($this->imageResized);
- imagesavealpha($this->imageResized, true);
- }
- }
- /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
- Shadow
- *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
- public function addShadow($shadowAngle=45, $blur=15, $bgColor='transparent')
- #
- # Author: Jarrod Oberto (Adapted from Pascal Naidon)
- # Ref: http://www.les-stooges.org/pascal/webdesign/vignettes/index.php?la=en
- # Purpose: Add a drop shadow to your image
- # Params in: (int) $angle: the angle of the shadow
- # (int) $blur: the blur distance
- # (mixed) $bgColor: the color of the background
- # Params out:
- # Notes:
- #
- {
- // *** A higher number results in a smoother shadow
- define('STEPS', $blur*2);
- // *** Set the shadow distance
- $shadowDistance = $blur*0.25;
- // *** Set blur width and height
- $blurWidth = $blurHeight = $blur;
- if ($shadowAngle == 0) {
- $distWidth = 0;
- $distHeight = 0;
- } else {
- $distWidth = $shadowDistance * cos(deg2rad($shadowAngle));
- $distHeight = $shadowDistance * sin(deg2rad($shadowAngle));
- }
- // *** Convert color
- if (fix_strtolower($bgColor) != 'transparent') {
- $rgbArray = $this->formatColor($bgColor);
- $r0 = $rgbArray['r'];
- $g0 = $rgbArray['g'];
- $b0 = $rgbArray['b'];
- }
- $image = $this->imageResized;
- $width = $this->width;
- $height = $this->height;
- $newImage = imagecreatetruecolor($width, $height);
- imagecopyresampled($newImage, $image, 0, 0, 0, 0, $width, $height, $width, $height);
- // *** RGB
- $rgb = imagecreatetruecolor($width+$blurWidth,$height+$blurHeight);
- $colour = imagecolorallocate($rgb, 0, 0, 0);
- imagefilledrectangle($rgb, 0, 0, $width+$blurWidth, $height+$blurHeight, $colour);
- $colour = imagecolorallocate($rgb, 255, 255, 255);
- //imagefilledrectangle($rgb, $blurWidth*0.5-$distWidth, $blurHeight*0.5-$distHeight, $width+$blurWidth*0.5-$distWidth, $height+$blurWidth*0.5-$distHeight, $colour);
- imagefilledrectangle($rgb, $blurWidth*0.5-$distWidth, $blurHeight*0.5-$distHeight, $width+$blurWidth*0.5-$distWidth, $height+$blurWidth*0.5-$distHeight, $colour);
- //imagecopymerge($rgb, $newImage, 1+$blurWidth*0.5-$distWidth, 1+$blurHeight*0.5-$distHeight, 0,0, $width, $height, 100);
- imagecopymerge($rgb, $newImage, $blurWidth*0.5-$distWidth, $blurHeight*0.5-$distHeight, 0,0, $width+$blurWidth, $height+$blurHeight, 100);
- // *** Shadow (alpha)
- $shadow = imagecreatetruecolor($width+$blurWidth,$height+$blurHeight);
- imagealphablending($shadow, false);
- $colour = imagecolorallocate($shadow, 0, 0, 0);
- imagefilledrectangle($shadow, 0, 0, $width+$blurWidth, $height+$blurHeight, $colour);
- for($i=0;$i<=STEPS;$i++) {
- $t = ((1.0*$i)/STEPS);
- $intensity = 255*$t*$t;
- $colour = imagecolorallocate($shadow, $intensity, $intensity, $intensity);
- $points = array(
- $blurWidth*$t, …
Large files files are truncated, but you can click here to view the full file