????

Your IP : 216.73.216.152


Current Path : /home2/morganrand/www/wp-content-bkp/plugins/jch-optimize/jchoptimize/libs/
Upload File :
Current File : /home2/morganrand/www/wp-content-bkp/plugins/jch-optimize/jchoptimize/libs/CssSpriteGen.php

<?php

namespace JchOptimize;

use Imagick;
use ImagickPixel;
use ImagickException;
use JchOptimizeHelper;
use JchPlatformUtility;
use JchPlatformProfiler;
use JchPlatformPaths;

/**
 * This is a modified version of the original class from the online css sprite generator found at
 * http://spritegen.website-performance.org/ 
 *
 * @copyright Copyright (C) 2007-2009, Project Fondue (Ed Eliot, Stuart Colville
 *              & Cyril Doussin). All rights reserved.
 * @license Software License Agreement (BSD License)
 */

/**
 * JCH Optimize - Plugin to aggregate and minify external resources for
 * optmized downloads
 * @author Samuel Marshall <sdmarshall73@gmail.com>
 * @copyright Copyright (c) 2010 Samuel Marshall
 * @license GNU/GPLv3, See LICENSE file
 * This program is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 *
 * This program is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 * GNU General Public License for more details.
 *
 * If LICENSE file missing, see <http://www.gnu.org/licenses/>.
 */

class CssSpriteGen
{

        public $aFormValues    = array();
        public $bTransparent;
        protected $aImageTypes = array();
        protected $aFormErrors = array();
        protected $sZipFolder  = '';
        protected $sCss;
        protected $sTempSpriteName;
        protected $bValidImages;
        protected $aBackground = array();
        protected $aPosition   = array();
        protected $oImageHandler;
        protected $params;
        protected $bBackend    = FALSE;

        public function __construct($ImageLibrary, $params, $bBackend = FALSE)
        {
                $this->bBackend = $bBackend;

                $this->params = $params;

                $class = 'JchOptimize\ImageHandler' . ucfirst($ImageLibrary);

                $this->oImageHandler = new $class($this->params, $this);

                $this->aImageTypes = $this->oImageHandler->getSupportedFormats();

                $this->aFormValues = array(
                        'path'                    => '',
                        'sub'                     => '',
                        'file-regex'              => '',
                        'wrap-columns'            => $this->params->get('csg_wrap_images', 'off'),
                        'build-direction'         => $this->params->get('csg_direction', 'vertical'),
                        'use-transparency'        => 'on',
                        'use-optipng'             => '',
                        'vertical-offset'         => 50,
                        'horizontal-offset'       => 50,
                        'background'              => '',
                        'image-output'            => 'PNG', //$this->params->get('csg_file_output'),
                        'image-num-colours'       => 'true-colour',
                        'image-quality'           => 100,
                        'width-resize'            => 100,
                        'height-resize'           => 100,
                        'ignore-duplicates'       => 'merge',
                        'class-prefix'            => '',
                        'selector-prefix'         => '',
                        'selector-suffix'         => '',
                        'add-width-height-to-css' => 'off',
                        'sprite-path'             => $this->params->get('sprite-path')
                );
        }

        public function GetImageTypes()
        {
                return $this->aImageTypes;
        }

        public function GetSpriteFormats()
        {
                return $this->oImageHandler->aSpriteFormats;
        }

        protected $iMaxWidth;
        protected $iMaxHeight;
        protected $aMaxRowHeight;
        protected $aMaxColumnWidth;
        protected $iMaxVOffset;
        protected $iMaxHOffset;

        public function CreateSprite($aFilePaths)
        {
                // set up variable defaults used when calculating offsets etc
                $aFilesInfo   = array();
                $aFilesMD5    = array();
                $bResize      = false;
                $aValidImages = array();

                if ($this->aFormValues['build-direction'] == 'horizontal')
                {
                        $iRowCount     = 1;
                        $iTotalWidth   = 0;
                        $iTotalHeight  = 0;
                        $aMaxRowHeight = array();
                        $iMaxVOffset   = 0;
                }
                else
                {
                        $iColumnCount    = 1;
                        $iTotalWidth     = 0;
                        $iTotalHeight    = 0;
                        $aMaxColumnWidth = array();
                        $iMaxHOffset     = 0;
                }
                $iMaxWidth     = 0;
                $iMaxHeight    = 0;
                $i             = 0;
                $k             = 0;
                $bValidImages  = false;
                $sOutputFormat = strtolower($this->aFormValues['image-output']);

                $optimize = FALSE;

                /*                 * **************************************** */
                /* this section calculates all offsets etc */
                /*                 * **************************************** */

                foreach ($aFilePaths as $sFile)
                {
                        JCH_DEBUG ? JchPlatformProfiler::start('CalculateSprite') : null;
                        
                        $sFilePath = str_replace(JchOptimizeHelper::cookieLessDomain($this->params, '', TRUE), '', $sFile);
                        $sFilePath = JchOptimizeHelper::getFilepath($sFilePath);

                        $bFileExists = TRUE;

                        if (@file_exists($sFilePath))
                        {

                                // do we want to scale down the source images
                                // scaling up isn't supported as that would result in poorer quality images
                                $bResize = ($this->aFormValues['width-resize'] != 100 && $this->aFormValues['height-resize'] != 100);

                                // grab path information
                                //$sFilePath = $sFolderMD5.$sFile;
                                $aPathParts = pathinfo($sFilePath);
                                $sFileBaseName      = $aPathParts['basename'];

                                $iImageType = exif_imagetype($sFilePath);
                                $aImageInfo = getimagesize($sFilePath);


                                $iWidth  = $aImageInfo[0];
                                $iHeight = $aImageInfo[1];


                                // are we matching filenames against a regular expression
                                // if so it's likely not all images from the ZIP file will end up in the generated sprite image
                                if (!empty($this->aFormValues['file-regex']))
                                {
                                        // forward slashes should be escaped - it's likely not doing this might be a security risk also
                                        // one might be able to break out and change the modifiers (to for example run PHP code)
                                        $this->aFormValues['file-regex'] = str_replace('/', '\/', $this->aFormValues['file-regex']);

                                        // if the regular expression matches grab the first match and store for use as the class name
                                        if (preg_match('/^' . $this->aFormValues['file-regex'] . '$/i', $sFileBaseName, $aMatches))
                                        {
                                                $sFileClass = $aMatches[1];
                                        }
                                        else
                                        {
                                                $sFileClass = '';
                                        }
                                }
                                else
                                { // not using regular expressions - set the class name to the base part of the filename (excluding extension)
                                        $sFileClass = $aPathParts['basename'];
                                }

                                // format the class name - it should only contain certain characters
                                // this strips out any which aren't
                                $sFileClass = $this->FormatClassName($sFileClass);
                        }
                        else
                        {
                                $bFileExists = FALSE;
                        }

                        // the file also isn't valid if its extension doesn't match one of the image formats supported by the tool
                        //discard images whose height or width is greater than 50px
                        if (
                                $bFileExists && !empty($sFileClass) && in_array(strtoupper($aPathParts['extension']), $this->aImageTypes)
                                && in_array($iImageType, array(IMAGETYPE_GIF, IMAGETYPE_JPEG, IMAGETYPE_PNG))
                                && substr($sFileBaseName, 0, 1) != '.'
                                && $iWidth < 50 && $iHeight < 50
                        )
                        {
                                // grab the file extension
                                $sExtension = $aPathParts['extension'];

                                // get MD5 of file (this can be used to compare if a file's content is exactly the same as another's)
                                $sFileMD5 = md5(file_get_contents($sFilePath));

                                // check if this file's MD5 already exists in array of MD5s recorded so far
                                // if so it's a duplicate of another file in the ZIP
                                if (($sKey = array_search($sFileMD5, $aFilesMD5)) !== false)
                                {
                                        // do we want to drop duplicate files and merge CSS rules
                                        // if so CSS will end up like .filename1, .filename2 { }
                                        if ($this->aFormValues['ignore-duplicates'] == 'merge')
                                        {

                                                if (isset($aFilesInfo[$sKey]['class']))
                                                {
                                                        $aFilesInfo[$sKey]['class'] = $aFilesInfo[$sKey]['class'] .
                                                                $this->aFormValues['selector-suffix'] . ', ' .
                                                                $this->aFormValues['selector-prefix'] . '.' .
                                                                $this->aFormValues['class-prefix'] . $sFileClass;
                                                }

                                                $this->aBackground[$k] = $sKey;
                                                $k++;

                                                continue;
                                        }
                                }
                                else
                                {
                                        $this->aBackground[$k] = $i;
                                        $k++;
                                }

                                // add MD5 to array to check future files against
                                $aFilesMD5[$i]          = $sFileMD5;
                                // store generated class selector details
                                //$aFilesInfo[$i]['class'] = ".{$this->aFormValues['class-prefix']}$sFileClass";
                                // store file path information and extension
                                $aFilesInfo[$i]['path'] = $sFilePath;
                                $aFilesInfo[$i]['ext']  = $sExtension;


                                if ($this->aFormValues['build-direction'] == 'horizontal')
                                {
                                        // get the current width of the sprite image - after images processed so far
                                        $iCurrentWidth = $iTotalWidth + $this->aFormValues['horizontal-offset'] + $iWidth;

                                        // store the maximum width reached so far
                                        // if we're on a new column current height might be less than the maximum
                                        if ($iMaxWidth < $iCurrentWidth)
                                        {
                                                $iMaxWidth = $iCurrentWidth;
                                        }
                                }
                                else
                                {
                                        // get the current height of the sprite image - after images processed so far
                                        $iCurrentHeight = $iTotalHeight + $this->aFormValues['vertical-offset'] + $iHeight;

                                        // store the maximum height reached so far
                                        // if we're on a new column current height might be less than the maximum
                                        if ($iMaxHeight < $iCurrentHeight)
                                        {
                                                $iMaxHeight = $iCurrentHeight;
                                        }
                                }

                                // store the original width and height of the image
                                // we'll need this later if the image is to be resized
                                $aFilesInfo[$i]['original-width']  = $iWidth;
                                $aFilesInfo[$i]['original-height'] = $iHeight;

                                // store the width and height of the image
                                // if we're resizing they'll be less than the original
                                $aFilesInfo[$i]['width']  = $bResize ? round(($iWidth / 100) * $this->aFormValues['width-resize']) : $iWidth;
                                $aFilesInfo[$i]['height'] = $bResize ? round(($iHeight / 100) * $this->aFormValues['height-resize']) : $iHeight;

                                if ($this->aFormValues['build-direction'] == 'horizontal')
                                {
                                        // opera (9.0 and below) has a bug which prevents it recognising  offsets of less than -2042px
                                        // all subsequent values are treated as -2042px
                                        // if we've hit 2000 pixels and we care about this (as set in the interface) then wrap to a new row
                                        // increment row count and reset current height
                                        if (
                                                ($iTotalWidth + $this->aFormValues['horizontal-offset']) >= 2000 && !empty($this->aFormValues['wrap-columns'])
                                        )
                                        {
                                                $iRowCount++;
                                                $iTotalWidth = 0;
                                        }

                                        // if the current image is higher than any other in the current row then set the maximum height to that
                                        // it will be used to set the height of the current row
                                        if ($aFilesInfo[$i]['height'] > $iMaxHeight)
                                        {
                                                $iMaxHeight = $aFilesInfo[$i]['height'];
                                        }

                                        // keep track of the height of rows added so far
                                        $aMaxRowHeight[$iRowCount] = $iMaxHeight;
                                        // calculate the current maximum vertical offset so far
                                        $iMaxVOffset               = $this->aFormValues['vertical-offset'] * ($iRowCount - 1);

                                        // get the x position of current image in overall sprite
                                        $aFilesInfo[$i]['x'] = $iTotalWidth;
                                        $iTotalWidth += ($aFilesInfo[$i]['width'] + $this->aFormValues['horizontal-offset']);
                                        // get the y position of current image in overall sprite
                                        if ($iRowCount == 1)
                                        {
                                                $aFilesInfo[$i]['y'] = 0;
                                        }
                                        else
                                        {
                                                $aFilesInfo[$i]['y'] = (
                                                        $this->aFormValues['vertical-offset'] * ($iRowCount - 1) + (array_sum($aMaxRowHeight) - $aMaxRowHeight[$iRowCount])
                                                        );
                                        }
                                        $aFilesInfo[$i]['currentCombinedWidth'] = $iTotalWidth;
                                        $aFilesInfo[$i]['rowNumber']            = $iRowCount;
                                }
                                else
                                {
                                        if (
                                        // opera (9.0 and below) has a bug which prevents it recognising  offsets of less than -2042px
                                        // all subsequent values are treated as -2042px
                                        // if we've hit 2000 pixels and we care about this (as set in the interface) then wrap to a new column
                                        // increment column count and reset current height
                                                ($iTotalHeight + $this->aFormValues['vertical-offset']) >= 2000 && !empty($this->aFormValues['wrap-columns'])
                                        )
                                        {
                                                $iColumnCount++;
                                                $iTotalHeight = 0;
                                        }

                                        // if the current image is wider than any other in the current column then set the maximum width to that
                                        // it will be used to set the width of the current column
                                        if ($aFilesInfo[$i]['width'] > $iMaxWidth)
                                        {
                                                $iMaxWidth = $aFilesInfo[$i]['width'];
                                        }

                                        // keep track of the width of columns added so far
                                        $aMaxColumnWidth[$iColumnCount] = $iMaxWidth;
                                        // calculate the current maximum horizontal offset so far
                                        $iMaxHOffset                    = $this->aFormValues['horizontal-offset'] * ($iColumnCount - 1);

                                        // get the y position of current image in overall sprite
                                        $aFilesInfo[$i]['y'] = $iTotalHeight;
                                        $iTotalHeight += ($aFilesInfo[$i]['height'] + $this->aFormValues['vertical-offset']);
                                        // get the x position of current image in overall sprite
                                        if ($iColumnCount == 1)
                                        {
                                                $aFilesInfo[$i]['x'] = 0;
                                        }
                                        else
                                        {
                                                $aFilesInfo[$i]['x'] = (
                                                        $this->aFormValues['horizontal-offset'] * ($iColumnCount - 1) + (array_sum($aMaxColumnWidth) -
                                                        $aMaxColumnWidth[$iColumnCount])
                                                        );
                                        }
                                        $aFilesInfo[$i]['currentCombinedHeight'] = $iTotalHeight;
                                        $aFilesInfo[$i]['columnNumber']          = $iColumnCount;
                                }

                                $i++;

                                $aValidImages[] = $sFile;
                        }
                        else
                        {
                                $this->aBackground[$k] = null;
                                $k++;
                        }

                        if ($i > 30)
                        {
                                break;
                        }
                }

                JCH_DEBUG ? JchPlatformProfiler::stop('CalculateSprite', TRUE) : null;

                if ($this->bBackend)
                {
                        return $aValidImages;
                }

                JCH_DEBUG ? JchPlatformProfiler::start('CreateSprite') : null;


                /*                 * **************************************** */
                /* this section generates the sprite image */
                /* and CSS rules                           */
                /*                 * **************************************** */
                // if $i is greater than 1 then we managed to generate enough info to create a sprite
                if (count($aFilesInfo) > 1)
                {

                        // if Imagick throws an exception we want the script to terminate cleanly so that
                        // temporary files are cleaned up
                        try
                        {
                                // get the sprite width and height
                                if ($this->aFormValues['build-direction'] == 'horizontal')
                                {
                                        $iSpriteWidth  = $iMaxWidth - $this->aFormValues['horizontal-offset'];
                                        $iSpriteHeight = array_sum($aMaxRowHeight) + $iMaxVOffset;
                                }
                                else
                                {
                                        $iSpriteHeight = $iMaxHeight - $this->aFormValues['vertical-offset'];
                                        $iSpriteWidth  = array_sum($aMaxColumnWidth) + $iMaxHOffset;
                                }

                                // get background colour - remove # if added
                                $sBgColour = str_replace('#', '', $this->aFormValues['background']);
                                // convert 3 digit hex values to 6 digit equivalent
                                if (strlen($sBgColour) == 3)
                                {
                                        $sBgColour = substr($sBgColour, 0, 1) .
                                                substr($sBgColour, 0, 1) .
                                                substr($sBgColour, 1, 1) .
                                                substr($sBgColour, 1, 1) .
                                                substr($sBgColour, 2, 1) .
                                                substr($sBgColour, 2, 1);
                                }
                                // should the image be transparent
                                $this->bTransparent = (
                                        !empty($this->aFormValues['use-transparency']) && in_array($this->aFormValues['image-output'],
                                                                                                   array('GIF', 'PNG'))
                                        );

                                $oSprite = $this->oImageHandler->createSprite($iSpriteWidth, $iSpriteHeight, $sBgColour, $sOutputFormat);

                                // initalise variable to store CSS rules
                                $this->aCss = array();

                                // loop through file info for valid images
                                for ($i = 0; $i < count($aFilesInfo); $i++)
                                {
                                        // create a new image object for current file
                                        if (!$oCurrentImage = $this->oImageHandler->createImage($aFilesInfo[$i]))
                                        {
                                                // if we've got here then a valid but corrupt image was found
                                                // at this stage we've already allocated space for the image so create
                                                // a blank one to fill the space instead
                                                // this should happen very rarely
                                                $oCurrentImage = $this->oImageHandler->createBlankImage($aFilesInfo[$i]);
                                        }

                                        // if resizing get image width and height and resample to new dimensions (percentage of original)
                                        // and copy to sprite image
                                        if ($bResize)
                                        {
                                                $this->oImageHandler->resizeImage($oSprite, $oCurrentImage, $aFilesInfo[$i]);
                                        }

                                        // copy image to sprite
                                        $this->oImageHandler->copyImageToSprite($oSprite, $oCurrentImage, $aFilesInfo[$i], $bResize);

                                        // get CSS x & y values
                                        $iX                  = $aFilesInfo[$i]['x'] != 0 ? '-' . $aFilesInfo[$i]['x'] . 'px' : '0';
                                        $iY                  = $aFilesInfo[$i]['y'] != 0 ? '-' . $aFilesInfo[$i]['y'] . 'px' : '0';
                                        $this->aPosition[$i] = $iX . ' ' . $iY;
                                        // create CSS rules and append to overall CSS rules
//                                        $this->sCss .= "{$this->aFormValues['selector-prefix']}{$aFilesInfo[$i]['class']} "
//                                                . "{$this->aFormValues['selector-suffix']}{ background-position: $iX $iY; ";
//
//                                        // If add widths and heights the sprite image width and height are added to the CSS
//                                        if ($this->aFormValues['add-width-height-to-css'] == 'on')
//                                        {
//                                                $this->sCss .= "width: {$aFilesInfo[$i]['width']}px; height: {$aFilesInfo[$i]['height']}px;";
//                                        }
//
//                                        $this->sCss .= " } \n";
                                        // destroy object created for current image to save memory
                                        $this->oImageHandler->destroy($oCurrentImage);
                                }


                                $path                  = $this->aFormValues['sprite-path'];
                                //See if image already exists
                                //
                                // create a unqiue filename for sprite image
                                $sSpriteMD5            = md5(implode($aFilesMD5) . implode($this->aFormValues));
                                $this->sTempSpriteName = $path . DIRECTORY_SEPARATOR . 'csg-' . $sSpriteMD5 . ".$sOutputFormat";

                                if (!file_exists($path))
                                {
                                        JchPlatformUtility::createFolder($path);
                                }

                                // write image to file
                                if (!file_exists($this->sTempSpriteName))
                                {
                                        $this->oImageHandler->writeImage($oSprite, $sOutputFormat, $this->sTempSpriteName);

                                        $optimize = TRUE;
                                }

                                

                                // destroy object created for sprite image to save memory
                                $this->oImageHandler->destroy($oSprite);

                                // set flag to indicate valid images created
                                $this->bValidImages = true;
                        }
                        catch (Exception $e)
                        {
                                JchOptimizeLogger::log($e->getMessage(), $this->params);
                        }
                        
                        JCH_DEBUG ? JchPlatformProfiler::stop('CreateSprite', TRUE) : null;
                }

                
                ##<procode>##
                if ($optimize)
                {
                        $curl_enabled    = function_exists('curl_version') && curl_version();
                        $allow_url_fopen = (bool) ini_get('allow_url_fopen');

                        if ($curl_enabled && $allow_url_fopen)
                        {
                                $this->optimizeImage($this->sTempSpriteName);
                        }
                }
                ##</procode>##
        }

        protected function FormatClassName($sClassName)
        {
                $aExtensions = array();

                foreach ($this->aImageTypes as $sType)
                {
                        $aExtensions[] = ".$sType";
                }

                return preg_replace("/[^a-z0-9_-]+/i", '', str_ireplace($aExtensions, '', $sClassName));
        }

        ##<procode>##

        protected function optimizeImage($sFileName)
        {
                JCH_DEBUG ? JchPlatformProfiler::start('OptimizeSprite') : null;

                $aQuery = array(
                        'dir'   => JchOptimizeHelper::prepareImageUrl($sFileName),
                        'async' => 1,
                        'task'  => 'optimize'
                );

                $url = JchPlatformPaths::ajaxUrl('optimizeimages');

                JchOptimizeHelper::postAsync($url, $this->params, $aQuery);

                JCH_DEBUG ? JchPlatformProfiler::stop('OptimizeSprite', TRUE) : null;
        }

        ##</procode>##

        public function ValidImages()
        {
                return $this->bValidImages;
        }

        public function GetSpriteFilename()
        {
                $aFileParts = pathinfo($this->sTempSpriteName);
                return $aFileParts['basename'];
        }

        public function GetSpriteHash()
        {
                //return md5($this->GetSpriteFilename().ConfigHelper::Get('/checksum'));
        }

        public function GetCss()
        {
                return $this->aCss;
        }

        public function GetAllErrors()
        {
                return $this->aFormErrors;
        }

        public function GetZipFolder()
        {
                return $this->sZipFolder;
        }

        public function GetCssBackground()
        {
                $aCssBackground = array();

                foreach ($this->aBackground as $background)
                {
                        $aCssBackground[] = @$this->aPosition[$background];
                }

                return $aCssBackground;
        }

}

class ImageHandlerImagick implements ImageHandlerInterface
{

        protected $params;
        protected $obj;
        public $aSpriteFormats = array();

        public function __construct($params, $obj)
        {
                $this->obj    = $obj;
                $this->params = $params;
        }

        public function getSupportedFormats()
        {
                $aImageTypes = array();

                try
                {
                        $oImagick      = new Imagick();
                        $aImageFormats = $oImagick->queryFormats();
                }
                catch (ImagickException $e)
                {
                        JchOptimizeLogger::log($e->getMessage(), $this->params);
                }

                // store supported formats for populating drop downs etc later
                if (in_array('PNG', $aImageFormats))
                {
                        $aImageTypes[] = 'PNG';

                        $this->aSpriteFormats[] = 'PNG';
                }
                if (in_array('GIF', $aImageFormats))
                {
                        $aImageTypes[] = 'GIF';

                        $this->aSpriteFormats[] = 'GIF';
                }
                if (in_array('JPG', $aImageFormats) || in_array('JPEG', $aImageFormats))
                {
                        $aImageTypes[] = 'JPG';
                }

                return $aImageTypes;
        }

        public function createSprite($iSpriteWidth, $iSpriteHeight, $sBgColour, $sOutputFormat)
        {
                $oSprite = new Imagick();
                // create a new image - set background according to transparency
                if (!empty($this->obj->aFormValues['background']))
                {
                        $oSprite->newImage($iSpriteWidth, $iSpriteHeight, new ImagickPixel("#$sBgColour"), $sOutputFormat);
                }
                else
                {
                        if ($this->obj->bTransparent)
                        {
                                $oSprite->newImage($iSpriteWidth, $iSpriteHeight, new ImagickPixel('#000000'), $sOutputFormat);
                        }
                        else
                        {
                                $oSprite->newImage($iSpriteWidth, $iSpriteHeight, new ImagickPixel('#ffffff'), $sOutputFormat);
                        }
                }

                // check for transparency option
                if ($this->obj->bTransparent)
                {
                        // set background colour to transparent
                        // if no background colour use black
                        if (!empty($this->obj->aFormValues['background']))
                        {
                                $oSprite->transparentPaintImage(new ImagickPixel("#$sBgColour"), 0.0, 0, false);
                        }
                        else
                        {
                                $oSprite->transparentPaintImage(new ImagickPixel("#000000"), 0.0, 0, false);
                        }
                }

                return $oSprite;
        }

        public function createBlankImage($aFileInfo)
        {
                $oCurrentImage = new Imagick();

                $oCurrentImage->newImage(
                        $aFileInfo['original-width'], $aFileInfo['original-height'], new ImagickPixel('#ffffff')
                );

                return $oCurrentImage;
        }

        public function resizeImage($oSprite, $oCurrentImage, $aFileInfo)
        {
                $oCurrentImage->thumbnailImage($aFileInfo['width'], $aFileInfo['height']);
        }

        public function copyImageToSprite($oSprite, $oCurrentImage, $aFileInfo, $bResize)
        {
                $oSprite->compositeImage(
                        $oCurrentImage, $oCurrentImage->getImageCompose(), $aFileInfo['x'], $aFileInfo['y']
                );
        }

        public function destroy($oImage)
        {
                $oImage->destroy();
        }

        public function createImage($aFileInfo)
        {
                // Imagick auto detects file extension when creating object from image
                $oImage = new Imagick();
                $oImage->readImage($aFileInfo['path']);

                return $oImage;
        }

        public function writeImage($oImage, $sExtension, $sFilename)
        {

                // check if we want to resample image to lower number of colours (to reduce file size)
                if (in_array($sExtension, array('gif', 'png')) && $this->obj->aFormValues['image-num-colours'] != 'true-colour')
                {
                        $oImage->quantizeImage($this->obj->aFormValues['image-num-colours'], Imagick::COLORSPACE_RGB, 0, false, false);
                }
                // if we're creating a JEPG set image quality - 0% - 100%
                if (in_array($sExtension, array('jpg', 'jpeg')))
                {
                        $oImage->setCompression(Imagick::COMPRESSION_JPEG);
                        $oImage->SetCompressionQuality($this->obj->aFormValues['image-quality']);
                }
                // write out image to file
                $oImage->writeImage($sFilename);
        }

}

class ImageHandlerGd implements ImageHandlerInterface
{

        protected $params;
        protected $obj;
        public $aSpriteFormats = array();

        public function __construct($params, $obj)
        {
                $this->obj    = $obj;
                $this->params = $params;
        }

        public function getSupportedFormats()
        {
                // get info about installed GD library to get image types (some versions of GD don't include GIF support)
                $oGD         = gd_info();
                $aImageTypes = array();
                // store supported formats for populating drop downs etc later
                if (@$oGD['PNG Support'])
                {
                        $aImageTypes[] = 'PNG';

                        $this->aSpriteFormats[] = 'PNG';
                }
                if (@$oGD['GIF Create Support'])
                {
                        $aImageTypes[] = 'GIF';
                }
                if (@$oGD['JPG Support'] || @$oGD['JPEG Support'])
                {
                        $aImageTypes[] = 'JPG';
                }

                return $aImageTypes;
        }

        public function createSprite($iSpriteWidth, $iSpriteHeight, $sBgColour, $sOutputFormat)
        {

                if ($this->obj->bTransparent && !empty($this->obj->aFormValues['background']))
                {
                        $oSprite = imagecreate($iSpriteWidth, $iSpriteHeight);
                }
                else
                {
                        $oSprite = imagecreatetruecolor($iSpriteWidth, $iSpriteHeight);
                }

                // check for transparency option
                if ($this->obj->bTransparent)
                {
                        if ($sOutputFormat == "png")
                        {
                                imagealphablending($oSprite, false);
                                $colorTransparent = imagecolorallocatealpha($oSprite, 0, 0, 0, 127);
                                imagefill($oSprite, 0, 0, $colorTransparent);
                                imagesavealpha($oSprite, true);
                        }
                        elseif ($sOutputFormat == "gif")
                        {
                                $iBgColour = imagecolorallocate($oSprite, 0, 0, 0);

                                imagecolortransparent($oSprite, $iBgColour);
                        }
                }
                else
                {
                        if (empty($sBgColour))
                        {
                                $sBgColour = 'ffffff';
                        }
                        $iBgColour = hexdec($sBgColour);
                        $iBgColour = imagecolorallocate(
                                $oSprite, 0xFF & ($iBgColour >> 0x10), 0xFF & ($iBgColour >> 0x8), 0xFF & $iBgColour
                        );
                        imagefill($oSprite, 0, 0, $iBgColour);
                }

                return $oSprite;
        }

        public function createBlankImage($aFileInfo)
        {
                $oCurrentImage = imagecreatetruecolor($aFileInfo['original-width'], $aFileInfo['original-height']);
                imagecolorallocate($oCurrentImage, 255, 255, 255);

                return $oCurrentImage;
        }

        public function resizeImage($oSprite, $oCurrentImage, $aFileInfo)
        {
                imagecopyresampled(
                        $oSprite, $oCurrentImage, $aFileInfo['x'], $aFileInfo['y'], 0, 0, $aFileInfo['width'], $aFileInfo['height'],
                        $aFileInfo['original-width'], $aFileInfo['original-height']
                );
        }

        public function copyImageToSprite($oSprite, $oCurrentImage, $aFileInfo, $bResize)
        {
                // if already resized the image will have been copied as part of the resize
                if (!$bResize)
                {
                        imagecopy(
                                $oSprite, $oCurrentImage, $aFileInfo['x'], $aFileInfo['y'], 0, 0, $aFileInfo['width'], $aFileInfo['height']
                        );
                }
        }

        public function destroy($oImage)
        {
                imagedestroy($oImage);
        }

        public function createImage($aFileInfo)
        {
                $sFile = $aFileInfo['path'];

                switch ($aFileInfo['ext'])
                {
                        case 'jpg':
                        case 'jpeg':
                                $oImage = @imagecreatefromjpeg($sFile);
                                break;
                        case 'gif':
                                $oImage = @imagecreatefromgif($sFile);
                                break;
                        case 'png':
                                $oImage = @imagecreatefrompng($sFile);
                                break;
                }

                return $oImage;
        }

        public function writeImage($oImage, $sExtension, $sFilename)
        {
                // check if we want to resample image to lower number of colours (to reduce file size)
                if (in_array($sExtension, array('gif', 'png')) && $this->obj->aFormValues['image-num-colours'] != 'true-colour')
                {
                        imagetruecolortopalette($oImage, true, $this->obj->aFormValues['image-num-colours']);
                }

                switch ($sExtension)
                {
                        case 'jpg':
                        case 'jpeg':
                                // GD takes quality setting in main creation function
                                imagejpeg($oImage, $sFilename, $this->obj->aFormValues['image-quality']);
                                break;
                        case 'gif':
                                // force colour palette to 256 colours if saving sprite image as GIF
                                // this will happen anyway (as GIFs can't be more than 256 colours)
                                // but the quality will be better if pre-forcing
                                if (
                                        $this->obj->bTransparent && (
                                        $this->obj->aFormValues['image-num-colours'] == -1 || $this->obj->aFormValues['image-num-colours'] > 256 || $this->obj->aFormValues['image-num-colours'] ==
                                        'true-colour'
                                        )
                                )
                                {
                                        imagetruecolortopalette($oImage, true, 256);
                                }
                                imagegif($oImage, $sFilename);
                                break;
                        case 'png':
                                imagepng($oImage, $sFilename, 0);
                                break;
                }
        }

}

interface ImageHandlerInterface
{

        public function getSupportedFormats();

        public function createSprite($iSpriteWidth, $iSpriteHeight, $sBgColour, $sOutputFormat);

        public function createBlankImage($aFileInfo);

        public function resizeImage($oSprite, $oCurrentImage, $aFileInfo);

        public function copyImageToSprite($oSprite, $oCurrentImage, $aFileInfo, $bResize);

        public function destroy($oImage);

        public function createImage($aFileInfo);

        public function writeImage($oImage, $sExtension, $sFilename);
}

?>