// +---------------------------------------------------------------------- // | ImageGd.class.php 2013-03-05 // +---------------------------------------------------------------------- namespace Think\Image\Driver; use Think\Image; class Gd { /** * 图像资源对象 * @var resource */ private $img; /** * 图像信息,包括width,height,type,mime,size * @var array */ private $info; /** * 构造方法,可用于打开一张图像 * @param string $imgname 图像路径 */ public function __construct($imgname = null) { $imgname && $this->open($imgname); } /** * 打开一张图像 * @param string $imgname 图像路径 */ public function open($imgname) { //检测图像文件 //当本地文件时才判断如下if语句,否则如果是http外网图片时不判断 if (substr($imgname, 0, 4) != 'http' && !is_file($imgname)) { E('不存在的图像文件'); } //获取图像信息 $info = getimagesize($imgname); //检测图像合法性 if (false === $info || (IMAGETYPE_GIF === $info[2] && empty($info['bits']))) { E('非法图像文件'); } //设置图像信息 $this->info = array( 'width' => $info[0], 'height' => $info[1], 'type' => image_type_to_extension($info[2], false), 'mime' => $info['mime'], ); //销毁已存在的图像 empty($this->img) || imagedestroy($this->img); //打开图像 if ('gif' == $this->info['type']) { $class = 'Think\\Image\\Driver\\GIF'; $this->gif = new $class($imgname); $this->img = imagecreatefromstring($this->gif->image()); } else { $fun = "imagecreatefrom{$this->info['type']}"; $this->img = $fun($imgname); } } /** * 保存图像 * @param string $imgname 图像保存名称 * @param string $type 图像类型 * @param integer $quality 图像质量 * @param boolean $interlace 是否对JPEG类型图像设置隔行扫描 */ public function save($imgname, $type = null, $quality = 80, $interlace = true) { if (empty($this->img)) { E('没有可以被保存的图像资源'); } //自动获取图像类型 if (is_null($type)) { $type = $this->info['type']; } else { $type = strtolower($type); } //保存图像 if ('jpeg' == $type || 'jpg' == $type) { //JPEG图像设置隔行扫描 imageinterlace($this->img, $interlace); imagejpeg($this->img, $imgname, $quality); } elseif ('gif' == $type && !empty($this->gif)) { $this->gif->save($imgname); } elseif ('png' == $type) { //设定保存完整的 alpha 通道信息 imagesavealpha($this->img, true); //ImagePNG生成图像的质量范围从0到9的 imagepng($this->img, $imgname, $quality / 10); } else { $fun = 'image' . $type; $fun($this->img, $imgname); } } /** * 返回图像宽度 * @return integer 图像宽度 */ public function width() { if (empty($this->img)) { E('没有指定图像资源'); } return $this->info['width']; } /** * 返回图像高度 * @return integer 图像高度 */ public function height() { if (empty($this->img)) { E('没有指定图像资源'); } return $this->info['height']; } /** * 返回图像类型 * @return string 图像类型 */ public function type() { if (empty($this->img)) { E('没有指定图像资源'); } return $this->info['type']; } /** * 返回图像MIME类型 * @return string 图像MIME类型 */ public function mime() { if (empty($this->img)) { E('没有指定图像资源'); } return $this->info['mime']; } /** * 返回图像尺寸数组 0 - 图像宽度,1 - 图像高度 * @return array 图像尺寸 */ public function size() { if (empty($this->img)) { E('没有指定图像资源'); } return array($this->info['width'], $this->info['height']); } /** * 裁剪图像 * @param integer $w 裁剪区域宽度 * @param integer $h 裁剪区域高度 * @param integer $x 裁剪区域x坐标 * @param integer $y 裁剪区域y坐标 * @param integer $width 图像保存宽度 * @param integer $height 图像保存高度 */ public function crop($w, $h, $x = 0, $y = 0, $width = null, $height = null) { if (empty($this->img)) { E('没有可以被裁剪的图像资源'); } //设置保存尺寸 empty($width) && $width = $w; empty($height) && $height = $h; do { //创建新图像 $img = imagecreatetruecolor($width, $height); // 调整默认颜色 $color = imagecolorallocate($img, 255, 255, 255); imagefill($img, 0, 0, $color); //取消默认的混色模式(优化原来生成的png图片为非透明的BUG) if ('png' == $this->info['type']) { imagealphablending($img, false); } //裁剪 imagecopyresampled($img, $this->img, 0, 0, $x, $y, $width, $height, $w, $h); imagedestroy($this->img); //销毁原图 //设置新图像 $this->img = $img; } while (!empty($this->gif) && $this->gifNext()); $this->info['width'] = $width; $this->info['height'] = $height; } /** * 生成缩略图 * @param integer $width 缩略图最大宽度 * @param integer $height 缩略图最大高度 * @param integer $type 缩略图裁剪类型 */ public function thumb($width, $height, $type = Image::IMAGE_THUMB_SCALE) { if (empty($this->img)) { E('没有可以被缩略的图像资源'); } //原图宽度和高度 $w = $this->info['width']; $h = $this->info['height']; /* 计算缩略图生成的必要参数 */ switch ($type) { /* 等比例缩放 */ case Image::IMAGE_THUMB_SCALE: //原图尺寸小于缩略图尺寸则不进行缩略 if ($w < $width && $h < $height) { return; } //计算缩放比例 $scale = min($width / $w, $height / $h); //设置缩略图的坐标及宽度和高度 $x = $y = 0; $width = $w * $scale; $height = $h * $scale; break; /* 居中裁剪 */ case Image::IMAGE_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 Image::IMAGE_THUMB_NORTHWEST: //计算缩放比例 $scale = max($width / $w, $height / $h); //设置缩略图的坐标及宽度和高度 $x = $y = 0; $w = $width / $scale; $h = $height / $scale; break; /* 右下角裁剪 */ case Image::IMAGE_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 Image::IMAGE_THUMB_FILLED: //计算缩放比例 if ($w < $width && $h < $height) { $scale = 1; } else { $scale = min($width / $w, $height / $h); } //设置缩略图的坐标及宽度和高度 $neww = $w * $scale; $newh = $h * $scale; $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->img, $posx, $posy, $x, $y, $neww, $newh, $w, $h); imagedestroy($this->img); //销毁原图 $this->img = $img; } while (!empty($this->gif) && $this->gifNext()); $this->info['width'] = $width; $this->info['height'] = $height; return; /* 固定 */ case Image::IMAGE_THUMB_FIXED: $x = $y = 0; break; default: E('不支持的缩略图裁剪类型'); } /* 裁剪图像 */ $this->crop($w, $h, $x, $y, $width, $height); } /** * 添加水印 * @param string $source 水印图片路径 * @param integer $locate 水印位置 * @param integer $alpha 水印透明度 */ public function water($source, $locate = Image::IMAGE_WATER_SOUTHEAST, $alpha = 80) { //资源检测 if (empty($this->img)) { E('没有可以被添加水印的图像资源'); } if (!is_file($source)) { E('水印图像不存在'); } //获取水印图像信息 $info = getimagesize($source); if (false === $info || (IMAGETYPE_GIF === $info[2] && empty($info['bits']))) { E('非法水印文件'); } //创建水印图像资源 $fun = 'imagecreatefrom' . image_type_to_extension($info[2], false); $water = $fun($source); //设定水印图像的混色模式 imagealphablending($water, true); /* 设定水印位置 */ switch ($locate) { /* 右下角水印 */ case Image::IMAGE_WATER_SOUTHEAST: $x = $this->info['width'] - $info[0]; $y = $this->info['height'] - $info[1]; break; /* 左下角水印 */ case Image::IMAGE_WATER_SOUTHWEST: $x = 0; $y = $this->info['height'] - $info[1]; break; /* 左上角水印 */ case Image::IMAGE_WATER_NORTHWEST: $x = $y = 0; break; /* 右上角水印 */ case Image::IMAGE_WATER_NORTHEAST: $x = $this->info['width'] - $info[0]; $y = 0; break; /* 居中水印 */ case Image::IMAGE_WATER_CENTER: $x = ($this->info['width'] - $info[0]) / 2; $y = ($this->info['height'] - $info[1]) / 2; break; /* 下居中水印 */ case Image::IMAGE_WATER_SOUTH: $x = ($this->info['width'] - $info[0]) / 2; $y = $this->info['height'] - $info[1]; break; /* 右居中水印 */ case Image::IMAGE_WATER_EAST: $x = $this->info['width'] - $info[0]; $y = ($this->info['height'] - $info[1]) / 2; break; /* 上居中水印 */ case Image::IMAGE_WATER_NORTH: $x = ($this->info['width'] - $info[0]) / 2; $y = 0; break; /* 左居中水印 */ case Image::IMAGE_WATER_WEST: $x = 0; $y = ($this->info['height'] - $info[1]) / 2; break; default: /* 自定义水印坐标 */ if (is_array($locate)) { list($x, $y) = $locate; } else { E('不支持的水印位置类型'); } } do { //添加水印 $src = imagecreatetruecolor($info[0], $info[1]); // 调整默认颜色 $color = imagecolorallocate($src, 255, 255, 255); imagefill($src, 0, 0, $color); imagecopy($src, $this->img, 0, 0, $x, $y, $info[0], $info[1]); imagecopy($src, $water, 0, 0, 0, 0, $info[0], $info[1]); imagecopymerge($this->img, $src, $x, $y, 0, 0, $info[0], $info[1], $alpha); //销毁零时图片资源 imagedestroy($src); } while (!empty($this->gif) && $this->gifNext()); //销毁水印资源 imagedestroy($water); } /** * 图像添加文字 * @param string $text 添加的文字 * @param string $font 字体路径 * @param integer $size 字号 * @param string $color 文字颜色 * @param integer $locate 文字写入位置 * @param integer $offset 文字相对当前位置的偏移量 * @param integer $angle 文字倾斜角度 */ public function text($text, $font, $size, $color = '#00000000', $locate = Image::IMAGE_WATER_SOUTHEAST, $offset = 0, $angle = 0) { //资源检测 if (empty($this->img)) { E('没有可以被写入文字的图像资源'); } if (!is_file($font)) { E("不存在的字体文件:{$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 Image::IMAGE_WATER_SOUTHEAST: $x += $this->info['width'] - $w; $y += $this->info['height'] - $h; break; /* 左下角文字 */ case Image::IMAGE_WATER_SOUTHWEST: $y += $this->info['height'] - $h; break; /* 左上角文字 */ case Image::IMAGE_WATER_NORTHWEST: // 起始坐标即为左上角坐标,无需调整 break; /* 右上角文字 */ case Image::IMAGE_WATER_NORTHEAST: $x += $this->info['width'] - $w; break; /* 居中文字 */ case Image::IMAGE_WATER_CENTER: $x += ($this->info['width'] - $w) / 2; $y += ($this->info['height'] - $h) / 2; break; /* 下居中文字 */ case Image::IMAGE_WATER_SOUTH: $x += ($this->info['width'] - $w) / 2; $y += $this->info['height'] - $h; break; /* 右居中文字 */ case Image::IMAGE_WATER_EAST: $x += $this->info['width'] - $w; $y += ($this->info['height'] - $h) / 2; break; /* 上居中文字 */ case Image::IMAGE_WATER_NORTH: $x += ($this->info['width'] - $w) / 2; break; /* 左居中文字 */ case Image::IMAGE_WATER_WEST: $y += ($this->info['height'] - $h) / 2; break; default: /* 自定义文字坐标 */ if (is_array($locate)) { list($posx, $posy) = $locate; $x += $posx; $y += $posy; } else { E('不支持的文字位置类型'); } } /* 设置偏移量 */ if (is_array($offset)) { $offset = array_map('intval', $offset); list($ox, $oy) = $offset; } else { $offset = intval($offset); $ox = $oy = $offset; } /* 设置颜色 */ 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)) { E('错误的颜色值'); } do { /* 写入文字 */ $col = imagecolorallocatealpha($this->img, $color[0], $color[1], $color[2], $color[3]); imagettftext($this->img, $size, $angle, $x + $ox, $y + $oy, $col, $font, $text); } while (!empty($this->gif) && $this->gifNext()); } /* 切换到GIF的下一帧并保存当前帧,内部使用 */ private function gifNext() { ob_start(); ob_implicit_flush(0); imagegif($this->img); $img = ob_get_clean(); $this->gif->image($img); $next = $this->gif->nextImage(); if ($next) { $this->img = imagecreatefromstring($next); return $next; } else { $this->img = imagecreatefromstring($this->gif->image()); return false; } } /** * 析构方法,用于销毁图像资源 */ public function __destruct() { empty($this->img) || imagedestroy($this->img); } }