首頁 php教程 PHP源码 图片类 支持GD和Imagick,gif动画类型

图片类 支持GD和Imagick,gif动画类型

May 23, 2016 am 08:39 AM
php





1. [文件]     Image.class.php                                

<?php
namespace YBL;
class Image{
    /* 驱动相关常量定义 */
    const IMAGE_GD              =   0; //常量,标识GD库类型
    const IMAGE_IMAGICK         =   1; //常量,标识imagick库类型

    /* 缩略图相关常量定义 */
    const IMAGE_THUMB_SCALE     =   0 ; //常量,标识缩略图等比例缩放类型
    const IMAGE_THUMB_FILLED    =   1 ; //常量,标识缩略图缩放后填充类型
    const IMAGE_THUMB_CENTER    =   2 ; //常量,标识缩略图居中裁剪类型
    const IMAGE_THUMB_NORTHWEST =   3 ; //常量,标识缩略图左上角裁剪类型
    const IMAGE_THUMB_SOUTHEAST =   4 ; //常量,标识缩略图右下角裁剪类型
    const IMAGE_THUMB_FIXED     =   5 ; //常量,标识缩略图固定尺寸缩放类型

    /* 水印相关常量定义 */ 
	const IMAGE_WATER_CENTER    =   0 ; //常量,标识居中水印
    const IMAGE_WATER_NORTHWEST =   1 ; //常量,标识左上角水印
    const IMAGE_WATER_NORTH     =   2 ; //常量,标识上居中水印
    const IMAGE_WATER_NORTHEAST =   3 ; //常量,标识右上角水印
    const IMAGE_WATER_EAST      =   4 ; //常量,标识右居中水印
    const IMAGE_WATER_SOUTHEAST =   5 ; //常量,标识右下角水印	
    const IMAGE_WATER_SOUTH     =   6 ; //常量,标识下居中水印	
    const IMAGE_WATER_SOUTHWEST =   7 ; //常量,标识左下角水印
    const IMAGE_WATER_WEST      =   8 ; //常量,标识左居中水印


    /**
     * 图片资源
     * @var resource
     */
    private $img;

    /**
     * 构造方法,用于实例化一个图片处理对象
     * @param string $type 要使用的类库,默认使用GD库
     */
    public function __construct($gif_proess=false,$type = self::IMAGE_GD){
        /* 判断调用库的类型 */
        switch ($type) {
            case self::IMAGE_GD:
                $class = &#39;Gd&#39;;
                break;
            case self::IMAGE_IMAGICK:
                $class =&#39;Imagick&#39;;
                break;
            default:
                exit(&#39;不支持的图片处理库类型&#39;);
        }

        /* 引入处理库,实例化图片处理对象 */
		include_once($class.&#39;.class.php&#39;);
		$class = __NAMESPACE__.&#39;\\&#39;.$class;
        $this->img = new $class($gif_proess);

    }

    /**
     * 打开一幅图像
     * @param  string $imgname 图片路径
     * @return Object          当前图片处理库对象
     */
    public function open($imgname){
        $this->img->open($imgname);
        return $this;
    }

    /**
     * 保存图片
     * @param  string  $imgname   图片保存名称
     * @param  string  $type      图片类型
     * @param  integer $quality   图像质量      
     * @param  boolean $interlace 是否对JPEG类型图片设置隔行扫描
     * @return Object             当前图片处理库对象
     */
    public function save($imgname, $type = null, $quality=80,$interlace = true){
        $this->img->save($imgname, $type, $quality,$interlace);
        return $this;
    }

    /**
     * 返回图片宽度
     * @return integer 图片宽度
     */
    public function width(){
        return $this->img->width();
    }

    /**
     * 返回图片高度
     * @return integer 图片高度
     */
    public function height(){
        return $this->img->height();
    }

    /**
     * 返回图像类型
     * @return string 图片类型
     */
    public function type(){
        return $this->img->type();
    }

    /**
     * 返回图像MIME类型
     * @return string 图像MIME类型
     */
    public function mime(){
        return $this->img->mime();
    }

    /**
     * 返回图像尺寸数组 0 - 图片宽度,1 - 图片高度
     * @return array 图片尺寸
     */
    public function size(){
        return $this->img->size();
    }

    /**
     * 裁剪图片
     * @param  integer $w      裁剪区域宽度
     * @param  integer $h      裁剪区域高度
     * @param  integer $x      裁剪区域x坐标
     * @param  integer $y      裁剪区域y坐标
     * @param  integer $width  图片保存宽度
     * @param  integer $height 图片保存高度
     * @return Object          当前图片处理库对象
     */
    public function crop($w, $h, $x = 0, $y = 0, $width = null, $height = null){
        $this->img->crop($w, $h, $x, $y, $width, $height);
        return $this;
    }

    /**
     * 生成缩略图
     * @param  integer $width  缩略图最大宽度
     * @param  integer $height 缩略图最大高度
     * @param  integer $type   缩略图裁剪类型
     * @return Object          当前图片处理库对象
     */
    public function thumb($width, $height, $type = self::IMAGE_THUMB_SCALE){
        $this->img->thumb($width, $height, $type);
        return $this;
    }

    /**
     * 添加水印
     * @param  string  $source 水印图片路径
     * @param  integer $locate 水印位置
     * @param  integer $alpha  水印透明度
     * @return Object          当前图片处理库对象
     */
    public function water($source, $locate = self::IMAGE_WATER_SOUTHEAST,$alpha=80){
        $this->img->water($source, $locate,$alpha);
        return $this;
    }

    /**
     * 图像添加文字
     * @param  string  $text   添加的文字
     * @param  string  $font   字体路径
     * @param  integer $size   字号
     * @param  string  $color  文字颜色
     * @param  integer $locate 文字写入位置
     * @param  integer $offset 文字相对当前位置的偏移量
     * @param  integer $angle  文字倾斜角度
     * @return Object          当前图片处理库对象
     */
    public function text($text, $font, $size, $color = &#39;#00000000&#39;,$locate = self::IMAGE_WATER_SOUTHEAST, $offset = 0, $angle = 0){
        $this->img->text($text, $font, $size, $color, $locate, $offset, $angle);
        return $this;
    }
}
登入後複製

2. [文件] Gd.class.php

<?php
namespace YBL;
class Gd{
    /**
     * 图像资源对象
     * @var resource
     */
    private $img;
    /**
     * 图像信息,包括width,height,type,mime,size
     * @var array
     */
    private $info;
    public $error;
    /**
     * 构造方法,可用于打开一张图像
     * @param string $imgname 图像路径
     */
    private $gif_proess;       
    public function __construct($gif_proess=true) {
       $this->gif_proess=$gif_proess; 
    }
    /**
     * 打开一张图像
     * @param  string $imgname 图像路径
     */
    public function open($imgname){
        //检测图像文件
        if(!is_file($imgname)){
      $this->error=&#39;不存在的图像文件&#39;;
         return false;
      } 
        //获取图像信息
        $info = getimagesize($imgname);
        //检测图像合法性
        if(false === $info || (IMAGETYPE_GIF === $info[2] && empty($info[&#39;bits&#39;]))){
           $this->error=&#39;非法图像文件&#39;;
         return false;
        }
        //设置图像信息
        $this->info = array(
            &#39;width&#39;  => $info[0],
            &#39;height&#39; => $info[1],
            &#39;type&#39;   => image_type_to_extension($info[2], false),
            &#39;mime&#39;   => $info[&#39;mime&#39;],
        );
        //销毁已存在的图像
        empty($this->img) || imagedestroy($this->img);
        //打开图像
        if((&#39;gif&#39; == $this->info[&#39;type&#39;])&&$this->gif_proess){
            $this->gif = new GIF($imgname);
            $this->img = imagecreatefromstring($this->gif->image());
        } else {
            $fun = "imagecreatefrom{$this->info[&#39;type&#39;]}";
            $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)){
         $this->error=&#39;没有可以被保存的图像资源&#39;;
         return false;
      } 
        //自动获取图像类型
        if(is_null($type)){
            $type = $this->info[&#39;type&#39;];
        } else {
            $type = strtolower($type);
        }
        //保存图像
        if(&#39;jpeg&#39; == $type || &#39;jpg&#39; == $type){
            //JPEG图像设置隔行扫描
            imageinterlace($this->img, $interlace);
            imagejpeg($this->img, $imgname,$quality);
        }elseif((&#39;gif&#39; == $type)&& !empty($this->gif)){
            $this->gif->save($imgname);
        }else{
            $fun  =   &#39;image&#39;.$type;
            $fun($this->img, $imgname);
        }
    }
    /**
     * 返回图像宽度
     * @return integer 图像宽度
     */
    public function width(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      }
        return $this->info[&#39;width&#39;];
    }
    /**
     * 返回图像高度
     * @return integer 图像高度
     */
    public function height(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      }
        return $this->info[&#39;height&#39;];
    }
    /**
     * 返回图像类型
     * @return string 图像类型
     */
    public function type(){
        if(empty($this->img)){$this->error=&#39;没有指定图像资源&#39;;
         return false;}
        return $this->info[&#39;type&#39;];
    }
    /**
     * 返回图像MIME类型
     * @return string 图像MIME类型
     */
    public function mime(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      } 
        return $this->info[&#39;mime&#39;];
    }
    /**
     * 返回图像尺寸数组 0 - 图像宽度,1 - 图像高度
     * @return array 图像尺寸
     */
    public function size(){
        if(empty($this->img)) {
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      }
        return array($this->info[&#39;width&#39;], $this->info[&#39;height&#39;]);
    }
    /**
     * 裁剪图像
     * @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)){
         $this->error=&#39;没有可以被裁剪的图像资源&#39;;
         return false;
      }
        //设置保存尺寸
        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->img, 0, 0, $x, $y, $width, $height, $w, $h);
            imagedestroy($this->img); //销毁原图
            //设置新图像
            $this->img = $img;
        } while(!empty($this->gif)&&$this->gifNext());
        $this->info[&#39;width&#39;]  = $width;
        $this->info[&#39;height&#39;] = $height;
    }
    /**
     * 生成缩略图
     * @param  integer $width  缩略图最大宽度
     * @param  integer $height 缩略图最大高度
     * @param  integer $type   缩略图裁剪类型
     */
    public function thumb($width, $height, $type = Image::IMAGE_THUMB_SCALE){
        if(empty($this->img)){
         $this->error=&#39;没有可以被缩略的图像资源&#39;;
         return false;
      }
        //原图宽度和高度
        $w = $this->info[&#39;width&#39;];
        $h = $this->info[&#39;height&#39;];
        /* 计算缩略图生成的必要参数 */
        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[&#39;width&#39;] - $w)/2;
                $y = ($this->info[&#39;height&#39;] - $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[&#39;width&#39;] - $w;
                $y = $this->info[&#39;height&#39;] - $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[&#39;width&#39;]  = $width;
                $this->info[&#39;height&#39;] = $height;
                return;
            /* 固定 */
            case Image::IMAGE_THUMB_FIXED:
                $x = $y = 0;
                break;
            default:
               $this->error=&#39;不支持的缩略图裁剪类型&#39;;
         return false;
        }
        /* 裁剪图像 */
        $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)) {
           $this->error=&#39;没有可以被添加水印的图像资源&#39;;
         return false;
      }
        if(!is_file($source)) {
         $this->error=&#39;水印图像不存在&#39;;
         return false;
      }
        //获取水印图像信息
        $info = getimagesize($source);
        if(false === $info || (IMAGETYPE_GIF === $info[2] && empty($info[&#39;bits&#39;]))){
            $this->error=&#39;非法水印文件&#39;;
         return false;
        }
        //创建水印图像资源
        $fun   = &#39;imagecreatefrom&#39; . image_type_to_extension($info[2], false);
        $water = $fun($source);
        //设定水印图像的混色模式
        imagealphablending($water, true);
        /* 设定水印位置 */
        switch ($locate) {
            /* 右下角水印 */
            case Image::IMAGE_WATER_SOUTHEAST:
                $x = $this->info[&#39;width&#39;] - $info[0];
                $y = $this->info[&#39;height&#39;] - $info[1];
                break;
            /* 左下角水印 */
            case Image::IMAGE_WATER_SOUTHWEST:
                $x = 0;
                $y = $this->info[&#39;height&#39;] - $info[1];
                break;
            /* 左上角水印 */
            case Image::IMAGE_WATER_NORTHWEST:
                $x = $y = 0;
                break;
            /* 右上角水印 */
            case Image::IMAGE_WATER_NORTHEAST:
                $x = $this->info[&#39;width&#39;] - $info[0];
                $y = 0;
                break;
            /* 居中水印 */
            case Image::IMAGE_WATER_CENTER:
                $x = ($this->info[&#39;width&#39;] - $info[0])/2;
                $y = ($this->info[&#39;height&#39;] - $info[1])/2;
                break;
            /* 下居中水印 */
            case Image::IMAGE_WATER_SOUTH:
                $x = ($this->info[&#39;width&#39;] - $info[0])/2;
                $y = $this->info[&#39;height&#39;] - $info[1];
                break;
            /* 右居中水印 */
            case Image::IMAGE_WATER_EAST:
                $x = $this->info[&#39;width&#39;] - $info[0];
                $y = ($this->info[&#39;height&#39;] - $info[1])/2;
                break;
            /* 上居中水印 */
            case Image::IMAGE_WATER_NORTH:
                $x = ($this->info[&#39;width&#39;] - $info[0])/2;
                $y = 0;
                break;
            /* 左居中水印 */
            case Image::IMAGE_WATER_WEST:
                $x = 0;
                $y = ($this->info[&#39;height&#39;] - $info[1])/2;
                break;
            default:
                /* 自定义水印坐标 */
                if(is_array($locate)){
                    list($x, $y) = $locate;
                } else {$this->error=&#39;不支持的水印位置类型&#39;;
         return false;
               
                }
        }
        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 = &#39;#00000000&#39;, 
        $locate = Image::IMAGE_WATER_SOUTHEAST, $offset = 0, $angle = 0){
        //资源检测
        if(empty($this->img)){
         $this->error=&#39;没有可以被写入文字的图像资源&#39;;
         return false;
      } 
        if(!is_file($font)){
         $this->error="不存在的字体文件:{$font}";
         return false;
      }
        //获取文字信息
        $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[&#39;width&#39;]  - $w;
                $y += $this->info[&#39;height&#39;] - $h;
                break;
            /* 左下角文字 */
            case Image::IMAGE_WATER_SOUTHWEST:
                $y += $this->info[&#39;height&#39;] - $h;
                break;
            /* 左上角文字 */
            case Image::IMAGE_WATER_NORTHWEST:
                // 起始坐标即为左上角坐标,无需调整
                break;
            /* 右上角文字 */
            case Image::IMAGE_WATER_NORTHEAST:
                $x += $this->info[&#39;width&#39;] - $w;
                break;
            /* 居中文字 */
            case Image::IMAGE_WATER_CENTER:
                $x += ($this->info[&#39;width&#39;]  - $w)/2;
                $y += ($this->info[&#39;height&#39;] - $h)/2;
                break;
            /* 下居中文字 */
            case Image::IMAGE_WATER_SOUTH:
                $x += ($this->info[&#39;width&#39;] - $w)/2;
                $y += $this->info[&#39;height&#39;] - $h;
                break;
            /* 右居中文字 */
            case Image::IMAGE_WATER_EAST:
                $x += $this->info[&#39;width&#39;] - $w;
                $y += ($this->info[&#39;height&#39;] - $h)/2;
                break;
            /* 上居中文字 */
            case Image::IMAGE_WATER_NORTH:
                $x += ($this->info[&#39;width&#39;] - $w)/2;
                break;
            /* 左居中文字 */
            case Image::IMAGE_WATER_WEST:
                $y += ($this->info[&#39;height&#39;] - $h)/2;
                break;
            default:
                /* 自定义文字坐标 */
                if(is_array($locate)){
                    list($posx, $posy) = $locate;
                    $x += $posx;
                    $y += $posy;
                } else {$this->error=&#39;不支持的文字位置类型&#39;;
         return false;
                }
        }
        /* 设置偏移量 */
        if(is_array($offset)){
            $offset = array_map(&#39;intval&#39;, $offset);
            list($ox, $oy) = $offset;
        } else{
            $offset = intval($offset);
            $ox = $oy = $offset;
        }
        /* 设置颜色 */
        if(is_string($color) && 0 === strpos($color, &#39;#&#39;)){
            $color = str_split(substr($color, 1), 2);
            $color = array_map(&#39;hexdec&#39;, $color);
            if(empty($color[3]) || $color[3] > 127){
                $color[3] = 0;
            }
        } elseif (!is_array($color)) {
            $this->error=&#39;错误的颜色值&#39;;
         return false;
        }
        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);
    }
}
class GIF{
   /**
    * GIF帧列表
    * @var array
    */
   private $frames = array();
   /**
    * 每帧等待时间列表
    * @var array
    */
   private $delays = array();
   /**
    * 构造方法,用于解码GIF图片
    * @param string $src GIF图片数据
    * @param string $mod 图片数据类型
    */
   public function __construct($src = null, $mod = &#39;url&#39;) {
      if(!is_null($src)){
         if(&#39;url&#39; == $mod && is_file($src)){
            $src = file_get_contents($src);
         }
         
         /* 解码GIF图片 */
         try{
            $de = new GIFDecoder($src);
            $this->frames = $de->GIFGetFrames();
            $this->delays = $de->GIFGetDelays();
         } catch(Exception $e){
            $this->error=&#39;解码GIF图片出错&#39;;
         return false;
         }
      }
   }
   /**
    * 设置或获取当前帧的数据
    * @param  string $stream 二进制数据流
    * @return boolean        获取到的数据
    */
   public function image($stream = null){
      if(is_null($stream)){
         $current = current($this->frames);
         return false === $current ? reset($this->frames) : $current;
      } else {
         $this->frames[key($this->frames)] = $stream;
      }
   }
   /**
    * 将当前帧移动到下一帧
    * @return string 当前帧数据
    */
   public function nextImage(){
      return next($this->frames);
   }
   /**
    * 编码并保存当前GIF图片
    * @param  string $gifname 图片名称
    */
   public function save($gifname){
      $gif = new GIFEncoder($this->frames, $this->delays, 0, 2, 0, 0, 0, &#39;bin&#39;);
      file_put_contents($gifname, $gif->GetAnimation());
   }
}
/*
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
::
:: GIFEncoder Version 2.0 by László Zsidi, http://gifs.hu
::
:: This class is a rewritten &#39;GifMerge.class.php&#39; version.
::
::  Modification:
::   - Simplified and easy code,
::   - Ultra fast encoding,
::   - Built-in errors,
::   - Stable working
::
::
:: Updated at 2007. 02. 13. &#39;00.05.AM&#39;
::
::
::
::  Try on-line GIFBuilder Form demo based on GIFEncoder.
::
::  http://gifs.hu/phpclasses/demos/GifBuilder/
::
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
*/
Class GIFEncoder {
   private $GIF = "GIF89a";      /* GIF header 6 bytes  */
   private $VER = "GIFEncoder V2.05"; /* Encoder version    */
   private $BUF = Array ( );
   private $LOP =  0;
   private $DIS =  2;
   private $COL = -1;
   private $IMG = -1;
   private $ERR = Array (
      &#39;ERR00&#39;    => "Does not supported function for only one image!",
      &#39;ERR01&#39;    => "Source is not a GIF image!",
      &#39;ERR02&#39;    => "Unintelligible flag ",
      &#39;ERR03&#39;    => "Does not make animation from animated GIF source",
   );
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFEncoder...
   ::
   */
   public function __construct($GIF_src, $GIF_dly, $GIF_lop, $GIF_dis,$GIF_red, $GIF_grn, $GIF_blu, $GIF_mod) {
      if ( ! is_array ( $GIF_src ) && ! is_array ( $GIF_dly ) ) {
         printf ( "%s: %s", $this->VER, $this->ERR [ &#39;ERR00&#39; ] );
         exit   ( 0 );
      }
      $this->LOP = ( $GIF_lop > -1 ) ? $GIF_lop : 0;
      $this->DIS = ( $GIF_dis > -1 ) ? ( ( $GIF_dis < 3 ) ? $GIF_dis : 3 ) : 2;
      $this->COL = ( $GIF_red > -1 && $GIF_grn > -1 && $GIF_blu > -1 ) ?
                  ( $GIF_red | ( $GIF_grn << 8 ) | ( $GIF_blu << 16 ) ) : -1;
      for ( $i = 0; $i < count ( $GIF_src ); $i++ ) {
         if ( strToLower ( $GIF_mod ) == "url" ) {
            $this->BUF [ ] = fread ( fopen ( $GIF_src [ $i ], "rb" ), filesize ( $GIF_src [ $i ] ) );
         }
         else if ( strToLower ( $GIF_mod ) == "bin" ) {
            $this->BUF [ ] = $GIF_src [ $i ];
         }
         else {
            printf ( "%s: %s ( %s )!", $this->VER, $this->ERR [ &#39;ERR02&#39; ], $GIF_mod );
            exit   ( 0 );
         }
         if ( substr ( $this->BUF [ $i ], 0, 6 ) != "GIF87a" && substr ( $this->BUF [ $i ], 0, 6 ) != "GIF89a" ) {
            printf ( "%s: %d %s", $this->VER, $i, $this->ERR [ &#39;ERR01&#39; ] );
            exit   ( 0 );
         }
         for ( $j = ( 13 + 3 * ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) ) ), $k = TRUE; $k; $j++ ) {
            switch ( $this->BUF [ $i ] { $j } ) {
               case "!":
                  if ( ( substr ( $this->BUF [ $i ], ( $j + 3 ), 8 ) ) == "NETSCAPE" ) {
                     printf ( "%s: %s ( %s source )!", $this->VER, $this->ERR [ &#39;ERR03&#39; ], ( $i + 1 ) );
                     exit   ( 0 );
                  }
                  break;
               case ";":
                  $k = FALSE;
                  break;
            }
         }
      }
      $this->GIFAddHeader ( );
      for ( $i = 0; $i < count ( $this->BUF ); $i++ ) {
         $this->GIFAddFrames ( $i, $GIF_dly [ $i ] );
      }
      $this->GIFAddFooter ( );
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFAddHeader...
   ::
   */
   private function GIFAddHeader ( ) {
      $cmap = 0;
      if ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x80 ) {
         $cmap = 3 * ( 2 << ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x07 ) );
         $this->GIF .= substr ( $this->BUF [ 0 ], 6, 7     );
         $this->GIF .= substr ( $this->BUF [ 0 ], 13, $cmap );
         $this->GIF .= "!\377\13NETSCAPE2.0\3\1" . $this->GIFWord ( $this->LOP ) . "\0";
      }
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFAddFrames...
   ::
   */
   private function GIFAddFrames ( $i, $d ) {
      $Locals_str = 13 + 3 * ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) );
      $Locals_end = strlen ( $this->BUF [ $i ] ) - $Locals_str - 1;
      $Locals_tmp = substr ( $this->BUF [ $i ], $Locals_str, $Locals_end );
      $Global_len = 2 << ( ord ( $this->BUF [ 0  ] { 10 } ) & 0x07 );
      $Locals_len = 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 );
      $Global_rgb = substr ( $this->BUF [ 0  ], 13,
                     3 * ( 2 << ( ord ( $this->BUF [ 0  ] { 10 } ) & 0x07 ) ) );
      $Locals_rgb = substr ( $this->BUF [ $i ], 13,
                     3 * ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) ) );
      $Locals_ext = "!\xF9\x04" . chr ( ( $this->DIS << 2 ) + 0 ) .
                  chr ( ( $d >> 0 ) & 0xFF ) . chr ( ( $d >> 8 ) & 0xFF ) . "\x0\x0";
      if ( $this->COL > -1 && ord ( $this->BUF [ $i ] { 10 } ) & 0x80 ) {
         for ( $j = 0; $j < ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) ); $j++ ) {
            if (
                  ord ( $Locals_rgb { 3 * $j + 0 } ) == ( ( $this->COL >> 16 ) & 0xFF ) &&
                  ord ( $Locals_rgb { 3 * $j + 1 } ) == ( ( $this->COL >>  8 ) & 0xFF ) &&
                  ord ( $Locals_rgb { 3 * $j + 2 } ) == ( ( $this->COL >>  0 ) & 0xFF )
               ) {
               $Locals_ext = "!\xF9\x04" . chr ( ( $this->DIS << 2 ) + 1 ) .
                           chr ( ( $d >> 0 ) & 0xFF ) . chr ( ( $d >> 8 ) & 0xFF ) . chr ( $j ) . "\x0";
               break;
            }
         }
      }
      switch ( $Locals_tmp { 0 } ) {
         case "!":
            $Locals_img = substr ( $Locals_tmp, 8, 10 );
            $Locals_tmp = substr ( $Locals_tmp, 18, strlen ( $Locals_tmp ) - 18 );
            break;
         case ",":
            $Locals_img = substr ( $Locals_tmp, 0, 10 );
            $Locals_tmp = substr ( $Locals_tmp, 10, strlen ( $Locals_tmp ) - 10 );
            break;
      }
      if ( ord ( $this->BUF [ $i ] { 10 } ) & 0x80 && $this->IMG > -1 ) {
         if ( $Global_len == $Locals_len ) {
            if ( $this->GIFBlockCompare ( $Global_rgb, $Locals_rgb, $Global_len ) ) {
               $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_tmp );
            }
            else {
               $byte  = ord ( $Locals_img { 9 } );
               $byte |= 0x80;
               $byte &= 0xF8;
               $byte |= ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x07 );
               $Locals_img { 9 } = chr ( $byte );
               $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_rgb . $Locals_tmp );
            }
         }
         else {
            $byte  = ord ( $Locals_img { 9 } );
            $byte |= 0x80;
            $byte &= 0xF8;
            $byte |= ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 );
            $Locals_img { 9 } = chr ( $byte );
            $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_rgb . $Locals_tmp );
         }
      }
      else {
         $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_tmp );
      }
      $this->IMG  = 1;
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFAddFooter...
   ::
   */
   private function GIFAddFooter ( ) {
      $this->GIF .= ";";
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFBlockCompare...
   ::
   */
   private function GIFBlockCompare ( $GlobalBlock, $LocalBlock, $Len ) {
      for ( $i = 0; $i < $Len; $i++ ) {
         if (
               $GlobalBlock { 3 * $i + 0 } != $LocalBlock { 3 * $i + 0 } ||
               $GlobalBlock { 3 * $i + 1 } != $LocalBlock { 3 * $i + 1 } ||
               $GlobalBlock { 3 * $i + 2 } != $LocalBlock { 3 * $i + 2 }
            ) {
               return ( 0 );
         }
      }
      return ( 1 );
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFWord...
   ::
   */
   private function GIFWord ( $int ) {
      return ( chr ( $int & 0xFF ) . chr ( ( $int >> 8 ) & 0xFF ) );
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GetAnimation...
   ::
   */
   public function GetAnimation ( ) {
      return ( $this->GIF );
   }
}
/*
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
::
:: GIFDecoder Version 2.0 by László Zsidi, http://gifs.hu
::
:: Created at 2007. 02. 01. &#39;07.47.AM&#39;
::
::
::
::
::  Try on-line GIFBuilder Form demo based on GIFDecoder.
::
::  http://gifs.hu/phpclasses/demos/GifBuilder/
::
:::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
*/
Class GIFDecoder {
   private $GIF_buffer = Array ( );
   private $GIF_arrays = Array ( );
   private $GIF_delays = Array ( );
   private $GIF_stream = "";
   private $GIF_string = "";
   private $GIF_bfseek =  0;
   private $GIF_screen = Array ( );
   private $GIF_global = Array ( );
   private $GIF_sorted;
   private $GIF_colorS;
   private $GIF_colorC;
   private $GIF_colorF;
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFDecoder ( $GIF_pointer )
   ::
   */
   public function __construct ( $GIF_pointer ) {
      $this->GIF_stream = $GIF_pointer;
      $this->GIFGetByte ( 6 );   // GIF89a
      $this->GIFGetByte ( 7 );   // Logical Screen Descriptor
      $this->GIF_screen = $this->GIF_buffer;
      $this->GIF_colorF = $this->GIF_buffer [ 4 ] & 0x80 ? 1 : 0;
      $this->GIF_sorted = $this->GIF_buffer [ 4 ] & 0x08 ? 1 : 0;
      $this->GIF_colorC = $this->GIF_buffer [ 4 ] & 0x07;
      $this->GIF_colorS = 2 << $this->GIF_colorC;
      if ( $this->GIF_colorF == 1 ) {
         $this->GIFGetByte ( 3 * $this->GIF_colorS );
         $this->GIF_global = $this->GIF_buffer;
      }
      /*
       *
       *  05.06.2007.
       *  Made a little modification
       *
       *
       - for ( $cycle = 1; $cycle; ) {
       +    if ( GIFDecoder::GIFGetByte ( 1 ) ) {
       -       switch ( $this->GIF_buffer [ 0 ] ) {
       -          case 0x21:
       -             GIFDecoder::GIFReadExtensions ( );
       -             break;
       -          case 0x2C:
       -             GIFDecoder::GIFReadDescriptor ( );
       -             break;
       -          case 0x3B:
       -             $cycle = 0;
       -             break;
       -       }
       -    }
       +    else {
       +       $cycle = 0;
       +    }
       - }
      */
      for ( $cycle = 1; $cycle; ) {
         if ( $this->GIFGetByte ( 1 ) ) {
            switch ( $this->GIF_buffer [ 0 ] ) {
               case 0x21:
                  $this->GIFReadExtensions ( );
                  break;
               case 0x2C:
                  $this->GIFReadDescriptor ( );
                  break;
               case 0x3B:
                  $cycle = 0;
                  break;
            }
         }
         else {
            $cycle = 0;
         }
      }
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFReadExtension ( )
   ::
   */
   private function GIFReadExtensions ( ) {
      $this->GIFGetByte ( 1 );
      for ( ; ; ) {
         $this->GIFGetByte ( 1 );
         if ( ( $u = $this->GIF_buffer [ 0 ] ) == 0x00 ) {
            break;
         }
         $this->GIFGetByte ( $u );
         /*
          * 07.05.2007.
          * Implemented a new line for a new function
          * to determine the originaly delays between
          * frames.
          *
          */
         if ( $u == 4 ) {
            $this->GIF_delays [ ] = ( $this->GIF_buffer [ 1 ] | $this->GIF_buffer [ 2 ] << 8 );
         }
      }
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFReadExtension ( )
   ::
   */
   private function GIFReadDescriptor ( ) {
      $GIF_screen    = Array ( );
      $this->GIFGetByte ( 9 );
      $GIF_screen = $this->GIF_buffer;
      $GIF_colorF = $this->GIF_buffer [ 8 ] & 0x80 ? 1 : 0;
      if ( $GIF_colorF ) {
         $GIF_code = $this->GIF_buffer [ 8 ] & 0x07;
         $GIF_sort = $this->GIF_buffer [ 8 ] & 0x20 ? 1 : 0;
      }
      else {
         $GIF_code = $this->GIF_colorC;
         $GIF_sort = $this->GIF_sorted;
      }
      $GIF_size = 2 << $GIF_code;
      $this->GIF_screen [ 4 ] &= 0x70;
      $this->GIF_screen [ 4 ] |= 0x80;
      $this->GIF_screen [ 4 ] |= $GIF_code;
      if ( $GIF_sort ) {
         $this->GIF_screen [ 4 ] |= 0x08;
      }
      $this->GIF_string = "GIF87a";
      $this->GIFPutByte ( $this->GIF_screen );
      if ( $GIF_colorF == 1 ) {
         $this->GIFGetByte ( 3 * $GIF_size );
         $this->GIFPutByte ( $this->GIF_buffer );
      }
      else {
         $this->GIFPutByte ( $this->GIF_global );
      }
      $this->GIF_string .= chr ( 0x2C );
      $GIF_screen [ 8 ] &= 0x40;
      $this->GIFPutByte ( $GIF_screen );
      $this->GIFGetByte ( 1 );
      $this->GIFPutByte ( $this->GIF_buffer );
      for ( ; ; ) {
         $this->GIFGetByte ( 1 );
         $this->GIFPutByte ( $this->GIF_buffer );
         if ( ( $u = $this->GIF_buffer [ 0 ] ) == 0x00 ) {
            break;
         }
         $this->GIFGetByte ( $u );
         $this->GIFPutByte ( $this->GIF_buffer );
      }
      $this->GIF_string .= chr ( 0x3B );
      /*
         Add frames into $GIF_stream array...
      */
      $this->GIF_arrays [ ] = $this->GIF_string;
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFGetByte ( $len )
   ::
   */
   /*
    *
    *  05.06.2007.
    *  Made a little modification
    *
    *
    - function GIFGetByte ( $len ) {
    -    $this->GIF_buffer = Array ( );
    -
    -    for ( $i = 0; $i < $len; $i++ ) {
    +       if ( $this->GIF_bfseek > strlen ( $this->GIF_stream ) ) {
    +          return 0;
    +       }
    -       $this->GIF_buffer [ ] = ord ( $this->GIF_stream { $this->GIF_bfseek++ } );
    -    }
    +    return 1;
    - }
    */
   private function GIFGetByte ( $len ) {
      $this->GIF_buffer = Array ( );
      for ( $i = 0; $i < $len; $i++ ) {
         if ( $this->GIF_bfseek > strlen ( $this->GIF_stream ) ) {
            return 0;
         }
         $this->GIF_buffer [ ] = ord ( $this->GIF_stream { $this->GIF_bfseek++ } );
      }
      return 1;
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFPutByte ( $bytes )
   ::
   */
   private function GIFPutByte ( $bytes ) {
      for ( $i = 0; $i < count ( $bytes ); $i++ ) {
         $this->GIF_string .= chr ( $bytes [ $i ] );
      }
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: PUBLIC FUNCTIONS
   ::
   ::
   :: GIFGetFrames ( )
   ::
   */
   public function GIFGetFrames ( ) {
      return ( $this->GIF_arrays );
   }
   /*
   :::::::::::::::::::::::::::::::::::::::::::::::::::
   ::
   :: GIFGetDelays ( )
   ::
   */
   public function GIFGetDelays ( ) {
      return ( $this->GIF_delays );
   }
}
登入後複製

3. [文件] Imagick.class.php

<?php
namespace YBL;
class Imagick{
    /**
     * 图像资源对象
     * @var resource
     */
    private $img;
    /**
     * 图像信息,包括width,height,type,mime,size
     * @var array
     */
    private $info;
    private $gif_proess;   
    public $error;
   
    /**
     * 构造方法,可用于打开一张图像
     * @param string $imgname 图像路径
     */
    public function __construct($gif_proess=true) {
          $this->gif_proess=$gif_proess;
    }
    /**
     * 打开一张图像
     * @param  string $imgname 图像路径
     */
    public function open($imgname){
        //检测图像文件
        if(!is_file($imgname)){
         $this->error=&#39;不存在的图像文件&#39;;
         return false;
      } 
        //销毁已存在的图像
        empty($this->img) || $this->img->destroy();
        //载入图像
        $this->img = new Imagick(realpath($imgname));
        //设置图像信息
        $this->info = array(
            &#39;width&#39;  => $this->img->getImageWidth(),
            &#39;height&#39; => $this->img->getImageHeight(),
            &#39;type&#39;   => strtolower($this->img->getImageFormat()),
            &#39;mime&#39;   => $this->img->getImageMimeType(),
        );
    }
    /**
     * 保存图像
     * @param  string  $imgname   图像保存名称
     * @param  string  $type      图像类型
     * @param  integer $quality   JPEG图像质量      
     * @param  boolean $interlace 是否对JPEG类型图像设置隔行扫描
     */
    public function save($imgname, $type = null, $quality=80,$interlace = true){
        if(empty($this->img)){
         $this->error=&#39;没有可以被保存的图像资源&#39;;
         return false;
      } 
        //设置图片类型
        if(is_null($type)){
            $type = $this->info[&#39;type&#39;];
        } else {
            $type = strtolower($type);
            $this->img->setImageFormat($type);
        }
        //JPEG图像设置隔行扫描
        if(&#39;jpeg&#39; == $type || &#39;jpg&#39; == $type){
            $this->img->setImageInterlaceScheme(1);
        }
        // 设置图像质量
        $this->img->setImageCompressionQuality($quality); 
        //去除图像配置信息
        $this->img->stripImage();
        //保存图像
        $imgname = realpath(dirname($imgname)) . &#39;/&#39; . basename($imgname); //强制绝对路径
        if ((&#39;gif&#39; == $type)&&$this->gif_proess) {
            $this->img->writeImages($imgname, true);
        } else {
            $this->img->writeImage($imgname);
        }
    }
    /**
     * 返回图像宽度
     * @return integer 图像宽度
     */
    public function width(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      } 
        return $this->info[&#39;width&#39;];
    }
    /**
     * 返回图像高度
     * @return integer 图像高度
     */
    public function height(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      } 
        return $this->info[&#39;height&#39;];
    }
    /**
     * 返回图像类型
     * @return string 图像类型
     */
    public function type(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      } 
        return $this->info[&#39;type&#39;];
    }
    /**
     * 返回图像MIME类型
     * @return string 图像MIME类型
     */
    public function mime(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      } 
        return $this->info[&#39;mime&#39;];
    }
    /**
     * 返回图像尺寸数组 0 - 图像宽度,1 - 图像高度
     * @return array 图像尺寸
     */
    public function size(){
        if(empty($this->img)){
         $this->error=&#39;没有指定图像资源&#39;;
         return false;
      } 
        return array($this->info[&#39;width&#39;], $this->info[&#39;height&#39;]);
    }
    /**
     * 裁剪图像
     * @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)){
         $this->error=&#39;没有可以被裁剪的图像资源&#39;;
         return false;
      } 
        //设置保存尺寸
        empty($width)  && $width  = $w;
        empty($height) && $height = $h;
        //裁剪图片
        if((&#39;gif&#39; == $this->info[&#39;type&#39;])&&$this->gif_proess){
            $img = $this->img->coalesceImages();
            $this->img->destroy(); //销毁原图
            //循环裁剪每一帧
            do {
                $this->_crop($w, $h, $x, $y, $width, $height, $img);
            } while ($img->nextImage());
            
            //压缩图片
            $this->img = $img->deconstructImages();
            $img->destroy(); //销毁零时图片
        } else {
            $this->_crop($w, $h, $x, $y, $width, $height);
        }
    }
    /* 裁剪图片,内部调用 */
    private function _crop($w, $h, $x, $y, $width, $height, $img = null){
        is_null($img) && $img = $this->img;
        //裁剪
        $info = $this->info;
        if($x != 0 || $y != 0 || $w != $info[&#39;width&#39;] || $h != $info[&#39;height&#39;]){
            $img->cropImage($w, $h, $x, $y);
            $img->setImagePage($w, $h, 0, 0); //调整画布和图片一致
        }
        
        //调整大小
        if($w != $width || $h != $height){
            $img->sampleImage($width, $height);
        }
        //设置缓存尺寸
        $this->info[&#39;width&#39;]  = $w;
        $this->info[&#39;height&#39;] = $h;
    }
    /**
     * 生成缩略图
     * @param  integer $width  缩略图最大宽度
     * @param  integer $height 缩略图最大高度
     * @param  integer $type   缩略图裁剪类型
     */
    public function thumb($width, $height, $type = Image::IMAGE_THUMB_SCALE){
        if(empty($this->img)){
         $this->error=&#39;没有可以被缩略的图像资源&#39;;
         return false;
      } 
        //原图宽度和高度
        $w = $this->info[&#39;width&#39;];
        $h = $this->info[&#39;height&#39;];
        /* 计算缩略图生成的必要参数 */
        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[&#39;width&#39;] - $w)/2;
                $y = ($this->info[&#39;height&#39;] - $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[&#39;width&#39;] - $w;
                $y = $this->info[&#39;height&#39;] - $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;
                //创建一张新图像
                $newimg = new Imagick();
                $newimg->newImage($width, $height, &#39;white&#39;, $this->info[&#39;type&#39;]);
                if((&#39;gif&#39; == $this->info[&#39;type&#39;])$this->gif_proess){
                    $imgs = $this->img->coalesceImages();
                    $img  = new Imagick();
                    $this->img->destroy(); //销毁原图
                    //循环填充每一帧
                    do {
                        //填充图像
                        $image = $this->_fill($newimg, $posx, $posy, $neww, $newh, $imgs);
                        
                        $img->addImage($image);
                        $img->setImageDelay($imgs->getImageDelay());
                        $img->setImagePage($width, $height, 0, 0);
                        $image->destroy(); //销毁零时图片
                    } while ($imgs->nextImage());
                    //压缩图片
                    $this->img->destroy();
                    $this->img = $img->deconstructImages();
                    $imgs->destroy(); //销毁零时图片
                    $img->destroy(); //销毁零时图片
                } else {
                    //填充图像
                    $img = $this->_fill($newimg, $posx, $posy, $neww, $newh);
                    //销毁原图
                    $this->img->destroy();
                    $this->img = $img;
                }
                //设置新图像属性
                $this->info[&#39;width&#39;]  = $width;
                $this->info[&#39;height&#39;] = $height;
                return;
            /* 固定 */
            case Image::IMAGE_THUMB_FIXED:
                $x = $y = 0;
                break;
            default:$this->error=&#39;不支持的缩略图裁剪类型&#39;;
         return false;
                
        }
        /* 裁剪图像 */
        $this->crop($w, $h, $x, $y, $width, $height);
    }
    /* 填充指定图像,内部使用 */
    private function _fill($newimg, $posx, $posy, $neww, $newh, $img = null){
        is_null($img) && $img = $this->img;
        /* 将指定图片绘入空白图片 */
        $draw  = new \ImagickDraw();
        $draw->composite($img->getImageCompose(), $posx, $posy, $neww, $newh, $img);
        $image = $newimg->clone();
        $image->drawImage($draw);
        $draw->destroy();
        return $image;
    }
    /**
     * 添加水印
     * @param  string  $source 水印图片路径
     * @param  integer $locate 水印位置
     * @param  integer $alpha  水印透明度
     */
    public function water($source, $locate = Image::IMAGE_WATER_SOUTHEAST,$alpha=80){
        //资源检测
        if(empty($this->img)){$this->error=&#39;没有可以被添加水印的图像资源&#39;;
         return false;}
        if(!is_file($source)){
         $this->error=&#39;水印图像不存在&#39;;
         return false;
      }
        //创建水印图像资源
        $water = new Imagick(realpath($source));
        $info  = array($water->getImageWidth(), $water->getImageHeight());
        /* 设定水印位置 */
        switch ($locate) {
            /* 右下角水印 */
            case Image::IMAGE_WATER_SOUTHEAST:
                $x = $this->info[&#39;width&#39;] - $info[0];
                $y = $this->info[&#39;height&#39;] - $info[1];
                break;
            /* 左下角水印 */
            case Image::IMAGE_WATER_SOUTHWEST:
                $x = 0;
                $y = $this->info[&#39;height&#39;] - $info[1];
                break;
            /* 左上角水印 */
            case Image::IMAGE_WATER_NORTHWEST:
                $x = $y = 0;
                break;
            /* 右上角水印 */
            case Image::IMAGE_WATER_NORTHEAST:
                $x = $this->info[&#39;width&#39;] - $info[0];
                $y = 0;
                break;
            /* 居中水印 */
            case Image::IMAGE_WATER_CENTER:
                $x = ($this->info[&#39;width&#39;] - $info[0])/2;
                $y = ($this->info[&#39;height&#39;] - $info[1])/2;
                break;
            /* 下居中水印 */
            case Image::IMAGE_WATER_SOUTH:
                $x = ($this->info[&#39;width&#39;] - $info[0])/2;
                $y = $this->info[&#39;height&#39;] - $info[1];
                break;
            /* 右居中水印 */
            case Image::IMAGE_WATER_EAST:
                $x = $this->info[&#39;width&#39;] - $info[0];
                $y = ($this->info[&#39;height&#39;] - $info[1])/2;
                break;
            /* 上居中水印 */
            case Image::IMAGE_WATER_NORTH:
                $x = ($this->info[&#39;width&#39;] - $info[0])/2;
                $y = 0;
                break;
            /* 左居中水印 */
            case Image::IMAGE_WATER_WEST:
                $x = 0;
                $y = ($this->info[&#39;height&#39;] - $info[1])/2;
                break;
            default:
                /* 自定义水印坐标 */
                if(is_array($locate)){
                    list($x, $y) = $locate;
                } else {
               $this->error=&#39;不支持的水印位置类型&#39;;
         return false;
                }
        }
        //创建绘图资源
        $draw = new \ImagickDraw();
        $draw->composite($water->getImageCompose(), $x, $y, $info[0], $info[1], $water);
        
        if((&#39;gif&#39; == $this->info[&#39;type&#39;])&&$this->gif_proess){
            $img = $this->img->coalesceImages();
            $this->img->destroy(); //销毁原图
            do{
                //添加水印
                $img->drawImage($draw);
            } while ($img->nextImage());
            //压缩图片
            $this->img = $img->deconstructImages();
            $img->destroy(); //销毁零时图片
        } else {
            //添加水印
            $this->img->drawImage($draw);
        }
        //销毁水印资源
        $draw->destroy();
        $water->destroy();
    }
    /**
     * 图像添加文字
     * @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 = &#39;#00000000&#39;, 
        $locate = Image::IMAGE_WATER_SOUTHEAST, $offset = 0, $angle = 0){
        //资源检测
        if(empty($this->img)){$this->error=&#39;没有可以被写入文字的图像资源&#39;;
         return false;} 
        if(!is_file($font)){
         $this->error="不存在的字体文件:{$font}";
         return false;
      }
        //获取颜色和透明度
        if(is_array($color)){
            $color = array_map(&#39;dechex&#39;, $color);
            foreach ($color as &$value) {
                $value = str_pad($value, 2, &#39;0&#39;, STR_PAD_LEFT);
            }
            $color = &#39;#&#39; . implode(&#39;&#39;, $color);
        } elseif(!is_string($color) || 0 !== strpos($color, &#39;#&#39;)) {
            $this->error=&#39;错误的颜色值&#39;;
         return false;
        }
        $col = substr($color, 0, 7);
        $alp = strlen($color) == 9 ? substr($color, -2) : 0;
        
        //获取文字信息
        $draw = new \ImagickDraw();
        $draw->setFont(realpath($font));
        $draw->setFontSize($size);
        $draw->setFillColor($col);
        $draw->setFillAlpha(1-hexdec($alp)/127);
        $draw->setTextAntialias(true);
        $draw->setStrokeAntialias(true);
        
        $metrics = $this->img->queryFontMetrics($draw, $text);
        /* 计算文字初始坐标和尺寸 */
        $x = 0;
        $y = $metrics[&#39;ascender&#39;];
        $w = $metrics[&#39;textWidth&#39;];
        $h = $metrics[&#39;textHeight&#39;];
        /* 设定文字位置 */
        switch ($locate) {
            /* 右下角文字 */
            case Image::IMAGE_WATER_SOUTHEAST:
                $x += $this->info[&#39;width&#39;]  - $w;
                $y += $this->info[&#39;height&#39;] - $h;
                break;
            /* 左下角文字 */
            case Image::IMAGE_WATER_SOUTHWEST:
                $y += $this->info[&#39;height&#39;] - $h;
                break;
            /* 左上角文字 */
            case Image::IMAGE_WATER_NORTHWEST:
                // 起始坐标即为左上角坐标,无需调整
                break;
            /* 右上角文字 */
            case Image::IMAGE_WATER_NORTHEAST:
                $x += $this->info[&#39;width&#39;] - $w;
                break;
            /* 居中文字 */
            case Image::IMAGE_WATER_CENTER:
                $x += ($this->info[&#39;width&#39;]  - $w)/2;
                $y += ($this->info[&#39;height&#39;] - $h)/2;
                break;
            /* 下居中文字 */
            case Image::IMAGE_WATER_SOUTH:
                $x += ($this->info[&#39;width&#39;] - $w)/2;
                $y += $this->info[&#39;height&#39;] - $h;
                break;
            /* 右居中文字 */
            case Image::IMAGE_WATER_EAST:
                $x += $this->info[&#39;width&#39;] - $w;
                $y += ($this->info[&#39;height&#39;] - $h)/2;
                break;
            /* 上居中文字 */
            case Image::IMAGE_WATER_NORTH:
                $x += ($this->info[&#39;width&#39;] - $w)/2;
                break;
            /* 左居中文字 */
            case Image::IMAGE_WATER_WEST:
                $y += ($this->info[&#39;height&#39;] - $h)/2;
                break;
            default:
                /* 自定义文字坐标 */
                if(is_array($locate)){
                    list($posx, $posy) = $locate;
                    $x += $posx;
                    $y += $posy;
                } else {
               $this->error=&#39;不支持的文字位置类型&#39;;
         return false;
                 
                }
        }
        /* 设置偏移量 */
        if(is_array($offset)){
            $offset = array_map(&#39;intval&#39;, $offset);
            list($ox, $oy) = $offset;
        } else{
            $offset = intval($offset);
            $ox = $oy = $offset;
        }
        /* 写入文字 */
        if((&#39;gif&#39; == $this->info[&#39;type&#39;])&&$this->gif_proess){
            $img = $this->img->coalesceImages();
            $this->img->destroy(); //销毁原图
            do{
                $img->annotateImage($draw, $x + $ox, $y + $oy, $angle, $text);
            } while ($img->nextImage());
            //压缩图片
            $this->img = $img->deconstructImages();
            $img->destroy(); //销毁零时图片
        } else {
            $this->img->annotateImage($draw, $x + $ox, $y + $oy, $angle, $text);
        }
        $draw->destroy();
    }
    /**
     * 析构方法,用于销毁图像资源
     */
    public function __destruct() {
        empty($this->img) || $this->img->destroy();
    }
}
登入後複製

4. [文件] image.php

<?php
namespace YBL;
set_time_limit(0);
require_once(&#39;../lib/image/Image.class.php&#39;);
$source=&#39;image/ybl.gif&#39;;
$image=new image(1);
//========================================================================
//打开一幅图像
$image->open($source);
echo &#39;图片宽度:&#39;.$image->width(),&#39;<br/>&#39;;
echo &#39;图片高度:&#39;.$image->height(),&#39;<br/>&#39;;
echo &#39;图片类型:&#39;.$image->type(),&#39;<br/>&#39;;
echo &#39;图像MIME类型:&#39;.$image->mime(),&#39;<br/>&#39;;
$size=$image->size();
echo &#39;图片宽度:&#39;.$size[0],&#39;<br/>&#39;;
echo &#39;图片高度:&#39;.$size[1],&#39;<br/>&#39;;
$savename=&#39;image/save.gif&#39;;
//保存图片
//参数 图片保存地址,图片类型,图像质量,是否对JPEG类型图片设置隔行扫描
 $image->save($savename);
$wh=2;
//======================================================================
//生成缩略图
//参数 缩略图最大宽度,缩略图最大高度,缩略图裁剪类型
$image->thumb($size[0]*$wh,$size[1]*$wh,5);
$savename=&#39;image/thumb.gif&#39;;
$image->save($savename);
//=================================================================================
//图像添加文字
$text=&#39;你好!!!文字&#39;;
//字体类型
$fonturl=&#39;image/simhei.ttf&#39;;
//字体大小
$fontsize=&#39;20&#39;;
//字体颜色
$fontcolor=&#39;#FFF&#39;;
//字体倾斜度
$fontxdu=20;
//参数 添加的文字,字体路径,字号,文字颜色,文字写入位置,文字相对当前位置的偏移量,文字倾斜角度
$image->text($text,$fonturl,$fontsize,$fontcolor,array(100,100),0,$fontxdu);
$savename=&#39;image/text.gif&#39;;
$image->save($savename);
//=================================================================================
//打水印
$alpha=80;
$water=&#39;image/water.png&#39;;
//数组表示 水印位置 
//数字 水印位置类型
$xy=array(120,20);
//参数  水印图片路径,水印位置,水印透明度
$image->water($water,$xy,$alpha);
$savename=&#39;image/water_aim.gif&#39;;
$image->save($savename);
//=================================================================================
//裁剪图像
//参数  裁剪区域宽度,裁剪区域高度,裁剪区域x坐标,裁剪区域y坐标,图片保存宽度,图片保存高度
$image->crop(100,100,100,100);
$savename=&#39;image/crop.gif&#39;;
$image->save($savename);
登入後複製

                               


                   

本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1677
14
CakePHP 教程
1431
52
Laravel 教程
1334
25
PHP教程
1280
29
C# 教程
1257
24
繼續使用PHP:耐力的原因 繼續使用PHP:耐力的原因 Apr 19, 2025 am 12:23 AM

PHP仍然流行的原因是其易用性、靈活性和強大的生態系統。 1)易用性和簡單語法使其成為初學者的首選。 2)與web開發緊密結合,處理HTTP請求和數據庫交互出色。 3)龐大的生態系統提供了豐富的工具和庫。 4)活躍的社區和開源性質使其適應新需求和技術趨勢。

如果session_start()被多次調用會發生什麼? 如果session_start()被多次調用會發生什麼? Apr 25, 2025 am 12:06 AM

多次調用session_start()會導致警告信息和可能的數據覆蓋。 1)PHP會發出警告,提示session已啟動。 2)可能導致session數據意外覆蓋。 3)使用session_status()檢查session狀態,避免重複調用。

IIS和PHP的兼容性:深度潛水 IIS和PHP的兼容性:深度潛水 Apr 22, 2025 am 12:01 AM

IIS和PHP可以兼容,通過FastCGI實現。 1.IIS通過配置文件將.php文件請求轉發給FastCGI模塊。 2.FastCGI模塊啟動PHP進程處理請求,提高性能和穩定性。 3.實際應用中需注意配置細節、錯誤調試和性能優化。

session_start()函數的意義是什麼? session_start()函數的意義是什麼? May 03, 2025 am 12:18 AM

session_start()iscucialinphpformanagingusersessions.1)ItInitiateSanewsessionifnoneexists,2)resumesanexistingsessions,and3)setsasesessionCookieforContinuityActinuityAccontinuityAcconActInityAcconActInityAcconAccRequests,EnablingApplicationsApplicationsLikeUseAppericationLikeUseAthenticationalticationaltication and PersersonalizedContentent。

作曲家:通過AI的幫助開發PHP 作曲家:通過AI的幫助開發PHP Apr 29, 2025 am 12:27 AM

AI可以幫助優化Composer的使用,具體方法包括:1.依賴管理優化:AI分析依賴關係,建議最佳版本組合,減少衝突。 2.自動化代碼生成:AI生成符合最佳實踐的composer.json文件。 3.代碼質量提升:AI檢測潛在問題,提供優化建議,提高代碼質量。這些方法通過機器學習和自然語言處理技術實現,幫助開發者提高效率和代碼質量。

使用Laravel:使用PHP簡化Web開發 使用Laravel:使用PHP簡化Web開發 Apr 19, 2025 am 12:18 AM

Laravel優化Web開發流程的方法包括:1.使用路由系統管理URL結構;2.利用Blade模板引擎簡化視圖開發;3.通過隊列處理耗時任務;4.使用EloquentORM簡化數據庫操作;5.遵循最佳實踐提高代碼質量和可維護性。

PHP和IIS:讓他們一起工作 PHP和IIS:讓他們一起工作 Apr 21, 2025 am 12:06 AM

在IIS上配置和運行PHP需要以下步驟:1)下載並安裝PHP,2)配置IIS並添加FastCGI模塊,3)創建並設置應用池,4)創建網站並綁定到應用池。通過這些步驟,你可以在Windows服務器上輕鬆部署PHP應用,並通過配置擴展和優化性能來提升應用的穩定性和效率。

H5:HTML5的關鍵改進 H5:HTML5的關鍵改進 Apr 28, 2025 am 12:26 AM

HTML5帶來了五個關鍵改進:1.語義化標籤提升了代碼清晰度和SEO效果;2.多媒體支持簡化了視頻和音頻嵌入;3.表單增強簡化了驗證;4.離線與本地存儲提高了用戶體驗;5.畫布與圖形功能增強了網頁的可視化效果。

See all articles