Imagick.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664
  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. // | ImageImagick.class.php 2013-03-06
  12. // +----------------------------------------------------------------------
  13. class Image_Driver_Imagick {
  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. empty($this->img) || $this->img->destroy();
  64. //载入图像
  65. $this->img = new \Imagick(realpath($imgname));
  66. //设置图像信息
  67. $this->info = array(
  68. 'width' => $this->img->getImageWidth(),
  69. 'height' => $this->img->getImageHeight(),
  70. 'type' => strtolower($this->img->getImageFormat()),
  71. 'mime' => $this->img->getImageMimeType(),
  72. );
  73. }
  74. /**
  75. * 保存图像
  76. *
  77. * @param string $imgname 图像保存名称
  78. * @param string $type 图像类型
  79. * @param integer $quality JPEG图像质量
  80. * @param boolean $interlace 是否对JPEG类型图像设置隔行扫描
  81. */
  82. public function save($imgname, $type = null, $quality = 80, $interlace = true) {
  83. if (empty($this->img)) {
  84. throw new PhalApi_Exception_BadRequest(T('没有可以被保存的图像资源'));
  85. }
  86. //设置图片类型
  87. if (is_null($type)) {
  88. $type = $this->info['type'];
  89. } else {
  90. $type = strtolower($type);
  91. $this->img->setImageFormat($type);
  92. }
  93. //JPEG图像设置隔行扫描
  94. if ('jpeg' == $type || 'jpg' == $type) {
  95. $this->img->setImageInterlaceScheme(1);
  96. }
  97. // 设置图像质量
  98. $this->img->setImageCompressionQuality($quality);
  99. //去除图像配置信息
  100. $this->img->stripImage();
  101. //保存图像
  102. $imgname = realpath(dirname($imgname)) . '/' . basename($imgname); //强制绝对路径
  103. if ('gif' == $type) {
  104. $this->img->writeImages($imgname, true);
  105. } else {
  106. $this->img->writeImage($imgname);
  107. }
  108. }
  109. /**
  110. * 返回图像宽度
  111. * @return integer 图像宽度
  112. */
  113. public function width() {
  114. if (empty($this->img)) {
  115. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  116. }
  117. return $this->info['width'];
  118. }
  119. /**
  120. * 返回图像高度
  121. * @return integer 图像高度
  122. */
  123. public function height() {
  124. if (empty($this->img)) {
  125. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  126. }
  127. return $this->info['height'];
  128. }
  129. /**
  130. * 返回图像类型
  131. * @return string 图像类型
  132. */
  133. public function type() {
  134. if (empty($this->img)) {
  135. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  136. }
  137. return $this->info['type'];
  138. }
  139. /**
  140. * 返回图像MIME类型
  141. * @return string 图像MIME类型
  142. */
  143. public function mime() {
  144. if (empty($this->img)) {
  145. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  146. }
  147. return $this->info['mime'];
  148. }
  149. /**
  150. * 返回图像尺寸数组 0 - 图像宽度,1 - 图像高度
  151. * @return array 图像尺寸
  152. */
  153. public function size() {
  154. if (empty($this->img)) {
  155. throw new PhalApi_Exception_BadRequest(T('没有指定图像资源'));
  156. }
  157. return array($this->info['width'], $this->info['height']);
  158. }
  159. /**
  160. * 裁剪图像
  161. *
  162. * @param integer $w 裁剪区域宽度
  163. * @param integer $h 裁剪区域高度
  164. * @param integer $x 裁剪区域x坐标
  165. * @param integer $y 裁剪区域y坐标
  166. * @param integer $width 图像保存宽度
  167. * @param integer $height 图像保存高度
  168. */
  169. public function crop($w, $h, $x = 0, $y = 0, $width = null, $height = null) {
  170. if (empty($this->img)) {
  171. throw new PhalApi_Exception_BadRequest(T('没有可以被裁剪的图像资源'));
  172. }
  173. //设置保存尺寸
  174. empty($width) && $width = $w;
  175. empty($height) && $height = $h;
  176. //裁剪图片
  177. if ('gif' == $this->info['type']) {
  178. $img = $this->img->coalesceImages();
  179. $this->img->destroy(); //销毁原图
  180. //循环裁剪每一帧
  181. do {
  182. $this->_crop($w, $h, $x, $y, $width, $height, $img);
  183. } while ($img->nextImage());
  184. //压缩图片
  185. $this->img = $img->deconstructImages();
  186. $img->destroy(); //销毁零时图片
  187. } else {
  188. $this->_crop($w, $h, $x, $y, $width, $height);
  189. }
  190. }
  191. /* 裁剪图片,内部调用 */
  192. private function _crop($w, $h, $x, $y, $width, $height, $img = null) {
  193. is_null($img) && $img = $this->img;
  194. //裁剪
  195. $info = $this->info;
  196. if ($x != 0 || $y != 0 || $w != $info['width'] || $h != $info['height']) {
  197. $img->cropImage($w, $h, $x, $y);
  198. $img->setImagePage($w, $h, 0, 0); //调整画布和图片一致
  199. }
  200. //调整大小
  201. if ($w != $width || $h != $height) {
  202. $img->sampleImage($width, $height);
  203. }
  204. //设置缓存尺寸
  205. $this->info['width'] = $width;
  206. $this->info['height'] = $height;
  207. }
  208. /**
  209. * 生成缩略图
  210. *
  211. * @param integer $width 缩略图最大宽度
  212. * @param integer $height 缩略图最大高度
  213. * @param integer $type 缩略图裁剪类型
  214. */
  215. public function thumb($width, $height, $type = self::IMAGE_THUMB_SCALE) {
  216. if (empty($this->img)) {
  217. throw new PhalApi_Exception_BadRequest(T('没有可以被缩略的图像资源'));
  218. }
  219. //原图宽度和高度
  220. $w = $this->info['width'];
  221. $h = $this->info['height'];
  222. /* 计算缩略图生成的必要参数 */
  223. switch ($type) {
  224. /* 等比例缩放 */
  225. case self::IMAGE_THUMB_SCALE:
  226. //原图尺寸小于缩略图尺寸则不进行缩略
  227. if ($w < $width && $h < $height) {
  228. return;
  229. }
  230. //计算缩放比例
  231. $scale = min($width / $w, $height / $h);
  232. //设置缩略图的坐标及宽度和高度
  233. $x = $y = 0;
  234. $width = $w * $scale;
  235. $height = $h * $scale;
  236. break;
  237. /* 居中裁剪 */
  238. case self::IMAGE_THUMB_CENTER:
  239. //计算缩放比例
  240. $scale = max($width / $w, $height / $h);
  241. //设置缩略图的坐标及宽度和高度
  242. $w = $width / $scale;
  243. $h = $height / $scale;
  244. $x = ($this->info['width'] - $w) / 2;
  245. $y = ($this->info['height'] - $h) / 2;
  246. break;
  247. /* 左上角裁剪 */
  248. case self::IMAGE_THUMB_NORTHWEST:
  249. //计算缩放比例
  250. $scale = max($width / $w, $height / $h);
  251. //设置缩略图的坐标及宽度和高度
  252. $x = $y = 0;
  253. $w = $width / $scale;
  254. $h = $height / $scale;
  255. break;
  256. /* 右下角裁剪 */
  257. case self::IMAGE_THUMB_SOUTHEAST:
  258. //计算缩放比例
  259. $scale = max($width / $w, $height / $h);
  260. //设置缩略图的坐标及宽度和高度
  261. $w = $width / $scale;
  262. $h = $height / $scale;
  263. $x = $this->info['width'] - $w;
  264. $y = $this->info['height'] - $h;
  265. break;
  266. /* 填充 */
  267. case self::IMAGE_THUMB_FILLED:
  268. //计算缩放比例
  269. if ($w < $width && $h < $height) {
  270. $scale = 1;
  271. } else {
  272. $scale = min($width / $w, $height / $h);
  273. }
  274. //设置缩略图的坐标及宽度和高度
  275. $neww = $w * $scale;
  276. $newh = $h * $scale;
  277. $posx = ($width - $w * $scale) / 2;
  278. $posy = ($height - $h * $scale) / 2;
  279. //创建一张新图像
  280. $newimg = new \Imagick();
  281. $newimg->newImage($width, $height, 'white', $this->info['type']);
  282. if ('gif' == $this->info['type']) {
  283. $imgs = $this->img->coalesceImages();
  284. $img = new \Imagick();
  285. $this->img->destroy(); //销毁原图
  286. //循环填充每一帧
  287. do {
  288. //填充图像
  289. $image = $this->_fill($newimg, $posx, $posy, $neww, $newh, $imgs);
  290. $img->addImage($image);
  291. $img->setImageDelay($imgs->getImageDelay());
  292. $img->setImagePage($width, $height, 0, 0);
  293. $image->destroy(); //销毁零时图片
  294. } while ($imgs->nextImage());
  295. //压缩图片
  296. $this->img->destroy();
  297. $this->img = $img->deconstructImages();
  298. $imgs->destroy(); //销毁零时图片
  299. $img->destroy(); //销毁零时图片
  300. } else {
  301. //填充图像
  302. $img = $this->_fill($newimg, $posx, $posy, $neww, $newh);
  303. //销毁原图
  304. $this->img->destroy();
  305. $this->img = $img;
  306. }
  307. //设置新图像属性
  308. $this->info['width'] = $width;
  309. $this->info['height'] = $height;
  310. return;
  311. /* 固定 */
  312. case self::IMAGE_THUMB_FIXED:
  313. $x = $y = 0;
  314. break;
  315. default:
  316. throw new PhalApi_Exception_BadRequest(T('不支持的缩略图裁剪类型'));
  317. }
  318. /* 裁剪图像 */
  319. $this->crop($w, $h, $x, $y, $width, $height);
  320. }
  321. /* 填充指定图像,内部使用 */
  322. private function _fill($newimg, $posx, $posy, $neww, $newh, $img = null) {
  323. is_null($img) && $img = $this->img;
  324. /* 将指定图片绘入空白图片 */
  325. $draw = new \ImagickDraw();
  326. $draw->composite($img->getImageCompose(), $posx, $posy, $neww, $newh, $img);
  327. $image = $newimg->clone();
  328. $image->drawImage($draw);
  329. $draw->destroy();
  330. return $image;
  331. }
  332. /**
  333. * 添加水印
  334. *
  335. * @param string $source 水印图片路径
  336. * @param integer $locate 水印位置
  337. * @param integer $alpha 水印透明度
  338. */
  339. public function water($source, $locate = self::IMAGE_WATER_SOUTHEAST, $alpha = 80) {
  340. //资源检测
  341. if (empty($this->img)) {
  342. throw new PhalApi_Exception_BadRequest(T('没有可以被添加水印的图像资源'));
  343. }
  344. if (!is_file($source)) {
  345. throw new PhalApi_Exception_BadRequest(T('水印图像不存在'));
  346. }
  347. //创建水印图像资源
  348. $water = new \Imagick(realpath($source));
  349. $info = array($water->getImageWidth(), $water->getImageHeight());
  350. /* 设定水印位置 */
  351. switch ($locate) {
  352. /* 右下角水印 */
  353. case self::IMAGE_WATER_SOUTHEAST:
  354. $x = $this->info['width'] - $info[0];
  355. $y = $this->info['height'] - $info[1];
  356. break;
  357. /* 左下角水印 */
  358. case self::IMAGE_WATER_SOUTHWEST:
  359. $x = 0;
  360. $y = $this->info['height'] - $info[1];
  361. break;
  362. /* 左上角水印 */
  363. case self::IMAGE_WATER_NORTHWEST:
  364. $x = $y = 0;
  365. break;
  366. /* 右上角水印 */
  367. case self::IMAGE_WATER_NORTHEAST:
  368. $x = $this->info['width'] - $info[0];
  369. $y = 0;
  370. break;
  371. /* 居中水印 */
  372. case self::IMAGE_WATER_CENTER:
  373. $x = ($this->info['width'] - $info[0]) / 2;
  374. $y = ($this->info['height'] - $info[1]) / 2;
  375. break;
  376. /* 下居中水印 */
  377. case self::IMAGE_WATER_SOUTH:
  378. $x = ($this->info['width'] - $info[0]) / 2;
  379. $y = $this->info['height'] - $info[1];
  380. break;
  381. /* 右居中水印 */
  382. case self::IMAGE_WATER_EAST:
  383. $x = $this->info['width'] - $info[0];
  384. $y = ($this->info['height'] - $info[1]) / 2;
  385. break;
  386. /* 上居中水印 */
  387. case self::IMAGE_WATER_NORTH:
  388. $x = ($this->info['width'] - $info[0]) / 2;
  389. $y = 0;
  390. break;
  391. /* 左居中水印 */
  392. case self::IMAGE_WATER_WEST:
  393. $x = 0;
  394. $y = ($this->info['height'] - $info[1]) / 2;
  395. break;
  396. default:
  397. /* 自定义水印坐标 */
  398. if (is_array($locate)) {
  399. list($x, $y) = $locate;
  400. } else {
  401. throw new PhalApi_Exception_BadRequest(T('不支持的水印位置类型'));
  402. }
  403. }
  404. //创建绘图资源
  405. $draw = new \ImagickDraw();
  406. $draw->composite($water->getImageCompose(), $x, $y, $info[0], $info[1], $water);
  407. if ('gif' == $this->info['type']) {
  408. $img = $this->img->coalesceImages();
  409. $this->img->destroy(); //销毁原图
  410. do {
  411. //添加水印
  412. $img->drawImage($draw);
  413. } while ($img->nextImage());
  414. //压缩图片
  415. $this->img = $img->deconstructImages();
  416. $img->destroy(); //销毁零时图片
  417. } else {
  418. //添加水印
  419. $this->img->drawImage($draw);
  420. }
  421. //销毁水印资源
  422. $draw->destroy();
  423. $water->destroy();
  424. }
  425. /**
  426. * 图像添加文字
  427. *
  428. * @param string $text 添加的文字
  429. * @param string $font 字体路径
  430. * @param integer $size 字号
  431. * @param string $color 文字颜色
  432. * @param integer $locate 文字写入位置
  433. * @param integer $offset 文字相对当前位置的偏移量
  434. * @param integer $angle 文字倾斜角度
  435. */
  436. public function text($text, $font, $size, $color = '#00000000', $locate = self::IMAGE_WATER_SOUTHEAST, $offset = 0, $angle = 0) {
  437. //资源检测
  438. if (empty($this->img)) {
  439. throw new PhalApi_Exception_BadRequest(T('没有可以被写入文字的图像资源'));
  440. }
  441. if (!is_file($font)) {
  442. throw new PhalApi_Exception_BadRequest(T("不存在的字体文件:{$font}"));
  443. }
  444. //获取颜色和透明度
  445. if (is_array($color)) {
  446. $color = array_map('dechex', $color);
  447. foreach ($color as &$value) {
  448. $value = str_pad($value, 2, '0', STR_PAD_LEFT);
  449. }
  450. $color = '#' . implode('', $color);
  451. } elseif (!is_string($color) || 0 !== strpos($color, '#')) {
  452. throw new PhalApi_Exception_BadRequest(T('错误的颜色值'));
  453. }
  454. $col = substr($color, 0, 7);
  455. $alp = strlen($color) == 9 ? substr($color, -2) : 0;
  456. //获取文字信息
  457. $draw = new \ImagickDraw();
  458. $draw->setFont(realpath($font));
  459. $draw->setFontSize($size);
  460. $draw->setFillColor($col);
  461. $draw->setFillAlpha(1 - hexdec($alp) / 127);
  462. $draw->setTextAntialias(true);
  463. $draw->setStrokeAntialias(true);
  464. $metrics = $this->img->queryFontMetrics($draw, $text);
  465. /* 计算文字初始坐标和尺寸 */
  466. $x = 0;
  467. $y = $metrics['ascender'];
  468. $w = $metrics['textWidth'];
  469. $h = $metrics['textHeight'];
  470. /* 设定文字位置 */
  471. switch ($locate) {
  472. /* 右下角文字 */
  473. case self::IMAGE_WATER_SOUTHEAST:
  474. $x += $this->info['width'] - $w;
  475. $y += $this->info['height'] - $h;
  476. break;
  477. /* 左下角文字 */
  478. case self::IMAGE_WATER_SOUTHWEST:
  479. $y += $this->info['height'] - $h;
  480. break;
  481. /* 左上角文字 */
  482. case self::IMAGE_WATER_NORTHWEST:
  483. // 起始坐标即为左上角坐标,无需调整
  484. break;
  485. /* 右上角文字 */
  486. case self::IMAGE_WATER_NORTHEAST:
  487. $x += $this->info['width'] - $w;
  488. break;
  489. /* 居中文字 */
  490. case self::IMAGE_WATER_CENTER:
  491. $x += ($this->info['width'] - $w) / 2;
  492. $y += ($this->info['height'] - $h) / 2;
  493. break;
  494. /* 下居中文字 */
  495. case self::IMAGE_WATER_SOUTH:
  496. $x += ($this->info['width'] - $w) / 2;
  497. $y += $this->info['height'] - $h;
  498. break;
  499. /* 右居中文字 */
  500. case self::IMAGE_WATER_EAST:
  501. $x += $this->info['width'] - $w;
  502. $y += ($this->info['height'] - $h) / 2;
  503. break;
  504. /* 上居中文字 */
  505. case self::IMAGE_WATER_NORTH:
  506. $x += ($this->info['width'] - $w) / 2;
  507. break;
  508. /* 左居中文字 */
  509. case self::IMAGE_WATER_WEST:
  510. $y += ($this->info['height'] - $h) / 2;
  511. break;
  512. default:
  513. /* 自定义文字坐标 */
  514. if (is_array($locate)) {
  515. list($posx, $posy) = $locate;
  516. $x += $posx;
  517. $y += $posy;
  518. } else {
  519. throw new PhalApi_Exception_BadRequest(T('不支持的文字位置类型'));
  520. }
  521. }
  522. /* 设置偏移量 */
  523. if (is_array($offset)) {
  524. $offset = array_map('intval', $offset);
  525. list($ox, $oy) = $offset;
  526. } else {
  527. $offset = intval($offset);
  528. $ox = $oy = $offset;
  529. }
  530. /* 写入文字 */
  531. if ('gif' == $this->info['type']) {
  532. $img = $this->img->coalesceImages();
  533. $this->img->destroy(); //销毁原图
  534. do {
  535. $img->annotateImage($draw, $x + $ox, $y + $oy, $angle, $text);
  536. } while ($img->nextImage());
  537. //压缩图片
  538. $this->img = $img->deconstructImages();
  539. $img->destroy(); //销毁零时图片
  540. } else {
  541. $this->img->annotateImage($draw, $x + $ox, $y + $oy, $angle, $text);
  542. }
  543. $draw->destroy();
  544. }
  545. /**
  546. * 析构方法,用于销毁图像资源
  547. */
  548. public function __destruct() {
  549. empty($this->img) || $this->img->destroy();
  550. }
  551. }