Total Complexity | 62 |
Total Lines | 643 |
Duplicated Lines | 0 % |
Changes | 1 | ||
Bugs | 0 | Features | 0 |
Complex classes like Image_GD often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use Image_GD, and based on these observations, apply Extract Interface, too.
1 | <?php |
||
17 | class Image_GD extends Image implements DriverInterface |
||
18 | { |
||
19 | |||
20 | // Which GD functions are available? |
||
21 | const IMAGEROTATE = 'imagerotate'; |
||
22 | const IMAGECONVOLUTION = 'imageconvolution'; |
||
23 | const IMAGEFILTER = 'imagefilter'; |
||
24 | const IMAGELAYEREFFECT = 'imagelayereffect'; |
||
25 | protected static $_available_functions = array(); |
||
26 | |||
27 | /** |
||
28 | * Checks if GD is enabled and verify that key methods exist, some of which require GD to |
||
29 | * be bundled with PHP. Exceptions will be thrown from those methods when GD is not |
||
30 | * bundled. |
||
31 | * |
||
32 | * @return boolean |
||
33 | * @throws ErrorException |
||
34 | */ |
||
35 | public static function check() |
||
36 | { |
||
37 | if (!function_exists('gd_info')) { |
||
38 | throw new ErrorException('GD is either not installed or not enabled, check your configuration'); |
||
39 | } |
||
40 | $functions = array( |
||
41 | Image_GD::IMAGEROTATE, |
||
42 | Image_GD::IMAGECONVOLUTION, |
||
43 | Image_GD::IMAGEFILTER, |
||
44 | Image_GD::IMAGELAYEREFFECT |
||
45 | ); |
||
46 | foreach ($functions as $function) { |
||
47 | Image_GD::$_available_functions[$function] = function_exists($function); |
||
48 | } |
||
49 | |||
50 | if (defined('GD_VERSION')) { |
||
51 | // Get the version via a constant, available in PHP 5.2.4+ |
||
52 | $version = GD_VERSION; |
||
53 | } else { |
||
54 | // Get the version information |
||
55 | $info = gd_info(); |
||
56 | |||
57 | // Extract the version number |
||
58 | preg_match('/\d+\.\d+(?:\.\d+)?/', $info['GD Version'], $matches); |
||
59 | |||
60 | // Get the major version |
||
61 | $version = $matches[0]; |
||
62 | } |
||
63 | |||
64 | if (!version_compare($version, '2.0.1', '>=')) { |
||
65 | throw new ErrorException(sprintf('Image_GD requires GD version 2.0.1 or greater, you have %s', $version)); |
||
66 | } |
||
67 | |||
68 | return Image_GD::$_checked = true; |
||
69 | } |
||
70 | |||
71 | /* @var resource Temporary image resource */ |
||
72 | protected $_image; |
||
73 | |||
74 | /* @var string Function name to open Image */ |
||
75 | protected $_create_function; |
||
76 | |||
77 | /** |
||
78 | * Runs [Image_GD::check] and loads the image. |
||
79 | * |
||
80 | * @param string $file image file path |
||
81 | * @return void |
||
82 | * @throws ErrorException |
||
83 | */ |
||
84 | public function __construct($file) |
||
85 | { |
||
86 | if (!Image_GD::$_checked) { |
||
87 | // Run the install check |
||
88 | Image_GD::check(); |
||
89 | } |
||
90 | |||
91 | parent::__construct($file); |
||
92 | |||
93 | // Set the image creation function name |
||
94 | switch ($this->type) { |
||
95 | case IMAGETYPE_JPEG: |
||
96 | $create = 'imagecreatefromjpeg'; |
||
97 | break; |
||
98 | case IMAGETYPE_GIF: |
||
99 | $create = 'imagecreatefromgif'; |
||
100 | break; |
||
101 | case IMAGETYPE_PNG: |
||
102 | $create = 'imagecreatefrompng'; |
||
103 | break; |
||
104 | } |
||
105 | |||
106 | if (!isset($create) || !function_exists($create)) { |
||
107 | throw new ErrorException(sprintf('Installed GD does not support %s images', image_type_to_extension($this->type, false))); |
||
108 | } |
||
109 | |||
110 | // Save function for future use |
||
111 | $this->_create_function = $create; |
||
112 | |||
113 | // Save filename for lazy loading |
||
114 | $this->_image = $this->file; |
||
|
|||
115 | } |
||
116 | |||
117 | /** |
||
118 | * Destroys the loaded image to free up resources. |
||
119 | * |
||
120 | * @return void |
||
121 | */ |
||
122 | public function __destruct() |
||
123 | { |
||
124 | if (is_resource($this->_image)) { |
||
125 | // Free all resources |
||
126 | imagedestroy($this->_image); |
||
127 | } |
||
128 | } |
||
129 | |||
130 | /** |
||
131 | * Loads an image into GD. |
||
132 | * |
||
133 | * @return void |
||
134 | */ |
||
135 | protected function _load_image() |
||
136 | { |
||
137 | if (!is_resource($this->_image)) { |
||
138 | // Gets create function |
||
139 | $create = $this->_create_function; |
||
140 | |||
141 | // Open the temporary image |
||
142 | $this->_image = $create($this->file); |
||
143 | |||
144 | // Preserve transparency when saving |
||
145 | imagesavealpha($this->_image, true); |
||
146 | } |
||
147 | } |
||
148 | |||
149 | /** |
||
150 | * Execute a resize. |
||
151 | * |
||
152 | * @param integer $width new width |
||
153 | * @param integer $height new height |
||
154 | * @return void |
||
155 | */ |
||
156 | public function _do_resize($width, $height) |
||
199 | } |
||
200 | } |
||
201 | |||
202 | /** |
||
203 | * Adaptation the image. |
||
204 | * |
||
205 | * @param integer $width image width |
||
206 | * @param integer $height image height |
||
207 | * @param integer $bg_width background width |
||
208 | * @param integer $bg_height background height |
||
209 | * @param integer $offset_x offset from the left |
||
210 | * @param integer $offset_y offset from the top |
||
211 | */ |
||
212 | public function _do_adapt($width, $height, $bg_width, $bg_height, $offset_x, $offset_y) |
||
213 | { |
||
214 | $this->_load_image(); |
||
215 | $image = $this->_image; |
||
216 | $this->_image = $this->_create($bg_width, $bg_height); |
||
217 | $this->width = $bg_width; |
||
218 | $this->height = $bg_height; |
||
219 | imagealphablending($this->_image, false); |
||
220 | $col = imagecolorallocatealpha($this->_image, 0, 255, 0, 127); |
||
221 | imagefilledrectangle($this->_image, 0, 0, $bg_width, $bg_height, $col); |
||
222 | imagealphablending($this->_image, true); |
||
223 | imagecopy($this->_image, $image, $offset_x, $offset_y, 0, 0, $width, $height); |
||
224 | imagealphablending($this->_image, false); |
||
225 | imagesavealpha($this->_image, true); |
||
226 | imagedestroy($image); |
||
227 | } |
||
228 | |||
229 | /** |
||
230 | * Execute a crop. |
||
231 | * |
||
232 | * @param integer $width new width |
||
233 | * @param integer $height new height |
||
234 | * @param integer $offset_x offset from the left |
||
235 | * @param integer $offset_y offset from the top |
||
236 | * @return void |
||
237 | */ |
||
238 | public function _do_crop($width, $height, $offset_x, $offset_y) |
||
239 | { |
||
240 | // Create the temporary image to copy to |
||
241 | $image = $this->_create($width, $height); |
||
242 | |||
243 | // Loads image if not yet loaded |
||
244 | $this->_load_image(); |
||
245 | |||
246 | // Execute the crop |
||
247 | if (imagecopyresampled($image, $this->_image, 0, 0, $offset_x, $offset_y, $width, $height, $width, $height)) { |
||
248 | // Swap the new image for the old one |
||
249 | imagedestroy($this->_image); |
||
250 | $this->_image = $image; |
||
251 | |||
252 | // Reset the width and height |
||
253 | $this->width = imagesx($image); |
||
254 | $this->height = imagesy($image); |
||
255 | } |
||
256 | } |
||
257 | |||
258 | /** |
||
259 | * Execute a rotation. |
||
260 | * |
||
261 | * @param integer $degrees degrees to rotate |
||
262 | * @return void |
||
263 | * @throws ErrorException |
||
264 | */ |
||
265 | public function _do_rotate($degrees) |
||
266 | { |
||
267 | if (empty(Image_GD::$_available_functions[Image_GD::IMAGEROTATE])) { |
||
268 | throw new ErrorException('This method requires imagerotate, which is only available in the bundled version of GD'); |
||
269 | } |
||
270 | |||
271 | // Loads image if not yet loaded |
||
272 | $this->_load_image(); |
||
273 | |||
274 | // Transparent black will be used as the background for the uncovered region |
||
275 | $transparent = imagecolorallocatealpha($this->_image, 0, 0, 0, 127); |
||
276 | |||
277 | // Rotate, setting the transparent color |
||
278 | $image = imagerotate($this->_image, 360 - $degrees, $transparent, 1); |
||
279 | |||
280 | // Save the alpha of the rotated image |
||
281 | imagesavealpha($image, true); |
||
282 | |||
283 | // Get the width and height of the rotated image |
||
284 | $width = imagesx($image); |
||
285 | $height = imagesy($image); |
||
286 | |||
287 | if (imagecopymerge($this->_image, $image, 0, 0, 0, 0, $width, $height, 100)) { |
||
288 | // Swap the new image for the old one |
||
289 | imagedestroy($this->_image); |
||
290 | $this->_image = $image; |
||
291 | |||
292 | // Reset the width and height |
||
293 | $this->width = $width; |
||
294 | $this->height = $height; |
||
295 | } |
||
296 | } |
||
297 | |||
298 | /** |
||
299 | * Execute a flip. |
||
300 | * |
||
301 | * @param integer $direction direction to flip |
||
302 | * @return void |
||
303 | */ |
||
304 | public function _do_flip($direction) |
||
331 | } |
||
332 | |||
333 | /** |
||
334 | * Execute a sharpen. |
||
335 | * |
||
336 | * @param integer $amount amount to sharpen |
||
337 | * @return void |
||
338 | * @throws ErrorException |
||
339 | */ |
||
340 | public function _do_sharpen($amount) |
||
341 | { |
||
342 | if (empty(Image_GD::$_available_functions[Image_GD::IMAGECONVOLUTION])) { |
||
343 | throw new ErrorException('This method requires imageconvolution, which is only available in the bundled version of GD'); |
||
344 | } |
||
345 | |||
346 | // Loads image if not yet loaded |
||
347 | $this->_load_image(); |
||
348 | |||
349 | // Amount should be in the range of 18-10 |
||
350 | $amount = round(abs(-18 + ($amount * 0.08)), 2); |
||
351 | |||
352 | // Gaussian blur matrix |
||
353 | $matrix = array |
||
354 | ( |
||
355 | array(-1, -1, -1), |
||
356 | array(-1, $amount, -1), |
||
357 | array(-1, -1, -1), |
||
358 | ); |
||
359 | |||
360 | // Perform the sharpen |
||
361 | if (imageconvolution($this->_image, $matrix, $amount - 8, 0)) { |
||
362 | // Reset the width and height |
||
363 | $this->width = imagesx($this->_image); |
||
364 | $this->height = imagesy($this->_image); |
||
365 | } |
||
366 | } |
||
367 | |||
368 | /** |
||
369 | * Execute a reflection. |
||
370 | * |
||
371 | * @param integer $height reflection height |
||
372 | * @param integer $opacity reflection opacity |
||
373 | * @param boolean $fade_in true to fade out, false to fade in |
||
374 | * @return void |
||
375 | * @throws ErrorException |
||
376 | */ |
||
377 | public function _do_reflection($height, $opacity, $fade_in) |
||
378 | { |
||
379 | if (empty(Image_GD::$_available_functions[Image_GD::IMAGEFILTER])) { |
||
380 | throw new ErrorException('This method requires imagefilter, which is only available in the bundled version of GD'); |
||
381 | } |
||
382 | |||
383 | // Loads image if not yet loaded |
||
384 | $this->_load_image(); |
||
385 | |||
386 | // Convert an opacity range of 0-100 to 127-0 |
||
387 | $opacity = round(abs(($opacity * 127 / 100) - 127)); |
||
388 | |||
389 | if ($opacity < 127) { |
||
390 | // Calculate the opacity stepping |
||
391 | $stepping = (127 - $opacity) / $height; |
||
392 | } else { |
||
393 | // Avoid a "divide by zero" error |
||
394 | $stepping = 127 / $height; |
||
395 | } |
||
396 | |||
397 | // Create the reflection image |
||
398 | $reflection = $this->_create($this->width, $this->height + $height); |
||
399 | |||
400 | // Copy the image to the reflection |
||
401 | imagecopy($reflection, $this->_image, 0, 0, 0, 0, $this->width, $this->height); |
||
402 | |||
403 | for ($offset = 0; $height >= $offset; $offset++) { |
||
404 | // Read the next line down |
||
405 | $src_y = $this->height - $offset - 1; |
||
406 | |||
407 | // Place the line at the bottom of the reflection |
||
408 | $dst_y = $this->height + $offset; |
||
409 | |||
410 | if ($fade_in === true) { |
||
411 | // Start with the most transparent line first |
||
412 | $dst_opacity = round($opacity + ($stepping * ($height - $offset))); |
||
413 | } else { |
||
414 | // Start with the most opaque line first |
||
415 | $dst_opacity = round($opacity + ($stepping * $offset)); |
||
416 | } |
||
417 | |||
418 | // Create a single line of the image |
||
419 | $line = $this->_create($this->width, 1); |
||
420 | |||
421 | // Copy a single line from the current image into the line |
||
422 | imagecopy($line, $this->_image, 0, 0, 0, $src_y, $this->width, 1); |
||
423 | |||
424 | // Colorize the line to add the correct alpha level |
||
425 | imagefilter($line, IMG_FILTER_COLORIZE, 0, 0, 0, $dst_opacity); |
||
426 | |||
427 | // Copy a the line into the reflection |
||
428 | imagecopy($reflection, $line, 0, $dst_y, 0, 0, $this->width, 1); |
||
429 | } |
||
430 | |||
431 | // Swap the new image for the old one |
||
432 | imagedestroy($this->_image); |
||
433 | $this->_image = $reflection; |
||
434 | |||
435 | // Reset the width and height |
||
436 | $this->width = imagesx($reflection); |
||
437 | $this->height = imagesy($reflection); |
||
438 | } |
||
439 | |||
440 | /** |
||
441 | * Execute a watermarking. |
||
442 | * |
||
443 | * @param \image\components\Kohana\Image $watermark watermarking Kohana_Image |
||
444 | * @param integer $offset_x offset from the left |
||
445 | * @param integer $offset_y offset from the top |
||
446 | * @param integer $opacity opacity of watermark |
||
447 | * @return void |
||
448 | * @throws ErrorException |
||
449 | */ |
||
450 | public function _do_watermark(\image\components\Kohana\Image $watermark, $offset_x, $offset_y, $opacity) |
||
451 | { |
||
452 | if (empty(Image_GD::$_available_functions[Image_GD::IMAGELAYEREFFECT])) { |
||
453 | throw new ErrorException('This method requires imagelayereffect, which is only available in the bundled version of GD'); |
||
454 | } |
||
455 | |||
456 | // Loads image if not yet loaded |
||
457 | $this->_load_image(); |
||
458 | |||
459 | // Create the watermark image resource |
||
460 | $overlay = imagecreatefromstring($watermark->render()); |
||
461 | |||
462 | imagesavealpha($overlay, true); |
||
463 | |||
464 | // Get the width and height of the watermark |
||
465 | $width = imagesx($overlay); |
||
466 | $height = imagesy($overlay); |
||
467 | |||
468 | if ($opacity < 100) { |
||
469 | // Convert an opacity range of 0-100 to 127-0 |
||
470 | $opacity = round(abs(($opacity * 127 / 100) - 127)); |
||
471 | |||
472 | // Allocate transparent gray |
||
473 | $color = imagecolorallocatealpha($overlay, 127, 127, 127, $opacity); |
||
474 | |||
475 | // The transparent image will overlay the watermark |
||
476 | imagelayereffect($overlay, IMG_EFFECT_OVERLAY); |
||
477 | |||
478 | // Fill the background with the transparent color |
||
479 | imagefilledrectangle($overlay, 0, 0, $width, $height, $color); |
||
480 | } |
||
481 | |||
482 | // Alpha blending must be enabled on the background! |
||
483 | imagealphablending($this->_image, true); |
||
484 | |||
485 | if (imagecopy($this->_image, $overlay, $offset_x, $offset_y, 0, 0, $width, $height)) { |
||
486 | // Destroy the overlay image |
||
487 | imagedestroy($overlay); |
||
488 | } |
||
489 | } |
||
490 | |||
491 | /** |
||
492 | * Execute a background. |
||
493 | * |
||
494 | * @param integer $r red |
||
495 | * @param integer $g green |
||
496 | * @param integer $b blue |
||
497 | * @param integer $opacity opacity |
||
498 | * @return void |
||
499 | */ |
||
500 | public function _do_background($r, $g, $b, $opacity) |
||
501 | { |
||
502 | // Loads image if not yet loaded |
||
503 | $this->_load_image(); |
||
504 | |||
505 | // Convert an opacity range of 0-100 to 127-0 |
||
506 | $opacity = round(abs(($opacity * 127 / 100) - 127)); |
||
507 | |||
508 | // Create a new background |
||
509 | $background = $this->_create($this->width, $this->height); |
||
510 | |||
511 | // Allocate the color |
||
512 | $color = imagecolorallocatealpha($background, $r, $g, $b, $opacity); |
||
513 | |||
514 | // Fill the image with white |
||
515 | imagefilledrectangle($background, 0, 0, $this->width, $this->height, $color); |
||
516 | |||
517 | // Alpha blending must be enabled on the background! |
||
518 | imagealphablending($background, true); |
||
519 | |||
520 | // Copy the image onto a white background to remove all transparency |
||
521 | if (imagecopy($background, $this->_image, 0, 0, 0, 0, $this->width, $this->height)) { |
||
522 | // Swap the new image for the old one |
||
523 | imagedestroy($this->_image); |
||
524 | $this->_image = $background; |
||
525 | } |
||
526 | } |
||
527 | |||
528 | /** |
||
529 | * Execute a save. |
||
530 | * |
||
531 | * @param string $file new image filename |
||
532 | * @param integer $quality quality |
||
533 | * @return boolean |
||
534 | * @throws ErrorException |
||
535 | */ |
||
536 | public function _do_save($file, $quality) |
||
561 | } |
||
562 | |||
563 | /** |
||
564 | * Execute a render. |
||
565 | * |
||
566 | * @param string $type image type: png, jpg, gif, etc |
||
567 | * @param integer $quality quality |
||
568 | * @return string |
||
569 | * @throws ErrorException |
||
570 | */ |
||
571 | public function _do_render($type, $quality) |
||
572 | { |
||
573 | // Loads image if not yet loaded |
||
574 | $this->_load_image(); |
||
575 | |||
576 | // Get the save function and IMAGETYPE |
||
577 | list($save, $type) = $this->_save_function($type, $quality); |
||
578 | |||
579 | // Capture the output |
||
580 | ob_start(); |
||
581 | |||
582 | // Render the image |
||
583 | $status = isset($quality) ? $save($this->_image, null, $quality) : $save($this->_image, null); |
||
584 | |||
585 | if ($status === true && $type !== $this->type) { |
||
586 | // Reset the image type and mime type |
||
587 | $this->type = $type; |
||
588 | $this->mime = image_type_to_mime_type($type); |
||
589 | } |
||
590 | |||
591 | return ob_get_clean(); |
||
592 | } |
||
593 | |||
594 | /** |
||
595 | * Get the GD saving function and image type for this extension. |
||
596 | * Also normalizes the quality setting |
||
597 | * |
||
598 | * @param string $extension image type: png, jpg, etc |
||
599 | * @param integer $quality image quality |
||
600 | * @return array save function, IMAGETYPE_* constant |
||
601 | * @throws ErrorException |
||
602 | */ |
||
603 | protected function _save_function($extension, & $quality) |
||
639 | } |
||
640 | |||
641 | /** |
||
642 | * Create an empty image with the given width and height. |
||
643 | * |
||
644 | * @param integer $width image width |
||
645 | * @param integer $height image height |
||
646 | * @return resource |
||
647 | */ |
||
648 | protected function _create($width, $height) |
||
660 | } |
||
661 | |||
662 | } // End Image_GD |
||
663 |
Our type inference engine has found an assignment to a property that is incompatible with the declared type of that property.
Either this assignment is in error or the assigned type should be added to the documentation/type hint for that property..