GIF.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568
  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. // | GIF.class.php 2013-03-09
  12. // +----------------------------------------------------------------------
  13. class Image_Driver_GIF{
  14. /**
  15. * GIF帧列表
  16. * @var array
  17. */
  18. private $frames = array();
  19. /**
  20. * 每帧等待时间列表
  21. * @var array
  22. */
  23. private $delays = array();
  24. /**
  25. * 构造方法,用于解码GIF图片
  26. * @param string $src GIF图片数据
  27. * @param string $mod 图片数据类型
  28. */
  29. public function __construct($src = null, $mod = 'url') {
  30. if(!is_null($src)){
  31. if('url' == $mod && is_file($src)){
  32. $src = file_get_contents($src);
  33. }
  34. /* 解码GIF图片 */
  35. try{
  36. $de = new GIFDecoder($src);
  37. $this->frames = $de->GIFGetFrames();
  38. $this->delays = $de->GIFGetDelays();
  39. } catch(\Exception $e){
  40. throw new PhalApi_Exception_BadRequest(T('解码GIF图片出错'));
  41. }
  42. }
  43. }
  44. /**
  45. * 设置或获取当前帧的数据
  46. * @param string $stream 二进制数据流
  47. * @return boolean 获取到的数据
  48. */
  49. public function image($stream = null){
  50. if(is_null($stream)){
  51. $current = current($this->frames);
  52. return false === $current ? reset($this->frames) : $current;
  53. } else {
  54. $this->frames[key($this->frames)] = $stream;
  55. }
  56. }
  57. /**
  58. * 将当前帧移动到下一帧
  59. * @return string 当前帧数据
  60. */
  61. public function nextImage(){
  62. return next($this->frames);
  63. }
  64. /**
  65. * 编码并保存当前GIF图片
  66. * @param string $gifname 图片名称
  67. */
  68. public function save($gifname){
  69. $gif = new GIFEncoder($this->frames, $this->delays, 0, 2, 0, 0, 0, 'bin');
  70. file_put_contents($gifname, $gif->GetAnimation());
  71. }
  72. }
  73. /*
  74. :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
  75. ::
  76. :: GIFEncoder Version 2.0 by László Zsidi, http://gifs.hu
  77. ::
  78. :: This class is a rewritten 'GifMerge.class.php' version.
  79. ::
  80. :: Modification:
  81. :: - Simplified and easy code,
  82. :: - Ultra fast encoding,
  83. :: - Built-in errors,
  84. :: - Stable working
  85. ::
  86. ::
  87. :: Updated at 2007. 02. 13. '00.05.AM'
  88. ::
  89. ::
  90. ::
  91. :: Try on-line GIFBuilder Form demo based on GIFEncoder.
  92. ::
  93. :: http://gifs.hu/phpclasses/demos/GifBuilder/
  94. ::
  95. :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
  96. */
  97. Class GIFEncoder {
  98. private $GIF = "GIF89a"; /* GIF header 6 bytes */
  99. private $VER = "GIFEncoder V2.05"; /* Encoder version */
  100. private $BUF = Array ( );
  101. private $LOP = 0;
  102. private $DIS = 2;
  103. private $COL = -1;
  104. private $IMG = -1;
  105. private $ERR = Array (
  106. 'ERR00' => "Does not supported function for only one image!",
  107. 'ERR01' => "Source is not a GIF image!",
  108. 'ERR02' => "Unintelligible flag ",
  109. 'ERR03' => "Does not make animation from animated GIF source",
  110. );
  111. /*
  112. :::::::::::::::::::::::::::::::::::::::::::::::::::
  113. ::
  114. :: GIFEncoder...
  115. ::
  116. */
  117. public function __construct($GIF_src, $GIF_dly, $GIF_lop, $GIF_dis,$GIF_red, $GIF_grn, $GIF_blu, $GIF_mod) {
  118. if ( ! is_array ( $GIF_src ) && ! is_array ( $GIF_dly ) ) {
  119. printf ( "%s: %s", $this->VER, $this->ERR [ 'ERR00' ] );
  120. exit ( 0 );
  121. }
  122. $this->LOP = ( $GIF_lop > -1 ) ? $GIF_lop : 0;
  123. $this->DIS = ( $GIF_dis > -1 ) ? ( ( $GIF_dis < 3 ) ? $GIF_dis : 3 ) : 2;
  124. $this->COL = ( $GIF_red > -1 && $GIF_grn > -1 && $GIF_blu > -1 ) ?
  125. ( $GIF_red | ( $GIF_grn << 8 ) | ( $GIF_blu << 16 ) ) : -1;
  126. for ( $i = 0; $i < count ( $GIF_src ); $i++ ) {
  127. if ( strToLower ( $GIF_mod ) == "url" ) {
  128. $this->BUF [ ] = fread ( fopen ( $GIF_src [ $i ], "rb" ), filesize ( $GIF_src [ $i ] ) );
  129. }
  130. else if ( strToLower ( $GIF_mod ) == "bin" ) {
  131. $this->BUF [ ] = $GIF_src [ $i ];
  132. }
  133. else {
  134. printf ( "%s: %s ( %s )!", $this->VER, $this->ERR [ 'ERR02' ], $GIF_mod );
  135. exit ( 0 );
  136. }
  137. if ( substr ( $this->BUF [ $i ], 0, 6 ) != "GIF87a" && substr ( $this->BUF [ $i ], 0, 6 ) != "GIF89a" ) {
  138. printf ( "%s: %d %s", $this->VER, $i, $this->ERR [ 'ERR01' ] );
  139. exit ( 0 );
  140. }
  141. for ( $j = ( 13 + 3 * ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) ) ), $k = TRUE; $k; $j++ ) {
  142. switch ( $this->BUF [ $i ] { $j } ) {
  143. case "!":
  144. if ( ( substr ( $this->BUF [ $i ], ( $j + 3 ), 8 ) ) == "NETSCAPE" ) {
  145. printf ( "%s: %s ( %s source )!", $this->VER, $this->ERR [ 'ERR03' ], ( $i + 1 ) );
  146. exit ( 0 );
  147. }
  148. break;
  149. case ";":
  150. $k = FALSE;
  151. break;
  152. }
  153. }
  154. }
  155. $this->GIFAddHeader ( );
  156. for ( $i = 0; $i < count ( $this->BUF ); $i++ ) {
  157. $this->GIFAddFrames ( $i, $GIF_dly [ $i ] );
  158. }
  159. $this->GIFAddFooter ( );
  160. }
  161. /*
  162. :::::::::::::::::::::::::::::::::::::::::::::::::::
  163. ::
  164. :: GIFAddHeader...
  165. ::
  166. */
  167. private function GIFAddHeader ( ) {
  168. $cmap = 0;
  169. if ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x80 ) {
  170. $cmap = 3 * ( 2 << ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x07 ) );
  171. $this->GIF .= substr ( $this->BUF [ 0 ], 6, 7 );
  172. $this->GIF .= substr ( $this->BUF [ 0 ], 13, $cmap );
  173. $this->GIF .= "!\377\13NETSCAPE2.0\3\1" . $this->GIFWord ( $this->LOP ) . "\0";
  174. }
  175. }
  176. /*
  177. :::::::::::::::::::::::::::::::::::::::::::::::::::
  178. ::
  179. :: GIFAddFrames...
  180. ::
  181. */
  182. private function GIFAddFrames ( $i, $d ) {
  183. $Locals_str = 13 + 3 * ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) );
  184. $Locals_end = strlen ( $this->BUF [ $i ] ) - $Locals_str - 1;
  185. $Locals_tmp = substr ( $this->BUF [ $i ], $Locals_str, $Locals_end );
  186. $Global_len = 2 << ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x07 );
  187. $Locals_len = 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 );
  188. $Global_rgb = substr ( $this->BUF [ 0 ], 13,
  189. 3 * ( 2 << ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x07 ) ) );
  190. $Locals_rgb = substr ( $this->BUF [ $i ], 13,
  191. 3 * ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) ) );
  192. $Locals_ext = "!\xF9\x04" . chr ( ( $this->DIS << 2 ) + 0 ) .
  193. chr ( ( $d >> 0 ) & 0xFF ) . chr ( ( $d >> 8 ) & 0xFF ) . "\x0\x0";
  194. if ( $this->COL > -1 && ord ( $this->BUF [ $i ] { 10 } ) & 0x80 ) {
  195. for ( $j = 0; $j < ( 2 << ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 ) ); $j++ ) {
  196. if (
  197. ord ( $Locals_rgb { 3 * $j + 0 } ) == ( ( $this->COL >> 16 ) & 0xFF ) &&
  198. ord ( $Locals_rgb { 3 * $j + 1 } ) == ( ( $this->COL >> 8 ) & 0xFF ) &&
  199. ord ( $Locals_rgb { 3 * $j + 2 } ) == ( ( $this->COL >> 0 ) & 0xFF )
  200. ) {
  201. $Locals_ext = "!\xF9\x04" . chr ( ( $this->DIS << 2 ) + 1 ) .
  202. chr ( ( $d >> 0 ) & 0xFF ) . chr ( ( $d >> 8 ) & 0xFF ) . chr ( $j ) . "\x0";
  203. break;
  204. }
  205. }
  206. }
  207. switch ( $Locals_tmp { 0 } ) {
  208. case "!":
  209. $Locals_img = substr ( $Locals_tmp, 8, 10 );
  210. $Locals_tmp = substr ( $Locals_tmp, 18, strlen ( $Locals_tmp ) - 18 );
  211. break;
  212. case ",":
  213. $Locals_img = substr ( $Locals_tmp, 0, 10 );
  214. $Locals_tmp = substr ( $Locals_tmp, 10, strlen ( $Locals_tmp ) - 10 );
  215. break;
  216. }
  217. if ( ord ( $this->BUF [ $i ] { 10 } ) & 0x80 && $this->IMG > -1 ) {
  218. if ( $Global_len == $Locals_len ) {
  219. if ( $this->GIFBlockCompare ( $Global_rgb, $Locals_rgb, $Global_len ) ) {
  220. $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_tmp );
  221. }
  222. else {
  223. $byte = ord ( $Locals_img { 9 } );
  224. $byte |= 0x80;
  225. $byte &= 0xF8;
  226. $byte |= ( ord ( $this->BUF [ 0 ] { 10 } ) & 0x07 );
  227. $Locals_img { 9 } = chr ( $byte );
  228. $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_rgb . $Locals_tmp );
  229. }
  230. }
  231. else {
  232. $byte = ord ( $Locals_img { 9 } );
  233. $byte |= 0x80;
  234. $byte &= 0xF8;
  235. $byte |= ( ord ( $this->BUF [ $i ] { 10 } ) & 0x07 );
  236. $Locals_img { 9 } = chr ( $byte );
  237. $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_rgb . $Locals_tmp );
  238. }
  239. }
  240. else {
  241. $this->GIF .= ( $Locals_ext . $Locals_img . $Locals_tmp );
  242. }
  243. $this->IMG = 1;
  244. }
  245. /*
  246. :::::::::::::::::::::::::::::::::::::::::::::::::::
  247. ::
  248. :: GIFAddFooter...
  249. ::
  250. */
  251. private function GIFAddFooter ( ) {
  252. $this->GIF .= ";";
  253. }
  254. /*
  255. :::::::::::::::::::::::::::::::::::::::::::::::::::
  256. ::
  257. :: GIFBlockCompare...
  258. ::
  259. */
  260. private function GIFBlockCompare ( $GlobalBlock, $LocalBlock, $Len ) {
  261. for ( $i = 0; $i < $Len; $i++ ) {
  262. if (
  263. $GlobalBlock { 3 * $i + 0 } != $LocalBlock { 3 * $i + 0 } ||
  264. $GlobalBlock { 3 * $i + 1 } != $LocalBlock { 3 * $i + 1 } ||
  265. $GlobalBlock { 3 * $i + 2 } != $LocalBlock { 3 * $i + 2 }
  266. ) {
  267. return ( 0 );
  268. }
  269. }
  270. return ( 1 );
  271. }
  272. /*
  273. :::::::::::::::::::::::::::::::::::::::::::::::::::
  274. ::
  275. :: GIFWord...
  276. ::
  277. */
  278. private function GIFWord ( $int ) {
  279. return ( chr ( $int & 0xFF ) . chr ( ( $int >> 8 ) & 0xFF ) );
  280. }
  281. /*
  282. :::::::::::::::::::::::::::::::::::::::::::::::::::
  283. ::
  284. :: GetAnimation...
  285. ::
  286. */
  287. public function GetAnimation ( ) {
  288. return ( $this->GIF );
  289. }
  290. }
  291. /*
  292. :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
  293. ::
  294. :: GIFDecoder Version 2.0 by László Zsidi, http://gifs.hu
  295. ::
  296. :: Created at 2007. 02. 01. '07.47.AM'
  297. ::
  298. ::
  299. ::
  300. ::
  301. :: Try on-line GIFBuilder Form demo based on GIFDecoder.
  302. ::
  303. :: http://gifs.hu/phpclasses/demos/GifBuilder/
  304. ::
  305. :::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
  306. */
  307. Class GIFDecoder {
  308. private $GIF_buffer = Array ( );
  309. private $GIF_arrays = Array ( );
  310. private $GIF_delays = Array ( );
  311. private $GIF_stream = "";
  312. private $GIF_string = "";
  313. private $GIF_bfseek = 0;
  314. private $GIF_screen = Array ( );
  315. private $GIF_global = Array ( );
  316. private $GIF_sorted;
  317. private $GIF_colorS;
  318. private $GIF_colorC;
  319. private $GIF_colorF;
  320. /*
  321. :::::::::::::::::::::::::::::::::::::::::::::::::::
  322. ::
  323. :: GIFDecoder ( $GIF_pointer )
  324. ::
  325. */
  326. public function __construct ( $GIF_pointer ) {
  327. $this->GIF_stream = $GIF_pointer;
  328. $this->GIFGetByte ( 6 ); // GIF89a
  329. $this->GIFGetByte ( 7 ); // Logical Screen Descriptor
  330. $this->GIF_screen = $this->GIF_buffer;
  331. $this->GIF_colorF = $this->GIF_buffer [ 4 ] & 0x80 ? 1 : 0;
  332. $this->GIF_sorted = $this->GIF_buffer [ 4 ] & 0x08 ? 1 : 0;
  333. $this->GIF_colorC = $this->GIF_buffer [ 4 ] & 0x07;
  334. $this->GIF_colorS = 2 << $this->GIF_colorC;
  335. if ( $this->GIF_colorF == 1 ) {
  336. $this->GIFGetByte ( 3 * $this->GIF_colorS );
  337. $this->GIF_global = $this->GIF_buffer;
  338. }
  339. /*
  340. *
  341. * 05.06.2007.
  342. * Made a little modification
  343. *
  344. *
  345. - for ( $cycle = 1; $cycle; ) {
  346. + if ( GIFDecoder::GIFGetByte ( 1 ) ) {
  347. - switch ( $this->GIF_buffer [ 0 ] ) {
  348. - case 0x21:
  349. - GIFDecoder::GIFReadExtensions ( );
  350. - break;
  351. - case 0x2C:
  352. - GIFDecoder::GIFReadDescriptor ( );
  353. - break;
  354. - case 0x3B:
  355. - $cycle = 0;
  356. - break;
  357. - }
  358. - }
  359. + else {
  360. + $cycle = 0;
  361. + }
  362. - }
  363. */
  364. for ( $cycle = 1; $cycle; ) {
  365. if ( $this->GIFGetByte ( 1 ) ) {
  366. switch ( $this->GIF_buffer [ 0 ] ) {
  367. case 0x21:
  368. $this->GIFReadExtensions ( );
  369. break;
  370. case 0x2C:
  371. $this->GIFReadDescriptor ( );
  372. break;
  373. case 0x3B:
  374. $cycle = 0;
  375. break;
  376. }
  377. }
  378. else {
  379. $cycle = 0;
  380. }
  381. }
  382. }
  383. /*
  384. :::::::::::::::::::::::::::::::::::::::::::::::::::
  385. ::
  386. :: GIFReadExtension ( )
  387. ::
  388. */
  389. private function GIFReadExtensions ( ) {
  390. $this->GIFGetByte ( 1 );
  391. for ( ; ; ) {
  392. $this->GIFGetByte ( 1 );
  393. if ( ( $u = $this->GIF_buffer [ 0 ] ) == 0x00 ) {
  394. break;
  395. }
  396. $this->GIFGetByte ( $u );
  397. /*
  398. * 07.05.2007.
  399. * Implemented a new line for a new function
  400. * to determine the originaly delays between
  401. * frames.
  402. *
  403. */
  404. if ( $u == 4 ) {
  405. $this->GIF_delays [ ] = ( $this->GIF_buffer [ 1 ] | $this->GIF_buffer [ 2 ] << 8 );
  406. }
  407. }
  408. }
  409. /*
  410. :::::::::::::::::::::::::::::::::::::::::::::::::::
  411. ::
  412. :: GIFReadExtension ( )
  413. ::
  414. */
  415. private function GIFReadDescriptor ( ) {
  416. $GIF_screen = Array ( );
  417. $this->GIFGetByte ( 9 );
  418. $GIF_screen = $this->GIF_buffer;
  419. $GIF_colorF = $this->GIF_buffer [ 8 ] & 0x80 ? 1 : 0;
  420. if ( $GIF_colorF ) {
  421. $GIF_code = $this->GIF_buffer [ 8 ] & 0x07;
  422. $GIF_sort = $this->GIF_buffer [ 8 ] & 0x20 ? 1 : 0;
  423. }
  424. else {
  425. $GIF_code = $this->GIF_colorC;
  426. $GIF_sort = $this->GIF_sorted;
  427. }
  428. $GIF_size = 2 << $GIF_code;
  429. $this->GIF_screen [ 4 ] &= 0x70;
  430. $this->GIF_screen [ 4 ] |= 0x80;
  431. $this->GIF_screen [ 4 ] |= $GIF_code;
  432. if ( $GIF_sort ) {
  433. $this->GIF_screen [ 4 ] |= 0x08;
  434. }
  435. $this->GIF_string = "GIF87a";
  436. $this->GIFPutByte ( $this->GIF_screen );
  437. if ( $GIF_colorF == 1 ) {
  438. $this->GIFGetByte ( 3 * $GIF_size );
  439. $this->GIFPutByte ( $this->GIF_buffer );
  440. }
  441. else {
  442. $this->GIFPutByte ( $this->GIF_global );
  443. }
  444. $this->GIF_string .= chr ( 0x2C );
  445. $GIF_screen [ 8 ] &= 0x40;
  446. $this->GIFPutByte ( $GIF_screen );
  447. $this->GIFGetByte ( 1 );
  448. $this->GIFPutByte ( $this->GIF_buffer );
  449. for ( ; ; ) {
  450. $this->GIFGetByte ( 1 );
  451. $this->GIFPutByte ( $this->GIF_buffer );
  452. if ( ( $u = $this->GIF_buffer [ 0 ] ) == 0x00 ) {
  453. break;
  454. }
  455. $this->GIFGetByte ( $u );
  456. $this->GIFPutByte ( $this->GIF_buffer );
  457. }
  458. $this->GIF_string .= chr ( 0x3B );
  459. /*
  460. Add frames into $GIF_stream array...
  461. */
  462. $this->GIF_arrays [ ] = $this->GIF_string;
  463. }
  464. /*
  465. :::::::::::::::::::::::::::::::::::::::::::::::::::
  466. ::
  467. :: GIFGetByte ( $len )
  468. ::
  469. */
  470. /*
  471. *
  472. * 05.06.2007.
  473. * Made a little modification
  474. *
  475. *
  476. - function GIFGetByte ( $len ) {
  477. - $this->GIF_buffer = Array ( );
  478. -
  479. - for ( $i = 0; $i < $len; $i++ ) {
  480. + if ( $this->GIF_bfseek > strlen ( $this->GIF_stream ) ) {
  481. + return 0;
  482. + }
  483. - $this->GIF_buffer [ ] = ord ( $this->GIF_stream { $this->GIF_bfseek++ } );
  484. - }
  485. + return 1;
  486. - }
  487. */
  488. private function GIFGetByte ( $len ) {
  489. $this->GIF_buffer = Array ( );
  490. for ( $i = 0; $i < $len; $i++ ) {
  491. if ( $this->GIF_bfseek > strlen ( $this->GIF_stream ) ) {
  492. return 0;
  493. }
  494. $this->GIF_buffer [ ] = ord ( $this->GIF_stream { $this->GIF_bfseek++ } );
  495. }
  496. return 1;
  497. }
  498. /*
  499. :::::::::::::::::::::::::::::::::::::::::::::::::::
  500. ::
  501. :: GIFPutByte ( $bytes )
  502. ::
  503. */
  504. private function GIFPutByte ( $bytes ) {
  505. for ( $i = 0; $i < count ( $bytes ); $i++ ) {
  506. $this->GIF_string .= chr ( $bytes [ $i ] );
  507. }
  508. }
  509. /*
  510. :::::::::::::::::::::::::::::::::::::::::::::::::::
  511. ::
  512. :: PUBLIC FUNCTIONS
  513. ::
  514. ::
  515. :: GIFGetFrames ( )
  516. ::
  517. */
  518. public function GIFGetFrames ( ) {
  519. return ( $this->GIF_arrays );
  520. }
  521. /*
  522. :::::::::::::::::::::::::::::::::::::::::::::::::::
  523. ::
  524. :: GIFGetDelays ( )
  525. ::
  526. */
  527. public function GIFGetDelays ( ) {
  528. return ( $this->GIF_delays );
  529. }
  530. }