<?php N2Loader::import('libraries.cache.image'); class N2Image extends N2CacheImage { public function __construct($group) { parent::__construct($group, true); } public static function resizeImage($group, $imageUrl, $targetWidth, $targetHeight, $lazy = false, $mode = 'cover', $backgroundColor = false, $resizeRemote = false, $quality = 100, $optimize = false, $x = 50, $y = 50) { if (strpos($imageUrl, N2Filesystem::getBasePath()) === 0) { $imageUrl = N2Uri::pathToUri($imageUrl); } if ($targetWidth <= 0 || $targetHeight <= 0 || !function_exists('imagecreatefrompng')) { return N2Filesystem::pathToAbsoluteURL($imageUrl); } $quality = max(0, min(100, $quality)); $originalImageUrl = $imageUrl; if (substr($imageUrl, 0, 2) == '//') { $imageUrl = parse_url(N2Uri::getFullUri(), PHP_URL_SCHEME) . ':' . $imageUrl; } if (strpos($imageUrl, N2Filesystem::getBasePath()) !== 0) { $imageUrl = N2Uri::relativetoabsolute($imageUrl); $imagePath = N2Filesystem::absoluteURLToPath($imageUrl); } else { $imagePath = $imageUrl; } $cache = new self($group); if ($lazy) { $cache->setLazy(true); } if ($imagePath == $imageUrl) { // The image is not local if (!$resizeRemote) { return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } $pathInfo = pathinfo(parse_url($imageUrl, PHP_URL_PATH)); $extension = self::validateExtension($pathInfo['extension']); if (!$extension) { return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } $resizedPath = $cache->makeCache($extension, array( $cache, '_resizeRemoteImage' ), array( $extension, $imageUrl, $targetWidth, $targetHeight, $mode, $backgroundColor, $quality, $optimize, $x, $y )); if ($resizedPath === $originalImageUrl) { return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } return N2Filesystem::pathToAbsoluteURL($resizedPath); } else { $extension = false; $imageType = @N2Image::exif_imagetype($imagePath); switch ($imageType) { case IMAGETYPE_JPEG: $extension = 'jpg'; break; case IMAGETYPE_PNG: $extension = 'png'; break; } if (!$extension) { return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } return N2Filesystem::pathToAbsoluteURL($cache->makeCache($extension, array( $cache, '_resizeImage' ), array( $extension, $imagePath, $targetWidth, $targetHeight, $mode, $backgroundColor, $quality, $optimize, $x, $y ))); } } protected function _resizeRemoteImage($targetFile, $extension, $imageUrl, $targetWidth, $targetHeight, $mode, $backgroundColor, $quality, $optimize, $x, $y) { return $this->_resizeImage($targetFile, $extension, $imageUrl, $targetWidth, $targetHeight, $mode, $backgroundColor, $quality, $optimize, $x, $y); } protected function _resizeImage($targetFile, $extension, $imagePath, $targetWidth, $targetHeight, $mode, $backgroundColor, $quality = 100, $optimize = false, $x = 50, $y = 50) { $targetDir = dirname($targetFile); if ($extension == 'png') { $image = @imagecreatefrompng($imagePath); } else if ($extension == 'jpg') { $image = @imagecreatefromjpeg($imagePath); if (function_exists("exif_read_data")) { $exif = @exif_read_data($imagePath); $rotated = $this->getOrientation($exif, $image); if ($rotated) { imagedestroy($image); $image = $rotated; } } } if (isset($image) && $image) { $originalWidth = imagesx($image); $originalHeight = imagesy($image); if ($optimize) { if ($originalWidth <= $targetWidth || $originalHeight <= $targetHeight) { if (!N2Filesystem::existsFolder($targetDir)) { N2Filesystem::createFolder($targetDir); } if ($extension == 'png') { imagepng($image, $targetFile); } else if ($extension == 'jpg') { imagejpeg($image, $targetFile, $quality); } imagedestroy($image); return true; } if ($originalWidth / $targetWidth > $originalHeight / $targetHeight) { $targetWidth = $originalWidth / ($originalHeight / $targetHeight); } else { $targetHeight = $originalHeight / ($originalWidth / $targetWidth); } } if ($rotated || $originalWidth != $targetWidth || $originalHeight != $targetHeight) { $newImage = imagecreatetruecolor($targetWidth, $targetHeight); if ($extension == 'png') { imagesavealpha($newImage, true); imagealphablending($newImage, false); $transparent = imagecolorallocatealpha($newImage, 255, 255, 255, 127); imagefilledrectangle($image, 0, 0, $targetWidth, $targetHeight, $transparent); } else if ($extension == 'jpg' && $backgroundColor) { $rgb = N2Color::hex2rgb($backgroundColor); $background = imagecolorallocate($newImage, $rgb[0], $rgb[1], $rgb[2]); imagefilledrectangle($newImage, 0, 0, $targetWidth, $targetHeight, $background); } list($dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h) = $this->imageMode($targetWidth, $targetHeight, $originalWidth, $originalHeight, $mode, $x, $y); imagecopyresampled($newImage, $image, $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h); imagedestroy($image); } else { $newImage = $image; } if (!N2Filesystem::existsFolder($targetDir)) { N2Filesystem::createFolder($targetDir); } if ($extension == 'png') { imagepng($newImage, $targetFile); } else if ($extension == 'jpg') { imagejpeg($newImage, $targetFile, $quality); } imagedestroy($newImage); return true; } throw new Exception('Unable to resize image: ' . $imagePath); } public static function scaleImage($group, $imageUrl, $scale = 1, $resizeRemote = false, $quality = 100) { $quality = max(0, min(100, $quality)); $originalImageUrl = $imageUrl; $imageUrl = N2ImageHelper::fixed($imageUrl); if ($scale > 0 && function_exists('imagecreatefrompng')) { if (substr($imageUrl, 0, 2) == '//') { $imageUrl = parse_url(N2Uri::getFullUri(), PHP_URL_SCHEME) . ':' . $imageUrl; } if (strpos($imageUrl, N2Filesystem::getBasePath()) !== 0) { $imageUrl = N2Uri::relativetoabsolute($imageUrl); $imagePath = N2Filesystem::absoluteURLToPath($imageUrl); } else { $imagePath = $imageUrl; } $cache = new self($group); if ($imagePath == $imageUrl) { // The image is not local if (!$resizeRemote) { return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } $pathInfo = pathinfo(parse_url($imageUrl, PHP_URL_PATH)); $extension = self::validateExtension($pathInfo['extension']); if (!$extension) { return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } return N2ImageHelper::dynamic(N2Filesystem::pathToAbsoluteURL($cache->makeCache($extension, array( $cache, '_scaleRemoteImage' ), array( $extension, $imageUrl, $scale, $quality )))); } else { $extension = false; $imageType = @N2Image::exif_imagetype($imagePath); switch ($imageType) { case IMAGETYPE_JPEG: $extension = 'jpg'; break; case IMAGETYPE_PNG: $extension = 'png'; $fp = fopen($imagePath, 'r'); fseek($fp, 25); $data = fgets($fp, 2); fclose($fp); if (ord($data) == 3) { // GD cannot resize palette PNG so we return the original image return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } break; } if (!$extension) { return N2Filesystem::pathToAbsoluteURL($originalImageUrl); } return N2ImageHelper::dynamic(N2Filesystem::pathToAbsoluteURL($cache->makeCache($extension, array( $cache, '_scaleImage' ), array( $extension, $imagePath, $scale, $quality )))); } } } protected function _scaleRemoteImage($targetFile, $extension, $imageUrl, $scale, $quality) { return $this->_scaleImage($targetFile, $extension, $imageUrl, $scale, $quality); } protected function _scaleImage($targetFile, $extension, $imagePath, $scale, $quality = 100) { $targetDir = dirname($targetFile); if ($extension == 'png') { $image = @imagecreatefrompng($imagePath); } else if ($extension == 'jpg') { $image = @imagecreatefromjpeg($imagePath); if (function_exists("exif_read_data")) { $exif = @exif_read_data($imagePath); $rotated = $this->getOrientation($exif, $image); if ($rotated) { imagedestroy($image); $image = $rotated; } } } if ($image) { $originalWidth = imagesx($image); $originalHeight = imagesy($image); $targetWidth = $originalWidth * $scale; $targetHeight = $originalHeight * $scale; if ((isset($rotated) && $rotated) || $originalWidth != $targetWidth || $originalHeight != $targetHeight) { $newImage = imagecreatetruecolor($targetWidth, $targetHeight); if ($extension == 'png') { imagesavealpha($newImage, true); imagealphablending($newImage, false); $transparent = imagecolorallocatealpha($newImage, 255, 255, 255, 127); imagefilledrectangle($image, 0, 0, $targetWidth, $targetHeight, $transparent); } list($dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h) = $this->imageMode($targetWidth, $targetHeight, $originalWidth, $originalHeight); imagecopyresampled($newImage, $image, $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h); imagedestroy($image); } else { $newImage = $image; } if (!N2Filesystem::existsFolder($targetDir)) { N2Filesystem::createFolder($targetDir); } if ($extension == 'png') { imagepng($newImage, $targetFile); } else if ($extension == 'jpg') { imagejpeg($newImage, $targetFile, $quality); } imagedestroy($newImage); return true; } throw new Exception('Unable to resize image: ' . $imagePath); } private function getOrientation($exif, $image) { if ($exif && !empty($exif['Orientation'])) { $rotated = false; switch ($exif['Orientation']) { case 3: $rotated = imagerotate($image, 180, 0); break; case 6: $rotated = imagerotate($image, -90, 0); break; case 8: $rotated = imagerotate($image, 90, 0); break; } return $rotated; } return false; } private function imageMode($width, $height, $originalWidth, $OriginalHeight, $mode = 'cover', $x = 50, $y = 50) { $dst_x = 0; $dst_y = 0; $src_x = 0; $src_y = 0; $dst_w = $width; $dst_h = $height; $src_w = $originalWidth; $src_h = $OriginalHeight; $horizontalRatio = $width / $originalWidth; $verticalRatio = $height / $OriginalHeight; if ($mode == 'cover') { if ($horizontalRatio > $verticalRatio) { $new_h = $horizontalRatio * $OriginalHeight; $dst_y = ($height - $new_h) / 2 * $y / 50; $dst_h = $new_h; } else { $new_w = $verticalRatio * $originalWidth; $dst_x = ($width - $new_w) / 2 * $x / 50; $dst_w = $new_w; } } else if ($mode == 'contain') { if ($horizontalRatio < $verticalRatio) { $new_h = $horizontalRatio * $OriginalHeight; $dst_y = ($height - $new_h) / 2 * $y / 50; $dst_h = $new_h; } else { $new_w = $verticalRatio * $originalWidth; $dst_x = ($width - $new_w) / 2 * $x / 50; $dst_w = $new_w; } } return array( $dst_x, $dst_y, $src_x, $src_y, $dst_w, $dst_h, $src_w, $src_h ); } private static function validateExtension($extension) { static $validExtensions = array( 'png' => 'png', 'jpg' => 'jpg', 'jpeg' => 'jpg', 'gif' => 'gif', 'webp' => 'webp', 'svg' => 'svg' ); $extension = strtolower($extension); if (isset($validExtensions[$extension])) { return $validExtensions[$extension]; } return false; } public static function base64Transparent() { return ''; } public static function base64($imagePath, $image) { $pathInfo = pathinfo(parse_url($imagePath, PHP_URL_PATH)); $extension = self::validateExtension($pathInfo['extension']); if ($extension) { return 'data:image/' . $extension . ';base64,' . n2_base64_encode(N2Filesystem::readFile($imagePath)); } return N2ImageHelper::fixed($image); } public static function exif_imagetype($filename) { if (!function_exists('exif_imagetype')) { if ((list($width, $height, $type, $attr) = getimagesize($filename)) !== false) { return $type; } return false; } return exif_imagetype($filename); } }