Seditio Source
Root |
./othercms/ips_4.3.4/system/Image/Image.php
<?php
/**
 * @brief        Image Class
 * @author        <a href='https://www.invisioncommunity.com'>Invision Power Services, Inc.</a>
 * @copyright    (c) Invision Power Services, Inc.
 * @license        https://www.invisioncommunity.com/legal/standards/
 * @package        Invision Community
 * @since        19 Feb 2013
 */

namespace IPS;

/* To prevent PHP errors (extending class does not exist) revealing path */
if ( !defined( '\IPS\SUITE_UNIQUE_KEY' ) )
{
   
header( ( isset( $_SERVER['SERVER_PROTOCOL'] ) ? $_SERVER['SERVER_PROTOCOL'] : 'HTTP/1.0' ) . ' 403 Forbidden' );
    exit;
}

/**
 * Image Class
 */
abstract class _Image
{
   
/**
     * @brief    Image Extensions
     */
   
public static $imageExtensions = array( 'gif', 'jpeg', 'jpe', 'jpg', 'png' );
   
   
/**
     * @brief    Image Mime Types
     */
   
public static $imageMimes = array( 'image/gif', 'image/jpeg', 'image/pjpeg', 'image/png' );
   
   
/**
     * Determine if EXIF extraction is supported
     *
     * @return    bool
     */
   
public static function exifSupported()
    {
        return
function_exists( 'exif_read_data' );
    }
   
   
/**
     * @brief    Has the image been automatically rotated?
     */
   
public $hasBeenRotated = FALSE;

   
/**
     * Create Object
     *
     * @param    string    $contents    Contents
     * @return    \IPS\Image
     * @throws    \InvalidArgumentException
     */
   
public static function create( $contents )
    {
       
/* Work out the type */
       
$imageType = NULL;
       
$isAnimated = FALSE;
       
$signatures = array(
           
'gif'    => array(
               
'47' . '49' . '46' . '38' . '37' . '61',
               
'47' . '49' . '46' . '38' . '39' . '61'
           
),
           
'jpeg'    => array(
               
'ff' . 'd8' . 'ff'
           
),
           
'png'    => array(
               
'89' . '50' . '4e' . '47' . '0d' . '0a' . '1a' . '0a'
           
)
        );        
       
$bytesNeeded = max( array_map( 'strlen', array_map( 'hex2bin', call_user_func_array( 'array_merge', $signatures ) ) ) );
       
$fileHeader = \substr( $contents, 0, $bytesNeeded );
        foreach (
$signatures as $type => $_signatures )
        {
            foreach (
$_signatures as $signature )
            {
                if (
bin2hex( \substr( $fileHeader, 0, \strlen( hex2bin( $signature ) ) ) ) === $signature )
                {
                   
$imageType = $type;
                    break
2;
                }
            }
        }
        if (
$imageType === NULL )
        {
            throw new \
InvalidArgumentException;
        }
               
       
/* Create object */
       
if( \IPS\Settings::i()->image_suite == 'imagemagick' and class_exists( 'Imagick', FALSE ) )
        {
           
$obj = new Image\Imagemagick( $contents );
        }
        else
        {
           
$obj = new Image\Gd( $contents );
        }
       
$obj->type = $imageType;
       
       
/* Animated? */
       
if ( $obj->type === 'gif' and (boolean) preg_match( '#(\x00\x21\xF9\x04.{4}\x00\x2C.*){2,}#s', $contents ) )
        {
           
$obj->isAnimatedGif = TRUE;
           
$obj->contents = $contents;
        }
               
       
/* Set EXIF data immediately */
       
if( static::exifSupported() )
        {
           
$obj->setExifData( $contents );
       
           
/* If the image is misoriented, attempt to automatically reorient */
           
$orientation = $obj->getImageOrientation();
           
           
/* Differences in orientation between GD and ImageMagick can cause auto-reorient to not work properly */
           
if ( !( $obj instanceof Image\Imagemagick ) )
            {
                switch (
$orientation )
                {
                    case
3:
                       
$obj->rotate( 180 );
                       
$obj->hasBeenRotated = TRUE;
                    break;

                    case
6:
                       
$obj->rotate( -90 );
                       
$obj->hasBeenRotated = TRUE;
                    break;
       
                    case
8:
                       
$obj->rotate( 90 );
                       
$obj->hasBeenRotated = TRUE;
                    break;
                }
            }
            else
            {
                switch(
$orientation )
                {
                    case
3:
                       
$obj->rotate( 180 );
                       
$obj->hasBeenRotated = TRUE;
                    break;
                   
                    case
6:
                       
$obj->rotate( 90 );
                       
$obj->hasBeenRotated = TRUE;
                    break;
                   
                    case
8:
                       
$obj->rotate( -90 );
                       
$obj->hasBeenRotated = TRUE;
                    break;
                }
            }
           
           
/* ImageMagick requires orientation be reset when rotated */
           
$obj->setImageOrientation( 1 );
        }
       
       
/* Return */
       
return $obj;
    }
   
   
/**
     * @brief    Type ('png', 'jpeg' or 'gif')
     */
   
public $type;
   
   
/**
     * @brief    Width
     */
   
public $width;
   
   
/**
     * @brief    Height
     */
   
public $height;
       
   
/**
     * @brief    Is this an animated gif?
     */
   
public $isAnimatedGif    = FALSE;
   
   
/**
     * @brief    Contents of the image file when animated gif
     */
   
public $contents            = NULL;

   
/**
     * @brief    EXIF data - has to be pulled and stored before GD manipulates image
     */
   
protected $exif                = array();
   
   
/**
     * Resize to maximum
     *
     * @param    int|NULL        $maxWidth        Max Width (in pixels) or NULL
     * @param    int|NULL        $maxHeight        Max Height (in pixels) or NULL
     * @param    bool            $retainRatio    If TRUE, the image will keep it's current width/height ratio rather than being squashed
     * @return    bool
     */
   
public function resizeToMax( $maxWidth=NULL, $maxHeight=NULL, $retainRatio=TRUE )
    {
       
/* If the image is smaller than max we can skip */
       
if( ( $maxWidth == NULL or $this->width < $maxWidth ) and ( $maxHeight == NULL or $this->height < $maxHeight ) )
        {
            return
TRUE;
        }

       
/* Work out the maximum width/height */
       
$width = ( $maxWidth !== NULL and $this->width > $maxWidth ) ? $maxWidth : $this->width;
       
$height = ( $maxHeight !== NULL and $this->height > $maxHeight ) ? $maxHeight : $this->height;
               
        if (
$width != $this->width or $height != $this->height )
        {
           
/* Adjust the width/height as necessary if we want to keep the ratio */
           
if ( $retainRatio === TRUE )
            {
                if ( (
$this->height - $height ) <= ( $this->width - $width ) )
                {
                   
$ratio = $this->height / $this->width;
                   
$height = $width * $ratio;
                }
                else
                {
                   
$ratio = $this->width / $this->height;
                   
$width = $height * $ratio;
                }
            }

           
/* And resize */
           
return $this->resize( $width, $height );
        }
   
        return
TRUE;
    }
   
   
/**
     * Add Watermark
     *
     * @param    \IPS\Image    $watermark    The watermark
     * @return    void
     */
   
public function watermark( \IPS\Image $watermark )
    {
       
/* If it's too big, resize the watermark */
       
$watermark->resizeToMax( $this->width, $this->height );

       
/* Impose */        
       
$this->impose( $watermark, $this->width - $watermark->width, $this->height - $watermark->height );
    }

   
/**
     * Parse file object to extract EXIF data
     *
     * @return    array
     * @throws    \LogicException
     */
   
public function parseExif()
    {
        if( !static::
exifSupported() )
        {
            throw new \
LogicException( 'NO_EXIF' );
        }

        if( !
in_array( $this->type, array( 'jpeg', 'jpg', 'jpe' ) ) )
        {
            return array();
        }

       
$result    = array();

       
/* Read the data and store in an array */
       
if( $values = $this->getExifData() )
        {
            foreach(
$values as $section => $data )
            {
                foreach(
$data as $name => $value )
                {
                   
$result[ $section . '.' . $name ]    = $value;
                }
            }
        }

       
/* Return the EXIF data */
       
return $result;
    }
   
   
/**
     * Get EXIF data, if possible
     *
     * @return    array
     */
   
public function getExifData()
    {
        return
$this->exif;
    }

   
/**
     * Get EXIF data, if possible
     *
     * @param    string    $image    Image contents
     * @return    void
     */
   
public function setExifData( $contents )
    {
        if( !
in_array( $this->type, array( 'jpeg', 'jpg', 'jpe' ) ) )
        {
            return;
        }

       
/* Exif requires a file on disk, so write it temporarily */
       
$temporary    = tempnam( \IPS\TEMP_DIRECTORY, 'exif' );
        \
file_put_contents( $temporary, $contents );

       
$result    = @exif_read_data( $temporary, NULL, TRUE );

       
/* Remove the temporary file */
       
if( @is_file( $temporary ) )
        {
            @
unlink( $temporary );
        }

       
$this->exif    = $result;
    }

   
/**
     * Create a new blank canvas image
     *
     * @param    int        $width    Width
     * @param    int        $height    Height
     * @param    array     $rgb    Color to use for bg
     * @return    \IPS\Image
     */
   
public static function newImageCanvas( $width, $height, $rgb )
    {
        if( \
IPS\Settings::i()->image_suite == 'imagemagick' and class_exists( 'Imagick', FALSE ) )
        {
            return
Image\Imagemagick::newImageCanvas( $width, $height, $rgb );
        }
        else
        {
            return
Image\Gd::newImageCanvas( $width, $height, $rgb );
        }
    }

   
/**
     * Write text on our image
     *
     * @param    string    $text    Text
     * @param    string    $font    Path to font to use
     * @param    int        $size    Size of text
     * @return    void
     */
   
abstract public function write( $text, $font, $size );

   
/**
     * Get Contents
     *
     * @return    string
     */
   
abstract public function __toString();
   
   
/**
     * Resize
     *
     * @param    int        $width            Width (in pixels)
     * @param    int        $height            Height (in pixels)
     * @return    void
     */
   
abstract public function resize( $width, $height );

   
/**
     * Crop to a given width and height (will attempt to downsize first)
     *
     * @param    int        $width            Width (in pixels)
     * @param    int        $height            Height (in pixels)
     * @return    void
     */
   
abstract public function crop( $width, $height );
   
   
/**
     * Crop at specific points
     *
     * @param    int        $point1X        x-point for top-left corner
     * @param    int        $point1Y        y-point for top-left corner
     * @param    int        $point2X        x-point for bottom-right corner
     * @param    int        $point2Y        y-point for bottom-right corner
     * @return    void
     */
   
abstract public function cropToPoints( $point1X, $point1Y, $point2X, $point2Y );
   
   
/**
     * Impose image
     *
     * @param    \IPS\Image    $image    Image to impose
     * @param    int            $x        Location to impose to, x axis
     * @param    int            $y        Location to impose to, y axis
     * @return    void
     */
   
abstract public function impose( $image, $x=0, $y=0 );

   
/**
     * Rotate image
     *
     * @param    int        $angle    Angle of rotation
     * @return    void
     */
   
abstract public function rotate( $angle );
   
   
/**
     * Get Image Orientation
     *
     * @return    int|NULL
     */
   
abstract public function getImageOrientation();
   
   
/**
     * Set Image Orientation
     *
     * @param    int        $orientation The orientation
     * @return    void
     */
   
abstract public function setImageOrientation( $orientation );

   
/**
     * Can we write text reliably on an image?
     *
     * @return    bool
     */
   
public static function canWriteText()
    {
       
/* Create object */
       
if( \IPS\Settings::i()->image_suite == 'imagemagick' and class_exists( 'Imagick', FALSE ) )
        {
            return
Image\Imagemagick::canWriteText();
        }
        else
        {
            return
Image\Gd::canWriteText();
        }
    }
}