643 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
			
		
		
	
	
			643 lines
		
	
	
		
			22 KiB
		
	
	
	
		
			PHP
		
	
	
		
			Executable File
		
	
	
| <?php
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | ThinkPHP [ WE CAN DO IT JUST THINK IT ]
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Copyright (c) 2006-2015 http://thinkphp.cn All rights reserved.
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
 | ||
| // +----------------------------------------------------------------------
 | ||
| // | Author: yunwuxin <448901948@qq.com>
 | ||
| // +----------------------------------------------------------------------
 | ||
| 
 | ||
| namespace think;
 | ||
| 
 | ||
| use think\image\Exception as ImageException;
 | ||
| use think\image\gif\Gif;
 | ||
| 
 | ||
| class Image
 | ||
| {
 | ||
| 
 | ||
|     /* 缩略图相关常量定义 */
 | ||
|     const THUMB_SCALING   = 1; //常量,标识缩略图等比例缩放类型
 | ||
|     const THUMB_FILLED    = 2; //常量,标识缩略图缩放后填充类型
 | ||
|     const THUMB_CENTER    = 3; //常量,标识缩略图居中裁剪类型
 | ||
|     const THUMB_NORTHWEST = 4; //常量,标识缩略图左上角裁剪类型
 | ||
|     const THUMB_SOUTHEAST = 5; //常量,标识缩略图右下角裁剪类型
 | ||
|     const THUMB_FIXED     = 6; //常量,标识缩略图固定尺寸缩放类型
 | ||
|     /* 水印相关常量定义 */
 | ||
|     const WATER_NORTHWEST = 1; //常量,标识左上角水印
 | ||
|     const WATER_NORTH     = 2; //常量,标识上居中水印
 | ||
|     const WATER_NORTHEAST = 3; //常量,标识右上角水印
 | ||
|     const WATER_WEST      = 4; //常量,标识左居中水印
 | ||
|     const WATER_CENTER    = 5; //常量,标识居中水印
 | ||
|     const WATER_EAST      = 6; //常量,标识右居中水印
 | ||
|     const WATER_SOUTHWEST = 7; //常量,标识左下角水印
 | ||
|     const WATER_SOUTH     = 8; //常量,标识下居中水印
 | ||
|     const WATER_SOUTHEAST = 9; //常量,标识右下角水印
 | ||
|     /* 翻转相关常量定义 */
 | ||
|     const FLIP_X = 1; //X轴翻转
 | ||
|     const FLIP_Y = 2; //Y轴翻转
 | ||
| 
 | ||
|     /**
 | ||
|      * 图像资源对象
 | ||
|      *
 | ||
|      * @var resource
 | ||
|      */
 | ||
|     protected $im;
 | ||
| 
 | ||
|     /** @var  Gif */
 | ||
|     protected $gif;
 | ||
| 
 | ||
|     /**
 | ||
|      * 图像信息,包括 width, height, type, mime, size
 | ||
|      *
 | ||
|      * @var array
 | ||
|      */
 | ||
|     protected $info;
 | ||
| 
 | ||
|     protected function __construct(\SplFileInfo $file)
 | ||
|     {
 | ||
|         //获取图像信息
 | ||
|         $info = @getimagesize($file->getPathname());
 | ||
| 
 | ||
|         //检测图像合法性
 | ||
|         if (false === $info || (IMAGETYPE_GIF === $info[2] && empty($info['bits']))) {
 | ||
|             throw new ImageException('Illegal image file');
 | ||
|         }
 | ||
| 
 | ||
|         //设置图像信息
 | ||
|         $this->info = [
 | ||
|             'width'  => $info[0],
 | ||
|             'height' => $info[1],
 | ||
|             'type'   => image_type_to_extension($info[2], false),
 | ||
|             'mime'   => $info['mime'],
 | ||
|         ];
 | ||
| 
 | ||
|         //打开图像
 | ||
|         if ('gif' == $this->info['type']) {
 | ||
|             $this->gif = new Gif($file->getPathname());
 | ||
|             $this->im  = @imagecreatefromstring($this->gif->image());
 | ||
|         } else {
 | ||
|             $fun      = "imagecreatefrom{$this->info['type']}";
 | ||
|             $this->im = @$fun($file->getPathname());
 | ||
|         }
 | ||
| 
 | ||
|         if (empty($this->im)) {
 | ||
|             throw new ImageException('Failed to create image resources!');
 | ||
|         }
 | ||
| 
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 打开一个图片文件
 | ||
|      * @param \SplFileInfo|string $file
 | ||
|      * @return Image
 | ||
|      */
 | ||
|     public static function open($file)
 | ||
|     {
 | ||
|         if (is_string($file)) {
 | ||
|             $file = new \SplFileInfo($file);
 | ||
|         }
 | ||
|         if (!$file->isFile()) {
 | ||
|             throw new ImageException('image file not exist');
 | ||
|         }
 | ||
|         return new self($file);
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 保存图像
 | ||
|      * @param string      $pathname  图像保存路径名称
 | ||
|      * @param null|string $type      图像类型
 | ||
|      * @param int         $quality   图像质量
 | ||
|      * @param bool        $interlace 是否对JPEG类型图像设置隔行扫描
 | ||
|      * @return $this
 | ||
|      */
 | ||
|     public function save($pathname, $type = null, $quality = 80, $interlace = true)
 | ||
|     {
 | ||
|         //自动获取图像类型
 | ||
|         if (is_null($type)) {
 | ||
|             $type = $this->info['type'];
 | ||
|         } else {
 | ||
|             $type = strtolower($type);
 | ||
|         }
 | ||
|         //保存图像
 | ||
|         if ('jpeg' == $type || 'jpg' == $type) {
 | ||
|             //JPEG图像设置隔行扫描
 | ||
|             imageinterlace($this->im, $interlace);
 | ||
|             imagejpeg($this->im, $pathname, $quality);
 | ||
|         } elseif ('gif' == $type && !empty($this->gif)) {
 | ||
|             $this->gif->save($pathname);
 | ||
|         } elseif ('png' == $type) {
 | ||
|             //设定保存完整的 alpha 通道信息
 | ||
|             imagesavealpha($this->im, true);
 | ||
|             //ImagePNG生成图像的质量范围从0到9的
 | ||
|             imagepng($this->im, $pathname, min((int) ($quality / 10), 9));
 | ||
|         } else {
 | ||
|             $fun = 'image' . $type;
 | ||
|             $fun($this->im, $pathname);
 | ||
|         }
 | ||
| 
 | ||
|         return $this;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 返回图像宽度
 | ||
|      * @return int 图像宽度
 | ||
|      */
 | ||
|     public function width()
 | ||
|     {
 | ||
|         return $this->info['width'];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 返回图像高度
 | ||
|      * @return int 图像高度
 | ||
|      */
 | ||
|     public function height()
 | ||
|     {
 | ||
|         return $this->info['height'];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 返回图像类型
 | ||
|      * @return string 图像类型
 | ||
|      */
 | ||
|     public function type()
 | ||
|     {
 | ||
|         return $this->info['type'];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 返回图像MIME类型
 | ||
|      * @return string 图像MIME类型
 | ||
|      */
 | ||
|     public function mime()
 | ||
|     {
 | ||
|         return $this->info['mime'];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 返回图像尺寸数组 0 - 图像宽度,1 - 图像高度
 | ||
|      * @return array 图像尺寸
 | ||
|      */
 | ||
|     public function size()
 | ||
|     {
 | ||
|         return [$this->info['width'], $this->info['height']];
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 旋转图像
 | ||
|      * @param int $degrees 顺时针旋转的度数
 | ||
|      * @return $this
 | ||
|      */
 | ||
|     public function rotate($degrees = 90)
 | ||
|     {
 | ||
|         do {
 | ||
|             $img = imagerotate($this->im, -$degrees, imagecolorallocatealpha($this->im, 0, 0, 0, 127));
 | ||
|             imagedestroy($this->im);
 | ||
|             $this->im = $img;
 | ||
|         } while (!empty($this->gif) && $this->gifNext());
 | ||
| 
 | ||
|         $this->info['width']  = imagesx($this->im);
 | ||
|         $this->info['height'] = imagesy($this->im);
 | ||
| 
 | ||
|         return $this;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 翻转图像
 | ||
|      * @param integer $direction 翻转轴,X或者Y
 | ||
|      * @return $this
 | ||
|      */
 | ||
|     public function flip($direction = self::FLIP_X)
 | ||
|     {
 | ||
|         //原图宽度和高度
 | ||
|         $w = $this->info['width'];
 | ||
|         $h = $this->info['height'];
 | ||
| 
 | ||
|         do {
 | ||
| 
 | ||
|             $img = imagecreatetruecolor($w, $h);
 | ||
| 
 | ||
|             switch ($direction) {
 | ||
|                 case self::FLIP_X:
 | ||
|                     for ($y = 0; $y < $h; $y++) {
 | ||
|                         imagecopy($img, $this->im, 0, $h - $y - 1, 0, $y, $w, 1);
 | ||
|                     }
 | ||
|                     break;
 | ||
|                 case self::FLIP_Y:
 | ||
|                     for ($x = 0; $x < $w; $x++) {
 | ||
|                         imagecopy($img, $this->im, $w - $x - 1, 0, $x, 0, 1, $h);
 | ||
|                     }
 | ||
|                     break;
 | ||
|                 default:
 | ||
|                     throw new ImageException('不支持的翻转类型');
 | ||
|             }
 | ||
| 
 | ||
|             imagedestroy($this->im);
 | ||
|             $this->im = $img;
 | ||
| 
 | ||
|         } while (!empty($this->gif) && $this->gifNext());
 | ||
| 
 | ||
|         return $this;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 裁剪图像
 | ||
|      *
 | ||
|      * @param  integer $w      裁剪区域宽度
 | ||
|      * @param  integer $h      裁剪区域高度
 | ||
|      * @param  integer $x      裁剪区域x坐标
 | ||
|      * @param  integer $y      裁剪区域y坐标
 | ||
|      * @param  integer $width  图像保存宽度
 | ||
|      * @param  integer $height 图像保存高度
 | ||
|      *
 | ||
|      * @return $this
 | ||
|      */
 | ||
|     public function crop($w, $h, $x = 0, $y = 0, $width = null, $height = null)
 | ||
|     {
 | ||
|         //设置保存尺寸
 | ||
|         empty($width) && $width   = $w;
 | ||
|         empty($height) && $height = $h;
 | ||
|         do {
 | ||
|             //创建新图像
 | ||
|             $img = imagecreatetruecolor($width, $height);
 | ||
|             // 调整默认颜色
 | ||
|             $color = imagecolorallocate($img, 255, 255, 255);
 | ||
|             imagefill($img, 0, 0, $color);
 | ||
|             //裁剪
 | ||
|             imagecopyresampled($img, $this->im, 0, 0, $x, $y, $width, $height, $w, $h);
 | ||
|             imagedestroy($this->im); //销毁原图
 | ||
|             //设置新图像
 | ||
|             $this->im = $img;
 | ||
|         } while (!empty($this->gif) && $this->gifNext());
 | ||
|         $this->info['width']  = (int) $width;
 | ||
|         $this->info['height'] = (int) $height;
 | ||
|         return $this;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 生成缩略图
 | ||
|      *
 | ||
|      * @param  integer $width  缩略图最大宽度
 | ||
|      * @param  integer $height 缩略图最大高度
 | ||
|      * @param int      $type   缩略图裁剪类型
 | ||
|      *
 | ||
|      * @return $this
 | ||
|      */
 | ||
|     public function thumb($width, $height, $type = self::THUMB_SCALING)
 | ||
|     {
 | ||
|         //原图宽度和高度
 | ||
|         $w = $this->info['width'];
 | ||
|         $h = $this->info['height'];
 | ||
|         /* 计算缩略图生成的必要参数 */
 | ||
|         switch ($type) {
 | ||
|             /* 等比例缩放 */
 | ||
|             case self::THUMB_SCALING:
 | ||
|                 //原图尺寸小于缩略图尺寸则不进行缩略
 | ||
|                 if ($w < $width && $h < $height) {
 | ||
|                     return $this;
 | ||
|                 }
 | ||
|                 //计算缩放比例
 | ||
|                 $scale = min($width / $w, $height / $h);
 | ||
|                 //设置缩略图的坐标及宽度和高度
 | ||
|                 $x      = $y      = 0;
 | ||
|                 $width  = $w * $scale;
 | ||
|                 $height = $h * $scale;
 | ||
|                 break;
 | ||
|             /* 居中裁剪 */
 | ||
|             case self::THUMB_CENTER:
 | ||
|                 //计算缩放比例
 | ||
|                 $scale = max($width / $w, $height / $h);
 | ||
|                 //设置缩略图的坐标及宽度和高度
 | ||
|                 $w = $width / $scale;
 | ||
|                 $h = $height / $scale;
 | ||
|                 $x = ($this->info['width'] - $w) / 2;
 | ||
|                 $y = ($this->info['height'] - $h) / 2;
 | ||
|                 break;
 | ||
|             /* 左上角裁剪 */
 | ||
|             case self::THUMB_NORTHWEST:
 | ||
|                 //计算缩放比例
 | ||
|                 $scale = max($width / $w, $height / $h);
 | ||
|                 //设置缩略图的坐标及宽度和高度
 | ||
|                 $x = $y = 0;
 | ||
|                 $w = $width / $scale;
 | ||
|                 $h = $height / $scale;
 | ||
|                 break;
 | ||
|             /* 右下角裁剪 */
 | ||
|             case self::THUMB_SOUTHEAST:
 | ||
|                 //计算缩放比例
 | ||
|                 $scale = max($width / $w, $height / $h);
 | ||
|                 //设置缩略图的坐标及宽度和高度
 | ||
|                 $w = $width / $scale;
 | ||
|                 $h = $height / $scale;
 | ||
|                 $x = $this->info['width'] - $w;
 | ||
|                 $y = $this->info['height'] - $h;
 | ||
|                 break;
 | ||
|             /* 填充 */
 | ||
|             case self::THUMB_FILLED:
 | ||
|                 //计算缩放比例
 | ||
|                 if ($w < $width && $h < $height) {
 | ||
|                     $scale = 1;
 | ||
|                 } else {
 | ||
|                     $scale = min($width / $w, $height / $h);
 | ||
|                 }
 | ||
|                 //设置缩略图的坐标及宽度和高度
 | ||
|                 $neww = $w * $scale;
 | ||
|                 $newh = $h * $scale;
 | ||
|                 $x    = $this->info['width'] - $w;
 | ||
|                 $y    = $this->info['height'] - $h;
 | ||
|                 $posx = ($width - $w * $scale) / 2;
 | ||
|                 $posy = ($height - $h * $scale) / 2;
 | ||
|                 do {
 | ||
|                     //创建新图像
 | ||
|                     $img = imagecreatetruecolor($width, $height);
 | ||
|                     // 调整默认颜色
 | ||
|                     $color = imagecolorallocate($img, 255, 255, 255);
 | ||
|                     imagefill($img, 0, 0, $color);
 | ||
|                     //裁剪
 | ||
|                     imagecopyresampled($img, $this->im, $posx, $posy, $x, $y, $neww, $newh, $w, $h);
 | ||
|                     imagedestroy($this->im); //销毁原图
 | ||
|                     $this->im = $img;
 | ||
|                 } while (!empty($this->gif) && $this->gifNext());
 | ||
|                 $this->info['width']  = (int) $width;
 | ||
|                 $this->info['height'] = (int) $height;
 | ||
|                 return $this;
 | ||
|             /* 固定 */
 | ||
|             case self::THUMB_FIXED:
 | ||
|                 $x = $y = 0;
 | ||
|                 break;
 | ||
|             default:
 | ||
|                 throw new ImageException('不支持的缩略图裁剪类型');
 | ||
|         }
 | ||
|         /* 裁剪图像 */
 | ||
|         return $this->crop($w, $h, $x, $y, $width, $height);
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 添加水印
 | ||
|      *
 | ||
|      * @param string    $source 水印图片路径
 | ||
|      * @param int|array $locate 水印位置
 | ||
|      * @param int       $alpha  透明度
 | ||
|      * @return $this
 | ||
|      */
 | ||
|     public function water($source, $locate = self::WATER_SOUTHEAST, $alpha = 100)
 | ||
|     {
 | ||
|         if (!is_file($source)) {
 | ||
|             throw new ImageException('水印图像不存在');
 | ||
|         }
 | ||
|         //获取水印图像信息
 | ||
|         $info = getimagesize($source);
 | ||
|         if (false === $info || (IMAGETYPE_GIF === $info[2] && empty($info['bits']))) {
 | ||
|             throw new ImageException('非法水印文件');
 | ||
|         }
 | ||
|         //创建水印图像资源
 | ||
|         $fun   = 'imagecreatefrom' . image_type_to_extension($info[2], false);
 | ||
|         $water = $fun($source);
 | ||
|         //设定水印图像的混色模式
 | ||
|         imagealphablending($water, true);
 | ||
|         /* 设定水印位置 */
 | ||
|         switch ($locate) {
 | ||
|             /* 右下角水印 */
 | ||
|             case self::WATER_SOUTHEAST:
 | ||
|                 $x = $this->info['width'] - $info[0];
 | ||
|                 $y = $this->info['height'] - $info[1];
 | ||
|                 break;
 | ||
|             /* 左下角水印 */
 | ||
|             case self::WATER_SOUTHWEST:
 | ||
|                 $x = 0;
 | ||
|                 $y = $this->info['height'] - $info[1];
 | ||
|                 break;
 | ||
|             /* 左上角水印 */
 | ||
|             case self::WATER_NORTHWEST:
 | ||
|                 $x = $y = 0;
 | ||
|                 break;
 | ||
|             /* 右上角水印 */
 | ||
|             case self::WATER_NORTHEAST:
 | ||
|                 $x = $this->info['width'] - $info[0];
 | ||
|                 $y = 0;
 | ||
|                 break;
 | ||
|             /* 居中水印 */
 | ||
|             case self::WATER_CENTER:
 | ||
|                 $x = ($this->info['width'] - $info[0]) / 2;
 | ||
|                 $y = ($this->info['height'] - $info[1]) / 2;
 | ||
|                 break;
 | ||
|             /* 下居中水印 */
 | ||
|             case self::WATER_SOUTH:
 | ||
|                 $x = ($this->info['width'] - $info[0]) / 2;
 | ||
|                 $y = $this->info['height'] - $info[1];
 | ||
|                 break;
 | ||
|             /* 右居中水印 */
 | ||
|             case self::WATER_EAST:
 | ||
|                 $x = $this->info['width'] - $info[0];
 | ||
|                 $y = ($this->info['height'] - $info[1]) / 2;
 | ||
|                 break;
 | ||
|             /* 上居中水印 */
 | ||
|             case self::WATER_NORTH:
 | ||
|                 $x = ($this->info['width'] - $info[0]) / 2;
 | ||
|                 $y = 0;
 | ||
|                 break;
 | ||
|             /* 左居中水印 */
 | ||
|             case self::WATER_WEST:
 | ||
|                 $x = 0;
 | ||
|                 $y = ($this->info['height'] - $info[1]) / 2;
 | ||
|                 break;
 | ||
|             default:
 | ||
|                 /* 自定义水印坐标 */
 | ||
|                 if (is_array($locate)) {
 | ||
|                     list($x, $y) = $locate;
 | ||
|                 } else {
 | ||
|                     throw new ImageException('不支持的水印位置类型');
 | ||
|                 }
 | ||
|         }
 | ||
|         do {
 | ||
|             //添加水印
 | ||
|             $src = imagecreatetruecolor($info[0], $info[1]);
 | ||
|             // 调整默认颜色
 | ||
|             $color = imagecolorallocate($src, 255, 255, 255);
 | ||
|             imagefill($src, 0, 0, $color);
 | ||
|             imagecopy($src, $this->im, 0, 0, $x, $y, $info[0], $info[1]);
 | ||
|             imagecopy($src, $water, 0, 0, 0, 0, $info[0], $info[1]);
 | ||
|             imagecopymerge($this->im, $src, $x, $y, 0, 0, $info[0], $info[1], $alpha);
 | ||
|             //销毁临时图片资源
 | ||
|             imagedestroy($src);
 | ||
|         } while (!empty($this->gif) && $this->gifNext());
 | ||
|         //销毁水印资源
 | ||
|         imagedestroy($water);
 | ||
|         return $this;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 图像添加文字
 | ||
|      *
 | ||
|      * @param string    $text   添加的文字
 | ||
|      * @param string    $font   字体路径
 | ||
|      * @param integer   $size   字号
 | ||
|      * @param string    $color  文字颜色
 | ||
|      * @param int|array $locate 文字写入位置
 | ||
|      * @param integer   $offset 文字相对当前位置的偏移量
 | ||
|      * @param integer   $angle  文字倾斜角度
 | ||
|      *
 | ||
|      * @return $this
 | ||
|      * @throws ImageException
 | ||
|      */
 | ||
|     public function text($text, $font, $size, $color = '#00000000',
 | ||
|         $locate = self::WATER_SOUTHEAST, $offset = 0, $angle = 0) {
 | ||
| 
 | ||
|         if (!is_file($font)) {
 | ||
|             throw new ImageException("不存在的字体文件:{$font}");
 | ||
|         }
 | ||
|         //获取文字信息
 | ||
|         $info = imagettfbbox($size, $angle, $font, $text);
 | ||
|         $minx = min($info[0], $info[2], $info[4], $info[6]);
 | ||
|         $maxx = max($info[0], $info[2], $info[4], $info[6]);
 | ||
|         $miny = min($info[1], $info[3], $info[5], $info[7]);
 | ||
|         $maxy = max($info[1], $info[3], $info[5], $info[7]);
 | ||
|         /* 计算文字初始坐标和尺寸 */
 | ||
|         $x = $minx;
 | ||
|         $y = abs($miny);
 | ||
|         $w = $maxx - $minx;
 | ||
|         $h = $maxy - $miny;
 | ||
|         /* 设定文字位置 */
 | ||
|         switch ($locate) {
 | ||
|             /* 右下角文字 */
 | ||
|             case self::WATER_SOUTHEAST:
 | ||
|                 $x += $this->info['width'] - $w;
 | ||
|                 $y += $this->info['height'] - $h;
 | ||
|                 break;
 | ||
|             /* 左下角文字 */
 | ||
|             case self::WATER_SOUTHWEST:
 | ||
|                 $y += $this->info['height'] - $h;
 | ||
|                 break;
 | ||
|             /* 左上角文字 */
 | ||
|             case self::WATER_NORTHWEST:
 | ||
|                 // 起始坐标即为左上角坐标,无需调整
 | ||
|                 break;
 | ||
|             /* 右上角文字 */
 | ||
|             case self::WATER_NORTHEAST:
 | ||
|                 $x += $this->info['width'] - $w;
 | ||
|                 break;
 | ||
|             /* 居中文字 */
 | ||
|             case self::WATER_CENTER:
 | ||
|                 $x += ($this->info['width'] - $w) / 2;
 | ||
|                 $y += ($this->info['height'] - $h) / 2;
 | ||
|                 break;
 | ||
|             /* 下居中文字 */
 | ||
|             case self::WATER_SOUTH:
 | ||
|                 $x += ($this->info['width'] - $w) / 2;
 | ||
|                 $y += $this->info['height'] - $h;
 | ||
|                 break;
 | ||
|             /* 右居中文字 */
 | ||
|             case self::WATER_EAST:
 | ||
|                 $x += $this->info['width'] - $w;
 | ||
|                 $y += ($this->info['height'] - $h) / 2;
 | ||
|                 break;
 | ||
|             /* 上居中文字 */
 | ||
|             case self::WATER_NORTH:
 | ||
|                 $x += ($this->info['width'] - $w) / 2;
 | ||
|                 break;
 | ||
|             /* 左居中文字 */
 | ||
|             case self::WATER_WEST:
 | ||
|                 $y += ($this->info['height'] - $h) / 2;
 | ||
|                 break;
 | ||
|             default:
 | ||
|                 /* 自定义文字坐标 */
 | ||
|                 if (is_array($locate)) {
 | ||
|                     list($posx, $posy) = $locate;
 | ||
|                     $x += $posx;
 | ||
|                     $y += $posy;
 | ||
|                 } else {
 | ||
|                     throw new ImageException('不支持的文字位置类型');
 | ||
|                 }
 | ||
|         }
 | ||
|         /* 设置偏移量 */
 | ||
|         if (is_array($offset)) {
 | ||
|             $offset        = array_map('intval', $offset);
 | ||
|             list($ox, $oy) = $offset;
 | ||
|         } else {
 | ||
|             $offset = intval($offset);
 | ||
|             $ox     = $oy     = $offset;
 | ||
|         }
 | ||
|         /* 图片黑白检测 */
 | ||
|         if ($color == "auto") {
 | ||
|             //X方向采集宽度:单英文字符占据宽度约为字体大小/1.6,单中文字符占据宽度约为字体大小*4/3;Y方向采集宽度:英文字符高度约为字体大小,中文会高一些。
 | ||
|             //使用保守宽度,以免在纯英文情况下采集区域超出图像范围,并且精度完全可以满足本功能。
 | ||
|             $pickX = intval(mb_strwidth($text) * ($size / 1.6));
 | ||
|             $pickY = $size;
 | ||
| 
 | ||
|             $brightness = 0;
 | ||
|             for ($i = $x + $ox; $i < $pickX + $x + $ox; $i++) { //根据文字基线确定要进行遍历的像素
 | ||
|                 for ($j = $y + $oy - $pickY; $j < $y + $oy; $j++) { //基线修正
 | ||
|                     $brightness += self::getBrightnessOfPixel($i, $j);
 | ||
|                 }
 | ||
|             }
 | ||
| 
 | ||
|             $color = $brightness / ($pickX * $pickY) > 127 ? '#00000000' : '#ffffffff';
 | ||
|         }
 | ||
|         /* 设置颜色 */
 | ||
|         if (is_string($color) && 0 === strpos($color, '#')) {
 | ||
|             $color = str_split(substr($color, 1), 2);
 | ||
|             $color = array_map('hexdec', $color);
 | ||
|             if (empty($color[3]) || $color[3] > 127) {
 | ||
|                 $color[3] = 0;
 | ||
|             }
 | ||
|         } elseif (!is_array($color)) {
 | ||
|             throw new ImageException('错误的颜色值');
 | ||
|         }
 | ||
|         do {
 | ||
|             /* 写入文字 */
 | ||
|             $col = imagecolorallocatealpha($this->im, $color[0], $color[1], $color[2], $color[3]);
 | ||
|             imagettftext($this->im, $size, $angle, $x + $ox, $y + $oy, $col, $font, $text);
 | ||
|         } while (!empty($this->gif) && $this->gifNext());
 | ||
|         return $this;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 获取图片指定像素点的亮度值
 | ||
|      */
 | ||
|     private function getBrightnessOfPixel($x, $y)
 | ||
|     {
 | ||
|         $rgb = imagecolorat($this->im, $x, $y);
 | ||
|         $r = ($rgb >> 16) & 0xFF;
 | ||
|         $g = ($rgb >> 8) & 0xFF;
 | ||
|         $b = $rgb & 0xFF;
 | ||
| 
 | ||
|         //红绿蓝能量不同,亮度不同,对应系数也不同(参考https://www.w3.org/TR/AERT/#color-contrast)
 | ||
|         $brightness = intval($r * 0.299 + $g * 0.587 + $b * 0.114);
 | ||
| 
 | ||
|         return $brightness;
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 切换到GIF的下一帧并保存当前帧
 | ||
|      */
 | ||
|     protected function gifNext()
 | ||
|     {
 | ||
|         ob_start();
 | ||
|         ob_implicit_flush(0);
 | ||
|         imagegif($this->im);
 | ||
|         $img = ob_get_clean();
 | ||
|         $this->gif->image($img);
 | ||
|         $next = $this->gif->nextImage();
 | ||
|         if ($next) {
 | ||
|             imagedestroy($this->im);
 | ||
|             $this->im = imagecreatefromstring($next);
 | ||
|             return $next;
 | ||
|         } else {
 | ||
|             imagedestroy($this->im);
 | ||
|             $this->im = imagecreatefromstring($this->gif->image());
 | ||
|             return false;
 | ||
|         }
 | ||
|     }
 | ||
| 
 | ||
|     /**
 | ||
|      * 析构方法,用于销毁图像资源
 | ||
|      */
 | ||
|     public function __destruct()
 | ||
|     {
 | ||
|         empty($this->im) || imagedestroy($this->im);
 | ||
|     }
 | ||
| 
 | ||
| }
 |