最近在做一个项目,需要裁减图片,由于图片需要自动裁减成指定大小的图片并保存下来。
一、思路
裁减思路是配合伪静态来实现的,比如 https://yefengs.com/wp-content/20200620/abc.jpg
若我需要裁减成 600x300px的图片,那么我直接访问图片https://yefengs.com/wp-content/20200620/abc.jpg_600x300.jpg
就可以显示600x300px的图片。
其中中间的原理是利用伪静态实现,首先NGINX或者Apache访问https://yefengs.com/wp-content/20200620/abc.jpg_600x300.jpg
发现不存在此路径,于是,交给伪静态处理,而这个伪静态重写规则是
rewrite ^/.*?(wp-content/\d/.*.+?)\_([\d]+)x([\d]+)\.jpg$ /plugin/thumb/?src=/$1&w=$2&h=$3&zc=1 last;
//写法没有测试,可能有问题,大概思想是这也样子的,我以NGINX的伪静态规则为例
经过伪静态处理后,图片交给了/plugin/thumb/index.php来处理,传入了图片正确地址和图片的尺寸,index.php自己输入的路径和参数生成一个完整的https://yefengs.com/wp-content/20200620/abc.jpg_600x300.jpg
图片路径,生成成功后当前连接就直接跳转到https://yefengs.com/wp-content/20200620/abc.jpg_600x300.jpg
这个真实的路径上了。
这种方法有个缺点就是需要加载一次图片才会生成图片,倘若不加载,就不会生成该图片,好处是一次生成终生存在。
二、图片的裁减函数
我先在这里做个代码摘抄,思路是这个思路,可能会用到下面的代码
代码1. 一个简单的图片裁减和函数,我比较关注的是图片的裁切形式,比如是左上角还是中心裁减,这个当比例不合适时,应该以哪种方式裁减。
/**
* @param $srcImage 原图片路径
* @param $toFile 裁剪后图片路径
* @param int $maxWidth 裁剪最大的宽度
* @param int $maxHeight 裁剪后最大的高度
* @param int $imgQuality 图片的质量
* @return bool|string|void
*/
function resize($srcImage,$toFile,$maxWidth = 640,$maxHeight = 480,$imgQuality=100)
{
list($width, $height, $type, $attr) = getimagesize($srcImage);
if($width < $maxWidth || $height < $maxHeight) return ;
switch ($type) {//判断图片类别
case 1: $img = imagecreatefromgif($srcImage); break;
case 2: $img = imagecreatefromjpeg($srcImage); break;
case 3: $img = imagecreatefrompng($srcImage); break;
}
$scale = min($maxWidth/$width, $maxHeight/$height);
if($scale < 1) {
$newWidth = floor($scale*$width);
$newHeight = floor($scale*$height);
$newImg = imagecreatetruecolor($newWidth, $newHeight);
imagecopyresampled($newImg, $img, 0, 0, 0, 0, $newWidth, $newHeight, $width, $height);
$newName = "";
$toFile = preg_replace("/(.gif|.jpg|.jpeg|.png)/i","",$toFile);
switch($type) {
case 1: if(imagegif($newImg, "$toFile$newName.gif", $imgQuality))
return "$newName.gif"; break;
case 2: if(imagejpeg($newImg, "$toFile$newName.jpg", $imgQuality))
return "$newName.jpg"; break;
case 3: if(imagepng($newImg, "$toFile$newName.png", $imgQuality))
return "$newName.png"; break;
default: if(imagejpeg($newImg, "$toFile$newName.jpg", $imgQuality))
return "$newName.jpg"; break;
}
imagedestroy($newImg);
}
imagedestroy($img);
return false;
}
代码2.
<?php
class Image
{
private $info;
private $image;
public $type;
public function __construct($src)
{
$this->info=getimagesize($src);
$this->type=image_type_to_extension($this->info['2'],false);
$fun="imagecreatefrom{$this->type}";
$this->image=$fun($src);
}
/**
* 文字水印
* @param [type] $font 字体
* @param [type] $content 内容
* @param [type] $size 文字大小
* @param [type] $col 文字颜色(四元数组)
* @param array $location 位置
* @param integer $angle 倾斜角度
* @return [type]
*/
public function fontMark($font,$content,$size,$col,$location,$angle=0){
$col=imagecolorallocatealpha($this->image, $col['0'], $col['1'], $col['2'],$col['3']);
imagettftext($this->image, $size, $angle, $location['0'], $location['1'], $col,$font,$content);
}
/**
* 图片水印
* @param [type] $imageMark 水印图片地址
* @param [type] $dst 水印图片在原图片中的位置
* @param [type] $pct 透明度
* @return [type]
*/
public function imageMark($imageMark,$dst,$pct){
$info2=getimagesize($imageMark);
$type=image_type_to_extension($info2['2'],false);
$func2="imagecreatefrom".$type;
$water=$func2($imageMark);
imagecopymerge($this->image, $water, $dst[0], $dst[1], 0, 0, $info2['0'], $info2['1'], $pct);
imagedestroy($water);
}
/**
* 压缩图片
* @param [type] $thumbSize 压缩图片大小
* @return [type] [description]
*/
public function thumb($thumbSize){
$imageThumb=imagecreatetruecolor($thumbSize[0], $thumbSize[1]);
imagecopyresampled($imageThumb, $this->image, 0, 0, 0, 0, $thumbSize[0], $thumbSize[1], $this->info['0'], $this->info['1']);
imagedestroy($this->image);
$this->image=$imageThumb;
}
/**
* 裁剪图片
* @param [type] $cutSize 裁剪大小
* @param [type] $location 裁剪位置
* @return [type] [description]
*/
public function cut($cutSize,$location){
$imageCut=imagecreatetruecolor($cutSize[0],$cutSize[1]);
imagecopyresampled($imageCut, $this->image, 0, 0, $location[0], $location[1],$cutSize[0],$cutSize[1],$cutSize[0],$cutSize[1]);
imagedestroy($this->image);
$this->image=$imageCut;
}
/**
* 展现图片
* @return [type] [description]
*/
public function show(){
header("content-type:".$this->info['mime']);
$funn="image".$this->type;
$funn($this->image);
}
/**
* 保存图片
* @param [type] $newname 新图片名
* @return [type] [description]
*/
public function save($newname){
header("content-type:".$this->info['mime']);
$funn="image".$this->type;
$funn($this->image,$newname.'.'.$this->type);
}
public function __destruct(){
imagedestroy($this->image);
}
}
?>
以上代码的说明
php对图片文件的操作主要是利用GD库扩展。当我们频繁利用php对图片进行操作时,会自然封装很多函数,否则会写太多重复的代码。当有很多对图片的相关函数的时候,我们可以考虑将这些函数也整理一下,因而就有了封装成类的想法。
操作图片主要历经四个步骤:
1、打开图片 2、操作图片 3、输出图片 4、销毁图片
1,3,4三个步骤每次都要写,每次又都差不多。真正需要变通的只有操作图片的这一步骤了。操作图片又往往通过1或多个主要的GD函数来完成。