2009-01-22 20:35:09 +09:00
< ? php
/**
2009-08-26 07:12:20 +09:00
* StatusNet , the distributed open - source microblogging tool
2009-01-22 20:35:09 +09:00
*
* Abstraction for an image file
*
* PHP version 5
*
* LICENCE : This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU Affero 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 Affero General Public License for more details .
*
* You should have received a copy of the GNU Affero General Public License
* along with this program . If not , see < http :// www . gnu . org / licenses />.
*
* @ category Image
2009-08-26 07:12:20 +09:00
* @ package StatusNet
2009-08-26 07:19:04 +09:00
* @ author Evan Prodromou < evan @ status . net >
* @ author Zach Copley < zach @ status . net >
2009-08-26 07:12:20 +09:00
* @ copyright 2008 - 2009 StatusNet , Inc .
2009-01-22 20:35:09 +09:00
* @ license http :// www . fsf . org / licensing / licenses / agpl - 3.0 . html GNU Affero General Public License version 3.0
2009-08-26 07:16:46 +09:00
* @ link http :// status . net /
2009-01-22 20:35:09 +09:00
*/
2014-04-22 03:39:28 +09:00
if ( ! defined ( 'GNUSOCIAL' )) { exit ( 1 ); }
2009-01-22 20:35:09 +09:00
/**
* A wrapper on uploaded files
*
* Makes it slightly easier to accept an image file from upload .
*
* @ category Image
2009-08-26 07:12:20 +09:00
* @ package StatusNet
2009-08-26 07:19:04 +09:00
* @ author Evan Prodromou < evan @ status . net >
* @ author Zach Copley < zach @ status . net >
2009-01-22 20:35:09 +09:00
* @ license http :// www . fsf . org / licensing / licenses / agpl - 3.0 . html GNU Affero General Public License version 3.0
2009-08-26 07:16:46 +09:00
* @ link http :// status . net /
2009-01-22 20:35:09 +09:00
*/
class ImageFile
{
2009-02-05 09:32:15 +09:00
var $id ;
var $filepath ;
2015-01-25 10:27:02 +09:00
var $filename ;
2009-02-05 09:32:15 +09:00
var $type ;
var $height ;
var $width ;
2014-05-12 21:33:41 +09:00
var $rotate = 0 ; // degrees to rotate for properly oriented image (extrapolated from EXIF etc.)
2015-01-26 06:45:25 +09:00
var $animated = null ; // Animated image? (has more than 1 frame). null means untested
2015-02-25 09:33:32 +09:00
var $mimetype = null ; // The _ImageFile_ mimetype, _not_ the originating File object
2009-01-22 20:35:09 +09:00
2015-03-04 21:12:42 +09:00
protected $fileRecord = null ;
2015-02-25 09:31:44 +09:00
function __construct ( $id , $filepath )
2009-01-22 20:35:09 +09:00
{
2009-02-05 09:32:15 +09:00
$this -> id = $id ;
2015-03-04 21:12:42 +09:00
if ( ! empty ( $this -> id )) {
2015-03-04 21:29:18 +09:00
$this -> fileRecord = new File ();
$this -> fileRecord -> id = $this -> id ;
2015-03-04 21:46:51 +09:00
if ( ! $this -> fileRecord -> find ( true )) {
2015-03-04 21:29:18 +09:00
// If we have set an ID, we need that ID to exist!
throw new NoResultException ( $this -> fileRecord );
2015-03-04 21:12:42 +09:00
}
}
2015-03-04 21:29:18 +09:00
// These do not have to be the same as fileRecord->filename for example,
// since we may have generated an image source file from something else!
2009-02-05 09:32:15 +09:00
$this -> filepath = $filepath ;
2015-01-25 10:27:02 +09:00
$this -> filename = basename ( $filepath );
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
$info = @ getimagesize ( $this -> filepath );
2010-03-12 10:12:32 +09:00
if ( ! (
( $info [ 2 ] == IMAGETYPE_GIF && function_exists ( 'imagecreatefromgif' )) ||
( $info [ 2 ] == IMAGETYPE_JPEG && function_exists ( 'imagecreatefromjpeg' )) ||
$info [ 2 ] == IMAGETYPE_BMP ||
( $info [ 2 ] == IMAGETYPE_WBMP && function_exists ( 'imagecreatefromwbmp' )) ||
( $info [ 2 ] == IMAGETYPE_XBM && function_exists ( 'imagecreatefromxbm' )) ||
( $info [ 2 ] == IMAGETYPE_PNG && function_exists ( 'imagecreatefrompng' )))) {
2011-01-29 08:33:13 +09:00
// TRANS: Exception thrown when trying to upload an unsupported image file format.
2014-04-17 02:14:26 +09:00
throw new UnsupportedMediaException ( _ ( 'Unsupported image format.' ), $this -> filepath );
2010-03-12 10:12:32 +09:00
}
2015-02-25 09:31:44 +09:00
$this -> width = $info [ 0 ];
$this -> height = $info [ 1 ];
$this -> type = $info [ 2 ];
2015-02-25 09:33:32 +09:00
$this -> mimetype = $info [ 'mime' ];
2014-05-12 21:33:41 +09:00
2016-04-18 22:04:03 +09:00
if ( $this -> type === IMAGETYPE_JPEG && function_exists ( 'exif_read_data' )) {
2014-05-12 21:45:23 +09:00
// Orientation value to rotate thumbnails properly
2016-03-21 10:56:47 +09:00
$exif = @ exif_read_data ( $this -> filepath );
2014-05-12 21:45:23 +09:00
if ( is_array ( $exif ) && isset ( $exif [ 'Orientation' ])) {
switch (( int ) $exif [ 'Orientation' ]) {
case 1 : // top is top
$this -> rotate = 0 ;
break ;
case 3 : // top is bottom
$this -> rotate = 180 ;
break ;
case 6 : // top is right
$this -> rotate = - 90 ;
break ;
case 8 : // top is left
$this -> rotate = 90 ;
break ;
}
// If we ever write this back, Orientation should be set to '1'
2014-05-12 21:33:41 +09:00
}
2015-01-26 06:45:25 +09:00
} elseif ( $this -> type === IMAGETYPE_GIF ) {
$this -> animated = $this -> isAnimatedGif ();
2014-05-12 21:33:41 +09:00
}
2014-06-05 19:59:38 +09:00
Event :: handle ( 'FillImageFileMetadata' , array ( $this ));
2009-01-22 20:35:09 +09:00
}
2014-04-22 19:09:24 +09:00
public static function fromFileObject ( File $file )
{
$imgPath = null ;
$media = common_get_mime_media ( $file -> mimetype );
if ( Event :: handle ( 'CreateFileImageThumbnailSource' , array ( $file , & $imgPath , $media ))) {
2015-10-02 05:14:49 +09:00
if ( empty ( $file -> filename ) && ! file_exists ( $imgPath )) {
2015-01-26 09:16:28 +09:00
throw new UnsupportedMediaException ( _ ( 'File without filename could not get a thumbnail source.' ));
}
2015-03-05 19:59:11 +09:00
// First some mimetype specific exceptions
switch ( $file -> mimetype ) {
case 'image/svg+xml' :
2016-05-04 18:34:50 +09:00
throw new UseFileAsThumbnailException ( $file );
2015-03-05 19:59:11 +09:00
}
// And we'll only consider it an image if it has such a media type
2014-04-22 19:09:24 +09:00
switch ( $media ) {
case 'image' :
$imgPath = $file -> getPath ();
break ;
default :
throw new UnsupportedMediaException ( _ ( 'Unsupported media format.' ), $file -> getPath ());
}
}
if ( ! file_exists ( $imgPath )) {
2016-07-15 20:19:16 +09:00
throw new FileNotFoundException ( $imgPath );
2014-04-22 19:09:24 +09:00
}
try {
2016-02-10 12:40:10 +09:00
$image = new ImageFile ( $file -> getID (), $imgPath );
2016-07-15 19:52:01 +09:00
} catch ( Exception $e ) {
2014-04-22 19:09:24 +09:00
// Avoid deleting the original
2016-03-08 06:33:34 +09:00
try {
2016-07-15 19:52:01 +09:00
if ( strlen ( $imgPath ) > 0 && $imgPath !== $file -> getPath ()) {
common_debug ( __METHOD__ . ': Deleting temporary file that was created as image file thumbnail source: ' . _ve ( $imgPath ));
2016-03-08 06:33:34 +09:00
@ unlink ( $imgPath );
}
} catch ( FileNotFoundException $e ) {
// File reported (via getPath) that the original file
// doesn't exist anyway, so it's safe to delete $imgPath
@ unlink ( $imgPath );
2014-04-22 19:09:24 +09:00
}
2017-08-23 01:48:04 +09:00
common_debug ( sprintf ( 'Exception %s caught when creating ImageFile for File id==%s and imgPath==%s: %s' , get_class ( $e ), _ve ( $file -> id ), _ve ( $imgPath ), _ve ( $e -> getMessage ())));
2016-07-21 07:23:27 +09:00
throw $e ;
2014-04-22 19:09:24 +09:00
}
return $image ;
}
2014-04-17 02:14:26 +09:00
public function getPath ()
{
if ( ! file_exists ( $this -> filepath )) {
2015-02-25 09:14:27 +09:00
throw new FileNotFoundException ( $this -> filepath );
2014-04-17 02:14:26 +09:00
}
return $this -> filepath ;
}
2009-01-22 20:35:09 +09:00
static function fromUpload ( $param = 'upload' )
{
switch ( $_FILES [ $param ][ 'error' ]) {
2009-02-10 02:06:06 +09:00
case UPLOAD_ERR_OK : // success, jump out
2009-01-22 20:35:09 +09:00
break ;
2014-04-17 02:14:26 +09:00
2009-02-10 02:06:06 +09:00
case UPLOAD_ERR_INI_SIZE :
case UPLOAD_ERR_FORM_SIZE :
2010-10-31 07:34:28 +09:00
// TRANS: Exception thrown when too large a file is uploaded.
// TRANS: %s is the maximum file size, for example "500b", "10kB" or "2MB".
2014-04-17 02:14:26 +09:00
throw new Exception ( sprintf ( _ ( 'That file is too big. The maximum file size is %s.' ), ImageFile :: maxFileSize ()));
2009-02-10 02:06:06 +09:00
case UPLOAD_ERR_PARTIAL :
2009-01-22 20:35:09 +09:00
@ unlink ( $_FILES [ $param ][ 'tmp_name' ]);
2011-01-29 08:33:13 +09:00
// TRANS: Exception thrown when uploading an image and that action could not be completed.
2009-01-22 20:35:09 +09:00
throw new Exception ( _ ( 'Partial upload.' ));
2014-04-17 02:14:26 +09:00
2009-11-03 02:08:26 +09:00
case UPLOAD_ERR_NO_FILE :
// No file; probably just a non-AJAX submission.
2015-07-17 02:21:12 +09:00
throw new ClientException ( _ ( 'No file uploaded.' ));
2009-02-10 02:06:06 +09:00
default :
2014-04-17 02:14:26 +09:00
common_log ( LOG_ERR , __METHOD__ . " : Unknown upload error " . $_FILES [ $param ][ 'error' ]);
2011-01-29 08:33:13 +09:00
// TRANS: Exception thrown when uploading an image fails for an unknown reason.
2009-01-22 20:35:09 +09:00
throw new Exception ( _ ( 'System error uploading file.' ));
}
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
$info = @ getimagesize ( $_FILES [ $param ][ 'tmp_name' ]);
2009-02-06 06:32:58 +09:00
2009-01-22 20:35:09 +09:00
if ( ! $info ) {
2009-02-05 09:32:15 +09:00
@ unlink ( $_FILES [ $param ][ 'tmp_name' ]);
2011-01-29 08:33:13 +09:00
// TRANS: Exception thrown when uploading a file as image that is not an image or is a corrupt file.
2014-04-17 02:14:26 +09:00
throw new UnsupportedMediaException ( _ ( 'Not an image or corrupt file.' ), '[deleted]' );
2009-01-22 20:35:09 +09:00
}
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
return new ImageFile ( null , $_FILES [ $param ][ 'tmp_name' ]);
}
2009-02-06 06:32:58 +09:00
2011-01-25 05:22:47 +09:00
/**
* Copy the image file to the given destination .
2014-06-17 18:50:42 +09:00
*
* This function may modify the resulting file . Please use the
* returned ImageFile object to read metadata ( width , height etc . )
2011-01-25 05:22:47 +09:00
*
* @ param string $outpath
2014-06-17 18:50:42 +09:00
* @ return ImageFile the image stored at target path
2011-01-25 05:22:47 +09:00
*/
function copyTo ( $outpath )
{
2014-06-17 18:50:42 +09:00
return new ImageFile ( null , $this -> resizeTo ( $outpath ));
2011-01-25 05:22:47 +09:00
}
2010-11-09 09:51:31 +09:00
/**
* Create and save a thumbnail image .
*
* @ param string $outpath
2014-06-05 19:59:38 +09:00
* @ param array $box width , height , boundary box ( x , y , w , h ) defaults to full image
2010-11-09 09:51:31 +09:00
* @ return string full local filesystem filename
*/
2014-06-05 19:59:38 +09:00
function resizeTo ( $outpath , array $box = array ())
2009-02-05 09:32:15 +09:00
{
2014-06-05 19:59:38 +09:00
$box [ 'width' ] = isset ( $box [ 'width' ]) ? intval ( $box [ 'width' ]) : $this -> width ;
$box [ 'height' ] = isset ( $box [ 'height' ]) ? intval ( $box [ 'height' ]) : $this -> height ;
$box [ 'x' ] = isset ( $box [ 'x' ]) ? intval ( $box [ 'x' ]) : 0 ;
$box [ 'y' ] = isset ( $box [ 'y' ]) ? intval ( $box [ 'y' ]) : 0 ;
$box [ 'w' ] = isset ( $box [ 'w' ]) ? intval ( $box [ 'w' ]) : $this -> width ;
$box [ 'h' ] = isset ( $box [ 'h' ]) ? intval ( $box [ 'h' ]) : $this -> height ;
2009-02-05 09:32:15 +09:00
if ( ! file_exists ( $this -> filepath )) {
2011-01-29 08:33:13 +09:00
// TRANS: Exception thrown during resize when image has been registered as present, but is no longer there.
2009-02-05 09:32:15 +09:00
throw new Exception ( _ ( 'Lost our file.' ));
}
2009-02-10 02:06:06 +09:00
2014-06-17 18:50:42 +09:00
// Don't rotate/crop/scale if it isn't necessary
2014-06-05 19:59:38 +09:00
if ( $box [ 'width' ] === $this -> width
2014-06-17 18:50:42 +09:00
&& $box [ 'height' ] === $this -> height
&& $box [ 'x' ] === 0
&& $box [ 'y' ] === 0
&& $box [ 'w' ] === $this -> width
&& $box [ 'h' ] === $this -> height
&& $this -> type == $this -> preferredType ()) {
if ( $this -> rotate == 0 ) {
// No rotational difference, just copy it as-is
@ copy ( $this -> filepath , $outpath );
return $outpath ;
} elseif ( abs ( $this -> rotate ) == 90 ) {
// Box is rotated 90 degrees in either direction,
// so we have to redefine x to y and vice versa.
$tmp = $box [ 'width' ];
$box [ 'width' ] = $box [ 'height' ];
$box [ 'height' ] = $tmp ;
$tmp = $box [ 'x' ];
$box [ 'x' ] = $box [ 'y' ];
$box [ 'y' ] = $tmp ;
$tmp = $box [ 'w' ];
$box [ 'w' ] = $box [ 'h' ];
$box [ 'h' ] = $tmp ;
}
2009-02-08 00:01:08 +09:00
}
2009-02-05 09:32:15 +09:00
2014-06-17 18:50:42 +09:00
2014-06-05 19:59:38 +09:00
if ( Event :: handle ( 'StartResizeImageFile' , array ( $this , $outpath , $box ))) {
$this -> resizeToFile ( $outpath , $box );
}
2014-06-17 18:54:05 +09:00
if ( ! file_exists ( $outpath )) {
2016-05-04 18:34:50 +09:00
if ( $this -> fileRecord instanceof File ) {
throw new UseFileAsThumbnailException ( $this -> fileRecord );
} else {
throw new UnsupportedMediaException ( 'No local File object exists for ImageFile.' );
}
2014-06-17 18:54:05 +09:00
}
2014-06-05 19:59:38 +09:00
return $outpath ;
}
protected function resizeToFile ( $outpath , array $box )
{
2009-02-05 09:32:15 +09:00
switch ( $this -> type ) {
2014-06-05 19:59:38 +09:00
case IMAGETYPE_GIF :
2009-02-05 09:32:15 +09:00
$image_src = imagecreatefromgif ( $this -> filepath );
2009-01-22 20:35:09 +09:00
break ;
2014-06-05 19:59:38 +09:00
case IMAGETYPE_JPEG :
2009-02-05 09:32:15 +09:00
$image_src = imagecreatefromjpeg ( $this -> filepath );
break ;
2014-06-05 19:59:38 +09:00
case IMAGETYPE_PNG :
2009-02-05 09:32:15 +09:00
$image_src = imagecreatefrompng ( $this -> filepath );
break ;
2014-06-05 19:59:38 +09:00
case IMAGETYPE_BMP :
2010-02-25 12:24:11 +09:00
$image_src = imagecreatefrombmp ( $this -> filepath );
break ;
2014-06-05 19:59:38 +09:00
case IMAGETYPE_WBMP :
2010-02-25 12:24:11 +09:00
$image_src = imagecreatefromwbmp ( $this -> filepath );
break ;
2014-06-05 19:59:38 +09:00
case IMAGETYPE_XBM :
2010-02-25 12:24:11 +09:00
$image_src = imagecreatefromxbm ( $this -> filepath );
break ;
2014-06-05 19:59:38 +09:00
default :
2011-01-29 08:33:13 +09:00
// TRANS: Exception thrown when trying to resize an unknown file type.
2009-02-05 09:32:15 +09:00
throw new Exception ( _ ( 'Unknown file type' ));
}
2014-05-12 21:33:41 +09:00
if ( $this -> rotate != 0 ) {
$image_src = imagerotate ( $image_src , $this -> rotate , 0 );
}
2014-06-05 19:59:38 +09:00
$image_dest = imagecreatetruecolor ( $box [ 'width' ], $box [ 'height' ]);
2009-02-06 06:32:58 +09:00
2010-02-25 12:24:11 +09:00
if ( $this -> type == IMAGETYPE_GIF || $this -> type == IMAGETYPE_PNG || $this -> type == IMAGETYPE_BMP ) {
2009-02-05 09:32:15 +09:00
$transparent_idx = imagecolortransparent ( $image_src );
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
if ( $transparent_idx >= 0 ) {
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
$transparent_color = imagecolorsforindex ( $image_src , $transparent_idx );
$transparent_idx = imagecolorallocate ( $image_dest , $transparent_color [ 'red' ], $transparent_color [ 'green' ], $transparent_color [ 'blue' ]);
imagefill ( $image_dest , 0 , 0 , $transparent_idx );
imagecolortransparent ( $image_dest , $transparent_idx );
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
} elseif ( $this -> type == IMAGETYPE_PNG ) {
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
imagealphablending ( $image_dest , false );
$transparent = imagecolorallocatealpha ( $image_dest , 0 , 0 , 0 , 127 );
imagefill ( $image_dest , 0 , 0 , $transparent );
imagesavealpha ( $image_dest , true );
2009-02-06 06:32:58 +09:00
2009-02-05 09:32:15 +09:00
}
}
2014-06-05 19:59:38 +09:00
imagecopyresampled ( $image_dest , $image_src , 0 , 0 , $box [ 'x' ], $box [ 'y' ], $box [ 'width' ], $box [ 'height' ], $box [ 'w' ], $box [ 'h' ]);
2009-02-05 09:32:15 +09:00
2014-06-05 19:59:38 +09:00
switch ( $this -> preferredType ()) {
2009-02-05 09:32:15 +09:00
case IMAGETYPE_GIF :
imagegif ( $image_dest , $outpath );
break ;
case IMAGETYPE_JPEG :
2013-10-01 18:37:59 +09:00
imagejpeg ( $image_dest , $outpath , common_config ( 'image' , 'jpegquality' ));
2009-02-05 09:32:15 +09:00
break ;
case IMAGETYPE_PNG :
imagepng ( $image_dest , $outpath );
break ;
default :
2011-01-29 08:33:13 +09:00
// TRANS: Exception thrown when trying resize an unknown file type.
2009-02-05 09:32:15 +09:00
throw new Exception ( _ ( 'Unknown file type' ));
2009-01-22 20:35:09 +09:00
}
2009-02-10 02:06:06 +09:00
2009-02-08 00:01:08 +09:00
imagedestroy ( $image_src );
imagedestroy ( $image_dest );
2010-11-09 09:51:31 +09:00
}
2014-06-05 19:59:38 +09:00
2010-11-09 09:51:31 +09:00
/**
* Several obscure file types should be normalized to PNG on resize .
*
2011-01-25 05:22:47 +09:00
* @ fixme consider flattening anything not GIF or JPEG to PNG
2010-11-09 09:51:31 +09:00
* @ return int
*/
2011-01-25 05:22:47 +09:00
function preferredType ()
2010-11-09 09:51:31 +09:00
{
2011-01-25 05:22:47 +09:00
if ( $this -> type == IMAGETYPE_BMP ) {
2010-11-09 09:51:31 +09:00
//we don't want to save BMP... it's an inefficient, rare, antiquated format
//save png instead
return IMAGETYPE_PNG ;
2011-01-25 05:22:47 +09:00
} else if ( $this -> type == IMAGETYPE_WBMP ) {
2010-11-09 09:51:31 +09:00
//we don't want to save WBMP... it's a rare format that we can't guarantee clients will support
//save png instead
return IMAGETYPE_PNG ;
2011-01-25 05:22:47 +09:00
} else if ( $this -> type == IMAGETYPE_XBM ) {
2010-11-09 09:51:31 +09:00
//we don't want to save XBM... it's a rare format that we can't guarantee clients will support
//save png instead
return IMAGETYPE_PNG ;
}
2011-01-25 05:22:47 +09:00
return $this -> type ;
2009-01-22 20:35:09 +09:00
}
function unlink ()
{
2014-04-17 03:08:21 +09:00
@ unlink ( $this -> filepath );
2009-01-22 20:35:09 +09:00
}
2009-02-06 06:32:58 +09:00
static function maxFileSize ()
2009-02-06 04:11:50 +09:00
{
2009-02-06 06:32:58 +09:00
$value = ImageFile :: maxFileSizeInt ();
if ( $value > 1024 * 1024 ) {
2010-10-31 07:34:28 +09:00
$value = $value / ( 1024 * 1024 );
// TRANS: Number of megabytes. %d is the number.
return sprintf ( _m ( '%dMB' , '%dMB' , $value ), $value );
2009-02-06 06:32:58 +09:00
} else if ( $value > 1024 ) {
2010-10-31 07:34:28 +09:00
$value = $value / 1024 ;
// TRANS: Number of kilobytes. %d is the number.
return sprintf ( _m ( '%dkB' , '%dkB' , $value ), $value );
2009-02-06 06:32:58 +09:00
} else {
2010-10-31 07:34:28 +09:00
// TRANS: Number of bytes. %d is the number.
return sprintf ( _m ( '%dB' , '%dB' , $value ), $value );
2009-02-06 05:01:44 +09:00
}
2009-02-06 04:11:50 +09:00
}
2009-02-06 06:32:58 +09:00
static function maxFileSizeInt ()
{
return min ( ImageFile :: strToInt ( ini_get ( 'post_max_size' )),
ImageFile :: strToInt ( ini_get ( 'upload_max_filesize' )),
ImageFile :: strToInt ( ini_get ( 'memory_limit' )));
}
2009-02-06 04:11:50 +09:00
static function strToInt ( $str )
{
$unit = substr ( $str , - 1 );
$num = substr ( $str , 0 , - 1 );
2009-02-06 06:32:58 +09:00
2009-02-06 04:11:50 +09:00
switch ( strtoupper ( $unit )){
2009-02-10 02:06:06 +09:00
case 'G' :
$num *= 1024 ;
case 'M' :
$num *= 1024 ;
case 'K' :
$num *= 1024 ;
2009-02-06 04:11:50 +09:00
}
2009-02-06 06:32:58 +09:00
2009-02-06 04:11:50 +09:00
return $num ;
}
2014-04-22 03:39:28 +09:00
public function scaleToFit ( $maxWidth = null , $maxHeight = null , $crop = null )
{
return self :: getScalingValues ( $this -> width , $this -> height ,
2014-05-12 21:33:41 +09:00
$maxWidth , $maxHeight , $crop , $this -> rotate );
2014-04-22 03:39:28 +09:00
}
/*
* Gets scaling values for images of various types . Cropping can be enabled .
*
* Values will scale _up_ to fit max values if cropping is enabled !
* With cropping disabled , the max value of each axis will be respected .
*
* @ param $width int Original width
* @ param $height int Original height
* @ param $maxW int Resulting max width
* @ param $maxH int Resulting max height
* @ param $crop int Crop to the size ( not preserving aspect ratio )
*/
public static function getScalingValues ( $width , $height ,
$maxW = null , $maxH = null ,
2014-05-12 21:33:41 +09:00
$crop = null , $rotate = 0 )
2014-04-22 03:39:28 +09:00
{
$maxW = $maxW ? : common_config ( 'thumbnail' , 'width' );
$maxH = $maxH ? : common_config ( 'thumbnail' , 'height' );
if ( $maxW < 1 || ( $maxH !== null && $maxH < 1 )) {
throw new ServerException ( 'Bad parameters for ImageFile::getScalingValues' );
} elseif ( $maxH === null ) {
// if maxH is null, we set maxH to equal maxW and enable crop
$maxH = $maxW ;
$crop = true ;
}
2014-05-12 21:33:41 +09:00
// Because GD doesn't understand EXIF orientation etc.
if ( abs ( $rotate ) == 90 ) {
$tmp = $width ;
$width = $height ;
$height = $tmp ;
}
2014-04-22 03:39:28 +09:00
// Cropping data (for original image size). Default values, 0 and null,
// imply no cropping and with preserved aspect ratio (per axis).
$cx = 0 ; // crop x
$cy = 0 ; // crop y
$cw = null ; // crop area width
$ch = null ; // crop area height
if ( $crop ) {
$s_ar = $width / $height ;
$t_ar = $maxW / $maxH ;
$rw = $maxW ;
$rh = $maxH ;
// Source aspect ratio differs from target, recalculate crop points!
if ( $s_ar > $t_ar ) {
$cx = floor ( $width / 2 - $height * $t_ar / 2 );
$cw = ceil ( $height * $t_ar );
} elseif ( $s_ar < $t_ar ) {
$cy = floor ( $height / 2 - $width / $t_ar / 2 );
$ch = ceil ( $width / $t_ar );
}
} else {
$rw = $maxW ;
$rh = ceil ( $height * $rw / $width );
// Scaling caused too large height, decrease to max accepted value
if ( $rh > $maxH ) {
$rh = $maxH ;
$rw = ceil ( $width * $rh / $height );
}
}
return array ( intval ( $rw ), intval ( $rh ),
intval ( $cx ), intval ( $cy ),
2014-04-22 05:31:13 +09:00
is_null ( $cw ) ? $width : intval ( $cw ),
is_null ( $ch ) ? $height : intval ( $ch ));
2014-04-22 03:39:28 +09:00
}
2015-01-26 06:45:25 +09:00
/**
* Animated GIF test , courtesy of frank at huddler dot com et al :
* http :// php . net / manual / en / function . imagecreatefromgif . php #104473
* Modified so avoid landing inside of a header ( and thus not matching our regexp ) .
*/
protected function isAnimatedGif ()
{
if ( ! ( $fh = @ fopen ( $this -> filepath , 'rb' ))) {
return false ;
}
$count = 0 ;
//an animated gif contains multiple "frames", with each frame having a
//header made up of:
// * a static 4-byte sequence (\x00\x21\xF9\x04)
// * 4 variable bytes
// * a static 2-byte sequence (\x00\x2C)
// In total the header is maximum 10 bytes.
// We read through the file til we reach the end of the file, or we've found
// at least 2 frame headers
while ( ! feof ( $fh ) && $count < 2 ) {
$chunk = fread ( $fh , 1024 * 100 ); //read 100kb at a time
$count += preg_match_all ( '#\x00\x21\xF9\x04.{4}\x00\x2C#s' , $chunk , $matches );
// rewind in case we ended up in the middle of the header, but avoid
// infinite loop (i.e. don't rewind if we're already in the end).
if ( ! feof ( $fh ) && ftell ( $fh ) >= 9 ) {
fseek ( $fh , - 9 , SEEK_CUR );
}
}
fclose ( $fh );
2016-05-04 18:57:55 +09:00
return $count >= 1 ; // number of animated frames apart from the original image
2015-01-26 06:45:25 +09:00
}
2015-03-04 21:12:42 +09:00
2016-02-10 12:40:10 +09:00
public function getFileThumbnail ( $width , $height , $crop , $upscale = false )
2015-03-04 21:12:42 +09:00
{
if ( ! $this -> fileRecord instanceof File ) {
throw new ServerException ( 'No File object attached to this ImageFile object.' );
}
if ( $width === null ) {
$width = common_config ( 'thumbnail' , 'width' );
$height = common_config ( 'thumbnail' , 'height' );
$crop = common_config ( 'thumbnail' , 'crop' );
}
2016-02-10 12:40:10 +09:00
if ( ! $upscale ) {
if ( $width > $this -> width ) {
$width = $this -> width ;
}
if ( ! is_null ( $height ) && $height > $this -> height ) {
$height = $this -> height ;
}
}
2015-03-04 21:12:42 +09:00
if ( $height === null ) {
$height = $width ;
$crop = true ;
}
// Get proper aspect ratio width and height before lookup
// We have to do it through an ImageFile object because of orientation etc.
// Only other solution would've been to rotate + rewrite uploaded files
// which we don't want to do because we like original, untouched data!
list ( $width , $height , $x , $y , $w , $h ) = $this -> scaleToFit ( $width , $height , $crop );
$thumb = File_thumbnail :: pkeyGet ( array (
2016-07-21 07:23:27 +09:00
'file_id' => $this -> fileRecord -> getID (),
2015-03-04 21:12:42 +09:00
'width' => $width ,
'height' => $height ,
));
if ( $thumb instanceof File_thumbnail ) {
return $thumb ;
}
$filename = $this -> fileRecord -> filehash ? : $this -> filename ; // Remote files don't have $this->filehash
$extension = File :: guessMimeExtension ( $this -> mimetype );
2016-07-21 07:23:27 +09:00
$outname = " thumb- { $this -> fileRecord -> getID () } - { $width } x { $height } - { $filename } . " . $extension ;
2015-03-04 21:12:42 +09:00
$outpath = File_thumbnail :: path ( $outname );
// The boundary box for our resizing
$box = array ( 'width' => $width , 'height' => $height ,
'x' => $x , 'y' => $y ,
'w' => $w , 'h' => $h );
// Doublecheck that parameters are sane and integers.
if ( $box [ 'width' ] < 1 || $box [ 'width' ] > common_config ( 'thumbnail' , 'maxsize' )
|| $box [ 'height' ] < 1 || $box [ 'height' ] > common_config ( 'thumbnail' , 'maxsize' )
|| $box [ 'w' ] < 1 || $box [ 'x' ] >= $this -> width
|| $box [ 'h' ] < 1 || $box [ 'y' ] >= $this -> height ) {
// Fail on bad width parameter. If this occurs, it's due to algorithm in ImageFile->scaleToFit
common_debug ( " Boundary box parameters for resize of { $this -> filepath } : " . var_export ( $box , true ));
throw new ServerException ( 'Bad thumbnail size parameters.' );
}
2016-07-21 07:23:27 +09:00
common_debug ( sprintf ( 'Generating a thumbnail of File id==%u of size %ux%u' , $this -> fileRecord -> getID (), $width , $height ));
2015-03-04 21:12:42 +09:00
// Perform resize and store into file
$this -> resizeTo ( $outpath , $box );
2016-03-08 06:33:34 +09:00
try {
// Avoid deleting the original
if ( ! in_array ( $this -> getPath (), [ File :: path ( $this -> filename ), File_thumbnail :: path ( $this -> filename )])) {
$this -> unlink ();
}
} catch ( FileNotFoundException $e ) {
// $this->getPath() says the file doesn't exist anyway, so no point in trying to delete it!
2015-03-04 21:12:42 +09:00
}
2016-03-08 06:33:34 +09:00
2016-02-10 12:40:10 +09:00
return File_thumbnail :: saveThumbnail ( $this -> fileRecord -> getID (),
null , // no url since we generated it ourselves and can dynamically generate the url
2015-03-04 21:12:42 +09:00
$width , $height ,
$outname );
}
2010-02-25 12:24:11 +09:00
}
//PHP doesn't (as of 2/24/2010) have an imagecreatefrombmp so conditionally define one
if ( ! function_exists ( 'imagecreatefrombmp' )){
//taken shamelessly from http://www.php.net/manual/en/function.imagecreatefromwbmp.php#86214
function imagecreatefrombmp ( $p_sFile )
{
// Load the image into a string
$file = fopen ( $p_sFile , " rb " );
$read = fread ( $file , 10 );
while ( ! feof ( $file ) && ( $read <> " " ))
$read .= fread ( $file , 1024 );
$temp = unpack ( " H* " , $read );
$hex = $temp [ 1 ];
$header = substr ( $hex , 0 , 108 );
// Process the header
// Structure: http://www.fastgraph.com/help/bmp_header_format.html
if ( substr ( $header , 0 , 4 ) == " 424d " )
{
// Cut it in parts of 2 bytes
$header_parts = str_split ( $header , 2 );
// Get the width 4 bytes
$width = hexdec ( $header_parts [ 19 ] . $header_parts [ 18 ]);
// Get the height 4 bytes
$height = hexdec ( $header_parts [ 23 ] . $header_parts [ 22 ]);
// Unset the header params
unset ( $header_parts );
}
// Define starting X and Y
$x = 0 ;
$y = 1 ;
// Create newimage
$image = imagecreatetruecolor ( $width , $height );
// Grab the body from the image
$body = substr ( $hex , 108 );
// Calculate if padding at the end-line is needed
// Divided by two to keep overview.
// 1 byte = 2 HEX-chars
$body_size = ( strlen ( $body ) / 2 );
$header_size = ( $width * $height );
// Use end-line padding? Only when needed
$usePadding = ( $body_size > ( $header_size * 3 ) + 4 );
// Using a for-loop with index-calculation instaid of str_split to avoid large memory consumption
// Calculate the next DWORD-position in the body
for ( $i = 0 ; $i < $body_size ; $i += 3 )
{
// Calculate line-ending and padding
if ( $x >= $width )
{
// If padding needed, ignore image-padding
// Shift i to the ending of the current 32-bit-block
if ( $usePadding )
$i += $width % 4 ;
// Reset horizontal position
$x = 0 ;
// Raise the height-position (bottom-up)
$y ++ ;
// Reached the image-height? Break the for-loop
if ( $y > $height )
break ;
}
// Calculation of the RGB-pixel (defined as BGR in image-data)
// Define $i_pos as absolute position in the body
$i_pos = $i * 2 ;
$r = hexdec ( $body [ $i_pos + 4 ] . $body [ $i_pos + 5 ]);
$g = hexdec ( $body [ $i_pos + 2 ] . $body [ $i_pos + 3 ]);
$b = hexdec ( $body [ $i_pos ] . $body [ $i_pos + 1 ]);
// Calculate and draw the pixel
$color = imagecolorallocate ( $image , $r , $g , $b );
imagesetpixel ( $image , $x , $height - $y , $color );
// Raise the horizontal position
$x ++ ;
}
// Unset the body / free the memory
unset ( $body );
// Return image-object
return $image ;
}
2014-04-22 03:39:28 +09:00
} // if(!function_exists('imagecreatefrombmp'))