Gd.php 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | TOPThink [ WE CAN DO IT JUST THINK ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2010 http://topthink.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed ( http://www.apache.org/licenses/LICENSE-2.0 )
  8. // +----------------------------------------------------------------------
  9. // | Author: 麦当苗儿 <zuojiazi@vip.qq.com> <http://www.zjzit.cn>
  10. // +----------------------------------------------------------------------
  11. // | ImageGd.class.php 2013-03-05
  12. // +----------------------------------------------------------------------
  13. class Image_Driver_Gd {
  14. /* 驱动相关常量定义 */
  15. const IMAGE_GD = 1; //常量,标识GD库类型
  16. const IMAGE_IMAGICK = 2; //常量,标识imagick库类型
  17. /* 缩略图相关常量定义 */
  18. const IMAGE_THUMB_SCALE = 1; //常量,标识缩略图等比例缩放类型
  19. const IMAGE_THUMB_FILLED = 2; //常量,标识缩略图缩放后填充类型
  20. const IMAGE_THUMB_CENTER = 3; //常量,标识缩略图居中裁剪类型
  21. const IMAGE_THUMB_NORTHWEST = 4; //常量,标识缩略图左上角裁剪类型
  22. const IMAGE_THUMB_SOUTHEAST = 5; //常量,标识缩略图右下角裁剪类型
  23. const IMAGE_THUMB_FIXED = 6; //常量,标识缩略图固定尺寸缩放类型
  24. /* 水印相关常量定义 */
  25. const IMAGE_WATER_NORTHWEST = 1; //常量,标识左上角水印
  26. const IMAGE_WATER_NORTH = 2; //常量,标识上居中水印
  27. const IMAGE_WATER_NORTHEAST = 3; //常量,标识右上角水印
  28. const IMAGE_WATER_WEST = 4; //常量,标识左居中水印
  29. const IMAGE_WATER_CENTER = 5; //常量,标识居中水印
  30. const IMAGE_WATER_EAST = 6; //常量,标识右居中水印
  31. const IMAGE_WATER_SOUTHWEST = 7; //常量,标识左下角水印
  32. const IMAGE_WATER_SOUTH = 8; //常量,标识下居中水印
  33. const IMAGE_WATER_SOUTHEAST = 9; //常量,标识右下角水印
  34. /**
  35. * 图像资源对象
  36. * @var resource
  37. */
  38. private $img;
  39. /**
  40. * 图像信息,包括width,height,type,mime,size
  41. * @var array
  42. */
  43. private $info;
  44. /**
  45. * 构造方法,可用于打开一张图像
  46. *
  47. * @param string $imgname 图像路径
  48. */
  49. public function __construct($imgname = null) {
  50. $imgname && $this->open($imgname);
  51. }
  52. /**
  53. * 打开一张图像
  54. *
  55. * @param string $imgname 图像路径
  56. */
  57. public function open($imgname) {
  58. //检测图像文件
  59. if (!is_file($imgname)) {
  60. throw new PhalApi_Exception_BadRequest(T('不存在的图像文件'));
  61. };
  62. //获取图像信息
  63. $info = getimagesize($imgname);
  64. //检测图像合法性
  65. if (false === $info || (IMAGETYPE_GIF === $info[2] && empty($info['bits']))) {
  66. throw new PhalApi_Exception_BadRequest(T('非法图像文件'));
  67. }
  68. //设置图像信息
  69. $this->info = array(
  70. 'width' => $info[0],
  71. 'height' => $info[1],
  72. 'type' => image_type_to_extension($info[2], false),
  73. 'mime' => $info['mime'],
  74. );
  75. //销毁已存在的图像
  76. empty($this->img) || imagedestroy($this->img);
  77. //打开图像
  78. if ('gif' == $this->info['type']) {
  79. $class = 'Image_Driver_GIF';
  80. $this->gif = new $class($imgname);
  81. $this->img = imagecreatefromstring($this->gif->image());
  82. } else {
  83. $fun = "imagecreatefrom{$this->info['type']}";
  84. $this->img = $fun($imgname);
  85. }
  86. }
  87. /**
  88. * 保存图像
  89. *
  90. * @param string $imgname 图像保存名称
  91. * @param string $type 图像类型
  92. * @param integer $quality 图像质量
  93. * @param boolean $interlace 是否对JPEG类型图像设置隔行扫描
  94. */
  95. public function save($imgname, $type = null, $quality = 80, $interlace = true) {
  96. if (empty($this->img)) {
  97. throw new PhalApi_Exception_BadRequest(T('没有可以被保存的图像资源'));
  98. }
  99. //自动获取图像类型
  100. if (is_null($type)) {
  101. $type = $this->info['type'];
  102. } else {
  103. $type = strtolower($type);
  104. }
  105. //保存图像
  106. if ('jpeg' == $type || 'jpg' == $type) {
  107. //JPEG图像设置隔行扫描
  108. imageinterlace($this->img, $interlace);
  109. imagejpeg($this->img, $imgname, $quality);
  110. } elseif ('gif' == $type && !empty($this->gif)) {
  111. $this->gif->save($imgname);
  112. } else {
  113. $fun = 'image' . $type;
  114. $fun($this->img, $imgname);
  115. }
  116. }
  117. /**
  118. * 返回图像宽度
  119. * @return integer 图像宽度
  120. */
  121. public function width() {
  122. if (empty($this->img)) {
  123. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  124. }
  125. return $this->info['width'];
  126. }
  127. /**
  128. * 返回图像高度
  129. * @return integer 图像高度
  130. */
  131. public function height() {
  132. if (empty($this->img)) {
  133. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  134. }
  135. return $this->info['height'];
  136. }
  137. /**
  138. * 返回图像类型
  139. * @return string 图像类型
  140. */
  141. public function type() {
  142. if (empty($this->img)) {
  143. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  144. }
  145. return $this->info['type'];
  146. }
  147. /**
  148. * 返回图像MIME类型
  149. * @return string 图像MIME类型
  150. */
  151. public function mime() {
  152. if (empty($this->img)) {
  153. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  154. }
  155. return $this->info['mime'];
  156. }
  157. /**
  158. * 返回图像尺寸数组 0 - 图像宽度,1 - 图像高度
  159. * @return array 图像尺寸
  160. */
  161. public function size() {
  162. if (empty($this->img)) {
  163. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  164. }
  165. return array($this->info['width'], $this->info['height']);
  166. }
  167. /**
  168. * 裁剪图像
  169. *
  170. * @param integer $w 裁剪区域宽度
  171. * @param integer $h 裁剪区域高度
  172. * @param integer $x 裁剪区域x坐标
  173. * @param integer $y 裁剪区域y坐标
  174. * @param integer $width 图像保存宽度
  175. * @param integer $height 图像保存高度
  176. */
  177. public function crop($w, $h, $x = 0, $y = 0, $width = null, $height = null) {
  178. if (empty($this->img)) {
  179. throw new PhalApi_Exception_BadRequest(T('没有可以被裁剪的图像资源'));
  180. }
  181. //设置保存尺寸
  182. empty($width) && $width = $w;
  183. empty($height) && $height = $h;
  184. do {
  185. //创建新图像
  186. $img = imagecreatetruecolor($width, $height);
  187. // 调整默认颜色
  188. $color = imagecolorallocate($img, 255, 255, 255);
  189. imagefill($img, 0, 0, $color);
  190. //裁剪
  191. imagecopyresampled($img, $this->img, 0, 0, $x, $y, $width, $height, $w, $h);
  192. imagedestroy($this->img); //销毁原图
  193. //设置新图像
  194. $this->img = $img;
  195. } while (!empty($this->gif) && $this->gifNext());
  196. $this->info['width'] = $width;
  197. $this->info['height'] = $height;
  198. }
  199. /**
  200. * 生成缩略图
  201. *
  202. * @param integer $width 缩略图最大宽度
  203. * @param integer $height 缩略图最大高度
  204. * @param integer $type 缩略图裁剪类型
  205. */
  206. public function thumb($width, $height, $type = self::IMAGE_THUMB_SCALE) {
  207. if (empty($this->img)) {
  208. throw new PhalApi_Exception_BadRequest(T('没有可以被缩略的图像资源'));
  209. }
  210. //原图宽度和高度
  211. $w = $this->info['width'];
  212. $h = $this->info['height'];
  213. /* 计算缩略图生成的必要参数 */
  214. switch ($type) {
  215. /* 等比例缩放 */
  216. case self::IMAGE_THUMB_SCALE:
  217. //原图尺寸小于缩略图尺寸则不进行缩略
  218. if ($w < $width && $h < $height) {
  219. return;
  220. }
  221. //计算缩放比例
  222. $scale = min($width / $w, $height / $h);
  223. //设置缩略图的坐标及宽度和高度
  224. $x = $y = 0;
  225. $width = $w * $scale;
  226. $height = $h * $scale;
  227. break;
  228. /* 居中裁剪 */
  229. case self::IMAGE_THUMB_CENTER:
  230. //计算缩放比例
  231. $scale = max($width / $w, $height / $h);
  232. //设置缩略图的坐标及宽度和高度
  233. $w = $width / $scale;
  234. $h = $height / $scale;
  235. $x = ($this->info['width'] - $w) / 2;
  236. $y = ($this->info['height'] - $h) / 2;
  237. break;
  238. /* 左上角裁剪 */
  239. case self::IMAGE_THUMB_NORTHWEST:
  240. //计算缩放比例
  241. $scale = max($width / $w, $height / $h);
  242. //设置缩略图的坐标及宽度和高度
  243. $x = $y = 0;
  244. $w = $width / $scale;
  245. $h = $height / $scale;
  246. break;
  247. /* 右下角裁剪 */
  248. case self::IMAGE_THUMB_SOUTHEAST:
  249. //计算缩放比例
  250. $scale = max($width / $w, $height / $h);
  251. //设置缩略图的坐标及宽度和高度
  252. $w = $width / $scale;
  253. $h = $height / $scale;
  254. $x = $this->info['width'] - $w;
  255. $y = $this->info['height'] - $h;
  256. break;
  257. /* 填充 */
  258. case self::IMAGE_THUMB_FILLED:
  259. //计算缩放比例
  260. if ($w < $width && $h < $height) {
  261. $scale = 1;
  262. } else {
  263. $scale = min($width / $w, $height / $h);
  264. }
  265. //设置缩略图的坐标及宽度和高度
  266. $neww = $w * $scale;
  267. $newh = $h * $scale;
  268. $posx = ($width - $w * $scale) / 2;
  269. $posy = ($height - $h * $scale) / 2;
  270. do {
  271. //创建新图像
  272. $img = imagecreatetruecolor($width, $height);
  273. // 调整默认颜色
  274. $color = imagecolorallocate($img, 255, 255, 255);
  275. imagefill($img, 0, 0, $color);
  276. //裁剪
  277. imagecopyresampled($img, $this->img, $posx, $posy, $x, $y, $neww, $newh, $w, $h);
  278. imagedestroy($this->img); //销毁原图
  279. $this->img = $img;
  280. } while (!empty($this->gif) && $this->gifNext());
  281. $this->info['width'] = $width;
  282. $this->info['height'] = $height;
  283. return;
  284. /* 固定 */
  285. case IMAGE_THUMB_FIXED:
  286. $x = $y = 0;
  287. break;
  288. default:
  289. throw new PhalApi_Exception_BadRequest(T('不支持的缩略图裁剪类型'));
  290. }
  291. /* 裁剪图像 */
  292. $this->crop($w, $h, $x, $y, $width, $height);
  293. }
  294. /**
  295. * 添加水印
  296. *
  297. * @param string $source 水印图片路径
  298. * @param integer $locate 水印位置
  299. * @param integer $alpha 水印透明度
  300. */
  301. public function water($source, $locate = self::IMAGE_WATER_SOUTHEAST, $alpha = 80) {
  302. //资源检测
  303. if (empty($this->img)) {
  304. throw new PhalApi_Exception_BadRequest(T('没有可以被添加水印的图像资源'));
  305. }
  306. if (!is_file($source)) {
  307. throw new PhalApi_Exception_BadRequest(T('水印图像不存在'));
  308. }
  309. //获取水印图像信息
  310. $info = getimagesize($source);
  311. if (false === $info || (IMAGETYPE_GIF === $info[2] && empty($info['bits']))) {
  312. throw new PhalApi_Exception_BadRequest(T('非法水印文件'));
  313. }
  314. //创建水印图像资源
  315. $fun = 'imagecreatefrom' . image_type_to_extension($info[2], false);
  316. $water = $fun($source);
  317. //设定水印图像的混色模式
  318. imagealphablending($water, true);
  319. /* 设定水印位置 */
  320. switch ($locate) {
  321. /* 右下角水印 */
  322. case this::IMAGE_WATER_SOUTHEAST:
  323. $x = $this->info['width'] - $info[0];
  324. $y = $this->info['height'] - $info[1];
  325. break;
  326. /* 左下角水印 */
  327. case self::IMAGE_WATER_SOUTHWEST:
  328. $x = 0;
  329. $y = $this->info['height'] - $info[1];
  330. break;
  331. /* 左上角水印 */
  332. case self::IMAGE_WATER_NORTHWEST:
  333. $x = $y = 0;
  334. break;
  335. /* 右上角水印 */
  336. case self::IMAGE_WATER_NORTHEAST:
  337. $x = $this->info['width'] - $info[0];
  338. $y = 0;
  339. break;
  340. /* 居中水印 */
  341. case self::IMAGE_WATER_CENTER:
  342. $x = ($this->info['width'] - $info[0]) / 2;
  343. $y = ($this->info['height'] - $info[1]) / 2;
  344. break;
  345. /* 下居中水印 */
  346. case self::IMAGE_WATER_SOUTH:
  347. $x = ($this->info['width'] - $info[0]) / 2;
  348. $y = $this->info['height'] - $info[1];
  349. break;
  350. /* 右居中水印 */
  351. case self::IMAGE_WATER_EAST:
  352. $x = $this->info['width'] - $info[0];
  353. $y = ($this->info['height'] - $info[1]) / 2;
  354. break;
  355. /* 上居中水印 */
  356. case self::IMAGE_WATER_NORTH:
  357. $x = ($this->info['width'] - $info[0]) / 2;
  358. $y = 0;
  359. break;
  360. /* 左居中水印 */
  361. case self::IMAGE_WATER_WEST:
  362. $x = 0;
  363. $y = ($this->info['height'] - $info[1]) / 2;
  364. break;
  365. default:
  366. /* 自定义水印坐标 */
  367. if (is_array($locate)) {
  368. list($x, $y) = $locate;
  369. } else {
  370. throw new PhalApi_Exception_BadRequest(T('不支持的水印位置类型'));
  371. }
  372. }
  373. do {
  374. //添加水印
  375. $src = imagecreatetruecolor($info[0], $info[1]);
  376. // 调整默认颜色
  377. $color = imagecolorallocate($src, 255, 255, 255);
  378. imagefill($src, 0, 0, $color);
  379. imagecopy($src, $this->img, 0, 0, $x, $y, $info[0], $info[1]);
  380. imagecopy($src, $water, 0, 0, 0, 0, $info[0], $info[1]);
  381. imagecopymerge($this->img, $src, $x, $y, 0, 0, $info[0], $info[1], $alpha);
  382. //销毁零时图片资源
  383. imagedestroy($src);
  384. } while (!empty($this->gif) && $this->gifNext());
  385. //销毁水印资源
  386. imagedestroy($water);
  387. }
  388. /**
  389. * 图像添加文字
  390. *
  391. * @param string $text 添加的文字
  392. * @param string $font 字体路径
  393. * @param integer $size 字号
  394. * @param string $color 文字颜色
  395. * @param integer $locate 文字写入位置
  396. * @param integer $offset 文字相对当前位置的偏移量
  397. * @param integer $angle 文字倾斜角度
  398. */
  399. public function text($text, $font, $size, $color = '#00000000', $locate = self::IMAGE_WATER_SOUTHEAST, $offset = 0, $angle = 0) {
  400. //资源检测
  401. if (empty($this->img)) {
  402. throw new PhalApi_Exception_BadRequest(T('没有可以被写入文字的图像资源'));
  403. }
  404. if (!is_file($font)) {
  405. throw new PhalApi_Exception_BadRequest(T("不存在的字体文件:{$font}"));
  406. }
  407. //获取文字信息
  408. $info = imagettfbbox($size, $angle, $font, $text);
  409. $minx = min($info[0], $info[2], $info[4], $info[6]);
  410. $maxx = max($info[0], $info[2], $info[4], $info[6]);
  411. $miny = min($info[1], $info[3], $info[5], $info[7]);
  412. $maxy = max($info[1], $info[3], $info[5], $info[7]);
  413. /* 计算文字初始坐标和尺寸 */
  414. $x = $minx;
  415. $y = abs($miny);
  416. $w = $maxx - $minx;
  417. $h = $maxy - $miny;
  418. /* 设定文字位置 */
  419. switch ($locate) {
  420. /* 右下角文字 */
  421. case self::IMAGE_WATER_SOUTHEAST:
  422. $x += $this->info['width'] - $w;
  423. $y += $this->info['height'] - $h;
  424. break;
  425. /* 左下角文字 */
  426. case self::IMAGE_WATER_SOUTHWEST:
  427. $y += $this->info['height'] - $h;
  428. break;
  429. /* 左上角文字 */
  430. case self::IMAGE_WATER_NORTHWEST:
  431. // 起始坐标即为左上角坐标,无需调整
  432. break;
  433. /* 右上角文字 */
  434. case self::IMAGE_WATER_NORTHEAST:
  435. $x += $this->info['width'] - $w;
  436. break;
  437. /* 居中文字 */
  438. case self::IMAGE_WATER_CENTER:
  439. $x += ($this->info['width'] - $w) / 2;
  440. $y += ($this->info['height'] - $h) / 2;
  441. break;
  442. /* 下居中文字 */
  443. case self::IMAGE_WATER_SOUTH:
  444. $x += ($this->info['width'] - $w) / 2;
  445. $y += $this->info['height'] - $h;
  446. break;
  447. /* 右居中文字 */
  448. case self::IMAGE_WATER_EAST:
  449. $x += $this->info['width'] - $w;
  450. $y += ($this->info['height'] - $h) / 2;
  451. break;
  452. /* 上居中文字 */
  453. case self::IMAGE_WATER_NORTH:
  454. $x += ($this->info['width'] - $w) / 2;
  455. break;
  456. /* 左居中文字 */
  457. case self::IMAGE_WATER_WEST:
  458. $y += ($this->info['height'] - $h) / 2;
  459. break;
  460. default:
  461. /* 自定义文字坐标 */
  462. if (is_array($locate)) {
  463. list($posx, $posy) = $locate;
  464. $x += $posx;
  465. $y += $posy;
  466. } else {
  467. throw new PhalApi_Exception_BadRequest(T('不支持的文字位置类型'));
  468. }
  469. }
  470. /* 设置偏移量 */
  471. if (is_array($offset)) {
  472. $offset = array_map('intval', $offset);
  473. list($ox, $oy) = $offset;
  474. } else {
  475. $offset = intval($offset);
  476. $ox = $oy = $offset;
  477. }
  478. /* 设置颜色 */
  479. if (is_string($color) && 0 === strpos($color, '#')) {
  480. $color = str_split(substr($color, 1), 2);
  481. $color = array_map('hexdec', $color);
  482. if (empty($color[3]) || $color[3] > 127) {
  483. $color[3] = 0;
  484. }
  485. } elseif (!is_array($color)) {
  486. throw new PhalApi_Exception_BadRequest(T('错误的颜色值'));
  487. }
  488. do {
  489. /* 写入文字 */
  490. $col = imagecolorallocatealpha($this->img, $color[0], $color[1], $color[2], $color[3]);
  491. imagettftext($this->img, $size, $angle, $x + $ox, $y + $oy, $col, $font, $text);
  492. } while (!empty($this->gif) && $this->gifNext());
  493. }
  494. /* 切换到GIF的下一帧并保存当前帧,内部使用 */
  495. private function gifNext() {
  496. ob_start();
  497. ob_implicit_flush(0);
  498. imagegif($this->img);
  499. $img = ob_get_clean();
  500. $this->gif->image($img);
  501. $next = $this->gif->nextImage();
  502. if ($next) {
  503. $this->img = imagecreatefromstring($next);
  504. return $next;
  505. } else {
  506. $this->img = imagecreatefromstring($this->gif->image());
  507. return false;
  508. }
  509. }
  510. /**
  511. * 析构方法,用于销毁图像资源
  512. */
  513. public function __destruct() {
  514. empty($this->img) || imagedestroy($this->img);
  515. }
  516. }