1.开始

在网站上传图片过程,经常用到缩略图功能。这里我自己写了一个图片处理的Image类,能生成缩略图,并且可以添加水印图。
2.如何生成缩略图
生成缩略图,关键的是如何计算缩放比率。
这里,我根据图片等比缩放,宽高的几种常见变化,得出一个算缩放比率算法是,使用新图(即缩略图)的宽高,分别除以原图的宽高,看哪个值大,就取它作为缩放比率:
缩放比率 = Max( { 新图高度 / 原图高度 , 新图宽度 / 原图宽度 } )
也就是:
If ( (新图高度 / 原图高度) > (新图宽度 / 原图宽度 ) ) {
缩放比率 = 新图高度 / 原图高度;
}ELSE {
缩放比率 = 新图宽度 / 原图宽度;
}
这里列出场景的图片缩放场景,及处理方法:
e.g
场景1,原图比新图大的情况, 缩放比率 = 新图宽度 / 原图宽度 :
场景2,原图比新图大的情况,b. 缩放比率 = 新图高度 / 原图高度 :
场景3,原图比新图大的情况,而且新图宽高相等,即新图形状是正方形,那么上面的缩放算法也是适用的。
场景4,如果 “新图宽度 >= 原图宽度” ,同时 “新图高度 >= 原图高度”,那么不缩放图片,也不放大图片,保持原图。
场景5,如果 “新图宽度 < 原图宽度”,同时 “新图高度 >= 原图高度” ,那么先设置 “新图高度= 原图高度”,再剪切。
场景6,如果 “新图高度 < 原图高度”,同时 “新图宽度 >= 原图宽度” ,那么先设置 “新图宽度= 原图宽度”,再剪切。
3.如何添加水印图片
添加水印很容易,我这里没考虑那么复杂,主要是控制水印位置在图片的右下角,和控制水印在图片中的大小。如,当目标图片与水印图大小接近,那么需要先等比缩放水印图片,再添加水印图片。
左边两幅图,上面是原图,下面是水印图,右边的缩放后加水印的新图。
4.类图
5.PHP代码
5.1. 构造函数 __construct()
在Image类中,除了构造函数__construct()是public,其它函数都为private.也就是在函数__construct()中,直接完成了生成缩略图和添加水印图的功能。如果,只生成缩略图而不需要添加水印,那么直接在__construct()的参数$markPath,设置为null即可。
其中,“$this->quality = $quality ? $quality : 75;” 控制输出为JPG图片时,控制图片质量(0-100),默认值为75;
/**
* Image constructor.
* @param string $imagePath 图片路径
* @param string $markPath 水印图片路径
* @param int $new_width 缩略图宽度
* @param int $new_height 缩略图高度
* @param int $quality JPG图片格输出质量
*/
public function __construct(string $imagePath,
string $markPath = null,
int $new_width = null,
int $new_height = null,
int $quality = 75)
{
$this->imgPath = $_SERVER['DOCUMENT_ROOT'] . $imagePath;
$this->waterMarkPath = $markPath;
$this->newWidth = $new_width ? $new_width : $this->width;
$this->newHeight = $new_height ? $new_height : $this->height;
$this->quality = $quality ? $quality : 75;
list($this->width, $this->height, $this->type) = getimagesize($this->imgPath);
$this->img = $this->_loadImg($this->imgPath, $this->type);
//生成缩略图
$this->_thumb();
//添加水印图片
if (!empty($this->waterMarkPath)) $this->_addWaterMark();
//输出图片
$this->_outputImg();
}
Note: 先生成缩略图,再在新图上添加水印 图片。
5.2. 生成缩略图函数_thumb()
/**
* 缩略图(按等比例,根据设置的宽度和高度进行裁剪)
*/
private function _thumb()
{
//如果原图本身小于缩略图,按原图长高
if ($this->newWidth > $this->width) $this->newWidth = $this->width;
if ($this->newHeight > $this->height) $this->newHeight = $this->height;
//背景图长高
$gd_width = $this->newWidth;
$gd_height = $this->newHeight;
//如果缩略图宽高,其中有一边等于原图的宽高,就直接裁剪
if ($gd_width == $this->width || $gd_height == $this->height) {
$this->newWidth = $this->width;
$this->newHeight = $this->height;
} else {
//计算缩放比率
$per = 1;
if (($this->newHeight / $this->height) > ($this->newWidth / $this->width)) {
$per = $this->newHeight / $this->height;
} else {
$per = $this->newWidth / $this->width;
}
if ($per < 1) {
$this->newWidth = $this->width * $per;
$this->newHeight = $this->height * $per;
}
}
$this->newImg = $this->_CreateImg($gd_width, $gd_height, $this->type);
imagecopyresampled($this->newImg, $this->img, 0, 0, 0, 0, $this->newWidth, $this->newHeight, $this->width, $this->height);
}
生成缩略图函数_thumb() ,是按照前面的分析来进行编码。
5.3. 添加水印图片函数 _addWaterMark()
/**
* 添加水印
*/
private function _addWaterMark()
{
$ratio = 1 / 5; //水印缩放比率
$Width = imagesx($this->newImg);
$Height = imagesy($this->newImg);
$n_width = $Width * $ratio;
$n_height = $Width * $ratio;
list($markWidth, $markHeight, $markType) = getimagesize($this->waterMarkPath);
if ($n_width > $markWidth) $n_width = $markWidth;
if ($n_height > $markHeight) $n_height = $markHeight;
$Img = $this->_loadImg($this->waterMarkPath, $markType);
$Img = $this->_thumb1($Img, $markWidth, $markHeight, $markType, $n_width, $n_height);
$markWidth = imagesx($Img);
$markHeight = imagesy($Img);
imagecopyresampled($this->newImg, $Img, $Width - $markWidth - 10, $Height - $markHeight - 10, 0, 0, $markWidth, $markHeight, $markWidth, $markHeight);
imagedestroy($Img);
}
在添加水印图片中,用到一个_thumb1()函数来缩放水印图片:
/**
* 缩略图(按等比例)
* @param resource $img 图像流
* @param int $width
* @param int $height
* @param int $type
* @param int $new_width
* @param int $new_height
* @return resource
*/
private function _thumb1($img, $width, $height, $type, $new_width, $new_height)
{
if ($width < $height) {
$new_width = ($new_height / $height) * $width;
} else {
$new_height = ($new_width / $width) * $height;
}
$newImg = $this->_CreateImg($new_width, $new_height, $type);
imagecopyresampled($newImg, $img, 0, 0, 0, 0, $new_width, $new_height, $width, $height);
return $newImg;
}
5.4. 完整代码:
<?php
/**
* 图片处理,生成缩略图和添加水印图片
* Created by PhpStorm.
* User: andy
* Date: 17-1-3
* Time: 上午11:55
*/
class Image
{
//原图
private $imgPath; //图片地址
private $width; //图片宽度
private $height; //图片高度
private $type; //图片类型
private $img; //图片(图像流)
//缩略图
private $newImg; //缩略图(图像流)
private $newWidth;
private $newHeight;
//水印图路径
private $waterMarkPath;
//输出图像质量,jpg有效
private $quality;
/**
* Image constructor.
* @param string $imagePath 图片路径
* @param string $markPath 水印图片路径
* @param int $new_width 缩略图宽度
* @param int $new_height 缩略图高度
* @param int $quality JPG图片格输出质量
*/
public function __construct(string $imagePath,
string $markPath = null,
int $new_width = null,
int $new_height = null,
int $quality = 75)
{
$this->imgPath = $_SERVER['DOCUMENT_ROOT'] . $imagePath;
$this->waterMarkPath = $markPath;
$this->newWidth = $new_width ? $new_width : $this->width;
$this->newHeight = $new_height ? $new_height : $this->height;
$this->quality = $quality ? $quality : 75;
list($this->width, $this->height, $this->type) = getimagesize($this->imgPath);
$this->img = $this->_loadImg($this->imgPath, $this->type);
//生成缩略图
$this->_thumb();
//添加水印图片
if (!empty($this->waterMarkPath)) $this->_addWaterMark();
//输出图片
$this->_outputImg();
}
/**
*图片输出
*/
private function _outputImg()
{
switch ($this->type) {
case 1: // GIF
imagegif($this->newImg, $this->imgPath);
break;
case 2: // JPG
if (intval($this->quality) < 0 || intval($this->quality) > 100) $this->quality = 75;
imagejpeg($this->newImg, $this->imgPath, $this->quality);
break;
case 3: // PNG
imagepng($this->newImg, $this->imgPath);
break;
}
imagedestroy($this->newImg);
imagedestroy($this->img);
}
/**
* 添加水印
*/
private function _addWaterMark()
{
$ratio = 1 / 5; //水印缩放比率
$Width = imagesx($this->newImg);
$Height = imagesy($this->newImg);
$n_width = $Width * $ratio;
$n_height = $Width * $ratio;
list($markWidth, $markHeight, $markType) = getimagesize($this->waterMarkPath);
if ($n_width > $markWidth) $n_width = $markWidth;
if ($n_height > $markHeight) $n_height = $markHeight;
$Img = $this->_loadImg($this->waterMarkPath, $markType);
$Img = $this->_thumb1($Img, $markWidth, $markHeight, $markType, $n_width, $n_height);
$markWidth = imagesx($Img);
$markHeight = imagesy($Img);
imagecopyresampled($this->newImg, $Img, $Width - $markWidth - 10, $Height - $markHeight - 10, 0, 0, $markWidth, $markHeight, $markWidth, $markHeight);
imagedestroy($Img);
}
/**
* 缩略图(按等比例,根据设置的宽度和高度进行裁剪)
*/
private function _thumb()
{
//如果原图本身小于缩略图,按原图长高
if ($this->newWidth > $this->width) $this->newWidth = $this->width;
if ($this->newHeight > $this->height) $this->newHeight = $this->height;
//背景图长高
$gd_width = $this->newWidth;
$gd_height = $this->newHeight;
//如果缩略图宽高,其中有一边等于原图的宽高,就直接裁剪
if ($gd_width == $this->width || $gd_height == $this->height) {
$this->newWidth = $this->width;
$this->newHeight = $this->height;
} else {
//计算缩放比率
$per = 1;
if (($this->newHeight / $this->height) > ($this->newWidth / $this->width)) {
$per = $this->newHeight / $this->height;
} else {
$per = $this->newWidth / $this->width;
}
if ($per < 1) {
$this->newWidth = $this->width * $per;
$this->newHeight = $this->height * $per;
}
}
$this->newImg = $this->_CreateImg($gd_width, $gd_height, $this->type);
imagecopyresampled($this->newImg, $this->img, 0, 0, 0, 0, $this->newWidth, $this->newHeight, $this->width, $this->height);
}
/**
* 缩略图(按等比例)
* @param resource $img 图像流
* @param int $width
* @param int $height
* @param int $type
* @param int $new_width
* @param int $new_height
* @return resource
*/
private function _thumb1($img, $width, $height, $type, $new_width, $new_height)
{
if ($width < $height) {
$new_width = ($new_height / $height) * $width;
} else {
$new_height = ($new_width / $width) * $height;
}
$newImg = $this->_CreateImg($new_width, $new_height, $type);
imagecopyresampled($newImg, $img, 0, 0, 0, 0, $new_width, $new_height, $width, $height);
return $newImg;
}
/**
* 加载图片
* @param string $imgPath
* @param int $type
* @return resource
*/
private function _loadImg($imgPath, $type)
{
switch ($type) {
case 1: // GIF
$img = imagecreatefromgif($imgPath);
break;
case 2: // JPG
$img = imagecreatefromjpeg($imgPath);
break;
case 3: // PNG
$img = imagecreatefrompng($imgPath);
break;
default: //其他类型
Tool::alertBack('不支持当前图片类型.' . $type);
break;
}
return $img;
}
/**
* 创建一个背景图像
* @param int $width
* @param int $height
* @param int $type
* @return resource
*/
private function _CreateImg($width, $height, $type)
{
$img = imagecreatetruecolor($width, $height);
switch ($type) {
case 3: //png
imagecolortransparent($img, 0); //设置背景为透明的
imagealphablending($img, false);
imagesavealpha($img, true);
break;
case 4://gif
imagecolortransparent($img, 0);
break;
}
return $img;
}
}
6.调用
调用非常简单,在引入类后,直接new 并输入对应参数即可:
e.g.
new Image($_path, MARK, 400, 200, 100);
7.小结
这个Image 类能够生成缩略图,不出现黑边,添加水印图,能根据图片的大小缩放水印图。当然有个缺点,就是不能缩放GIF的动画,因为涉及到帧的处理,比较麻烦。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
# PHP
# 缩略图
# 图片水印
# thinkPHP实现上传图片及生成缩略图功能示例
# PHP基于ffmpeg实现转换视频
# 截图及生成缩略图的方法
# PHP基于GD库实现的生成图片缩略图函数示例
# PHP基于正则批量替换Img中src内容实现获取缩略图的功能示例
# 使用ThinkPHP生成缩略图及显示
# php生成图片缩略图功能示例
# PHP生成图片缩略图类示例
# php生成缩略图质量较差解决方法代码示例
# 新图
# 长高
# 中有
# 等比例
# 按原
# 的是
# 图片处理
# 也不
# 是在
# 有个
# 也就
# 很容易
# 而不
# 写了
# 几种
# 不支持
# 涉及到
# 设置为
# 都为
# 创建一个
相关文章:
音乐网站服务器如何优化API响应速度?
如何通过商城免费建站系统源码自定义网站主题?
,巨量百应是干嘛的?
已有域名如何免费搭建网站?
一键网站制作软件,义乌购一件代发流程?
如何在Windows环境下新建FTP站点并设置权限?
如何用IIS7快速搭建并优化网站站点?
c# await 一个已经完成的Task会发生什么
python的本地网站制作,如何创建本地站点?
建站之星CMS建站配置指南:模板选择与SEO优化技巧
如何选择网络建站服务器?高效建站必看指南
如何自己制作一个网站链接,如何制作一个企业网站,建设网站的基本步骤有哪些?
移动端手机网站制作软件,掌上时代,移动端网站的谷歌SEO该如何做?
潍坊网站制作公司有哪些,潍坊哪家招聘网站好?
建站主机是否等同于虚拟主机?
建站之星与建站宝盒如何选择最佳方案?
,想在网上投简历,哪几个网站比较好?
南宁网站建设制作定制,南宁网站建设可以定制吗?
焦点电影公司作品,电影焦点结局是什么?
网站建设设计制作营销公司南阳,如何策划设计和建设网站?
如何自定义建站之星网站的导航菜单样式?
Dapper的Execute方法的返回值是什么意思 Dapper Execute返回值详解
建站之星备案流程有哪些注意事项?
建站主机解析:虚拟主机配置与服务器选择指南
建站之星手机一键生成:多端自适应+小程序开发快速建站指南
西安制作网站公司有哪些,西安货运司机用的最多的app或者网站是什么?
猪八戒网站制作视频,开发一个猪八戒网站,大约需要多少?或者自己请程序员,需要什么程序员,多少程序员能完成?
微课制作网站有哪些,微课网怎么进?
如何安全更换建站之星模板并保留数据?
如何在阿里云ECS服务器部署织梦CMS网站?
电商网站制作公司有哪些,1688网是什么意思?
如何通过建站之星自助学习解决操作问题?
网站制作说明怎么写,简述网页设计的流程并说明原因?
移民网站制作流程,怎么看加拿大移民官网?
如何在VPS电脑上快速搭建网站?
如何做网站制作流程,*游戏网站怎么搭建?
如何在宝塔面板创建新站点?
Python如何创建带属性的XML节点
Java解压缩zip - 解压缩多个文件或文件夹实例
网站制作软件有哪些,制图软件有哪些?
建站之星如何取消后台验证码生成?
宝塔建站助手安装配置与建站模板使用全流程解析
如何选择最佳自助建站系统?快速指南解析优劣
标准网站视频模板制作软件,现在有哪个网站的视频编辑素材最齐全的,背景音乐、音效等?
零服务器AI建站解决方案:快速部署与云端平台低成本实践
如何实现建站之星域名转发设置?
电影网站制作价格表,那些提供免费电影的网站,他们是怎么盈利的?
图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?
nginx修改上传文件大小限制的方法
网站制作新手教程,新手建设一个网站需要注意些什么?
*请认真填写需求信息,我们会在24小时内与您取得联系。