Code Duplication    Length = 3151-3151 lines in 2 locations

htdocs/class/xoopseditor/tinymce4/external_plugins/filemanager/include/php_image_magician.php 1 location

@@ 123-3273 (lines=3151) @@
120
#
121
# ========================================================================#
122
123
class imageLib
124
{
125
126
    private   $fileName;
127
    private   $image;
128
    protected $imageResized;
129
    private   $widthOriginal;     # Always be the original width
130
    private   $heightOriginal;
131
    private   $width;         # Current width (width after resize)
132
    private   $height;
133
    private   $imageSize;
134
    private   $fileExtension;
135
136
    private $isImage = false;
137
138
    private $debug      = true;
139
    private $errorArray = [];
140
141
    private $forceStretch        = true;
142
    private $aggresiveSharpening = false;
143
144
    private $transparentArray = ['.png', '.gif'];
145
    private $keepTransparency = true;
146
    private $fillColorArray   = ['r' => 255, 'g' => 255, 'b' => 255];
147
148
    private $sharpenArray = ['jpg'];
149
150
    private $psdReaderPath;
151
    private $filterOverlayPath;
152
153
    private $isInterlace;
154
155
    private $captionBoxPositionArray = [];
156
157
    private $fontDir = 'fonts';
158
159
    private $cropFromTopPercent = 10;
160
161
    ## --------------------------------------------------------
162
163
    public function __construct($fileName)
164
        # Author:     Jarrod Oberto
165
        # Date:       27-02-08
166
        # Purpose:    Constructor
167
        # Param in:   $fileName: File name and path.
168
        # Param out:  n/a
169
        # Reference:
170
        # Notes:
171
        #
172
    {
173
        if (!$this->testGDInstalled()) {
174
            if ($this->debug) {
175
                die('The GD Library is not installed.');
176
            } else {
177
                die();
178
            }
179
        }
180
181
        $this->initialise();
182
183
        // *** Save the image file name. Only store this incase you want to display it
184
        $this->fileName      = $fileName;
185
        $this->fileExtension = strtolower(strrchr($fileName, '.'));
186
187
        // *** Open up the file
188
        try {
189
            $this->image = $this->openImage($fileName);
190
        } catch (Exception $e) {
191
        }
192
193
        // *** Assign here so we don't modify the original
194
        $this->imageResized = $this->image;
195
196
        // *** If file is an image
197
        $this->isImage = $this->testIsImage();
198
199
        if ($this->isImage) {
200
            // *** Get width and height
201
            $this->width          = imagesx($this->image);
202
            $this->widthOriginal  = imagesx($this->image);
203
            $this->height         = imagesy($this->image);
204
            $this->heightOriginal = imagesy($this->image);
205
206
            /*  Added 15-09-08
207
             *  Get the filesize using this build in method.
208
             *  Stores an array of size
209
             *
210
             *  $this->imageSize[1] = width
211
             *  $this->imageSize[2] = height
212
             *  $this->imageSize[3] = width x height
213
             *
214
             */
215
            $this->imageSize = getimagesize($this->fileName);
216
        } else {
217
            $this->errorArray[] = 'File is not an image';
218
        }
219
    }
220
221
    ## --------------------------------------------------------
222
223
    private function initialise()
224
    {
225
        $this->psdReaderPath     = __DIR__ . '/classPhpPsdReader.php';
226
        $this->filterOverlayPath = __DIR__ . '/filters';
227
228
        // *** Set if image should be interlaced or not.
229
        $this->isInterlace = false;
230
    }
231
232
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
233
        Resize
234
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
235
236
    public function resizeImage($newWidth, $newHeight, $option = 0, $sharpen = false, $autoRotate = true)
237
        # Author:     Jarrod Oberto
238
        # Date:       27-02-08
239
        # Purpose:    Resizes the image
240
        # Param in:   $newWidth:
241
        #             $newHeight:
242
        #             $option:     0 / exact = defined size;
243
        #                          1 / portrait = keep aspect set height;
244
        #                          2 / landscape = keep aspect set width;
245
        #                          3 / auto = auto;
246
        #                          4 / crop= resize and crop;
247
        #
248
        #         $option can also be an array containing options for
249
        #         cropping. E.G., array('crop', 'r')
250
        #
251
        #         This array only applies to 'crop' and the 'r' refers to
252
        #         "crop right". Other value include; tl, t, tr, l, m (default),
253
        #         r, bl, b, br, or you can specify your own co-ords (which
254
        #         isn't recommended.
255
        #
256
        #       $sharpen:    true: sharpen (jpg only);
257
        #                false: don't sharpen
258
        # Param out:  n/a
259
        # Reference:
260
        # Notes:      To clarify the $option input:
261
        #               0 = The exact height and width dimensions you set.
262
        #               1 = Whatever height is passed in will be the height that
263
        #                   is set. The width will be calculated and set automatically
264
        #                   to a the value that keeps the original aspect ratio.
265
        #               2 = The same but based on the width. We try make the image the
266
        #                  biggest size we can while stil fitting inside the box size
267
        #               3 = Depending whether the image is landscape or portrait, this
268
        #                   will automatically determine whether to resize via
269
        #                   dimension 1,2 or 0
270
        #               4 = Will resize and then crop the image for best fit
271
        #
272
        #       forceStretch can be applied to options 1,2,3 and 4
273
        #
274
    {
275
        // *** We can pass in an array of options to change the crop position
276
        $cropPos = 'm';
277
        if (is_array($option) && 'crop' === strtolower($option[0])) {
278
            $cropPos = $option[1];         # get the crop option
279
        } elseif (false !== strpos($option, '-')) {
280
            // *** Or pass in a hyphen seperated option
281
            $optionPiecesArray = explode('-', $option);
282
            $cropPos           = end($optionPiecesArray);
283
        }
284
285
        // *** Check the option is valid
286
        try {
287
            $option = $this->prepOption($option);
288
        } catch (Exception $e) {
289
        }
290
291
        // *** Make sure the file passed in is valid
292
        if (!$this->image) {
293
            if ($this->debug) {
294
                die('file ' . $this->getFileName() . ' is missing or invalid');
295
            } else {
296
                die();
297
            }
298
        }
299
300
        // *** Get optimal width and height - based on $option
301
        $dimensionsArray = $this->getDimensions($newWidth, $newHeight, $option);
302
303
        $optimalWidth  = $dimensionsArray['optimalWidth'];
304
        $optimalHeight = $dimensionsArray['optimalHeight'];
305
306
        // *** Resample - create image canvas of x, y size
307
        $this->imageResized = imagecreatetruecolor($optimalWidth, $optimalHeight);
308
        $this->keepTransparancy($optimalWidth, $optimalHeight, $this->imageResized);
309
        imagecopyresampled($this->imageResized, $this->image, 0, 0, 0, 0, $optimalWidth, $optimalHeight, $this->width, $this->height);
310
311
        // *** If '4', then crop too
312
        if (4 == $option || 'crop' === $option) {
313
            if (($optimalWidth >= $newWidth && $optimalHeight >= $newHeight)) {
314
                $this->crop($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos);
315
            }
316
        }
317
318
        // *** If Rotate.
319
        if ($autoRotate) {
320
            try {
321
                $exifData = $this->getExif(false);
322
            } catch (Exception $e) {
323
            }
324
            if (count($exifData) > 0) {
325
                switch ($exifData['orientation']) {
326
                    case 8:
327
                        $this->imageResized = imagerotate($this->imageResized, 90, 0);
328
                        break;
329
                    case 3:
330
                        $this->imageResized = imagerotate($this->imageResized, 180, 0);
331
                        break;
332
                    case 6:
333
                        $this->imageResized = imagerotate($this->imageResized, -90, 0);
334
                        break;
335
                }
336
            }
337
        }
338
339
        // *** Sharpen image (if jpg and the user wishes to do so)
340
        if ($sharpen && in_array($this->fileExtension, $this->sharpenArray)) {
341
            // *** Sharpen
342
            try {
343
                $this->sharpen();
344
            } catch (Exception $e) {
345
            }
346
        }
347
    }
348
349
    ## --------------------------------------------------------
350
351
    public function cropImage($newWidth, $newHeight, $cropPos = 'm')
352
        # Author:     Jarrod Oberto
353
        # Date:       08-09-11
354
        # Purpose:    Crops the image
355
        # Param in:   $newWidth: crop with
356
        #             $newHeight: crop height
357
        #             $cropPos: Can be any of the following:
358
        #                   tl, t, tr, l, m, r, bl, b, br, auto
359
        #             Or:
360
        #                   a custom position such as '30x50'
361
        # Param out:  n/a
362
        # Reference:
363
        # Notes:
364
        #
365
    {
366
        // *** Make sure the file passed in is valid
367
        if (!$this->image) {
368
            if ($this->debug) {
369
                die('file ' . $this->getFileName() . ' is missing or invalid');
370
            } else {
371
                die();
372
            }
373
        }
374
375
        $this->imageResized = $this->image;
376
        $this->crop($this->width, $this->height, $newWidth, $newHeight, $cropPos);
377
    }
378
379
    ## --------------------------------------------------------
380
381
    private function keepTransparancy($width, $height, $im)
382
        # Author:     Jarrod Oberto
383
        # Date:       08-04-11
384
        # Purpose:    Keep transparency for png and gif image
385
        # Param in:
386
        # Param out:  n/a
387
        # Reference:
388
        # Notes:
389
        #
390
    {
391
        // *** If PNG, perform some transparency retention actions (gif untested)
392
        if (in_array($this->fileExtension, $this->transparentArray) && $this->keepTransparency) {
393
            imagealphablending($im, false);
394
            imagesavealpha($im, true);
395
            $transparent = imagecolorallocatealpha($im, 255, 255, 255, 127);
396
            imagefilledrectangle($im, 0, 0, $width, $height, $transparent);
397
        } else {
398
            $color = imagecolorallocate($im, $this->fillColorArray['r'], $this->fillColorArray['g'], $this->fillColorArray['b']);
399
            imagefilledrectangle($im, 0, 0, $width, $height, $color);
400
        }
401
    }
402
403
    ## --------------------------------------------------------
404
405
    private function crop($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos)
406
        # Author:     Jarrod Oberto
407
        # Date:       15-09-08
408
        # Purpose:    Crops the image
409
        # Param in:   $newWidth:
410
        #             $newHeight:
411
        # Param out:  n/a
412
        # Reference:
413
        # Notes:
414
        #
415
    {
416
        // *** Get cropping co-ordinates
417
        $cropArray  = $this->getCropPlacing($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos);
418
        $cropStartX = $cropArray['x'];
419
        $cropStartY = $cropArray['y'];
420
421
        // *** Crop this bad boy
422
        $crop = imagecreatetruecolor($newWidth, $newHeight);
423
        $this->keepTransparancy($optimalWidth, $optimalHeight, $crop);
424
        imagecopyresampled($crop, $this->imageResized, 0, 0, $cropStartX, $cropStartY, $newWidth, $newHeight, $newWidth, $newHeight);
425
426
        $this->imageResized = $crop;
427
428
        // *** Set new width and height to our variables
429
        $this->width  = $newWidth;
430
        $this->height = $newHeight;
431
    }
432
433
    ## --------------------------------------------------------
434
435
    private function getCropPlacing($optimalWidth, $optimalHeight, $newWidth, $newHeight, $pos = 'm')
436
        #
437
        # Author:   Jarrod Oberto
438
        # Date:   July 11
439
        # Purpose:  Set the cropping area.
440
        # Params in:
441
        # Params out: (array) the crop x and y co-ordinates.
442
        # Notes:    When specifying the exact pixel crop position (eg 10x15), be
443
        #       very careful as it's easy to crop out of the image leaving
444
        #       black borders.
445
        #
446
    {
447
        $pos = strtolower($pos);
448
449
        // *** If co-ords have been entered
450
        if (strstr($pos, 'x')) {
451
            $pos = str_replace(' ', '', $pos);
452
453
            $xyArray = explode('x', $pos);
454
            list($cropStartX, $cropStartY) = $xyArray;
455
        } else {
456
            switch ($pos) {
457
                case 'tl':
458
                    $cropStartX = 0;
459
                    $cropStartY = 0;
460
                    break;
461
462
                case 't':
463
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
464
                    $cropStartY = 0;
465
                    break;
466
467
                case 'tr':
468
                    $cropStartX = $optimalWidth - $newWidth;
469
                    $cropStartY = 0;
470
                    break;
471
472
                case 'l':
473
                    $cropStartX = 0;
474
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
475
                    break;
476
477
                case 'm':
478
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
479
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
480
                    break;
481
482
                case 'r':
483
                    $cropStartX = $optimalWidth - $newWidth;
484
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
485
                    break;
486
487
                case 'bl':
488
                    $cropStartX = 0;
489
                    $cropStartY = $optimalHeight - $newHeight;
490
                    break;
491
492
                case 'b':
493
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
494
                    $cropStartY = $optimalHeight - $newHeight;
495
                    break;
496
497
                case 'br':
498
                    $cropStartX = $optimalWidth - $newWidth;
499
                    $cropStartY = $optimalHeight - $newHeight;
500
                    break;
501
502
                case 'auto':
503
                    // *** If image is a portrait crop from top, not center. v1.5
504
                    if ($optimalHeight > $optimalWidth) {
505
                        $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
506
                        $cropStartY = ($this->cropFromTopPercent / 100) * $optimalHeight;
507
                    } else {
508
                        // *** Else crop from the center
509
                        $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
510
                        $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
511
                    }
512
                    break;
513
514
                default:
515
                    // *** Default to center
516
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
517
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
518
                    break;
519
            }
520
        }
521
522
        return ['x' => $cropStartX, 'y' => $cropStartY];
523
    }
524
525
    ## --------------------------------------------------------
526
527
    private function getDimensions($newWidth, $newHeight, $option)
528
        # Author:     Jarrod Oberto
529
        # Date:       17-11-09
530
        # Purpose:    Get new image dimensions based on user specificaions
531
        # Param in:   $newWidth:
532
        #             $newHeight:
533
        # Param out:  Array of new width and height values
534
        # Reference:
535
        # Notes:      If $option = 3 then this function is call recursivly
536
        #
537
        #       To clarify the $option input:
538
        #               0 = The exact height and width dimensions you set.
539
        #               1 = Whatever height is passed in will be the height that
540
        #                   is set. The width will be calculated and set automatically
541
        #                   to a the value that keeps the original aspect ratio.
542
        #               2 = The same but based on the width.
543
        #               3 = Depending whether the image is landscape or portrait, this
544
        #                   will automatically determine whether to resize via
545
        #                   dimension 1,2 or 0.
546
        #               4 = Resize the image as much as possible, then crop the
547
        #         remainder.
548
    {
549
        switch ((string)$option) {
550
            case '0':
551
            case 'exact':
552
                $optimalWidth  = $newWidth;
553
                $optimalHeight = $newHeight;
554
                break;
555
            case '1':
556
            case 'portrait':
557
                $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
558
                $optimalWidth    = $dimensionsArray['optimalWidth'];
559
                $optimalHeight   = $dimensionsArray['optimalHeight'];
560
                break;
561
            case '2':
562
            case 'landscape':
563
                $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
564
                $optimalWidth    = $dimensionsArray['optimalWidth'];
565
                $optimalHeight   = $dimensionsArray['optimalHeight'];
566
                break;
567
            case '3':
568
            case 'auto':
569
                $dimensionsArray = $this->getSizeByAuto($newWidth, $newHeight);
570
                $optimalWidth    = $dimensionsArray['optimalWidth'];
571
                $optimalHeight   = $dimensionsArray['optimalHeight'];
572
                break;
573
            case '4':
574
            case 'crop':
575
                $dimensionsArray = $this->getOptimalCrop($newWidth, $newHeight);
576
                $optimalWidth    = $dimensionsArray['optimalWidth'];
577
                $optimalHeight   = $dimensionsArray['optimalHeight'];
578
                break;
579
        }
580
581
        return ['optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight];
582
    }
583
584
    ## --------------------------------------------------------
585
586
    private function getSizeByFixedHeight($newWidth, $newHeight)
587
    {
588
        // *** If forcing is off...
589
        if (!$this->forceStretch) {
590
            // *** ...check if actual height is less than target height
591
            if ($this->height < $newHeight) {
592
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
593
            }
594
        }
595
596
        $ratio = $this->width / $this->height;
597
598
        $newWidth = $newHeight * $ratio;
599
600
        //return $newWidth;
601
        return ['optimalWidth' => $newWidth, 'optimalHeight' => $newHeight];
602
    }
603
604
    ## --------------------------------------------------------
605
606
    private function getSizeByFixedWidth($newWidth, $newHeight)
607
    {
608
        // *** If forcing is off...
609
        if (!$this->forceStretch) {
610
            // *** ...check if actual width is less than target width
611
            if ($this->width < $newWidth) {
612
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
613
            }
614
        }
615
616
        $ratio = $this->height / $this->width;
617
618
        $newHeight = $newWidth * $ratio;
619
620
        //return $newHeight;
621
        return ['optimalWidth' => $newWidth, 'optimalHeight' => $newHeight];
622
    }
623
624
    ## --------------------------------------------------------
625
626
    private function getSizeByAuto($newWidth, $newHeight)
627
        # Author:     Jarrod Oberto
628
        # Date:       19-08-08
629
        # Purpose:    Depending on the height, choose to resize by 0, 1, or 2
630
        # Param in:   The new height and new width
631
        # Notes:
632
        #
633
    {
634
        // *** If forcing is off...
635
        if (!$this->forceStretch) {
636
            // *** ...check if actual size is less than target size
637
            if ($this->width < $newWidth && $this->height < $newHeight) {
638
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
639
            }
640
        }
641
642
        if ($this->height < $this->width) // *** Image to be resized is wider (landscape)
643
        {
644
            //$optimalWidth = $newWidth;
645
            //$optimalHeight= $this->getSizeByFixedWidth($newWidth);
646
647
            $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
648
            $optimalWidth    = $dimensionsArray['optimalWidth'];
649
            $optimalHeight   = $dimensionsArray['optimalHeight'];
650
        } elseif ($this->height > $this->width) // *** Image to be resized is taller (portrait)
651
        {
652
            //$optimalWidth = $this->getSizeByFixedHeight($newHeight);
653
            //$optimalHeight= $newHeight;
654
655
            $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
656
            $optimalWidth    = $dimensionsArray['optimalWidth'];
657
            $optimalHeight   = $dimensionsArray['optimalHeight'];
658
        } else // *** Image to be resizerd is a square
659
        {
660
            if ($newHeight < $newWidth) {
661
                //$optimalWidth = $newWidth;
662
                //$optimalHeight= $this->getSizeByFixedWidth($newWidth);
663
                $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
664
                $optimalWidth    = $dimensionsArray['optimalWidth'];
665
                $optimalHeight   = $dimensionsArray['optimalHeight'];
666
            } elseif ($newHeight > $newWidth) {
667
                //$optimalWidth = $this->getSizeByFixedHeight($newHeight);
668
                //$optimalHeight= $newHeight;
669
                $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
670
                $optimalWidth    = $dimensionsArray['optimalWidth'];
671
                $optimalHeight   = $dimensionsArray['optimalHeight'];
672
            } else {
673
                // *** Sqaure being resized to a square
674
                $optimalWidth  = $newWidth;
675
                $optimalHeight = $newHeight;
676
            }
677
        }
678
679
        return ['optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight];
680
    }
681
682
    ## --------------------------------------------------------
683
684
    private function getOptimalCrop($newWidth, $newHeight)
685
        # Author:     Jarrod Oberto
686
        # Date:       17-11-09
687
        # Purpose:    Get optimal crop dimensions
688
        # Param in:   width and height as requested by user (fig 3)
689
        # Param out:  Array of optimal width and height (fig 2)
690
        # Reference:
691
        # Notes:      The optimal width and height return are not the same as the
692
        #             same as the width and height passed in. For example:
693
        #
694
        #
695
        #   |-----------------|     |------------|       |-------|
696
        #   |                 | =>  |**|      |**|   =>  |       |
697
        #   |                 |     |**|      |**|       |       |
698
        #   |                 |     |------------|       |-------|
699
        #   |-----------------|
700
        #        original                optimal             crop
701
        #          size                   size               size
702
        #  Fig      1                      2                  3
703
        #
704
        #       300 x 250           150 x 125          150 x 100
705
        #
706
        #  The optimal size is the smallest size (that is closest to the crop size)
707
        #  while retaining proportion/ratio.
708
        #
709
        #  The crop size is the optimal size that has been cropped on one axis to
710
        #  make the image the exact size specified by the user.
711
        #
712
        #      * represent cropped area
713
        #
714
    {
715
        // *** If forcing is off...
716
        if (!$this->forceStretch) {
717
            // *** ...check if actual size is less than target size
718
            if ($this->width < $newWidth && $this->height < $newHeight) {
719
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
720
            }
721
        }
722
723
        $heightRatio = $this->height / $newHeight;
724
        $widthRatio  = $this->width / $newWidth;
725
726
        if ($heightRatio < $widthRatio) {
727
            $optimalRatio = $heightRatio;
728
        } else {
729
            $optimalRatio = $widthRatio;
730
        }
731
732
        $optimalHeight = round($this->height / $optimalRatio);
733
        $optimalWidth  = round($this->width / $optimalRatio);
734
735
        return ['optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight];
736
    }
737
738
    ## --------------------------------------------------------
739
740
    private function sharpen()
741
        # Author:     Jarrod Oberto
742
        # Date:       08 04 2011
743
        # Purpose:    Sharpen image
744
        # Param in:   n/a
745
        # Param out:  n/a
746
        # Reference:
747
        # Notes:
748
        # Credit:   Incorporates Joe Lencioni (August 6, 2008) code
749
    {
750
        if (version_compare(PHP_VERSION, '5.1.0') >= 0) {
751
            // ***
752
            if ($this->aggresiveSharpening) { # A more aggressive sharpening solution
753
754
                $sharpenMatrix = [
755
                    [-1, -1, -1],
756
                    [-1, 16, -1],
757
                    [-1, -1, -1]
758
                ];
759
                $divisor       = 8;
760
                $offset        = 0;
761
762
                imageconvolution($this->imageResized, $sharpenMatrix, $divisor, $offset);
763
            } else # More subtle and personally more desirable
764
            {
765
                $sharpness = $this->findSharp($this->widthOriginal, $this->width);
766
767
                $sharpenMatrix = [
768
                    [-1, -2, -1],
769
                    [-2, $sharpness + 12, -2], //Lessen the effect of a filter by increasing the value in the center cell
770
                    [-1, -2, -1]
771
                ];
772
                $divisor       = $sharpness; // adjusts brightness
773
                $offset        = 0;
774
                imageconvolution($this->imageResized, $sharpenMatrix, $divisor, $offset);
775
            }
776
        } else {
777
            if ($this->debug) {
778
                die('Sharpening required PHP 5.1.0 or greater.');
779
            }
780
        }
781
    }
782
783
    ## --------------------------------------------------------
784
785
    private function sharpen2($level)
786
    {
787
        $sharpenMatrix = [
788
            [$level, $level, $level],
789
            [$level, (8 * $level) + 1, $level], //Lessen the effect of a filter by increasing the value in the center cell
790
            [$level, $level, $level]
791
        ];
792
    }
793
794
    ## --------------------------------------------------------
795
796
    private function findSharp($orig, $final)
797
        # Author:     Ryan Rud (http://adryrun.com)
798
        # Purpose:    Find optimal sharpness
799
        # Param in:   n/a
800
        # Param out:  n/a
801
        # Reference:
802
        # Notes:
803
        #
804
    {
805
        $final = $final * (750.0 / $orig);
806
        $a     = 52;
807
        $b     = -0.27810650887573124;
808
        $c     = .00047337278106508946;
809
810
        $result = $a + $b * $final + $c * $final * $final;
811
812
        return max(round($result), 0);
813
    }
814
815
    ## --------------------------------------------------------
816
817
    private function prepOption($option)
818
        # Author:     Jarrod Oberto
819
        # Purpose:    Prep option like change the passed in option to lowercase
820
        # Param in:   (str/int) $option: eg. 'exact', 'crop'. 0, 4
821
        # Param out:  lowercase string
822
        # Reference:
823
        # Notes:
824
        #
825
    {
826
        if (is_array($option)) {
827
            if ('crop' === strtolower($option[0]) && 2 == count($option)) {
828
                return 'crop';
829
            } else {
830
                die('Crop resize option array is badly formatted.');
831
            }
832
        } elseif (false !== strpos($option, 'crop')) {
833
            return 'crop';
834
        }
835
836
        if (is_string($option)) {
837
            return strtolower($option);
838
        }
839
840
        return $option;
841
    }
842
843
844
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
845
        Presets
846
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
847
848
    #
849
    # Preset are pre-defined templates you can apply to your image.
850
    #
851
    # These are inteded to be applied to thumbnail images.
852
    #
853
854
    public function borderPreset($preset)
855
    {
856
        switch ($preset) {
857
            case 'simple':
858
                $this->addBorder(7, '#fff');
859
                $this->addBorder(6, '#f2f1f0');
860
                $this->addBorder(2, '#fff');
861
                $this->addBorder(1, '#ccc');
862
                break;
863
            default:
864
                break;
865
        }
866
    }
867
868
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
869
        Draw border
870
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
871
872
    public function addBorder($thickness = 1, $rgbArray = [255, 255, 255])
873
        # Author:     Jarrod Oberto
874
        # Date:       05-05-11
875
        # Purpose:    Add a border to the image
876
        # Param in:
877
        # Param out:
878
        # Reference:
879
        # Notes:    This border is added to the INSIDE of the image
880
        #
881
    {
882
        if ($this->imageResized) {
883
            $rgbArray = $this->formatColor($rgbArray);
884
            $r        = $rgbArray['r'];
885
            $g        = $rgbArray['g'];
886
            $b        = $rgbArray['b'];
887
888
            $x1 = 0;
889
            $y1 = 0;
890
            $x2 = imagesx($this->imageResized) - 1;
891
            $y2 = imagesy($this->imageResized) - 1;
892
893
            $rgbArray = imagecolorallocate($this->imageResized, $r, $g, $b);
894
895
            for ($i = 0; $i < $thickness; ++$i) {
896
                imagerectangle($this->imageResized, ++$x1, ++$y1, $x2--, $y2--, $rgbArray);
897
            }
898
        }
899
    }
900
901
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
902
        Gray Scale
903
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
904
905
    public function greyScale()
906
        # Author:     Jarrod Oberto
907
        # Date:       07-05-2011
908
        # Purpose:    Make image greyscale
909
        # Param in:   n/a
910
        # Param out:
911
        # Reference:
912
        # Notes:
913
        #
914
    {
915
        if ($this->imageResized) {
916
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
917
        }
918
    }
919
920
    ## --------------------------------------------------------
921
922
    public function greyScaleEnhanced()
923
        # Author:     Jarrod Oberto
924
        # Date:       07-05-2011
925
        # Purpose:    Make image greyscale
926
        # Param in:   n/a
927
        # Param out:
928
        # Reference:
929
        # Notes:
930
        #
931
    {
932
        if ($this->imageResized) {
933
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
934
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -15);
935
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, 2);
936
            try {
937
                $this->sharpen($this->width);
938
            } catch (Exception $e) {
939
            }
940
        }
941
    }
942
943
    ## --------------------------------------------------------
944
945
    public function greyScaleDramatic()
946
        # Alias of gd_filter_monopin
947
    {
948
        $this->gd_filter_monopin();
949
    }
950
951
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
952
        Black 'n White
953
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
954
955
    public function blackAndWhite()
956
        # Author:     Jarrod Oberto
957
        # Date:       07-05-2011
958
        # Purpose:    Make image black and white
959
        # Param in:   n/a
960
        # Param out:
961
        # Reference:
962
        # Notes:
963
        #
964
    {
965
        if ($this->imageResized) {
966
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
967
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -1000);
968
        }
969
    }
970
971
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
972
        Negative
973
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
974
975
    public function negative()
976
        # Author:     Jarrod Oberto
977
        # Date:       07-05-2011
978
        # Purpose:    Make image negative
979
        # Param in:   n/a
980
        # Param out:
981
        # Reference:
982
        # Notes:
983
        #
984
    {
985
        if ($this->imageResized) {
986
            imagefilter($this->imageResized, IMG_FILTER_NEGATE);
987
        }
988
    }
989
990
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
991
        Sepia
992
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
993
994
    public function sepia()
995
        # Author:     Jarrod Oberto
996
        # Date:       07-05-2011
997
        # Purpose:    Make image sepia
998
        # Param in:   n/a
999
        # Param out:
1000
        # Reference:
1001
        # Notes:
1002
        #
1003
    {
1004
        if ($this->imageResized) {
1005
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
1006
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, -10);
1007
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -20);
1008
            imagefilter($this->imageResized, IMG_FILTER_COLORIZE, 60, 30, -15);
1009
        }
1010
    }
1011
1012
    ## --------------------------------------------------------
1013
1014
    public function sepia2()
1015
    {
1016
        if ($this->imageResized) {
1017
            $total = imagecolorstotal($this->imageResized);
1018
            for ($i = 0; $i < $total; ++$i) {
1019
                $index = imagecolorsforindex($this->imageResized, $i);
1020
                $red   = ($index['red'] * 0.393 + $index['green'] * 0.769 + $index['blue'] * 0.189) / 1.351;
1021
                $green = ($index['red'] * 0.349 + $index['green'] * 0.686 + $index['blue'] * 0.168) / 1.203;
1022
                $blue  = ($index['red'] * 0.272 + $index['green'] * 0.534 + $index['blue'] * 0.131) / 2.140;
1023
                imagecolorset($this->imageResized, $i, $red, $green, $blue);
1024
            }
1025
        }
1026
    }
1027
1028
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1029
        Vintage
1030
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1031
1032
    public function vintage()
1033
        # Alias of gd_filter_monopin
1034
    {
1035
        $this->gd_filter_vintage();
1036
    }
1037
1038
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1039
        Presets By Marc Hibbins
1040
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1041
1042
    /** Apply 'Monopin' preset */
1043
    public function gd_filter_monopin()
1044
    {
1045
        if ($this->imageResized) {
1046
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
1047
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, -15);
1048
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -15);
1049
            $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'vignette', 100);
1050
        }
1051
    }
1052
1053
    ## --------------------------------------------------------
1054
1055
    public function gd_filter_vintage()
1056
    {
1057
        if ($this->imageResized) {
1058
            $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'vignette', 45);
1059
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, 20);
1060
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -35);
1061
            imagefilter($this->imageResized, IMG_FILTER_COLORIZE, 60, -10, 35);
1062
            imagefilter($this->imageResized, IMG_FILTER_SMOOTH, 7);
1063
            $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'scratch', 10);
1064
        }
1065
    }
1066
1067
    ## --------------------------------------------------------
1068
1069
    /** Apply a PNG overlay
1070
     * @param $im
1071
     * @param $type
1072
     * @param $amount
1073
     * @return mixed
1074
     */
1075
    private function gd_apply_overlay($im, $type, $amount)
1076
        #
1077
        # Original Author:    Marc Hibbins
1078
        # License:  Attribution-ShareAlike 3.0
1079
        # Purpose:
1080
        # Params in:
1081
        # Params out:
1082
        # Notes:
1083
        #
1084
    {
1085
        $width  = imagesx($im);
1086
        $height = imagesy($im);
1087
        $filter = imagecreatetruecolor($width, $height);
1088
1089
        imagealphablending($filter, false);
1090
        imagesavealpha($filter, true);
1091
1092
        $transparent = imagecolorallocatealpha($filter, 255, 255, 255, 127);
1093
        imagefilledrectangle($filter, 0, 0, $width, $height, $transparent);
1094
1095
        // *** Resize overlay
1096
        $overlay = $this->filterOverlayPath . '/' . $type . '.png';
1097
        $png     = imagecreatefrompng($overlay);
1098
        imagecopyresampled($filter, $png, 0, 0, 0, 0, $width, $height, imagesx($png), imagesy($png));
1099
1100
        $comp = imagecreatetruecolor($width, $height);
1101
        imagecopy($comp, $im, 0, 0, 0, 0, $width, $height);
1102
        imagecopy($comp, $filter, 0, 0, 0, 0, $width, $height);
1103
        imagecopymerge($im, $comp, 0, 0, 0, 0, $width, $height, $amount);
1104
1105
        imagedestroy($comp);
1106
        return $im;
1107
    }
1108
1109
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1110
        Colorise
1111
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1112
1113
    public function image_colorize($rgb)
1114
    {
1115
        imagetruecolortopalette($this->imageResized, true, 256);
1116
        $numColors = imagecolorstotal($this->imageResized);
1117
1118
        for ($x = 0; $x < $numColors; ++$x) {
1119
            list($r, $g, $b) = array_values(imagecolorsforindex($this->imageResized, $x));
1120
1121
            // calculate grayscale in percent
1122
            $grayscale = ($r + $g + $b) / 3 / 0xff;
1123
1124
            imagecolorset(
1125
                $this->imageResized,
1126
                $x,
1127
                $grayscale * $rgb[0],
1128
                $grayscale * $rgb[1],
1129
                $grayscale * $rgb[2]
1130
            );
1131
        }
1132
1133
        return true;
1134
    }
1135
1136
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1137
        Reflection
1138
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1139
1140
    public function addReflection($reflectionHeight = 50, $startingTransparency = 30, $inside = false, $bgColor = '#fff', $stretch = false, $divider = 0)
1141
    {
1142
        // *** Convert color
1143
        $rgbArray = $this->formatColor($bgColor);
1144
        $r        = $rgbArray['r'];
1145
        $g        = $rgbArray['g'];
1146
        $b        = $rgbArray['b'];
1147
1148
        $im = $this->imageResized;
1149
        $li = imagecreatetruecolor($this->width, 1);
1150
1151
        $bgc = imagecolorallocate($li, $r, $g, $b);
1152
        imagefilledrectangle($li, 0, 0, $this->width, 1, $bgc);
1153
1154
        $bg = imagecreatetruecolor($this->width, $reflectionHeight);
1155
        $wh = imagecolorallocate($im, 255, 255, 255);
1156
1157
        $im = imagerotate($im, -180, $wh);
1158
        imagecopyresampled($bg, $im, 0, 0, 0, 0, $this->width, $this->height, $this->width, $this->height);
1159
1160
        $im = $bg;
1161
1162
        $bg = imagecreatetruecolor($this->width, $reflectionHeight);
1163
1164
        for ($x = 0; $x < $this->width; ++$x) {
1165
            imagecopy($bg, $im, $x, 0, $this->width - $x - 1, 0, 1, $reflectionHeight);
1166
        }
1167
        $im = $bg;
1168
1169
        $transaprencyAmount = $this->invertTransparency($startingTransparency, 100);
1170
1171
        // *** Fade
1172
        if ($stretch) {
1173
            $step = 100 / ($reflectionHeight + $startingTransparency);
1174
        } else {
1175
            $step = 100 / $reflectionHeight;
1176
        }
1177
        for ($i = 0; $i <= $reflectionHeight; ++$i) {
1178
            if ($startingTransparency > 100) {
1179
                $startingTransparency = 100;
1180
            }
1181
            if ($startingTransparency < 1) {
1182
                $startingTransparency = 1;
1183
            }
1184
            imagecopymerge($bg, $li, 0, $i, 0, 0, $this->width, 1, $startingTransparency);
1185
            $startingTransparency += $step;
1186
        }
1187
1188
        // *** Apply fade
1189
        imagecopymerge($im, $li, 0, 0, 0, 0, $this->width, $divider, 100); // Divider
1190
1191
        // *** width, height of reflection.
1192
        $x = imagesx($im);
1193
        $y = imagesy($im);
1194
1195
        // *** Determines if the reflection should be displayed inside or outside the image
1196
        if ($inside) {
1197
            // Create new blank image with sizes.
1198
            $final = imagecreatetruecolor($this->width, $this->height);
1199
1200
            imagecopymerge($final, $this->imageResized, 0, 0, 0, $reflectionHeight, $this->width, $this->height - $reflectionHeight, 100);
1201
            imagecopymerge($final, $im, 0, $this->height - $reflectionHeight, 0, 0, $x, $y, 100);
1202
        } else {
1203
            // Create new blank image with sizes.
1204
            $final = imagecreatetruecolor($this->width, $this->height + $y);
1205
1206
            imagecopymerge($final, $this->imageResized, 0, 0, 0, 0, $this->width, $this->height, 100);
1207
            imagecopymerge($final, $im, 0, $this->height, 0, 0, $x, $y, 100);
1208
        }
1209
1210
        $this->imageResized = $final;
1211
1212
        imagedestroy($li);
1213
        imagedestroy($im);
1214
    }
1215
1216
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1217
        Rotate
1218
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1219
1220
    public function rotate($value = 90, $bgColor = 'transparent')
1221
        # Author:     Jarrod Oberto
1222
        # Date:       07-05-2011
1223
        # Purpose:    Rotate image
1224
        # Param in:   (mixed) $degrees: (int) number of degress to rotate image
1225
        #             (str) param "left": rotate left
1226
        #             (str) param "right": rotate right
1227
        #             (str) param "upside": upside-down image
1228
        # Param out:
1229
        # Reference:
1230
        # Notes:    The default direction of imageRotate() is counter clockwise.
1231
        #
1232
    {
1233
        if ($this->imageResized) {
1234
            if (is_int($value)) {
1235
                $degrees = $value;
1236
            }
1237
1238
            // *** Convert color
1239
            $rgbArray = $this->formatColor($bgColor);
1240
            $r        = $rgbArray['r'];
1241
            $g        = $rgbArray['g'];
1242
            $b        = $rgbArray['b'];
1243
            if (isset($rgbArray['a'])) {
1244
                $a = $rgbArray['a'];
1245
            }
1246
1247
            if (is_string($value)) {
1248
                $value = strtolower($value);
1249
1250
                switch ($value) {
1251
                    case 'left':
1252
                        $degrees = 90;
1253
                        break;
1254
                    case 'right':
1255
                        $degrees = 270;
1256
                        break;
1257
                    case 'upside':
1258
                        $degrees = 180;
1259
                        break;
1260
                    default:
1261
                        break;
1262
                }
1263
            }
1264
1265
            // *** The default direction of imageRotate() is counter clockwise
1266
            //   * This makes it clockwise
1267
            $degrees = 360 - $degrees;
1268
1269
            // *** Create background color
1270
            $bg = imagecolorallocatealpha($this->imageResized, $r, $g, $b, $a);
1271
1272
            // *** Fill with background
1273
            imagefill($this->imageResized, 0, 0, $bg);
1274
1275
            // *** Rotate
1276
            $this->imageResized = imagerotate($this->imageResized, $degrees, $bg); // Rotate 45 degrees and allocated the transparent colour as the one to make transparent (obviously)
1277
1278
            // Ensure alpha transparency
1279
            ImageSaveAlpha($this->imageResized, true);
1280
        }
1281
    }
1282
1283
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1284
        Round corners
1285
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1286
1287
    public function roundCorners($radius = 5, $bgColor = 'transparent')
1288
        # Author:     Jarrod Oberto
1289
        # Date:       19-05-2011
1290
        # Purpose:    Create rounded corners on your image
1291
        # Param in:   (int) radius = the amount of curvature
1292
        #             (mixed) $bgColor = the corner background color
1293
        # Param out:  n/a
1294
        # Reference:
1295
        # Notes:
1296
        #
1297
    {
1298
        // *** Check if the user wants transparency
1299
        $isTransparent = false;
1300
        if (!is_array($bgColor)) {
1301
            if ('transparent' === strtolower($bgColor)) {
1302
                $isTransparent = true;
1303
            }
1304
        }
1305
1306
        // *** If we use transparency, we need to color our curved mask with a unique color
1307
        if ($isTransparent) {
1308
            $bgColor = $this->findUnusedGreen();
1309
        }
1310
1311
        // *** Convert color
1312
        $rgbArray = $this->formatColor($bgColor);
1313
        $r        = $rgbArray['r'];
1314
        $g        = $rgbArray['g'];
1315
        $b        = $rgbArray['b'];
1316
        if (isset($rgbArray['a'])) {
1317
            $a = $rgbArray['a'];
1318
        }
1319
1320
        // *** Create top-left corner mask (square)
1321
        $cornerImg = imagecreatetruecolor($radius, $radius);
1322
        //$cornerImg = imagecreate($radius, $radius);
1323
1324
        //imagealphablending($cornerImg, true);
1325
        //imagesavealpha($cornerImg, true);
1326
1327
        //imagealphablending($this->imageResized, false);
1328
        //imagesavealpha($this->imageResized, true);
1329
1330
        // *** Give it a color
1331
        $maskColor = imagecolorallocate($cornerImg, 0, 0, 0);
1332
1333
        // *** Replace the mask color (black) to transparent
1334
        imagecolortransparent($cornerImg, $maskColor);
1335
1336
        // *** Create the image background color
1337
        $imagebgColor = imagecolorallocate($cornerImg, $r, $g, $b);
1338
1339
        // *** Fill the corner area to the user defined color
1340
        imagefill($cornerImg, 0, 0, $imagebgColor);
1341
1342
        imagefilledellipse($cornerImg, $radius, $radius, $radius * 2, $radius * 2, $maskColor);
1343
1344
        // *** Map to top left corner
1345
        imagecopymerge($this->imageResized, $cornerImg, 0, 0, 0, 0, $radius, $radius, 100); #tl
1346
1347
        // *** Map rounded corner to other corners by rotating and applying the mask
1348
        $cornerImg = imagerotate($cornerImg, 90, 0);
1349
        imagecopymerge($this->imageResized, $cornerImg, 0, $this->height - $radius, 0, 0, $radius, $radius, 100); #bl
1350
1351
        $cornerImg = imagerotate($cornerImg, 90, 0);
1352
        imagecopymerge($this->imageResized, $cornerImg, $this->width - $radius, $this->height - $radius, 0, 0, $radius, $radius, 100); #br
1353
1354
        $cornerImg = imagerotate($cornerImg, 90, 0);
1355
        imagecopymerge($this->imageResized, $cornerImg, $this->width - $radius, 0, 0, 0, $radius, $radius, 100); #tr
1356
1357
        // *** If corners are to be transparent, we fill our chromakey color as transparent.
1358
        if ($isTransparent) {
1359
            //imagecolortransparent($this->imageResized, $imagebgColor);
1360
            $this->imageResized = $this->transparentImage($this->imageResized);
1361
            imagesavealpha($this->imageResized, true);
1362
        }
1363
    }
1364
1365
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1366
        Shadow
1367
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1368
1369
    public function addShadow($shadowAngle = 45, $blur = 15, $bgColor = 'transparent')
1370
        #
1371
        # Author:   Jarrod Oberto (Adapted from Pascal Naidon)
1372
        # Ref:    http://www.les-stooges.org/pascal/webdesign/vignettes/index.php?la=en
1373
        # Purpose:  Add a drop shadow to your image
1374
        # Params in:  (int) $angle: the angle of the shadow
1375
        #       (int) $blur: the blur distance
1376
        #       (mixed) $bgColor: the color of the background
1377
        # Params out:
1378
        # Notes:
1379
        #
1380
    {
1381
        // *** A higher number results in a smoother shadow
1382
        define('STEPS', $blur * 2);
1383
1384
        // *** Set the shadow distance
1385
        $shadowDistance = $blur * 0.25;
1386
1387
        // *** Set blur width and height
1388
        $blurWidth = $blurHeight = $blur;
1389
1390
        if (0 == $shadowAngle) {
1391
            $distWidth  = 0;
1392
            $distHeight = 0;
1393
        } else {
1394
            $distWidth  = $shadowDistance * cos(deg2rad($shadowAngle));
1395
            $distHeight = $shadowDistance * sin(deg2rad($shadowAngle));
1396
        }
1397
1398
        // *** Convert color
1399
        if ('transparent' !== strtolower($bgColor)) {
1400
            $rgbArray = $this->formatColor($bgColor);
1401
            $r0       = $rgbArray['r'];
1402
            $g0       = $rgbArray['g'];
1403
            $b0       = $rgbArray['b'];
1404
        }
1405
1406
        $image  = $this->imageResized;
1407
        $width  = $this->width;
1408
        $height = $this->height;
1409
1410
        $newImage = imagecreatetruecolor($width, $height);
1411
        imagecopyresampled($newImage, $image, 0, 0, 0, 0, $width, $height, $width, $height);
1412
1413
        // *** RGB
1414
        $rgb    = imagecreatetruecolor($width + $blurWidth, $height + $blurHeight);
1415
        $colour = imagecolorallocate($rgb, 0, 0, 0);
1416
        imagefilledrectangle($rgb, 0, 0, $width + $blurWidth, $height + $blurHeight, $colour);
1417
        $colour = imagecolorallocate($rgb, 255, 255, 255);
1418
        //imagefilledrectangle($rgb, $blurWidth*0.5-$distWidth, $blurHeight*0.5-$distHeight, $width+$blurWidth*0.5-$distWidth, $height+$blurWidth*0.5-$distHeight, $colour);
1419
        imagefilledrectangle($rgb, $blurWidth * 0.5 - $distWidth, $blurHeight * 0.5 - $distHeight, $width + $blurWidth * 0.5 - $distWidth, $height + $blurWidth * 0.5 - $distHeight, $colour);
1420
        //imagecopymerge($rgb, $newImage, 1+$blurWidth*0.5-$distWidth, 1+$blurHeight*0.5-$distHeight, 0,0, $width, $height, 100);
1421
        imagecopymerge($rgb, $newImage, $blurWidth * 0.5 - $distWidth, $blurHeight * 0.5 - $distHeight, 0, 0, $width + $blurWidth, $height + $blurHeight, 100);
1422
1423
        // *** Shadow (alpha)
1424
        $shadow = imagecreatetruecolor($width + $blurWidth, $height + $blurHeight);
1425
        imagealphablending($shadow, false);
1426
        $colour = imagecolorallocate($shadow, 0, 0, 0);
1427
        imagefilledrectangle($shadow, 0, 0, $width + $blurWidth, $height + $blurHeight, $colour);
1428
1429
        for ($i = 0; $i <= STEPS; ++$i) {
1430
            $t         = ((1.0 * $i) / STEPS);
1431
            $intensity = 255 * $t * $t;
1432
1433
            $colour = imagecolorallocate($shadow, $intensity, $intensity, $intensity);
1434
            $points = [
1435
                $blurWidth * $t,
1436
                $blurHeight,     // Point 1 (x, y)
1437
                $blurWidth,
1438
                $blurHeight * $t,  // Point 2 (x, y)
1439
                $width,
1440
                $blurHeight * $t,  // Point 3 (x, y)
1441
                $width + $blurWidth * (1 - $t),
1442
                $blurHeight,     // Point 4 (x, y)
1443
                $width + $blurWidth * (1 - $t),
1444
                $height,     // Point 5 (x, y)
1445
                $width,
1446
                $height + $blurHeight * (1 - $t),  // Point 6 (x, y)
1447
                $blurWidth,
1448
                $height + $blurHeight * (1 - $t),  // Point 7 (x, y)
1449
                $blurWidth * $t,
1450
                $height      // Point 8 (x, y)
1451
            ];
1452
            imagepolygon($shadow, $points, 8, $colour);
1453
        }
1454
1455
        for ($i = 0; $i <= STEPS; ++$i) {
1456
            $t         = ((1.0 * $i) / STEPS);
1457
            $intensity = 255 * $t * $t;
1458
1459
            $colour = imagecolorallocate($shadow, $intensity, $intensity, $intensity);
1460
            imagefilledarc($shadow, $blurWidth - 1, $blurHeight - 1, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 180, 268, $colour, IMG_ARC_PIE);
1461
            imagefilledarc($shadow, $width, $blurHeight - 1, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 270, 358, $colour, IMG_ARC_PIE);
1462
            imagefilledarc($shadow, $width, $height, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 0, 90, $colour, IMG_ARC_PIE);
1463
            imagefilledarc($shadow, $blurWidth - 1, $height, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 90, 180, $colour, IMG_ARC_PIE);
1464
        }
1465
1466
        $colour = imagecolorallocate($shadow, 255, 255, 255);
1467
        imagefilledrectangle($shadow, $blurWidth, $blurHeight, $width, $height, $colour);
1468
        imagefilledrectangle($shadow, $blurWidth * 0.5 - $distWidth, $blurHeight * 0.5 - $distHeight, $width + $blurWidth * 0.5 - 1 - $distWidth, $height + $blurHeight * 0.5 - 1 - $distHeight, $colour);
1469
1470
        // *** The magic
1471
        imagealphablending($rgb, false);
1472
1473
        for ($theX = 0; $theX < imagesx($rgb); $theX++) {
1474
            for ($theY = 0; $theY < imagesy($rgb); $theY++) {
1475
                // *** Get the RGB values for every pixel of the RGB image
1476
                $colArray = imagecolorat($rgb, $theX, $theY);
1477
                $r        = ($colArray >> 16) & 0xFF;
1478
                $g        = ($colArray >> 8) & 0xFF;
1479
                $b        = $colArray & 0xFF;
1480
1481
                // *** Get the alpha value for every pixel of the shadow image
1482
                $colArray = imagecolorat($shadow, $theX, $theY);
1483
                $a        = $colArray & 0xFF;
1484
                $a        = 127 - floor($a / 2);
1485
                $t        = $a / 128.0;
1486
1487
                // *** Create color
1488
                if ('transparent' === strtolower($bgColor)) {
1489
                    $myColour = imagecolorallocatealpha($rgb, $r, $g, $b, $a);
1490
                } else {
1491
                    $myColour = imagecolorallocate($rgb, $r * (1.0 - $t) + $r0 * $t, $g * (1.0 - $t) + $g0 * $t, $b * (1.0 - $t) + $b0 * $t);
1492
                }
1493
1494
                // *** Add color to new rgb image
1495
                imagesetpixel($rgb, $theX, $theY, $myColour);
1496
            }
1497
        }
1498
1499
        imagealphablending($rgb, true);
1500
        imagesavealpha($rgb, true);
1501
1502
        $this->imageResized = $rgb;
1503
1504
        imagedestroy($image);
1505
        imagedestroy($newImage);
1506
        imagedestroy($shadow);
1507
    }
1508
1509
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1510
        Add Caption Box
1511
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1512
1513
    public function addCaptionBox($side = 'b', $thickness = 50, $padding = 0, $bgColor = '#000', $transaprencyAmount = 30)
1514
        #
1515
        # Author:   Jarrod Oberto
1516
        # Date:   26 May 2011
1517
        # Purpose:  Add a caption box
1518
        # Params in:  (str) $side: the side to add the caption box (t, r, b, or l).
1519
        #       (int) $thickness: how thick you want the caption box to be.
1520
        #       (mixed) $bgColor: The color of the caption box.
1521
        #       (int) $transaprencyAmount: The amount of transparency to be
1522
        #       applied.
1523
        # Params out: n/a
1524
        # Notes:
1525
        #
1526
    {
1527
        $side = strtolower($side);
1528
1529
        // *** Convert color
1530
        $rgbArray = $this->formatColor($bgColor);
1531
        $r        = $rgbArray['r'];
1532
        $g        = $rgbArray['g'];
1533
        $b        = $rgbArray['b'];
1534
1535
        $positionArray = $this->calculateCaptionBoxPosition($side, $thickness, $padding);
1536
1537
        // *** Store incase we want to use method addTextToCaptionBox()
1538
        $this->captionBoxPositionArray = $positionArray;
1539
1540
        $transaprencyAmount = $this->invertTransparency($transaprencyAmount, 127, false);
1541
        $transparent        = imagecolorallocatealpha($this->imageResized, $r, $g, $b, $transaprencyAmount);
1542
        imagefilledrectangle($this->imageResized, $positionArray['x1'], $positionArray['y1'], $positionArray['x2'], $positionArray['y2'], $transparent);
1543
    }
1544
1545
    ## --------------------------------------------------------
1546
1547
    public function addTextToCaptionBox($text, $fontColor = '#fff', $fontSize = 12, $angle = 0, $font = null)
1548
        #
1549
        # Author:   Jarrod Oberto
1550
        # Date:   03 Aug 11
1551
        # Purpose:  Simplify adding text to a caption box by automatically
1552
        #       locating the center of the caption box
1553
        # Params in:  The usually text paams (less a couple)
1554
        # Params out: n/a
1555
        # Notes:
1556
        #
1557
    {
1558
        // *** Get the caption box measurements
1559
        if (4 == count($this->captionBoxPositionArray)) {
1560
            $x1 = $this->captionBoxPositionArray['x1'];
1561
            $x2 = $this->captionBoxPositionArray['x2'];
1562
            $y1 = $this->captionBoxPositionArray['y1'];
1563
            $y2 = $this->captionBoxPositionArray['y2'];
1564
        } else {
1565
            if ($this->debug) {
1566
                die('No caption box found.');
1567
            } else {
1568
                return false;
1569
            }
1570
        }
1571
1572
        // *** Get text font
1573
        try {
1574
            $font = $this->getTextFont($font);
1575
        } catch (Exception $e) {
1576
        }
1577
1578
        // *** Get text size
1579
        $textSizeArray = $this->getTextSize($fontSize, $angle, $font, $text);
1580
        $textWidth     = $textSizeArray['width'];
1581
        $textHeight    = $textSizeArray['height'];
1582
1583
        // *** Find the width/height middle points
1584
        $boxXMiddle = (($x2 - $x1) / 2);
1585
        $boxYMiddle = (($y2 - $y1) / 2);
1586
1587
        // *** Box middle - half the text width/height
1588
        $xPos = ($x1 + $boxXMiddle) - ($textWidth / 2);
1589
        $yPos = ($y1 + $boxYMiddle) - ($textHeight / 2);
1590
1591
        $pos = $xPos . 'x' . $yPos;
1592
1593
        $this->addText($text, $pos, $padding = 0, $fontColor, $fontSize, $angle, $font);
1594
    }
1595
1596
    ## --------------------------------------------------------
1597
1598
    private function calculateCaptionBoxPosition($side, $thickness, $padding)
1599
    {
1600
        $positionArray = [];
1601
1602
        switch ($side) {
1603
            case 't':
1604
                $positionArray['x1'] = 0;
1605
                $positionArray['y1'] = $padding;
1606
                $positionArray['x2'] = $this->width;
1607
                $positionArray['y2'] = $thickness + $padding;
1608
                break;
1609
            case 'r':
1610
                $positionArray['x1'] = $this->width - $thickness - $padding;
1611
                $positionArray['y1'] = 0;
1612
                $positionArray['x2'] = $this->width - $padding;
1613
                $positionArray['y2'] = $this->height;
1614
                break;
1615
            case 'b':
1616
                $positionArray['x1'] = 0;
1617
                $positionArray['y1'] = $this->height - $thickness - $padding;
1618
                $positionArray['x2'] = $this->width;
1619
                $positionArray['y2'] = $this->height - $padding;
1620
                break;
1621
            case 'l':
1622
                $positionArray['x1'] = $padding;
1623
                $positionArray['y1'] = 0;
1624
                $positionArray['x2'] = $thickness + $padding;
1625
                $positionArray['y2'] = $this->height;
1626
                break;
1627
1628
            default:
1629
                break;
1630
        }
1631
1632
        return $positionArray;
1633
    }
1634
1635
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1636
        Get EXIF Data
1637
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1638
1639
    public function getExif($debug = true)
1640
        # Author:     Jarrod Oberto
1641
        # Date:       07-05-2011
1642
        # Purpose:    Get image EXIF data
1643
        # Param in:   n/a
1644
        # Param out:  An associate array of EXIF data
1645
        # Reference:
1646
        # Notes:
1647
        # 23 May 13 : added orientation flag -jco
1648
        #
1649
    {
1650
        if (!$this->debug || !$debug) {
1651
            $debug = false;
1652
        }
1653
1654
        // *** Check all is good - check the EXIF library exists and the file exists, too.
1655
        if (!$this->testEXIFInstalled()) {
1656
            if ($debug) {
1657
                die('The EXIF Library is not installed.');
1658
            } else {
1659
                return [];
1660
            }
1661
        }
1662
        if (!file_exists($this->fileName)) {
1663
            if ($debug) {
1664
                die('Image not found.');
1665
            } else {
1666
                return [];
1667
            }
1668
        }
1669
        if ('.jpg' !== $this->fileExtension) {
1670
            if ($debug) {
1671
                die('Metadata not supported for this image type.');
1672
            } else {
1673
                return [];
1674
            }
1675
        }
1676
        $exifData = exif_read_data($this->fileName, 'IFD0');
1677
1678
        // *** Format the apperture value
1679
        $ev            = isset($exifData['ApertureValue']) ? $exifData['ApertureValue'] : '';
1680
        $apPeicesArray = explode('/', $ev);
1681
        if (2 == count($apPeicesArray)) {
1682
            $apertureValue = round($apPeicesArray[0] / $apPeicesArray[1], 2, PHP_ROUND_HALF_DOWN) . ' EV';
1683
        } else {
1684
            $apertureValue = '';
1685
        }
1686
1687
        // *** Format the focal length
1688
        $focalLength   = isset($exifData['FocalLength']) ? $exifData['FocalLength'] : '';
1689
        $flPeicesArray = explode('/', $focalLength);
1690
        if (2 == count($flPeicesArray)) {
1691
            $focalLength = $flPeicesArray[0] / $flPeicesArray[1] . '.0 mm';
1692
        } else {
1693
            $focalLength = '';
1694
        }
1695
1696
        // *** Format fNumber
1697
        $fNumber       = isset($exifData['FNumber']) ? $exifData['FNumber'] : '';
1698
        $fnPeicesArray = explode('/', $fNumber);
1699
        if (2 == count($fnPeicesArray)) {
1700
            $fNumber = $fnPeicesArray[0] / $fnPeicesArray[1];
1701
        } else {
1702
            $fNumber = '';
1703
        }
1704
1705
        // *** Resolve ExposureProgram
1706
        if (isset($exifData['ExposureProgram'])) {
1707
            $ep = $exifData['ExposureProgram'];
1708
        }
1709
        if (isset($ep)) {
1710
            $ep = $this->resolveExposureProgram($ep);
1711
        }
1712
1713
        // *** Resolve MeteringMode
1714
        $mm = isset($exifData['MeteringMode']) ? $exifData['MeteringMode'] : '';
1715
        $mm = $this->resolveMeteringMode($mm);
1716
1717
        // *** Resolve Flash
1718
        $flash = isset($exifData['Flash']) ? $exifData['Flash'] : '';
1719
        $flash = $this->resolveFlash($flash);
1720
1721
        if (isset($exifData['Make'])) {
1722
            $exifDataArray['make'] = $exifData['Make'];
1723
        } else {
1724
            $exifDataArray['make'] = '';
1725
        }
1726
1727
        if (isset($exifData['Model'])) {
1728
            $exifDataArray['model'] = $exifData['Model'];
1729
        } else {
1730
            $exifDataArray['model'] = '';
1731
        }
1732
1733
        if (isset($exifData['DateTime'])) {
1734
            $exifDataArray['date'] = $exifData['DateTime'];
1735
        } else {
1736
            $exifDataArray['date'] = '';
1737
        }
1738
1739
        if (isset($exifData['ExposureTime'])) {
1740
            $exifDataArray['exposure time'] = $exifData['ExposureTime'] . ' sec.';
1741
        } else {
1742
            $exifDataArray['exposure time'] = '';
1743
        }
1744
1745
        if ('' != $apertureValue) {
1746
            $exifDataArray['aperture value'] = $apertureValue;
1747
        } else {
1748
            $exifDataArray['aperture value'] = '';
1749
        }
1750
1751
        if (isset($exifData['COMPUTED']['ApertureFNumber'])) {
1752
            $exifDataArray['f-stop'] = $exifData['COMPUTED']['ApertureFNumber'];
1753
        } else {
1754
            $exifDataArray['f-stop'] = '';
1755
        }
1756
1757
        if (isset($exifData['FNumber'])) {
1758
            $exifDataArray['fnumber'] = $exifData['FNumber'];
1759
        } else {
1760
            $exifDataArray['fnumber'] = '';
1761
        }
1762
1763
        if ('' != $fNumber) {
1764
            $exifDataArray['fnumber value'] = $fNumber;
1765
        } else {
1766
            $exifDataArray['fnumber value'] = '';
1767
        }
1768
1769
        if (isset($exifData['ISOSpeedRatings'])) {
1770
            $exifDataArray['iso'] = $exifData['ISOSpeedRatings'];
1771
        } else {
1772
            $exifDataArray['iso'] = '';
1773
        }
1774
1775
        if ('' != $focalLength) {
1776
            $exifDataArray['focal length'] = $focalLength;
1777
        } else {
1778
            $exifDataArray['focal length'] = '';
1779
        }
1780
1781
        if (isset($ep)) {
1782
            $exifDataArray['exposure program'] = $ep;
1783
        } else {
1784
            $exifDataArray['exposure program'] = '';
1785
        }
1786
1787
        if ('' != $mm) {
1788
            $exifDataArray['metering mode'] = $mm;
1789
        } else {
1790
            $exifDataArray['metering mode'] = '';
1791
        }
1792
1793
        if ('' != $flash) {
1794
            $exifDataArray['flash status'] = $flash;
1795
        } else {
1796
            $exifDataArray['flash status'] = '';
1797
        }
1798
1799
        if (isset($exifData['Artist'])) {
1800
            $exifDataArray['creator'] = $exifData['Artist'];
1801
        } else {
1802
            $exifDataArray['creator'] = '';
1803
        }
1804
1805
        if (isset($exifData['Copyright'])) {
1806
            $exifDataArray['copyright'] = $exifData['Copyright'];
1807
        } else {
1808
            $exifDataArray['copyright'] = '';
1809
        }
1810
1811
        // *** Orientation
1812
        if (isset($exifData['Orientation'])) {
1813
            $exifDataArray['orientation'] = $exifData['Orientation'];
1814
        } else {
1815
            $exifDataArray['orientation'] = '';
1816
        }
1817
1818
        return $exifDataArray;
1819
    }
1820
1821
    ## --------------------------------------------------------
1822
1823
    private function resolveExposureProgram($ep)
1824
    {
1825
        switch ($ep) {
1826
            case 0:
1827
                $ep = '';
1828
                break;
1829
            case 1:
1830
                $ep = 'manual';
1831
                break;
1832
            case 2:
1833
                $ep = 'normal program';
1834
                break;
1835
            case 3:
1836
                $ep = 'aperture priority';
1837
                break;
1838
            case 4:
1839
                $ep = 'shutter priority';
1840
                break;
1841
            case 5:
1842
                $ep = 'creative program';
1843
                break;
1844
            case 6:
1845
                $ep = 'action program';
1846
                break;
1847
            case 7:
1848
                $ep = 'portrait mode';
1849
                break;
1850
            case 8:
1851
                $ep = 'landscape mode';
1852
                break;
1853
1854
            default:
1855
                break;
1856
        }
1857
1858
        return $ep;
1859
    }
1860
1861
    ## --------------------------------------------------------
1862
1863
    private function resolveMeteringMode($mm)
1864
    {
1865
        switch ($mm) {
1866
            case 0:
1867
                $mm = 'unknown';
1868
                break;
1869
            case 1:
1870
                $mm = 'average';
1871
                break;
1872
            case 2:
1873
                $mm = 'center weighted average';
1874
                break;
1875
            case 3:
1876
                $mm = 'spot';
1877
                break;
1878
            case 4:
1879
                $mm = 'multi spot';
1880
                break;
1881
            case 5:
1882
                $mm = 'pattern';
1883
                break;
1884
            case 6:
1885
                $mm = 'partial';
1886
                break;
1887
            case 255:
1888
                $mm = 'other';
1889
                break;
1890
1891
            default:
1892
                break;
1893
        }
1894
1895
        return $mm;
1896
    }
1897
1898
    ## --------------------------------------------------------
1899
1900
    private function resolveFlash($flash)
1901
    {
1902
        switch ($flash) {
1903
            case 0:
1904
                $flash = 'flash did not fire';
1905
                break;
1906
            case 1:
1907
                $flash = 'flash fired';
1908
                break;
1909
            case 5:
1910
                $flash = 'strobe return light not detected';
1911
                break;
1912
            case 7:
1913
                $flash = 'strobe return light detected';
1914
                break;
1915
            case 9:
1916
                $flash = 'flash fired, compulsory flash mode';
1917
                break;
1918
            case 13:
1919
                $flash = 'flash fired, compulsory flash mode, return light not detected';
1920
                break;
1921
            case 15:
1922
                $flash = 'flash fired, compulsory flash mode, return light detected';
1923
                break;
1924
            case 16:
1925
                $flash = 'flash did not fire, compulsory flash mode';
1926
                break;
1927
            case 24:
1928
                $flash = 'flash did not fire, auto mode';
1929
                break;
1930
            case 25:
1931
                $flash = 'flash fired, auto mode';
1932
                break;
1933
            case 29:
1934
                $flash = 'flash fired, auto mode, return light not detected';
1935
                break;
1936
            case 31:
1937
                $flash = 'flash fired, auto mode, return light detected';
1938
                break;
1939
            case 32:
1940
                $flash = 'no flash function';
1941
                break;
1942
            case 65:
1943
                $flash = 'flash fired, red-eye reduction mode';
1944
                break;
1945
            case 69:
1946
                $flash = 'flash fired, red-eye reduction mode, return light not detected';
1947
                break;
1948
            case 71:
1949
                $flash = 'flash fired, red-eye reduction mode, return light detected';
1950
                break;
1951
            case 73:
1952
                $flash = 'flash fired, compulsory flash mode, red-eye reduction mode';
1953
                break;
1954
            case 77:
1955
                $flash = 'flash fired, compulsory flash mode, red-eye reduction mode, return light not detected';
1956
                break;
1957
            case 79:
1958
                $flash = 'flash fired, compulsory flash mode, red-eye reduction mode, return light detected';
1959
                break;
1960
            case 89:
1961
                $flash = 'flash fired, auto mode, red-eye reduction mode';
1962
                break;
1963
            case 93:
1964
                $flash = 'flash fired, auto mode, return light not detected, red-eye reduction mode';
1965
                break;
1966
            case 95:
1967
                $flash = 'flash fired, auto mode, return light detected, red-eye reduction mode';
1968
                break;
1969
1970
            default:
1971
                break;
1972
        }
1973
1974
        return $flash;
1975
    }
1976
1977
1978
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1979
        Get IPTC Data
1980
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1981
1982
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1983
        Write IPTC Data
1984
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1985
1986
    public function writeIPTCcaption($value)
1987
        # Caption
1988
    {
1989
        $this->writeIPTC(120, $value);
1990
    }
1991
1992
    ## --------------------------------------------------------
1993
1994
    public function writeIPTCwriter($value)
1995
    {
1996
        //$this->writeIPTC(65, $value);
1997
    }
1998
1999
    ## --------------------------------------------------------
2000
2001
    private function writeIPTC($dat, $value)
2002
    {
2003
        # LIMIT TO JPG
2004
2005
        $caption_block = $this->iptc_maketag(2, $dat, $value);
2006
        $image_string  = iptcembed($caption_block, $this->fileName);
2007
        file_put_contents('iptc.jpg', $image_string);
2008
    }
2009
2010
    ## --------------------------------------------------------
2011
2012
    private function iptc_maketag($rec, $dat, $val)
2013
        # Author:   Thies C. Arntzen
2014
        # Purpose:    Function to format the new IPTC text
2015
        # Param in:   $rec: Application record. (We’re working with #2)
2016
        #       $dat: Index. (120 for caption, 118 for contact. See the IPTC IIM
2017
        #         specification:
2018
        #         http://www.iptc.org/std/IIM/4.1/specification/IIMV4.1.pdf
2019
        #       $val: Value/data/text. Make sure this is within the length
2020
        #         constraints of the IPTC IIM specification
2021
        # Ref:      http://blog.peterhaza.no/working-with-image-meta-data-in-exif-and-iptc-headers-from-php/
2022
        #       http://php.net/manual/en/function.iptcembed.php
2023
        #
2024
    {
2025
        $len = strlen($val);
2026
        if ($len < 0x8000) {
2027
            return chr(0x1c) . chr($rec) . chr($dat) . chr($len >> 8) . chr($len & 0xff) . $val;
2028
        } else {
2029
            return chr(0x1c) . chr($rec) . chr($dat) . chr(0x80) . chr(0x04) . chr(($len >> 24) & 0xff) . chr(($len >> 16) & 0xff) . chr(($len >> 8) & 0xff) . chr(($len) & 0xff) . $val;
2030
        }
2031
    }
2032
2033
2034
2035
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
2036
        Write XMP Data
2037
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
2038
2039
    //http://xmpphptoolkit.sourceforge.net/
2040
2041
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
2042
        Add Text
2043
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
2044
2045
    public function addText($text, $pos = '20x20', $padding = 0, $fontColor = '#fff', $fontSize = 12, $angle = 0, $font = null)
2046
        # Author:     Jarrod Oberto
2047
        # Date:       18-11-09
2048
        # Purpose:    Add text to an image
2049
        # Param in:
2050
        # Param out:
2051
        # Reference:  http://php.net/manual/en/function.imagettftext.php
2052
        # Notes:      Make sure you supply the font.
2053
        #
2054
    {
2055
        // *** Convert color
2056
        $rgbArray = $this->formatColor($fontColor);
2057
        $r        = $rgbArray['r'];
2058
        $g        = $rgbArray['g'];
2059
        $b        = $rgbArray['b'];
2060
2061
        // *** Get text font
2062
        try {
2063
            $font = $this->getTextFont($font);
2064
        } catch (Exception $e) {
2065
        }
2066
2067
        // *** Get text size
2068
        $textSizeArray = $this->getTextSize($fontSize, $angle, $font, $text);
2069
        $textWidth     = $textSizeArray['width'];
2070
        $textHeight    = $textSizeArray['height'];
2071
2072
        // *** Find co-ords to place text
2073
        $posArray = $this->calculatePosition($pos, $padding, $textWidth, $textHeight, false);
2074
        $x        = $posArray['width'];
2075
        $y        = $posArray['height'];
2076
2077
        $fontColor = imagecolorallocate($this->imageResized, $r, $g, $b);
2078
2079
        // *** Add text
2080
        imagettftext($this->imageResized, $fontSize, $angle, $x, $y, $fontColor, $font, $text);
2081
    }
2082
2083
    ## --------------------------------------------------------
2084
2085
    private function getTextFont($font)
2086
    {
2087
        // *** Font path (shou
2088
        $fontPath = __DIR__ . '/' . $this->fontDir;
2089
2090
        // *** The below is/may be needed depending on your version (see ref)
2091
        putenv('GDFONTPATH=' . realpath('.'));
2092
2093
        // *** Check if the passed in font exsits...
2094
        if (null == $font || !file_exists($font)) {
2095
            // *** ...If not, default to this font.
2096
            $font = $fontPath . '/arimo.ttf';
2097
2098
            // *** Check our default font exists...
2099
            if (!file_exists($font)) {
2100
                // *** If not, return false
2101
                if ($this->debug) {
2102
                    die('Font not found');
2103
                } else {
2104
                    return false;
2105
                }
2106
            }
2107
        }
2108
2109
        return $font;
2110
    }
2111
2112
    ## --------------------------------------------------------
2113
2114
    private function getTextSize($fontSize, $angle, $font, $text)
2115
    {
2116
        // *** Define box (so we can get the width)
2117
        $box = @imagettfbbox($fontSize, $angle, $font, $text);
2118
2119
        // ***  Get width of text from dimensions
2120
        $textWidth = abs($box[4] - $box[0]);
2121
2122
        // ***  Get height of text from dimensions (should also be same as $fontSize)
2123
        $textHeight = abs($box[5] - $box[1]);
2124
2125
        return ['height' => $textHeight, 'width' => $textWidth];
2126
    }
2127
2128
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
2129
        Add Watermark
2130
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
2131
2132
    public function addWatermark($watermarkImage, $pos, $padding = 0, $opacity = 0)
2133
        # Author:     Jarrod Oberto
2134
        # Date:       18-11-09
2135
        # Purpose:    Add watermark image
2136
        # Param in:   (str) $watermark: The watermark image
2137
        #             (str) $pos: Could be a pre-determined position such as:
2138
        #                 tl = top left,
2139
        #                 t  = top (middle),
2140
        #                 tr = top right,
2141
        #                 l  = left,
2142
        #                 m  = middle,
2143
        #                 r  = right,
2144
        #                 bl = bottom left,
2145
        #                 b  = bottom (middle),
2146
        #                 br = bottom right
2147
        #               Or, it could be a co-ordinate position such as: 50x100
2148
        #
2149
        #             (int) $padding: If using a pre-determined position you can
2150
        #               adjust the padding from the edges by passing an amount
2151
        #               in pixels. If using co-ordinates, this value is ignored.
2152
        # Param out:
2153
        # Reference:  http://www.php.net/manual/en/image.examples-watermark.php
2154
        # Notes:      Based on example in reference.
2155
        #
2156
        #
2157
    {
2158
        // Load the stamp and the photo to apply the watermark to
2159
        try {
2160
            $stamp = $this->openImage($watermarkImage);
2161
        } catch (Exception $e) {
2162
        }    # stamp
2163
        $im = $this->imageResized;            # photo
2164
2165
        // *** Get stamps width and height
2166
        $sx = imagesx($stamp);
2167
        $sy = imagesy($stamp);
2168
2169
        // *** Find co-ords to place image
2170
        $posArray = $this->calculatePosition($pos, $padding, $sx, $sy);
2171
        $x        = $posArray['width'];
2172
        $y        = $posArray['height'];
2173
2174
        // *** Set watermark opacity
2175
        if ('.png' === strtolower(strrchr($watermarkImage, '.'))) {
2176
            $opacity = $this->invertTransparency($opacity, 100);
2177
            $this->filterOpacity($stamp, $opacity);
2178
        }
2179
2180
        // Copy the watermark image onto our photo
2181
        imagecopy($im, $stamp, $x, $y, 0, 0, imagesx($stamp), imagesy($stamp));
2182
    }
2183
2184
    ## --------------------------------------------------------
2185
2186
    private function calculatePosition($pos, $padding, $assetWidth, $assetHeight, $upperLeft = true)
2187
        #
2188
        # Author:   Jarrod Oberto
2189
        # Date:   08-05-11
2190
        # Purpose:  Calculate the x, y pixel cordinates of the asset to place
2191
        # Params in:  (str) $pos: Either something like: "tl", "l", "br" or an
2192
        #         exact position like: "100x50"
2193
        #       (int) $padding: The amount of padding from the edge. Only
2194
        #         used for the predefined $pos.
2195
        #       (int) $assetWidth: The width of the asset to add to the image
2196
        #       (int) $assetHeight: The height of the asset to add to the image
2197
        #       (bol) $upperLeft: if true, the asset will be positioned based
2198
        #         on the upper left x, y coords. If false, it means you're
2199
        #         using the lower left as the basepoint and this will
2200
        #         convert it to the upper left position
2201
        # Params out:
2202
        # NOTE: this is done from the UPPER left corner!! But will convert lower
2203
        #   left basepoints to upper left if $upperleft is set to false
2204
        #
2205
        #
2206
    {
2207
        $pos = strtolower($pos);
2208
2209
        // *** If co-ords have been entered
2210
        if (strstr($pos, 'x')) {
2211
            $pos = str_replace(' ', '', $pos);
2212
2213
            $xyArray = explode('x', $pos);
2214
            list($width, $height) = $xyArray;
2215
        } else {
2216
            switch ($pos) {
2217
                case 'tl':
2218
                    $width  = 0 + $padding;
2219
                    $height = 0 + $padding;
2220
                    break;
2221
2222
                case 't':
2223
                    $width  = ($this->width / 2) - ($assetWidth / 2);
2224
                    $height = 0 + $padding;
2225
                    break;
2226
2227
                case 'tr':
2228
                    $width  = $this->width - $assetWidth - $padding;
2229
                    $height = 0 + $padding;
2230
                    break;
2231
2232
                case 'l':
2233
                    $width  = 0 + $padding;
2234
                    $height = ($this->height / 2) - ($assetHeight / 2);
2235
                    break;
2236
2237
                case 'm':
2238
                    $width  = ($this->width / 2) - ($assetWidth / 2);
2239
                    $height = ($this->height / 2) - ($assetHeight / 2);
2240
                    break;
2241
2242
                case 'r':
2243
                    $width  = $this->width - $assetWidth - $padding;
2244
                    $height = ($this->height / 2) - ($assetHeight / 2);
2245
                    break;
2246
2247
                case 'bl':
2248
                    $width  = 0 + $padding;
2249
                    $height = $this->height - $assetHeight - $padding;
2250
                    break;
2251
2252
                case 'b':
2253
                    $width  = ($this->width / 2) - ($assetWidth / 2);
2254
                    $height = $this->height - $assetHeight - $padding;
2255
                    break;
2256
2257
                case 'br':
2258
                    $width  = $this->width - $assetWidth - $padding;
2259
                    $height = $this->height - $assetHeight - $padding;
2260
                    break;
2261
2262
                default:
2263
                    $width  = 0;
2264
                    $height = 0;
2265
                    break;
2266
            }
2267
        }
2268
2269
        if (!$upperLeft) {
2270
            $height = $height + $assetHeight;
2271
        }
2272
2273
        return ['width' => $width, 'height' => $height];
2274
    }
2275
2276
    ## --------------------------------------------------------
2277
2278
    private function filterOpacity(&$img, $opacity = 75)
2279
        #
2280
        # Author:     aiden dot mail at freemail dot hu
2281
        # Author date:  29-03-08 08:16
2282
        # Date added:   08-05-11
2283
        # Purpose:    Change opacity of image
2284
        # Params in:    $img: Image resource id
2285
        #         (int) $opacity: the opacity amount: 0-100, 100 being not opaque.
2286
        # Params out:   (bool) true on success, else false
2287
        # Ref:      http://www.php.net/manual/en/function.imagefilter.php#82162
2288
        # Notes:      png only
2289
        #
2290
    {
2291
        if (!isset($opacity)) {
2292
            return false;
2293
        }
2294
2295
        if (100 == $opacity) {
2296
            return true;
2297
        }
2298
2299
        $opacity /= 100;
2300
2301
        //get image width and height
2302
        $w = imagesx($img);
2303
        $h = imagesy($img);
2304
2305
        //turn alpha blending off
2306
        imagealphablending($img, false);
2307
2308
        //find the most opaque pixel in the image (the one with the smallest alpha value)
2309
        $minalpha = 127;
2310
        for ($x = 0; $x < $w; ++$x) {
2311
            for ($y = 0; $y < $h; ++$y) {
2312
                $alpha = (imagecolorat($img, $x, $y) >> 24) & 0xFF;
2313
                if ($alpha < $minalpha) {
2314
                    $minalpha = $alpha;
2315
                }
2316
            }
2317
        }
2318
2319
        //loop through image pixels and modify alpha for each
2320
        for ($x = 0; $x < $w; ++$x) {
2321
            for ($y = 0; $y < $h; ++$y) {
2322
                //get current alpha value (represents the TANSPARENCY!)
2323
                $colorxy = imagecolorat($img, $x, $y);
2324
                $alpha   = ($colorxy >> 24) & 0xFF;
2325
                //calculate new alpha
2326
                if (127 !== $minalpha) {
2327
                    $alpha = 127 + 127 * $opacity * ($alpha - 127) / (127 - $minalpha);
2328
                } else {
2329
                    $alpha += 127 * $opacity;
2330
                }
2331
                //get the color index with new alpha
2332
                $alphacolorxy = imagecolorallocatealpha($img, ($colorxy >> 16) & 0xFF, ($colorxy >> 8) & 0xFF, $colorxy & 0xFF, $alpha);
2333
                //set pixel with the new color + opacity
2334
                if (!imagesetpixel($img, $x, $y, $alphacolorxy)) {
2335
                    return false;
2336
                }
2337
            }
2338
        }
2339
2340
        return true;
2341
    }
2342
2343
    ## --------------------------------------------------------
2344
2345
    private function openImage($file)
2346
        # Author:     Jarrod Oberto
2347
        # Date:       27-02-08
2348
        # Purpose:
2349
        # Param in:
2350
        # Param out:  n/a
2351
        # Reference:
2352
        # Notes:
2353
        #
2354
    {
2355
        if (!file_exists($file) && !$this->checkStringStartsWith('http://', $file)) {
2356
            if ($this->debug) {
2357
                die('Image not found.');
2358
            } else {
2359
                die();
2360
            }
2361
        }
2362
2363
        // *** Get extension
2364
        $extension = strrchr($file, '.');
2365
        $extension = strtolower($extension);
2366
2367
        switch ($extension) {
2368
            case '.jpg':
2369
            case '.jpeg':
2370
                $img = @imagecreatefromjpeg($file);
2371
                break;
2372
            case '.gif':
2373
                $img = @imagecreatefromgif($file);
2374
                break;
2375
            case '.png':
2376
                $img = @imagecreatefrompng($file);
2377
                break;
2378
            case '.bmp':
2379
                $img = @$this->ImageCreateFromBMP($file);
2380
                break;
2381
            case '.psd':
2382
                $img = @$this->imagecreatefrompsd($file);
2383
                break;
2384
2385
            // ... etc
2386
2387
            default:
2388
                $img = false;
2389
                break;
2390
        }
2391
2392
        return $img;
2393
    }
2394
2395
    ## --------------------------------------------------------
2396
2397
    public function reset()
2398
        #
2399
        # Author:   Jarrod Oberto
2400
        # Date:   30-08-11
2401
        # Purpose:  Reset the resource (allow further editing)
2402
        # Params in:
2403
        # Params out:
2404
        # Notes:
2405
        #
2406
    {
2407
        $this->__destruct();
2408
        $this->image        = null;
2409
        $this->imageResized = null;
2410
        gc_collect_cycles();
2411
        try {
2412
            $this->__construct($this->fileName);
2413
        } catch (Exception $e) {
2414
        }
2415
    }
2416
2417
    ## --------------------------------------------------------
2418
2419
    public function saveImage($savePath, $imageQuality = '100')
2420
        # Author:     Jarrod Oberto
2421
        # Date:       27-02-08
2422
        # Purpose:    Saves the image
2423
        # Param in:   $savePath: Where to save the image including filename:
2424
        #             $imageQuality: image quality you want the image saved at 0-100
2425
        # Param out:  n/a
2426
        # Reference:
2427
        # Notes:      * gif doesn't have a quality parameter
2428
        #             * jpg has a quality setting 0-100 (100 being the best)
2429
        #             * png has a quality setting 0-9 (0 being the best)
2430
        #
2431
        #             * bmp files have no native support for bmp files. We use a
2432
        #               third party class to save as bmp.
2433
    {
2434
        // *** Perform a check or two.
2435
        if (!is_resource($this->imageResized)) {
2436
            if ($this->debug) {
2437
                die('saveImage: This is not a resource.');
2438
            } else {
2439
                die();
2440
            }
2441
        }
2442
        $fileInfoArray = pathinfo($savePath);
2443
        clearstatcache();
2444
        if (!is_writable($fileInfoArray['dirname'])) {
2445
            if ($this->debug) {
2446
                die('The path is not writable. Please check your permissions.');
2447
            } else {
2448
                die();
2449
            }
2450
        }
2451
2452
        // *** Get extension
2453
        $extension = strrchr($savePath, '.');
2454
        $extension = strtolower($extension);
2455
2456
        $error = '';
2457
2458
        switch ($extension) {
2459
            case '.jpg':
2460
            case '.jpeg':
2461
                $this->checkInterlaceImage($this->isInterlace);
2462
                if (imagetypes() & IMG_JPG) {
2463
                    imagejpeg($this->imageResized, $savePath, $imageQuality);
2464
                } else {
2465
                    $error = 'jpg';
2466
                }
2467
                break;
2468
2469
            case '.gif':
2470
                $this->checkInterlaceImage($this->isInterlace);
2471
                if (imagetypes() & IMG_GIF) {
2472
                    imagegif($this->imageResized, $savePath);
2473
                } else {
2474
                    $error = 'gif';
2475
                }
2476
                break;
2477
2478
            case '.png':
2479
                // *** Scale quality from 0-100 to 0-9
2480
                $scaleQuality = round(($imageQuality / 100) * 9);
2481
2482
                // *** Invert qualit setting as 0 is best, not 9
2483
                $invertScaleQuality = 9 - $scaleQuality;
2484
2485
                $this->checkInterlaceImage($this->isInterlace);
2486
                if (imagetypes() & IMG_PNG) {
2487
                    imagepng($this->imageResized, $savePath, $invertScaleQuality);
2488
                } else {
2489
                    $error = 'png';
2490
                }
2491
                break;
2492
2493
            case '.bmp':
2494
                file_put_contents($savePath, $this->GD2BMPstring($this->imageResized));
2495
                break;
2496
2497
            // ... etc
2498
2499
            default:
2500
2501
                // *** No extension - No save.
2502
                $this->errorArray[] = 'This file type (' . $extension . ') is not supported. File not saved.';
2503
                break;
2504
        }
2505
2506
        //imagedestroy($this->imageResized);
2507
2508
        // *** Display error if a file type is not supported.
2509
        if ('' != $error) {
2510
            $this->errorArray[] = $error . ' support is NOT enabled. File not saved.';
2511
        }
2512
    }
2513
2514
    ## --------------------------------------------------------
2515
2516
    public function displayImage($fileType = 'jpg', $imageQuality = '100')
2517
        # Author:     Jarrod Oberto
2518
        # Date:       18-11-09
2519
        # Purpose:    Display images directly to the browser
2520
        # Param in:   The image type you want to display
2521
        # Param out:
2522
        # Reference:
2523
        # Notes:
2524
        #
2525
    {
2526
        if (!is_resource($this->imageResized)) {
2527
            if ($this->debug) {
2528
                die('saveImage: This is not a resource.');
2529
            } else {
2530
                die();
2531
            }
2532
        }
2533
2534
        switch ($fileType) {
2535
            case 'jpg':
2536
            case 'jpeg':
2537
                header('Content-type: image/jpeg');
2538
                imagejpeg($this->imageResized, '', $imageQuality);
2539
                break;
2540
            case 'gif':
2541
                header('Content-type: image/gif');
2542
                imagegif($this->imageResized);
2543
                break;
2544
            case 'png':
2545
                header('Content-type: image/png');
2546
2547
                // *** Scale quality from 0-100 to 0-9
2548
                $scaleQuality = round(($imageQuality / 100) * 9);
2549
2550
                // *** Invert qualit setting as 0 is best, not 9
2551
                $invertScaleQuality = 9 - $scaleQuality;
2552
2553
                imagepng($this->imageResized, '', $invertScaleQuality);
2554
                break;
2555
            case 'bmp':
2556
                echo 'bmp file format is not supported.';
2557
                break;
2558
2559
            // ... etc
2560
2561
            default:
2562
                // *** No extension - No save.
2563
                break;
2564
        }
2565
        //imagedestroy($this->imageResized);
2566
    }
2567
2568
    ## --------------------------------------------------------
2569
2570
    public function setTransparency($bool)
2571
        # Sep 2011
2572
    {
2573
        $this->keepTransparency = $bool;
2574
    }
2575
2576
    ## --------------------------------------------------------
2577
2578
    public function setFillColor($value)
2579
        # Sep 2011
2580
        # Param in:   (mixed) $value: (array) Could be an array of RGB
2581
        #             (str) Could be hex #ffffff or #fff, fff, ffffff
2582
        #
2583
        # If the keepTransparency is set to false, then no transparency is to be used.
2584
        # This is ideal when you want to save as jpg.
2585
        #
2586
        # this method allows you to set the background color to use instead of
2587
        # transparency.
2588
        #
2589
    {
2590
        $colorArray           = $this->formatColor($value);
2591
        $this->fillColorArray = $colorArray;
2592
    }
2593
2594
    ## --------------------------------------------------------
2595
2596
    public function setCropFromTop($value)
2597
        # Sep 2011
2598
    {
2599
        $this->cropFromTopPercent = $value;
2600
    }
2601
2602
    ## --------------------------------------------------------
2603
2604
    public function testGDInstalled()
2605
        # Author:     Jarrod Oberto
2606
        # Date:       27-02-08
2607
        # Purpose:    Test to see if GD is installed
2608
        # Param in:   n/a
2609
        # Param out:  (bool) True is gd extension loaded otherwise false
2610
        # Reference:
2611
        # Notes:
2612
        #
2613
    {
2614
        if (extension_loaded('gd') && function_exists('gd_info')) {
2615
            $gdInstalled = true;
2616
        } else {
2617
            $gdInstalled = false;
2618
        }
2619
2620
        return $gdInstalled;
2621
    }
2622
2623
    ## --------------------------------------------------------
2624
2625
    public function testEXIFInstalled()
2626
        # Author:     Jarrod Oberto
2627
        # Date:       08-05-11
2628
        # Purpose:    Test to see if EXIF is installed
2629
        # Param in:   n/a
2630
        # Param out:  (bool) True is exif extension loaded otherwise false
2631
        # Reference:
2632
        # Notes:
2633
        #
2634
    {
2635
        if (extension_loaded('exif')) {
2636
            $exifInstalled = true;
2637
        } else {
2638
            $exifInstalled = false;
2639
        }
2640
2641
        return $exifInstalled;
2642
    }
2643
2644
    ## --------------------------------------------------------
2645
2646
    public function testIsImage()
2647
        # Author:     Jarrod Oberto
2648
        # Date:       28 Nov 16
2649
        # Purpose:    Test if file is an image
2650
        # Param in:
2651
        # Param out:  n/a
2652
        # Reference:
2653
        # Notes: A simpler, less restrictive method would be to just check for
2654
        #           the 'image' part of 'image/gif', 'image/jpg', etc.
2655
        #
2656
    {
2657
        $file    = $this->fileName;
2658
        $isImage = false;
2659
2660
        $finfo    = finfo_open(FILEINFO_MIME_TYPE);
2661
        $mimeType = finfo_file($finfo, $file);
2662
        finfo_close($finfo);
2663
2664
        switch ($mimeType) {
2665
            case 'image/jpeg':
2666
            case 'image/gif':
2667
            case 'image/png':
2668
            case 'image/bmp':
2669
            case 'image/x-windows-bmp':
2670
                $isImage = true;
2671
                break;
2672
            default:
2673
                $isImage = false;
2674
        }
2675
2676
        return $isImage;
2677
    }
2678
2679
    ## --------------------------------------------------------
2680
2681
    public function getIsImage()
2682
        # Author:     Jarrod Oberto
2683
        # Date:       28 Nov 16
2684
        # Purpose:    Get testIsImage result
2685
        # Param in:
2686
        # Param out:  n/a
2687
        # Reference:
2688
        # Notes:
2689
        #
2690
    {
2691
        return $this->isImage;
2692
    }
2693
2694
    ## --------------------------------------------------------
2695
2696
    public function testFunct()
2697
        # Author:     Jarrod Oberto
2698
        # Date:       27-02-08
2699
        # Purpose:    Test Function
2700
        # Param in:   n/a
2701
        # Param out:  n/a
2702
        # Reference:
2703
        # Notes:
2704
        #
2705
    {
2706
        echo $this->height;
2707
    }
2708
2709
    ## --------------------------------------------------------
2710
2711
    public function setForceStretch($value)
2712
        # Author:     Jarrod Oberto
2713
        # Date:       23-12-10
2714
        # Purpose:
2715
        # Param in:   (bool) $value
2716
        # Param out:  n/a
2717
        # Reference:
2718
        # Notes:
2719
        #
2720
    {
2721
        $this->forceStretch = $value;
2722
    }
2723
2724
    ## --------------------------------------------------------
2725
2726
    public function setFile($fileName)
2727
        # Author:     Jarrod Oberto
2728
        # Date:       28-02-08
2729
        # Purpose:
2730
        # Param in:   n/a
2731
        # Param out:  n/a
2732
        # Reference:
2733
        # Notes:
2734
        #
2735
    {
2736
        try {
2737
            self::__construct($fileName);
2738
        } catch (Exception $e) {
2739
        }
2740
    }
2741
2742
    ## --------------------------------------------------------
2743
2744
    public function getFileName()
2745
        # Author:     Jarrod Oberto
2746
        # Date:       10-09-08
2747
        # Purpose:
2748
        # Param in:   n/a
2749
        # Param out:  n/a
2750
        # Reference:
2751
        # Notes:
2752
        #
2753
    {
2754
        return $this->fileName;
2755
    }
2756
2757
    ## --------------------------------------------------------
2758
2759
    public function getHeight()
2760
    {
2761
        return $this->height;
2762
    }
2763
2764
    ## --------------------------------------------------------
2765
2766
    public function getWidth()
2767
    {
2768
        return $this->width;
2769
    }
2770
2771
    ## --------------------------------------------------------
2772
2773
    public function getOriginalHeight()
2774
    {
2775
        return $this->heightOriginal;
2776
    }
2777
2778
    ## --------------------------------------------------------
2779
2780
    public function getOriginalWidth()
2781
    {
2782
        return $this->widthOriginal;
2783
    }
2784
2785
    ## --------------------------------------------------------
2786
2787
    public function getErrors()
2788
        # Author:     Jarrod Oberto
2789
        # Date:       19-11-09
2790
        # Purpose:    Returns the error array
2791
        # Param in:   n/a
2792
        # Param out:  Array of errors
2793
        # Reference:
2794
        # Notes:
2795
        #
2796
    {
2797
        return $this->errorArray;
2798
    }
2799
2800
    ## --------------------------------------------------------
2801
2802
    private function checkInterlaceImage($isEnabled)
2803
        # jpg will use progressive (they don't use interace)
2804
    {
2805
        if ($isEnabled) {
2806
            imageinterlace($this->imageResized, $isEnabled);
2807
        }
2808
    }
2809
2810
    ## --------------------------------------------------------
2811
2812
    protected function formatColor($value)
2813
        # Author:     Jarrod Oberto
2814
        # Date:       09-05-11
2815
        # Purpose:    Determine color method passed in and return color as RGB
2816
        # Param in:   (mixed) $value: (array) Could be an array of RGB
2817
        #             (str) Could be hex #ffffff or #fff, fff, ffffff
2818
        # Param out:
2819
        # Reference:
2820
        # Notes:
2821
        #
2822
    {
2823
        $rgbArray = [];
2824
2825
        // *** If it's an array it should be R, G, B
2826
        if (is_array($value)) {
2827
            if (0 == key($value) && 3 == count($value)) {
2828
                $rgbArray['r'] = $value[0];
2829
                $rgbArray['g'] = $value[1];
2830
                $rgbArray['b'] = $value[2];
2831
            } else {
2832
                $rgbArray = $value;
2833
            }
2834
        } elseif ('transparent' === strtolower($value)) {
2835
            $rgbArray = [
2836
                'r' => 255,
2837
                'g' => 255,
2838
                'b' => 255,
2839
                'a' => 127
2840
            ];
2841
        } else {
2842
            // *** ...Else it should be hex. Let's make it RGB
2843
            $rgbArray = $this->hex2dec($value);
2844
        }
2845
2846
        return $rgbArray;
2847
    }
2848
2849
    ## --------------------------------------------------------
2850
2851
    public function hex2dec($hex)
2852
        # Purpose:  Convert #hex color to RGB
2853
    {
2854
        $color = str_replace('#', '', $hex);
2855
2856
        if (3 == strlen($color)) {
2857
            $color = $color . $color;
2858
        }
2859
2860
        $rgb = [
2861
            'r' => hexdec(substr($color, 0, 2)),
2862
            'g' => hexdec(substr($color, 2, 2)),
2863
            'b' => hexdec(substr($color, 4, 2)),
2864
            'a' => 0
2865
        ];
2866
        return $rgb;
2867
    }
2868
2869
    ## --------------------------------------------------------
2870
2871
    private function createImageColor($colorArray)
2872
    {
2873
        $r = $colorArray['r'];
2874
        $g = $colorArray['g'];
2875
        $b = $colorArray['b'];
2876
2877
        return imagecolorallocate($this->imageResized, $r, $g, $b);
2878
    }
2879
2880
    ## --------------------------------------------------------
2881
2882
    private function testColorExists($colorArray)
2883
    {
2884
        $r = $colorArray['r'];
2885
        $g = $colorArray['g'];
2886
        $b = $colorArray['b'];
2887
2888
        if (-1 == imagecolorexact($this->imageResized, $r, $g, $b)) {
2889
            return false;
2890
        } else {
2891
            return true;
2892
        }
2893
    }
2894
2895
    ## --------------------------------------------------------
2896
2897
    private function findUnusedGreen()
2898
        # Purpose:  We find a green color suitable to use like green-screen effect.
2899
        #     Therefore, the color must not exist in the image.
2900
    {
2901
        $green = 255;
2902
2903
        do {
2904
            $greenChroma = [0, $green, 0];
2905
            $colorArray  = $this->formatColor($greenChroma);
2906
            $match       = $this->testColorExists($colorArray);
2907
            $green--;
2908
        } while (false == $match && $green > 0);
2909
2910
        // *** If no match, just bite the bullet and use green value of 255
2911
        if (!$match) {
2912
            $greenChroma = [0, $green, 0];
2913
        }
2914
2915
        return $greenChroma;
2916
    }
2917
2918
    ## --------------------------------------------------------
2919
2920
    private function findUnusedBlue()
2921
        # Purpose:  We find a green color suitable to use like green-screen effect.
2922
        #     Therefore, the color must not exist in the image.
2923
    {
2924
        $blue = 255;
2925
2926
        do {
2927
            $blueChroma = [0, 0, $blue];
2928
            $colorArray = $this->formatColor($blueChroma);
2929
            $match      = $this->testColorExists($colorArray);
2930
            $blue--;
2931
        } while (false == $match && $blue > 0);
2932
2933
        // *** If no match, just bite the bullet and use blue value of 255
2934
        if (!$match) {
2935
            $blueChroma = [0, 0, $blue];
2936
        }
2937
2938
        return $blueChroma;
2939
    }
2940
2941
    ## --------------------------------------------------------
2942
2943
    private function invertTransparency($value, $originalMax, $invert = true)
2944
        # Purpose:  This does two things:
2945
        #       1) Convert the range from 0-127 to 0-100
2946
        #       2) Inverts value to 100 is not transparent while 0 is fully
2947
        #          transparent (like Photoshop)
2948
    {
2949
        // *** Test max range
2950
        if ($value > $originalMax) {
2951
            $value = $originalMax;
2952
        }
2953
2954
        // *** Test min range
2955
        if ($value < 0) {
2956
            $value = 0;
2957
        }
2958
2959
        if ($invert) {
2960
            return $originalMax - (($value / 100) * $originalMax);
2961
        } else {
2962
            return ($value / 100) * $originalMax;
2963
        }
2964
    }
2965
2966
    ## --------------------------------------------------------
2967
2968
    private function transparentImage($src)
2969
    {
2970
        // *** making images with white bg transparent
2971
        $r1 = 0;
2972
        $g1 = 255;
2973
        $b1 = 0;
2974
        for ($x = 0; $x < imagesx($src); ++$x) {
2975
            for ($y = 0; $y < imagesy($src); ++$y) {
2976
                $color = imagecolorat($src, $x, $y);
2977
                $r     = ($color >> 16) & 0xFF;
2978
                $g     = ($color >> 8) & 0xFF;
2979
                $b     = $color & 0xFF;
2980
                for ($i = 0; $i < 270; ++$i) {
2981
                    //if ($r . $g . $b == ($r1 + $i) . ($g1 + $i) . ($b1 + $i)) {
2982
                    if (0 == $r && 255 == $g && 0 == $b) {
2983
                        //if ($g == 255) {
2984
                        $trans_colour = imagecolorallocatealpha($src, 0, 0, 0, 127);
2985
                        imagefill($src, $x, $y, $trans_colour);
2986
                    }
2987
                }
2988
            }
2989
        }
2990
2991
        return $src;
2992
    }
2993
2994
    ## --------------------------------------------------------
2995
2996
    public function checkStringStartsWith($needle, $haystack)
2997
        # Check if a string starts with a specific pattern
2998
    {
2999
        return (substr($haystack, 0, strlen($needle)) == $needle);
3000
    }
3001
3002
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
3003
        BMP SUPPORT (SAVING) - James Heinrich
3004
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
3005
3006
    private function GD2BMPstring(&$gd_image)
3007
        # Author:     James Heinrich
3008
        # Purpose:    Save file as type bmp
3009
        # Param in:   The image canvas (passed as ref)
3010
        # Param out:
3011
        # Reference:
3012
        # Notes:      This code was stripped out of two external files
3013
        #             (phpthumb.bmp.php,phpthumb.functions.php) and added below to
3014
        #             avoid dependancies.
3015
        #
3016
    {
3017
        $imageX = imagesx($gd_image);
3018
        $imageY = imagesy($gd_image);
3019
3020
        $BMP = '';
3021
        for ($y = ($imageY - 1); $y >= 0; $y--) {
3022
            $thisline = '';
3023
            for ($x = 0; $x < $imageX; ++$x) {
3024
                $argb     = $this->GetPixelColor($gd_image, $x, $y);
3025
                $thisline .= chr($argb['blue']) . chr($argb['green']) . chr($argb['red']);
3026
            }
3027
            while (strlen($thisline) % 4) {
3028
                $thisline .= "\x00";
3029
            }
3030
            $BMP .= $thisline;
3031
        }
3032
3033
        $bmpSize = strlen($BMP) + 14 + 40;
3034
        // BITMAPFILEHEADER [14 bytes] - http://msdn.microsoft.com/library/en-us/gdi/bitmaps_62uq.asp
3035
        $BITMAPFILEHEADER = 'BM';                                    // WORD    bfType;
3036
        $BITMAPFILEHEADER .= $this->LittleEndian2String($bmpSize, 4); // DWORD   bfSize;
3037
        $BITMAPFILEHEADER .= $this->LittleEndian2String(0, 2); // WORD    bfReserved1;
3038
        $BITMAPFILEHEADER .= $this->LittleEndian2String(0, 2); // WORD    bfReserved2;
3039
        $BITMAPFILEHEADER .= $this->LittleEndian2String(54, 4); // DWORD   bfOffBits;
3040
3041
        // BITMAPINFOHEADER - [40 bytes] http://msdn.microsoft.com/library/en-us/gdi/bitmaps_1rw2.asp
3042
        $BITMAPINFOHEADER = $this->LittleEndian2String(40, 4); // DWORD  biSize;
3043
        $BITMAPINFOHEADER .= $this->LittleEndian2String($imageX, 4); // LONG   biWidth;
3044
        $BITMAPINFOHEADER .= $this->LittleEndian2String($imageY, 4); // LONG   biHeight;
3045
        $BITMAPINFOHEADER .= $this->LittleEndian2String(1, 2); // WORD   biPlanes;
3046
        $BITMAPINFOHEADER .= $this->LittleEndian2String(24, 2); // WORD   biBitCount;
3047
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biCompression;
3048
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biSizeImage;
3049
        $BITMAPINFOHEADER .= $this->LittleEndian2String(2835, 4); // LONG   biXPelsPerMeter;
3050
        $BITMAPINFOHEADER .= $this->LittleEndian2String(2835, 4); // LONG   biYPelsPerMeter;
3051
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biClrUsed;
3052
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biClrImportant;
3053
3054
        return $BITMAPFILEHEADER . $BITMAPINFOHEADER . $BMP;
3055
    }
3056
3057
    ## --------------------------------------------------------
3058
3059
    private function GetPixelColor(&$img, $x, $y)
3060
        # Author:     James Heinrich
3061
        # Purpose:
3062
        # Param in:
3063
        # Param out:
3064
        # Reference:
3065
        # Notes:
3066
        #
3067
    {
3068
        if (!is_resource($img)) {
3069
            return false;
3070
        }
3071
        return @imagecolorsforindex($img, @imagecolorat($img, $x, $y));
3072
    }
3073
3074
    ## --------------------------------------------------------
3075
3076
    private function LittleEndian2String($number, $minbytes = 1)
3077
        # Author:     James Heinrich
3078
        # Purpose:    BMP SUPPORT (SAVING)
3079
        # Param in:
3080
        # Param out:
3081
        # Reference:
3082
        # Notes:
3083
        #
3084
    {
3085
        $intstring = '';
3086
        while ($number > 0) {
3087
            $intstring = $intstring . chr($number & 255);
3088
            $number    >>= 8;
3089
        }
3090
        return str_pad($intstring, $minbytes, "\x00", STR_PAD_RIGHT);
3091
    }
3092
3093
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
3094
        BMP SUPPORT (READING)
3095
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
3096
3097
    private function ImageCreateFromBMP($filename)
3098
        # Author:     DHKold
3099
        # Date:     The 15th of June 2005
3100
        # Version:    2.0B
3101
        # Purpose:    To create an image from a BMP file.
3102
        # Param in:   BMP file to open.
3103
        # Param out:  Return a resource like the other ImageCreateFrom functions
3104
        # Reference:  http://us3.php.net/manual/en/function.imagecreate.php#53879
3105
        # Bug fix:    Author:   domelca at terra dot es
3106
        #             Date:   06 March 2008
3107
        #             Fix:    Correct 16bit BMP support
3108
        # Notes:
3109
        #
3110
    {
3111
        //Ouverture du fichier en mode binaire
3112
        if (!$f1 = fopen($filename, 'rb')) {
3113
            return false;
3114
        }
3115
3116
        //1 : Chargement des ent�tes FICHIER
3117
        $FILE = unpack('vfile_type/Vfile_size/Vreserved/Vbitmap_offset', fread($f1, 14));
3118
        if (19778 != $FILE['file_type']) {
3119
            return false;
3120
        }
3121
3122
        //2 : Chargement des ent�tes BMP
3123
        $BMP           = unpack(
3124
            'Vheader_size/Vwidth/Vheight/vplanes/vbits_per_pixel' . '/Vcompression/Vsize_bitmap/Vhoriz_resolution' . '/Vvert_resolution/Vcolors_used/Vcolors_important',
3125
            fread($f1, 40)
3126
        );
3127
        $BMP['colors'] = pow(2, $BMP['bits_per_pixel']);
3128
3129
        if (0 == $BMP['size_bitmap']) {
3130
            $BMP['size_bitmap'] = $FILE['file_size'] - $FILE['bitmap_offset'];
3131
        }
3132
3133
        $BMP['bytes_per_pixel']  = $BMP['bits_per_pixel'] / 8;
3134
        $BMP['bytes_per_pixel2'] = ceil($BMP['bytes_per_pixel']);
3135
        $BMP['decal']            = ($BMP['width'] * $BMP['bytes_per_pixel'] / 4);
3136
        $BMP['decal']            -= floor($BMP['width'] * $BMP['bytes_per_pixel'] / 4);
3137
        $BMP['decal']            = 4 - (4 * $BMP['decal']);
3138
3139
        if (4 == $BMP['decal']) {
3140
            $BMP['decal'] = 0;
3141
        }
3142
3143
        //3 : Chargement des couleurs de la palette
3144
        $PALETTE = [];
3145
        if ($BMP['colors'] < 16777216) {
3146
            $PALETTE = unpack('V' . $BMP['colors'], fread($f1, $BMP['colors'] * 4));
3147
        }
3148
3149
        //4 : Cr�ation de l'image
3150
        $IMG  = fread($f1, $BMP['size_bitmap']);
3151
        $VIDE = chr(0);
3152
3153
        $res = imagecreatetruecolor($BMP['width'], $BMP['height']);
3154
        $P   = 0;
3155
        $Y   = $BMP['height'] - 1;
3156
        while ($Y >= 0) {
3157
            $X = 0;
3158
            while ($X < $BMP['width']) {
3159
                if (24 == $BMP['bits_per_pixel']) {
3160
                    $COLOR = unpack('V', substr($IMG, $P, 3) . $VIDE);
3161
                } elseif (16 == $BMP['bits_per_pixel']) {
3162
                    /*
3163
                     * BMP 16bit fix
3164
                     * =================
3165
                     *
3166
                     * Ref: http://us3.php.net/manual/en/function.imagecreate.php#81604
3167
                     *
3168
                     * Notes:
3169
                     * "don't work with bmp 16 bits_per_pixel. change pixel
3170
                     * generator for this."
3171
                     *
3172
                     */
3173
3174
                    // *** Original code (don't work)
3175
                    //$COLOR = unpack("n",substr($IMG,$P,2));
3176
                    //$COLOR[1] = $PALETTE[$COLOR[1]+1];
3177
3178
                    $COLOR    = unpack('v', substr($IMG, $P, 2));
3179
                    $blue     = ($COLOR[1] & 0x001f) << 3;
3180
                    $green    = ($COLOR[1] & 0x07e0) >> 3;
3181
                    $red      = ($COLOR[1] & 0xf800) >> 8;
3182
                    $COLOR[1] = $red * 65536 + $green * 256 + $blue;
3183
                } elseif (8 == $BMP['bits_per_pixel']) {
3184
                    $COLOR    = unpack('n', $VIDE . substr($IMG, $P, 1));
3185
                    $COLOR[1] = $PALETTE[$COLOR[1] + 1];
3186
                } elseif (4 == $BMP['bits_per_pixel']) {
3187
                    $COLOR = unpack('n', $VIDE . substr($IMG, floor($P), 1));
3188
                    if (0 == ($P * 2) % 2) {
3189
                        $COLOR[1] = ($COLOR[1] >> 4);
3190
                    } else {
3191
                        $COLOR[1] = ($COLOR[1] & 0x0F);
3192
                    }
3193
                    $COLOR[1] = $PALETTE[$COLOR[1] + 1];
3194
                } elseif (1 == $BMP['bits_per_pixel']) {
3195
                    $COLOR = unpack('n', $VIDE . substr($IMG, floor($P), 1));
3196
                    if (0 == ($P * 8) % 8) {
3197
                        $COLOR[1] = $COLOR[1] >> 7;
3198
                    } elseif (1 == ($P * 8) % 8) {
3199
                        $COLOR[1] = ($COLOR[1] & 0x40) >> 6;
3200
                    } elseif (2 == ($P * 8) % 8) {
3201
                        $COLOR[1] = ($COLOR[1] & 0x20) >> 5;
3202
                    } elseif (3 == ($P * 8) % 8) {
3203
                        $COLOR[1] = ($COLOR[1] & 0x10) >> 4;
3204
                    } elseif (4 == ($P * 8) % 8) {
3205
                        $COLOR[1] = ($COLOR[1] & 0x8) >> 3;
3206
                    } elseif (5 == ($P * 8) % 8) {
3207
                        $COLOR[1] = ($COLOR[1] & 0x4) >> 2;
3208
                    } elseif (6 == ($P * 8) % 8) {
3209
                        $COLOR[1] = ($COLOR[1] & 0x2) >> 1;
3210
                    } elseif (7 == ($P * 8) % 8) {
3211
                        $COLOR[1] = ($COLOR[1] & 0x1);
3212
                    }
3213
                    $COLOR[1] = $PALETTE[$COLOR[1] + 1];
3214
                } else {
3215
                    return false;
3216
                }
3217
3218
                imagesetpixel($res, $X, $Y, $COLOR[1]);
3219
                ++$X;
3220
                $P += $BMP['bytes_per_pixel'];
3221
            }
3222
3223
            $Y--;
3224
            $P += $BMP['decal'];
3225
        }
3226
3227
        //Fermeture du fichier
3228
        fclose($f1);
3229
3230
        return $res;
3231
    }
3232
3233
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
3234
        PSD SUPPORT (READING)
3235
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
3236
3237
    private function imagecreatefrompsd($fileName)
3238
        # Author:     Tim de Koning
3239
        # Version:    1.3
3240
        # Purpose:    To create an image from a PSD file.
3241
        # Param in:   PSD file to open.
3242
        # Param out:  Return a resource like the other ImageCreateFrom functions
3243
        # Reference:  http://www.kingsquare.nl/phppsdreader
3244
        # Notes:
3245
        #
3246
    {
3247
        if (file_exists($this->psdReaderPath)) {
3248
            include_once($this->psdReaderPath);
3249
3250
            $psdReader = new PhpPsdReader($fileName);
3251
3252
            if (isset($psdReader->infoArray['error'])) {
3253
                return '';
3254
            } else {
3255
                return $psdReader->getImage();
3256
            }
3257
        } else {
3258
            return false;
3259
        }
3260
    }
3261
3262
    ## --------------------------------------------------------
3263
3264
    public function __destruct()
3265
    {
3266
        if (is_resource($this->imageResized)) {
3267
            imagedestroy($this->imageResized);
3268
        }
3269
    }
3270
3271
    ## --------------------------------------------------------
3272
3273
}
3274
3275
3276

htdocs/class/xoopseditor/tinymce4bootstrap/external_plugins/filemanager/include/php_image_magician.php 1 location

@@ 123-3273 (lines=3151) @@
120
     #
121
     # ========================================================================#
122
123
class imageLib
124
{
125
126
    private   $fileName;
127
    private   $image;
128
    protected $imageResized;
129
    private   $widthOriginal;     # Always be the original width
130
    private   $heightOriginal;
131
    private   $width;         # Current width (width after resize)
132
    private   $height;
133
    private   $imageSize;
134
    private   $fileExtension;
135
136
    private $isImage = false;
137
138
    private $debug      = true;
139
    private $errorArray = [];
140
141
    private $forceStretch        = true;
142
    private $aggresiveSharpening = false;
143
144
    private $transparentArray = ['.png', '.gif'];
145
    private $keepTransparency = true;
146
    private $fillColorArray   = ['r' => 255, 'g' => 255, 'b' => 255];
147
148
    private $sharpenArray = ['jpg'];
149
150
    private $psdReaderPath;
151
    private $filterOverlayPath;
152
153
    private $isInterlace;
154
155
    private $captionBoxPositionArray = [];
156
157
    private $fontDir = 'fonts';
158
159
    private $cropFromTopPercent = 10;
160
161
    ## --------------------------------------------------------
162
163
    public function __construct($fileName)
164
        # Author:     Jarrod Oberto
165
        # Date:       27-02-08
166
        # Purpose:    Constructor
167
        # Param in:   $fileName: File name and path.
168
        # Param out:  n/a
169
        # Reference:
170
        # Notes:
171
        #
172
    {
173
        if (!$this->testGDInstalled()) {
174
            if ($this->debug) {
175
                die('The GD Library is not installed.');
176
            } else {
177
                die();
178
            }
179
        }
180
181
        $this->initialise();
182
183
        // *** Save the image file name. Only store this incase you want to display it
184
        $this->fileName      = $fileName;
185
        $this->fileExtension = strtolower(strrchr($fileName, '.'));
186
187
        // *** Open up the file
188
        try {
189
            $this->image = $this->openImage($fileName);
190
        } catch (Exception $e) {
191
        }
192
193
        // *** Assign here so we don't modify the original
194
        $this->imageResized = $this->image;
195
196
        // *** If file is an image
197
        $this->isImage = $this->testIsImage();
198
199
        if ($this->isImage) {
200
            // *** Get width and height
201
            $this->width          = imagesx($this->image);
202
            $this->widthOriginal  = imagesx($this->image);
203
            $this->height         = imagesy($this->image);
204
            $this->heightOriginal = imagesy($this->image);
205
206
            /*  Added 15-09-08
207
             *  Get the filesize using this build in method.
208
             *  Stores an array of size
209
             *
210
             *  $this->imageSize[1] = width
211
             *  $this->imageSize[2] = height
212
             *  $this->imageSize[3] = width x height
213
             *
214
             */
215
            $this->imageSize = getimagesize($this->fileName);
216
        } else {
217
            $this->errorArray[] = 'File is not an image';
218
        }
219
    }
220
221
    ## --------------------------------------------------------
222
223
    private function initialise()
224
    {
225
        $this->psdReaderPath     = __DIR__ . '/classPhpPsdReader.php';
226
        $this->filterOverlayPath = __DIR__ . '/filters';
227
228
        // *** Set if image should be interlaced or not.
229
        $this->isInterlace = false;
230
    }
231
232
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
233
        Resize
234
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
235
236
    public function resizeImage($newWidth, $newHeight, $option = 0, $sharpen = false, $autoRotate = true)
237
        # Author:     Jarrod Oberto
238
        # Date:       27-02-08
239
        # Purpose:    Resizes the image
240
        # Param in:   $newWidth:
241
        #             $newHeight:
242
        #             $option:     0 / exact = defined size;
243
        #                          1 / portrait = keep aspect set height;
244
        #                          2 / landscape = keep aspect set width;
245
        #                          3 / auto = auto;
246
        #                          4 / crop= resize and crop;
247
        #
248
        #         $option can also be an array containing options for
249
        #         cropping. E.G., array('crop', 'r')
250
        #
251
        #         This array only applies to 'crop' and the 'r' refers to
252
        #         "crop right". Other value include; tl, t, tr, l, m (default),
253
        #         r, bl, b, br, or you can specify your own co-ords (which
254
        #         isn't recommended.
255
        #
256
        #       $sharpen:    true: sharpen (jpg only);
257
        #                false: don't sharpen
258
        # Param out:  n/a
259
        # Reference:
260
        # Notes:      To clarify the $option input:
261
        #               0 = The exact height and width dimensions you set.
262
        #               1 = Whatever height is passed in will be the height that
263
        #                   is set. The width will be calculated and set automatically
264
        #                   to a the value that keeps the original aspect ratio.
265
        #               2 = The same but based on the width. We try make the image the
266
        #                  biggest size we can while stil fitting inside the box size
267
        #               3 = Depending whether the image is landscape or portrait, this
268
        #                   will automatically determine whether to resize via
269
        #                   dimension 1,2 or 0
270
        #               4 = Will resize and then crop the image for best fit
271
        #
272
        #       forceStretch can be applied to options 1,2,3 and 4
273
        #
274
    {
275
        // *** We can pass in an array of options to change the crop position
276
        $cropPos = 'm';
277
        if (is_array($option) && 'crop' === strtolower($option[0])) {
278
            $cropPos = $option[1];         # get the crop option
279
        } elseif (false !== strpos($option, '-')) {
280
            // *** Or pass in a hyphen seperated option
281
            $optionPiecesArray = explode('-', $option);
282
            $cropPos           = end($optionPiecesArray);
283
        }
284
285
        // *** Check the option is valid
286
        try {
287
            $option = $this->prepOption($option);
288
        } catch (Exception $e) {
289
        }
290
291
        // *** Make sure the file passed in is valid
292
        if (!$this->image) {
293
            if ($this->debug) {
294
                die('file ' . $this->getFileName() . ' is missing or invalid');
295
            } else {
296
                die();
297
            }
298
        }
299
300
        // *** Get optimal width and height - based on $option
301
        $dimensionsArray = $this->getDimensions($newWidth, $newHeight, $option);
302
303
        $optimalWidth  = $dimensionsArray['optimalWidth'];
304
        $optimalHeight = $dimensionsArray['optimalHeight'];
305
306
        // *** Resample - create image canvas of x, y size
307
        $this->imageResized = imagecreatetruecolor($optimalWidth, $optimalHeight);
308
        $this->keepTransparancy($optimalWidth, $optimalHeight, $this->imageResized);
309
        imagecopyresampled($this->imageResized, $this->image, 0, 0, 0, 0, $optimalWidth, $optimalHeight, $this->width, $this->height);
310
311
        // *** If '4', then crop too
312
        if (4 == $option || 'crop' === $option) {
313
            if (($optimalWidth >= $newWidth && $optimalHeight >= $newHeight)) {
314
                $this->crop($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos);
315
            }
316
        }
317
318
        // *** If Rotate.
319
        if ($autoRotate) {
320
            try {
321
                $exifData = $this->getExif(false);
322
            } catch (Exception $e) {
323
            }
324
            if (count($exifData) > 0) {
325
                switch ($exifData['orientation']) {
326
                    case 8:
327
                        $this->imageResized = imagerotate($this->imageResized, 90, 0);
328
                        break;
329
                    case 3:
330
                        $this->imageResized = imagerotate($this->imageResized, 180, 0);
331
                        break;
332
                    case 6:
333
                        $this->imageResized = imagerotate($this->imageResized, -90, 0);
334
                        break;
335
                }
336
            }
337
        }
338
339
        // *** Sharpen image (if jpg and the user wishes to do so)
340
        if ($sharpen && in_array($this->fileExtension, $this->sharpenArray)) {
341
            // *** Sharpen
342
            try {
343
                $this->sharpen();
344
            } catch (Exception $e) {
345
            }
346
        }
347
    }
348
349
    ## --------------------------------------------------------
350
351
    public function cropImage($newWidth, $newHeight, $cropPos = 'm')
352
        # Author:     Jarrod Oberto
353
        # Date:       08-09-11
354
        # Purpose:    Crops the image
355
        # Param in:   $newWidth: crop with
356
        #             $newHeight: crop height
357
        #             $cropPos: Can be any of the following:
358
        #                   tl, t, tr, l, m, r, bl, b, br, auto
359
        #             Or:
360
        #                   a custom position such as '30x50'
361
        # Param out:  n/a
362
        # Reference:
363
        # Notes:
364
        #
365
    {
366
        // *** Make sure the file passed in is valid
367
        if (!$this->image) {
368
            if ($this->debug) {
369
                die('file ' . $this->getFileName() . ' is missing or invalid');
370
            } else {
371
                die();
372
            }
373
        }
374
375
        $this->imageResized = $this->image;
376
        $this->crop($this->width, $this->height, $newWidth, $newHeight, $cropPos);
377
    }
378
379
    ## --------------------------------------------------------
380
381
    private function keepTransparancy($width, $height, $im)
382
        # Author:     Jarrod Oberto
383
        # Date:       08-04-11
384
        # Purpose:    Keep transparency for png and gif image
385
        # Param in:
386
        # Param out:  n/a
387
        # Reference:
388
        # Notes:
389
        #
390
    {
391
        // *** If PNG, perform some transparency retention actions (gif untested)
392
        if (in_array($this->fileExtension, $this->transparentArray) && $this->keepTransparency) {
393
            imagealphablending($im, false);
394
            imagesavealpha($im, true);
395
            $transparent = imagecolorallocatealpha($im, 255, 255, 255, 127);
396
            imagefilledrectangle($im, 0, 0, $width, $height, $transparent);
397
        } else {
398
            $color = imagecolorallocate($im, $this->fillColorArray['r'], $this->fillColorArray['g'], $this->fillColorArray['b']);
399
            imagefilledrectangle($im, 0, 0, $width, $height, $color);
400
        }
401
    }
402
403
    ## --------------------------------------------------------
404
405
    private function crop($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos)
406
        # Author:     Jarrod Oberto
407
        # Date:       15-09-08
408
        # Purpose:    Crops the image
409
        # Param in:   $newWidth:
410
        #             $newHeight:
411
        # Param out:  n/a
412
        # Reference:
413
        # Notes:
414
        #
415
    {
416
        // *** Get cropping co-ordinates
417
        $cropArray  = $this->getCropPlacing($optimalWidth, $optimalHeight, $newWidth, $newHeight, $cropPos);
418
        $cropStartX = $cropArray['x'];
419
        $cropStartY = $cropArray['y'];
420
421
        // *** Crop this bad boy
422
        $crop = imagecreatetruecolor($newWidth, $newHeight);
423
        $this->keepTransparancy($optimalWidth, $optimalHeight, $crop);
424
        imagecopyresampled($crop, $this->imageResized, 0, 0, $cropStartX, $cropStartY, $newWidth, $newHeight, $newWidth, $newHeight);
425
426
        $this->imageResized = $crop;
427
428
        // *** Set new width and height to our variables
429
        $this->width  = $newWidth;
430
        $this->height = $newHeight;
431
    }
432
433
    ## --------------------------------------------------------
434
435
    private function getCropPlacing($optimalWidth, $optimalHeight, $newWidth, $newHeight, $pos = 'm')
436
        #
437
        # Author:   Jarrod Oberto
438
        # Date:   July 11
439
        # Purpose:  Set the cropping area.
440
        # Params in:
441
        # Params out: (array) the crop x and y co-ordinates.
442
        # Notes:    When specifying the exact pixel crop position (eg 10x15), be
443
        #       very careful as it's easy to crop out of the image leaving
444
        #       black borders.
445
        #
446
    {
447
        $pos = strtolower($pos);
448
449
        // *** If co-ords have been entered
450
        if (strstr($pos, 'x')) {
451
            $pos = str_replace(' ', '', $pos);
452
453
            $xyArray = explode('x', $pos);
454
            list($cropStartX, $cropStartY) = $xyArray;
455
        } else {
456
            switch ($pos) {
457
                case 'tl':
458
                    $cropStartX = 0;
459
                    $cropStartY = 0;
460
                    break;
461
462
                case 't':
463
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
464
                    $cropStartY = 0;
465
                    break;
466
467
                case 'tr':
468
                    $cropStartX = $optimalWidth - $newWidth;
469
                    $cropStartY = 0;
470
                    break;
471
472
                case 'l':
473
                    $cropStartX = 0;
474
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
475
                    break;
476
477
                case 'm':
478
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
479
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
480
                    break;
481
482
                case 'r':
483
                    $cropStartX = $optimalWidth - $newWidth;
484
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
485
                    break;
486
487
                case 'bl':
488
                    $cropStartX = 0;
489
                    $cropStartY = $optimalHeight - $newHeight;
490
                    break;
491
492
                case 'b':
493
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
494
                    $cropStartY = $optimalHeight - $newHeight;
495
                    break;
496
497
                case 'br':
498
                    $cropStartX = $optimalWidth - $newWidth;
499
                    $cropStartY = $optimalHeight - $newHeight;
500
                    break;
501
502
                case 'auto':
503
                    // *** If image is a portrait crop from top, not center. v1.5
504
                    if ($optimalHeight > $optimalWidth) {
505
                        $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
506
                        $cropStartY = ($this->cropFromTopPercent / 100) * $optimalHeight;
507
                    } else {
508
                        // *** Else crop from the center
509
                        $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
510
                        $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
511
                    }
512
                    break;
513
514
                default:
515
                    // *** Default to center
516
                    $cropStartX = ($optimalWidth / 2) - ($newWidth / 2);
517
                    $cropStartY = ($optimalHeight / 2) - ($newHeight / 2);
518
                    break;
519
            }
520
        }
521
522
        return ['x' => $cropStartX, 'y' => $cropStartY];
523
    }
524
525
    ## --------------------------------------------------------
526
527
    private function getDimensions($newWidth, $newHeight, $option)
528
        # Author:     Jarrod Oberto
529
        # Date:       17-11-09
530
        # Purpose:    Get new image dimensions based on user specificaions
531
        # Param in:   $newWidth:
532
        #             $newHeight:
533
        # Param out:  Array of new width and height values
534
        # Reference:
535
        # Notes:      If $option = 3 then this function is call recursivly
536
        #
537
        #       To clarify the $option input:
538
        #               0 = The exact height and width dimensions you set.
539
        #               1 = Whatever height is passed in will be the height that
540
        #                   is set. The width will be calculated and set automatically
541
        #                   to a the value that keeps the original aspect ratio.
542
        #               2 = The same but based on the width.
543
        #               3 = Depending whether the image is landscape or portrait, this
544
        #                   will automatically determine whether to resize via
545
        #                   dimension 1,2 or 0.
546
        #               4 = Resize the image as much as possible, then crop the
547
        #         remainder.
548
    {
549
        switch ((string)$option) {
550
            case '0':
551
            case 'exact':
552
                $optimalWidth  = $newWidth;
553
                $optimalHeight = $newHeight;
554
                break;
555
            case '1':
556
            case 'portrait':
557
                $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
558
                $optimalWidth    = $dimensionsArray['optimalWidth'];
559
                $optimalHeight   = $dimensionsArray['optimalHeight'];
560
                break;
561
            case '2':
562
            case 'landscape':
563
                $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
564
                $optimalWidth    = $dimensionsArray['optimalWidth'];
565
                $optimalHeight   = $dimensionsArray['optimalHeight'];
566
                break;
567
            case '3':
568
            case 'auto':
569
                $dimensionsArray = $this->getSizeByAuto($newWidth, $newHeight);
570
                $optimalWidth    = $dimensionsArray['optimalWidth'];
571
                $optimalHeight   = $dimensionsArray['optimalHeight'];
572
                break;
573
            case '4':
574
            case 'crop':
575
                $dimensionsArray = $this->getOptimalCrop($newWidth, $newHeight);
576
                $optimalWidth    = $dimensionsArray['optimalWidth'];
577
                $optimalHeight   = $dimensionsArray['optimalHeight'];
578
                break;
579
        }
580
581
        return ['optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight];
582
    }
583
584
    ## --------------------------------------------------------
585
586
    private function getSizeByFixedHeight($newWidth, $newHeight)
587
    {
588
        // *** If forcing is off...
589
        if (!$this->forceStretch) {
590
            // *** ...check if actual height is less than target height
591
            if ($this->height < $newHeight) {
592
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
593
            }
594
        }
595
596
        $ratio = $this->width / $this->height;
597
598
        $newWidth = $newHeight * $ratio;
599
600
        //return $newWidth;
601
        return ['optimalWidth' => $newWidth, 'optimalHeight' => $newHeight];
602
    }
603
604
    ## --------------------------------------------------------
605
606
    private function getSizeByFixedWidth($newWidth, $newHeight)
607
    {
608
        // *** If forcing is off...
609
        if (!$this->forceStretch) {
610
            // *** ...check if actual width is less than target width
611
            if ($this->width < $newWidth) {
612
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
613
            }
614
        }
615
616
        $ratio = $this->height / $this->width;
617
618
        $newHeight = $newWidth * $ratio;
619
620
        //return $newHeight;
621
        return ['optimalWidth' => $newWidth, 'optimalHeight' => $newHeight];
622
    }
623
624
    ## --------------------------------------------------------
625
626
    private function getSizeByAuto($newWidth, $newHeight)
627
        # Author:     Jarrod Oberto
628
        # Date:       19-08-08
629
        # Purpose:    Depending on the height, choose to resize by 0, 1, or 2
630
        # Param in:   The new height and new width
631
        # Notes:
632
        #
633
    {
634
        // *** If forcing is off...
635
        if (!$this->forceStretch) {
636
            // *** ...check if actual size is less than target size
637
            if ($this->width < $newWidth && $this->height < $newHeight) {
638
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
639
            }
640
        }
641
642
        if ($this->height < $this->width) // *** Image to be resized is wider (landscape)
643
        {
644
            //$optimalWidth = $newWidth;
645
            //$optimalHeight= $this->getSizeByFixedWidth($newWidth);
646
647
            $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
648
            $optimalWidth    = $dimensionsArray['optimalWidth'];
649
            $optimalHeight   = $dimensionsArray['optimalHeight'];
650
        } elseif ($this->height > $this->width) // *** Image to be resized is taller (portrait)
651
        {
652
            //$optimalWidth = $this->getSizeByFixedHeight($newHeight);
653
            //$optimalHeight= $newHeight;
654
655
            $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
656
            $optimalWidth    = $dimensionsArray['optimalWidth'];
657
            $optimalHeight   = $dimensionsArray['optimalHeight'];
658
        } else // *** Image to be resizerd is a square
659
        {
660
            if ($newHeight < $newWidth) {
661
                //$optimalWidth = $newWidth;
662
                //$optimalHeight= $this->getSizeByFixedWidth($newWidth);
663
                $dimensionsArray = $this->getSizeByFixedWidth($newWidth, $newHeight);
664
                $optimalWidth    = $dimensionsArray['optimalWidth'];
665
                $optimalHeight   = $dimensionsArray['optimalHeight'];
666
            } elseif ($newHeight > $newWidth) {
667
                //$optimalWidth = $this->getSizeByFixedHeight($newHeight);
668
                //$optimalHeight= $newHeight;
669
                $dimensionsArray = $this->getSizeByFixedHeight($newWidth, $newHeight);
670
                $optimalWidth    = $dimensionsArray['optimalWidth'];
671
                $optimalHeight   = $dimensionsArray['optimalHeight'];
672
            } else {
673
                // *** Sqaure being resized to a square
674
                $optimalWidth  = $newWidth;
675
                $optimalHeight = $newHeight;
676
            }
677
        }
678
679
        return ['optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight];
680
    }
681
682
    ## --------------------------------------------------------
683
684
    private function getOptimalCrop($newWidth, $newHeight)
685
        # Author:     Jarrod Oberto
686
        # Date:       17-11-09
687
        # Purpose:    Get optimal crop dimensions
688
        # Param in:   width and height as requested by user (fig 3)
689
        # Param out:  Array of optimal width and height (fig 2)
690
        # Reference:
691
        # Notes:      The optimal width and height return are not the same as the
692
        #             same as the width and height passed in. For example:
693
        #
694
        #
695
        #   |-----------------|     |------------|       |-------|
696
        #   |                 | =>  |**|      |**|   =>  |       |
697
        #   |                 |     |**|      |**|       |       |
698
        #   |                 |     |------------|       |-------|
699
        #   |-----------------|
700
        #        original                optimal             crop
701
        #          size                   size               size
702
        #  Fig      1                      2                  3
703
        #
704
        #       300 x 250           150 x 125          150 x 100
705
        #
706
        #  The optimal size is the smallest size (that is closest to the crop size)
707
        #  while retaining proportion/ratio.
708
        #
709
        #  The crop size is the optimal size that has been cropped on one axis to
710
        #  make the image the exact size specified by the user.
711
        #
712
        #      * represent cropped area
713
        #
714
    {
715
        // *** If forcing is off...
716
        if (!$this->forceStretch) {
717
            // *** ...check if actual size is less than target size
718
            if ($this->width < $newWidth && $this->height < $newHeight) {
719
                return ['optimalWidth' => $this->width, 'optimalHeight' => $this->height];
720
            }
721
        }
722
723
        $heightRatio = $this->height / $newHeight;
724
        $widthRatio  = $this->width / $newWidth;
725
726
        if ($heightRatio < $widthRatio) {
727
            $optimalRatio = $heightRatio;
728
        } else {
729
            $optimalRatio = $widthRatio;
730
        }
731
732
        $optimalHeight = round($this->height / $optimalRatio);
733
        $optimalWidth  = round($this->width / $optimalRatio);
734
735
        return ['optimalWidth' => $optimalWidth, 'optimalHeight' => $optimalHeight];
736
    }
737
738
    ## --------------------------------------------------------
739
740
    private function sharpen()
741
        # Author:     Jarrod Oberto
742
        # Date:       08 04 2011
743
        # Purpose:    Sharpen image
744
        # Param in:   n/a
745
        # Param out:  n/a
746
        # Reference:
747
        # Notes:
748
        # Credit:   Incorporates Joe Lencioni (August 6, 2008) code
749
    {
750
        if (version_compare(PHP_VERSION, '5.1.0') >= 0) {
751
            // ***
752
            if ($this->aggresiveSharpening) { # A more aggressive sharpening solution
753
754
                $sharpenMatrix = [
755
                    [-1, -1, -1],
756
                    [-1, 16, -1],
757
                    [-1, -1, -1]
758
                ];
759
                $divisor       = 8;
760
                $offset        = 0;
761
762
                imageconvolution($this->imageResized, $sharpenMatrix, $divisor, $offset);
763
            } else # More subtle and personally more desirable
764
            {
765
                $sharpness = $this->findSharp($this->widthOriginal, $this->width);
766
767
                $sharpenMatrix = [
768
                    [-1, -2, -1],
769
                    [-2, $sharpness + 12, -2], //Lessen the effect of a filter by increasing the value in the center cell
770
                    [-1, -2, -1]
771
                ];
772
                $divisor       = $sharpness; // adjusts brightness
773
                $offset        = 0;
774
                imageconvolution($this->imageResized, $sharpenMatrix, $divisor, $offset);
775
            }
776
        } else {
777
            if ($this->debug) {
778
                die('Sharpening required PHP 5.1.0 or greater.');
779
            }
780
        }
781
    }
782
783
    ## --------------------------------------------------------
784
785
    private function sharpen2($level)
786
    {
787
        $sharpenMatrix = [
788
            [$level, $level, $level],
789
            [$level, (8 * $level) + 1, $level], //Lessen the effect of a filter by increasing the value in the center cell
790
            [$level, $level, $level]
791
        ];
792
    }
793
794
    ## --------------------------------------------------------
795
796
    private function findSharp($orig, $final)
797
        # Author:     Ryan Rud (http://adryrun.com)
798
        # Purpose:    Find optimal sharpness
799
        # Param in:   n/a
800
        # Param out:  n/a
801
        # Reference:
802
        # Notes:
803
        #
804
    {
805
        $final = $final * (750.0 / $orig);
806
        $a     = 52;
807
        $b     = -0.27810650887573124;
808
        $c     = .00047337278106508946;
809
810
        $result = $a + $b * $final + $c * $final * $final;
811
812
        return max(round($result), 0);
813
    }
814
815
    ## --------------------------------------------------------
816
817
    private function prepOption($option)
818
        # Author:     Jarrod Oberto
819
        # Purpose:    Prep option like change the passed in option to lowercase
820
        # Param in:   (str/int) $option: eg. 'exact', 'crop'. 0, 4
821
        # Param out:  lowercase string
822
        # Reference:
823
        # Notes:
824
        #
825
    {
826
        if (is_array($option)) {
827
            if ('crop' === strtolower($option[0]) && 2 == count($option)) {
828
                return 'crop';
829
            } else {
830
                die('Crop resize option array is badly formatted.');
831
            }
832
        } elseif (false !== strpos($option, 'crop')) {
833
            return 'crop';
834
        }
835
836
        if (is_string($option)) {
837
            return strtolower($option);
838
        }
839
840
        return $option;
841
    }
842
843
844
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
845
        Presets
846
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
847
848
    #
849
    # Preset are pre-defined templates you can apply to your image.
850
    #
851
    # These are inteded to be applied to thumbnail images.
852
    #
853
854
    public function borderPreset($preset)
855
    {
856
        switch ($preset) {
857
            case 'simple':
858
                $this->addBorder(7, '#fff');
859
                $this->addBorder(6, '#f2f1f0');
860
                $this->addBorder(2, '#fff');
861
                $this->addBorder(1, '#ccc');
862
                break;
863
            default:
864
                break;
865
        }
866
    }
867
868
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
869
        Draw border
870
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
871
872
    public function addBorder($thickness = 1, $rgbArray = [255, 255, 255])
873
        # Author:     Jarrod Oberto
874
        # Date:       05-05-11
875
        # Purpose:    Add a border to the image
876
        # Param in:
877
        # Param out:
878
        # Reference:
879
        # Notes:    This border is added to the INSIDE of the image
880
        #
881
    {
882
        if ($this->imageResized) {
883
            $rgbArray = $this->formatColor($rgbArray);
884
            $r        = $rgbArray['r'];
885
            $g        = $rgbArray['g'];
886
            $b        = $rgbArray['b'];
887
888
            $x1 = 0;
889
            $y1 = 0;
890
            $x2 = imagesx($this->imageResized) - 1;
891
            $y2 = imagesy($this->imageResized) - 1;
892
893
            $rgbArray = imagecolorallocate($this->imageResized, $r, $g, $b);
894
895
            for ($i = 0; $i < $thickness; ++$i) {
896
                imagerectangle($this->imageResized, ++$x1, ++$y1, $x2--, $y2--, $rgbArray);
897
            }
898
        }
899
    }
900
901
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
902
        Gray Scale
903
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
904
905
    public function greyScale()
906
        # Author:     Jarrod Oberto
907
        # Date:       07-05-2011
908
        # Purpose:    Make image greyscale
909
        # Param in:   n/a
910
        # Param out:
911
        # Reference:
912
        # Notes:
913
        #
914
    {
915
        if ($this->imageResized) {
916
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
917
        }
918
    }
919
920
    ## --------------------------------------------------------
921
922
    public function greyScaleEnhanced()
923
        # Author:     Jarrod Oberto
924
        # Date:       07-05-2011
925
        # Purpose:    Make image greyscale
926
        # Param in:   n/a
927
        # Param out:
928
        # Reference:
929
        # Notes:
930
        #
931
    {
932
        if ($this->imageResized) {
933
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
934
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -15);
935
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, 2);
936
            try {
937
                $this->sharpen($this->width);
938
            } catch (Exception $e) {
939
            }
940
        }
941
    }
942
943
    ## --------------------------------------------------------
944
945
    public function greyScaleDramatic()
946
        # Alias of gd_filter_monopin
947
    {
948
        $this->gd_filter_monopin();
949
    }
950
951
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
952
        Black 'n White
953
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
954
955
    public function blackAndWhite()
956
        # Author:     Jarrod Oberto
957
        # Date:       07-05-2011
958
        # Purpose:    Make image black and white
959
        # Param in:   n/a
960
        # Param out:
961
        # Reference:
962
        # Notes:
963
        #
964
    {
965
        if ($this->imageResized) {
966
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
967
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -1000);
968
        }
969
    }
970
971
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
972
        Negative
973
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
974
975
    public function negative()
976
        # Author:     Jarrod Oberto
977
        # Date:       07-05-2011
978
        # Purpose:    Make image negative
979
        # Param in:   n/a
980
        # Param out:
981
        # Reference:
982
        # Notes:
983
        #
984
    {
985
        if ($this->imageResized) {
986
            imagefilter($this->imageResized, IMG_FILTER_NEGATE);
987
        }
988
    }
989
990
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
991
        Sepia
992
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
993
994
    public function sepia()
995
        # Author:     Jarrod Oberto
996
        # Date:       07-05-2011
997
        # Purpose:    Make image sepia
998
        # Param in:   n/a
999
        # Param out:
1000
        # Reference:
1001
        # Notes:
1002
        #
1003
    {
1004
        if ($this->imageResized) {
1005
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
1006
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, -10);
1007
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -20);
1008
            imagefilter($this->imageResized, IMG_FILTER_COLORIZE, 60, 30, -15);
1009
        }
1010
    }
1011
1012
    ## --------------------------------------------------------
1013
1014
    public function sepia2()
1015
    {
1016
        if ($this->imageResized) {
1017
            $total = imagecolorstotal($this->imageResized);
1018
            for ($i = 0; $i < $total; ++$i) {
1019
                $index = imagecolorsforindex($this->imageResized, $i);
1020
                $red   = ($index['red'] * 0.393 + $index['green'] * 0.769 + $index['blue'] * 0.189) / 1.351;
1021
                $green = ($index['red'] * 0.349 + $index['green'] * 0.686 + $index['blue'] * 0.168) / 1.203;
1022
                $blue  = ($index['red'] * 0.272 + $index['green'] * 0.534 + $index['blue'] * 0.131) / 2.140;
1023
                imagecolorset($this->imageResized, $i, $red, $green, $blue);
1024
            }
1025
        }
1026
    }
1027
1028
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1029
        Vintage
1030
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1031
1032
    public function vintage()
1033
        # Alias of gd_filter_monopin
1034
    {
1035
        $this->gd_filter_vintage();
1036
    }
1037
1038
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1039
        Presets By Marc Hibbins
1040
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1041
1042
    /** Apply 'Monopin' preset */
1043
    public function gd_filter_monopin()
1044
    {
1045
        if ($this->imageResized) {
1046
            imagefilter($this->imageResized, IMG_FILTER_GRAYSCALE);
1047
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, -15);
1048
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -15);
1049
            $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'vignette', 100);
1050
        }
1051
    }
1052
1053
    ## --------------------------------------------------------
1054
1055
    public function gd_filter_vintage()
1056
    {
1057
        if ($this->imageResized) {
1058
            $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'vignette', 45);
1059
            imagefilter($this->imageResized, IMG_FILTER_BRIGHTNESS, 20);
1060
            imagefilter($this->imageResized, IMG_FILTER_CONTRAST, -35);
1061
            imagefilter($this->imageResized, IMG_FILTER_COLORIZE, 60, -10, 35);
1062
            imagefilter($this->imageResized, IMG_FILTER_SMOOTH, 7);
1063
            $this->imageResized = $this->gd_apply_overlay($this->imageResized, 'scratch', 10);
1064
        }
1065
    }
1066
1067
    ## --------------------------------------------------------
1068
1069
    /** Apply a PNG overlay
1070
     * @param $im
1071
     * @param $type
1072
     * @param $amount
1073
     * @return mixed
1074
     */
1075
    private function gd_apply_overlay($im, $type, $amount)
1076
        #
1077
        # Original Author:    Marc Hibbins
1078
        # License:  Attribution-ShareAlike 3.0
1079
        # Purpose:
1080
        # Params in:
1081
        # Params out:
1082
        # Notes:
1083
        #
1084
    {
1085
        $width  = imagesx($im);
1086
        $height = imagesy($im);
1087
        $filter = imagecreatetruecolor($width, $height);
1088
1089
        imagealphablending($filter, false);
1090
        imagesavealpha($filter, true);
1091
1092
        $transparent = imagecolorallocatealpha($filter, 255, 255, 255, 127);
1093
        imagefilledrectangle($filter, 0, 0, $width, $height, $transparent);
1094
1095
        // *** Resize overlay
1096
        $overlay = $this->filterOverlayPath . '/' . $type . '.png';
1097
        $png     = imagecreatefrompng($overlay);
1098
        imagecopyresampled($filter, $png, 0, 0, 0, 0, $width, $height, imagesx($png), imagesy($png));
1099
1100
        $comp = imagecreatetruecolor($width, $height);
1101
        imagecopy($comp, $im, 0, 0, 0, 0, $width, $height);
1102
        imagecopy($comp, $filter, 0, 0, 0, 0, $width, $height);
1103
        imagecopymerge($im, $comp, 0, 0, 0, 0, $width, $height, $amount);
1104
1105
        imagedestroy($comp);
1106
        return $im;
1107
    }
1108
1109
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1110
        Colorise
1111
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1112
1113
    public function image_colorize($rgb)
1114
    {
1115
        imagetruecolortopalette($this->imageResized, true, 256);
1116
        $numColors = imagecolorstotal($this->imageResized);
1117
1118
        for ($x = 0; $x < $numColors; ++$x) {
1119
            list($r, $g, $b) = array_values(imagecolorsforindex($this->imageResized, $x));
1120
1121
            // calculate grayscale in percent
1122
            $grayscale = ($r + $g + $b) / 3 / 0xff;
1123
1124
            imagecolorset(
1125
                $this->imageResized,
1126
                $x,
1127
                $grayscale * $rgb[0],
1128
                $grayscale * $rgb[1],
1129
                $grayscale * $rgb[2]
1130
            );
1131
        }
1132
1133
        return true;
1134
    }
1135
1136
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1137
        Reflection
1138
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1139
1140
    public function addReflection($reflectionHeight = 50, $startingTransparency = 30, $inside = false, $bgColor = '#fff', $stretch = false, $divider = 0)
1141
    {
1142
        // *** Convert color
1143
        $rgbArray = $this->formatColor($bgColor);
1144
        $r        = $rgbArray['r'];
1145
        $g        = $rgbArray['g'];
1146
        $b        = $rgbArray['b'];
1147
1148
        $im = $this->imageResized;
1149
        $li = imagecreatetruecolor($this->width, 1);
1150
1151
        $bgc = imagecolorallocate($li, $r, $g, $b);
1152
        imagefilledrectangle($li, 0, 0, $this->width, 1, $bgc);
1153
1154
        $bg = imagecreatetruecolor($this->width, $reflectionHeight);
1155
        $wh = imagecolorallocate($im, 255, 255, 255);
1156
1157
        $im = imagerotate($im, -180, $wh);
1158
        imagecopyresampled($bg, $im, 0, 0, 0, 0, $this->width, $this->height, $this->width, $this->height);
1159
1160
        $im = $bg;
1161
1162
        $bg = imagecreatetruecolor($this->width, $reflectionHeight);
1163
1164
        for ($x = 0; $x < $this->width; ++$x) {
1165
            imagecopy($bg, $im, $x, 0, $this->width - $x - 1, 0, 1, $reflectionHeight);
1166
        }
1167
        $im = $bg;
1168
1169
        $transaprencyAmount = $this->invertTransparency($startingTransparency, 100);
1170
1171
        // *** Fade
1172
        if ($stretch) {
1173
            $step = 100 / ($reflectionHeight + $startingTransparency);
1174
        } else {
1175
            $step = 100 / $reflectionHeight;
1176
        }
1177
        for ($i = 0; $i <= $reflectionHeight; ++$i) {
1178
            if ($startingTransparency > 100) {
1179
                $startingTransparency = 100;
1180
            }
1181
            if ($startingTransparency < 1) {
1182
                $startingTransparency = 1;
1183
            }
1184
            imagecopymerge($bg, $li, 0, $i, 0, 0, $this->width, 1, $startingTransparency);
1185
            $startingTransparency += $step;
1186
        }
1187
1188
        // *** Apply fade
1189
        imagecopymerge($im, $li, 0, 0, 0, 0, $this->width, $divider, 100); // Divider
1190
1191
        // *** width, height of reflection.
1192
        $x = imagesx($im);
1193
        $y = imagesy($im);
1194
1195
        // *** Determines if the reflection should be displayed inside or outside the image
1196
        if ($inside) {
1197
            // Create new blank image with sizes.
1198
            $final = imagecreatetruecolor($this->width, $this->height);
1199
1200
            imagecopymerge($final, $this->imageResized, 0, 0, 0, $reflectionHeight, $this->width, $this->height - $reflectionHeight, 100);
1201
            imagecopymerge($final, $im, 0, $this->height - $reflectionHeight, 0, 0, $x, $y, 100);
1202
        } else {
1203
            // Create new blank image with sizes.
1204
            $final = imagecreatetruecolor($this->width, $this->height + $y);
1205
1206
            imagecopymerge($final, $this->imageResized, 0, 0, 0, 0, $this->width, $this->height, 100);
1207
            imagecopymerge($final, $im, 0, $this->height, 0, 0, $x, $y, 100);
1208
        }
1209
1210
        $this->imageResized = $final;
1211
1212
        imagedestroy($li);
1213
        imagedestroy($im);
1214
    }
1215
1216
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1217
        Rotate
1218
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1219
1220
    public function rotate($value = 90, $bgColor = 'transparent')
1221
        # Author:     Jarrod Oberto
1222
        # Date:       07-05-2011
1223
        # Purpose:    Rotate image
1224
        # Param in:   (mixed) $degrees: (int) number of degress to rotate image
1225
        #             (str) param "left": rotate left
1226
        #             (str) param "right": rotate right
1227
        #             (str) param "upside": upside-down image
1228
        # Param out:
1229
        # Reference:
1230
        # Notes:    The default direction of imageRotate() is counter clockwise.
1231
        #
1232
    {
1233
        if ($this->imageResized) {
1234
            if (is_int($value)) {
1235
                $degrees = $value;
1236
            }
1237
1238
            // *** Convert color
1239
            $rgbArray = $this->formatColor($bgColor);
1240
            $r        = $rgbArray['r'];
1241
            $g        = $rgbArray['g'];
1242
            $b        = $rgbArray['b'];
1243
            if (isset($rgbArray['a'])) {
1244
                $a = $rgbArray['a'];
1245
            }
1246
1247
            if (is_string($value)) {
1248
                $value = strtolower($value);
1249
1250
                switch ($value) {
1251
                    case 'left':
1252
                        $degrees = 90;
1253
                        break;
1254
                    case 'right':
1255
                        $degrees = 270;
1256
                        break;
1257
                    case 'upside':
1258
                        $degrees = 180;
1259
                        break;
1260
                    default:
1261
                        break;
1262
                }
1263
            }
1264
1265
            // *** The default direction of imageRotate() is counter clockwise
1266
            //   * This makes it clockwise
1267
            $degrees = 360 - $degrees;
1268
1269
            // *** Create background color
1270
            $bg = imagecolorallocatealpha($this->imageResized, $r, $g, $b, $a);
1271
1272
            // *** Fill with background
1273
            imagefill($this->imageResized, 0, 0, $bg);
1274
1275
            // *** Rotate
1276
            $this->imageResized = imagerotate($this->imageResized, $degrees, $bg); // Rotate 45 degrees and allocated the transparent colour as the one to make transparent (obviously)
1277
1278
            // Ensure alpha transparency
1279
            ImageSaveAlpha($this->imageResized, true);
1280
        }
1281
    }
1282
1283
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1284
        Round corners
1285
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1286
1287
    public function roundCorners($radius = 5, $bgColor = 'transparent')
1288
        # Author:     Jarrod Oberto
1289
        # Date:       19-05-2011
1290
        # Purpose:    Create rounded corners on your image
1291
        # Param in:   (int) radius = the amount of curvature
1292
        #             (mixed) $bgColor = the corner background color
1293
        # Param out:  n/a
1294
        # Reference:
1295
        # Notes:
1296
        #
1297
    {
1298
        // *** Check if the user wants transparency
1299
        $isTransparent = false;
1300
        if (!is_array($bgColor)) {
1301
            if ('transparent' === strtolower($bgColor)) {
1302
                $isTransparent = true;
1303
            }
1304
        }
1305
1306
        // *** If we use transparency, we need to color our curved mask with a unique color
1307
        if ($isTransparent) {
1308
            $bgColor = $this->findUnusedGreen();
1309
        }
1310
1311
        // *** Convert color
1312
        $rgbArray = $this->formatColor($bgColor);
1313
        $r        = $rgbArray['r'];
1314
        $g        = $rgbArray['g'];
1315
        $b        = $rgbArray['b'];
1316
        if (isset($rgbArray['a'])) {
1317
            $a = $rgbArray['a'];
1318
        }
1319
1320
        // *** Create top-left corner mask (square)
1321
        $cornerImg = imagecreatetruecolor($radius, $radius);
1322
        //$cornerImg = imagecreate($radius, $radius);
1323
1324
        //imagealphablending($cornerImg, true);
1325
        //imagesavealpha($cornerImg, true);
1326
1327
        //imagealphablending($this->imageResized, false);
1328
        //imagesavealpha($this->imageResized, true);
1329
1330
        // *** Give it a color
1331
        $maskColor = imagecolorallocate($cornerImg, 0, 0, 0);
1332
1333
        // *** Replace the mask color (black) to transparent
1334
        imagecolortransparent($cornerImg, $maskColor);
1335
1336
        // *** Create the image background color
1337
        $imagebgColor = imagecolorallocate($cornerImg, $r, $g, $b);
1338
1339
        // *** Fill the corner area to the user defined color
1340
        imagefill($cornerImg, 0, 0, $imagebgColor);
1341
1342
        imagefilledellipse($cornerImg, $radius, $radius, $radius * 2, $radius * 2, $maskColor);
1343
1344
        // *** Map to top left corner
1345
        imagecopymerge($this->imageResized, $cornerImg, 0, 0, 0, 0, $radius, $radius, 100); #tl
1346
1347
        // *** Map rounded corner to other corners by rotating and applying the mask
1348
        $cornerImg = imagerotate($cornerImg, 90, 0);
1349
        imagecopymerge($this->imageResized, $cornerImg, 0, $this->height - $radius, 0, 0, $radius, $radius, 100); #bl
1350
1351
        $cornerImg = imagerotate($cornerImg, 90, 0);
1352
        imagecopymerge($this->imageResized, $cornerImg, $this->width - $radius, $this->height - $radius, 0, 0, $radius, $radius, 100); #br
1353
1354
        $cornerImg = imagerotate($cornerImg, 90, 0);
1355
        imagecopymerge($this->imageResized, $cornerImg, $this->width - $radius, 0, 0, 0, $radius, $radius, 100); #tr
1356
1357
        // *** If corners are to be transparent, we fill our chromakey color as transparent.
1358
        if ($isTransparent) {
1359
            //imagecolortransparent($this->imageResized, $imagebgColor);
1360
            $this->imageResized = $this->transparentImage($this->imageResized);
1361
            imagesavealpha($this->imageResized, true);
1362
        }
1363
    }
1364
1365
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1366
        Shadow
1367
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1368
1369
    public function addShadow($shadowAngle = 45, $blur = 15, $bgColor = 'transparent')
1370
        #
1371
        # Author:   Jarrod Oberto (Adapted from Pascal Naidon)
1372
        # Ref:    http://www.les-stooges.org/pascal/webdesign/vignettes/index.php?la=en
1373
        # Purpose:  Add a drop shadow to your image
1374
        # Params in:  (int) $angle: the angle of the shadow
1375
        #       (int) $blur: the blur distance
1376
        #       (mixed) $bgColor: the color of the background
1377
        # Params out:
1378
        # Notes:
1379
        #
1380
    {
1381
        // *** A higher number results in a smoother shadow
1382
        define('STEPS', $blur * 2);
1383
1384
        // *** Set the shadow distance
1385
        $shadowDistance = $blur * 0.25;
1386
1387
        // *** Set blur width and height
1388
        $blurWidth = $blurHeight = $blur;
1389
1390
        if (0 == $shadowAngle) {
1391
            $distWidth  = 0;
1392
            $distHeight = 0;
1393
        } else {
1394
            $distWidth  = $shadowDistance * cos(deg2rad($shadowAngle));
1395
            $distHeight = $shadowDistance * sin(deg2rad($shadowAngle));
1396
        }
1397
1398
        // *** Convert color
1399
        if ('transparent' !== strtolower($bgColor)) {
1400
            $rgbArray = $this->formatColor($bgColor);
1401
            $r0       = $rgbArray['r'];
1402
            $g0       = $rgbArray['g'];
1403
            $b0       = $rgbArray['b'];
1404
        }
1405
1406
        $image  = $this->imageResized;
1407
        $width  = $this->width;
1408
        $height = $this->height;
1409
1410
        $newImage = imagecreatetruecolor($width, $height);
1411
        imagecopyresampled($newImage, $image, 0, 0, 0, 0, $width, $height, $width, $height);
1412
1413
        // *** RGB
1414
        $rgb    = imagecreatetruecolor($width + $blurWidth, $height + $blurHeight);
1415
        $colour = imagecolorallocate($rgb, 0, 0, 0);
1416
        imagefilledrectangle($rgb, 0, 0, $width + $blurWidth, $height + $blurHeight, $colour);
1417
        $colour = imagecolorallocate($rgb, 255, 255, 255);
1418
        //imagefilledrectangle($rgb, $blurWidth*0.5-$distWidth, $blurHeight*0.5-$distHeight, $width+$blurWidth*0.5-$distWidth, $height+$blurWidth*0.5-$distHeight, $colour);
1419
        imagefilledrectangle($rgb, $blurWidth * 0.5 - $distWidth, $blurHeight * 0.5 - $distHeight, $width + $blurWidth * 0.5 - $distWidth, $height + $blurWidth * 0.5 - $distHeight, $colour);
1420
        //imagecopymerge($rgb, $newImage, 1+$blurWidth*0.5-$distWidth, 1+$blurHeight*0.5-$distHeight, 0,0, $width, $height, 100);
1421
        imagecopymerge($rgb, $newImage, $blurWidth * 0.5 - $distWidth, $blurHeight * 0.5 - $distHeight, 0, 0, $width + $blurWidth, $height + $blurHeight, 100);
1422
1423
        // *** Shadow (alpha)
1424
        $shadow = imagecreatetruecolor($width + $blurWidth, $height + $blurHeight);
1425
        imagealphablending($shadow, false);
1426
        $colour = imagecolorallocate($shadow, 0, 0, 0);
1427
        imagefilledrectangle($shadow, 0, 0, $width + $blurWidth, $height + $blurHeight, $colour);
1428
1429
        for ($i = 0; $i <= STEPS; ++$i) {
1430
            $t         = ((1.0 * $i) / STEPS);
1431
            $intensity = 255 * $t * $t;
1432
1433
            $colour = imagecolorallocate($shadow, $intensity, $intensity, $intensity);
1434
            $points = [
1435
                $blurWidth * $t,
1436
                $blurHeight,     // Point 1 (x, y)
1437
                $blurWidth,
1438
                $blurHeight * $t,  // Point 2 (x, y)
1439
                $width,
1440
                $blurHeight * $t,  // Point 3 (x, y)
1441
                $width + $blurWidth * (1 - $t),
1442
                $blurHeight,     // Point 4 (x, y)
1443
                $width + $blurWidth * (1 - $t),
1444
                $height,     // Point 5 (x, y)
1445
                $width,
1446
                $height + $blurHeight * (1 - $t),  // Point 6 (x, y)
1447
                $blurWidth,
1448
                $height + $blurHeight * (1 - $t),  // Point 7 (x, y)
1449
                $blurWidth * $t,
1450
                $height      // Point 8 (x, y)
1451
            ];
1452
            imagepolygon($shadow, $points, 8, $colour);
1453
        }
1454
1455
        for ($i = 0; $i <= STEPS; ++$i) {
1456
            $t         = ((1.0 * $i) / STEPS);
1457
            $intensity = 255 * $t * $t;
1458
1459
            $colour = imagecolorallocate($shadow, $intensity, $intensity, $intensity);
1460
            imagefilledarc($shadow, $blurWidth - 1, $blurHeight - 1, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 180, 268, $colour, IMG_ARC_PIE);
1461
            imagefilledarc($shadow, $width, $blurHeight - 1, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 270, 358, $colour, IMG_ARC_PIE);
1462
            imagefilledarc($shadow, $width, $height, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 0, 90, $colour, IMG_ARC_PIE);
1463
            imagefilledarc($shadow, $blurWidth - 1, $height, 2 * (1 - $t) * $blurWidth, 2 * (1 - $t) * $blurHeight, 90, 180, $colour, IMG_ARC_PIE);
1464
        }
1465
1466
        $colour = imagecolorallocate($shadow, 255, 255, 255);
1467
        imagefilledrectangle($shadow, $blurWidth, $blurHeight, $width, $height, $colour);
1468
        imagefilledrectangle($shadow, $blurWidth * 0.5 - $distWidth, $blurHeight * 0.5 - $distHeight, $width + $blurWidth * 0.5 - 1 - $distWidth, $height + $blurHeight * 0.5 - 1 - $distHeight, $colour);
1469
1470
        // *** The magic
1471
        imagealphablending($rgb, false);
1472
1473
        for ($theX = 0; $theX < imagesx($rgb); $theX++) {
1474
            for ($theY = 0; $theY < imagesy($rgb); $theY++) {
1475
                // *** Get the RGB values for every pixel of the RGB image
1476
                $colArray = imagecolorat($rgb, $theX, $theY);
1477
                $r        = ($colArray >> 16) & 0xFF;
1478
                $g        = ($colArray >> 8) & 0xFF;
1479
                $b        = $colArray & 0xFF;
1480
1481
                // *** Get the alpha value for every pixel of the shadow image
1482
                $colArray = imagecolorat($shadow, $theX, $theY);
1483
                $a        = $colArray & 0xFF;
1484
                $a        = 127 - floor($a / 2);
1485
                $t        = $a / 128.0;
1486
1487
                // *** Create color
1488
                if ('transparent' === strtolower($bgColor)) {
1489
                    $myColour = imagecolorallocatealpha($rgb, $r, $g, $b, $a);
1490
                } else {
1491
                    $myColour = imagecolorallocate($rgb, $r * (1.0 - $t) + $r0 * $t, $g * (1.0 - $t) + $g0 * $t, $b * (1.0 - $t) + $b0 * $t);
1492
                }
1493
1494
                // *** Add color to new rgb image
1495
                imagesetpixel($rgb, $theX, $theY, $myColour);
1496
            }
1497
        }
1498
1499
        imagealphablending($rgb, true);
1500
        imagesavealpha($rgb, true);
1501
1502
        $this->imageResized = $rgb;
1503
1504
        imagedestroy($image);
1505
        imagedestroy($newImage);
1506
        imagedestroy($shadow);
1507
    }
1508
1509
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1510
        Add Caption Box
1511
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1512
1513
    public function addCaptionBox($side = 'b', $thickness = 50, $padding = 0, $bgColor = '#000', $transaprencyAmount = 30)
1514
        #
1515
        # Author:   Jarrod Oberto
1516
        # Date:   26 May 2011
1517
        # Purpose:  Add a caption box
1518
        # Params in:  (str) $side: the side to add the caption box (t, r, b, or l).
1519
        #       (int) $thickness: how thick you want the caption box to be.
1520
        #       (mixed) $bgColor: The color of the caption box.
1521
        #       (int) $transaprencyAmount: The amount of transparency to be
1522
        #       applied.
1523
        # Params out: n/a
1524
        # Notes:
1525
        #
1526
    {
1527
        $side = strtolower($side);
1528
1529
        // *** Convert color
1530
        $rgbArray = $this->formatColor($bgColor);
1531
        $r        = $rgbArray['r'];
1532
        $g        = $rgbArray['g'];
1533
        $b        = $rgbArray['b'];
1534
1535
        $positionArray = $this->calculateCaptionBoxPosition($side, $thickness, $padding);
1536
1537
        // *** Store incase we want to use method addTextToCaptionBox()
1538
        $this->captionBoxPositionArray = $positionArray;
1539
1540
        $transaprencyAmount = $this->invertTransparency($transaprencyAmount, 127, false);
1541
        $transparent        = imagecolorallocatealpha($this->imageResized, $r, $g, $b, $transaprencyAmount);
1542
        imagefilledrectangle($this->imageResized, $positionArray['x1'], $positionArray['y1'], $positionArray['x2'], $positionArray['y2'], $transparent);
1543
    }
1544
1545
    ## --------------------------------------------------------
1546
1547
    public function addTextToCaptionBox($text, $fontColor = '#fff', $fontSize = 12, $angle = 0, $font = null)
1548
        #
1549
        # Author:   Jarrod Oberto
1550
        # Date:   03 Aug 11
1551
        # Purpose:  Simplify adding text to a caption box by automatically
1552
        #       locating the center of the caption box
1553
        # Params in:  The usually text paams (less a couple)
1554
        # Params out: n/a
1555
        # Notes:
1556
        #
1557
    {
1558
        // *** Get the caption box measurements
1559
        if (4 == count($this->captionBoxPositionArray)) {
1560
            $x1 = $this->captionBoxPositionArray['x1'];
1561
            $x2 = $this->captionBoxPositionArray['x2'];
1562
            $y1 = $this->captionBoxPositionArray['y1'];
1563
            $y2 = $this->captionBoxPositionArray['y2'];
1564
        } else {
1565
            if ($this->debug) {
1566
                die('No caption box found.');
1567
            } else {
1568
                return false;
1569
            }
1570
        }
1571
1572
        // *** Get text font
1573
        try {
1574
            $font = $this->getTextFont($font);
1575
        } catch (Exception $e) {
1576
        }
1577
1578
        // *** Get text size
1579
        $textSizeArray = $this->getTextSize($fontSize, $angle, $font, $text);
1580
        $textWidth     = $textSizeArray['width'];
1581
        $textHeight    = $textSizeArray['height'];
1582
1583
        // *** Find the width/height middle points
1584
        $boxXMiddle = (($x2 - $x1) / 2);
1585
        $boxYMiddle = (($y2 - $y1) / 2);
1586
1587
        // *** Box middle - half the text width/height
1588
        $xPos = ($x1 + $boxXMiddle) - ($textWidth / 2);
1589
        $yPos = ($y1 + $boxYMiddle) - ($textHeight / 2);
1590
1591
        $pos = $xPos . 'x' . $yPos;
1592
1593
        $this->addText($text, $pos, $padding = 0, $fontColor, $fontSize, $angle, $font);
1594
    }
1595
1596
    ## --------------------------------------------------------
1597
1598
    private function calculateCaptionBoxPosition($side, $thickness, $padding)
1599
    {
1600
        $positionArray = [];
1601
1602
        switch ($side) {
1603
            case 't':
1604
                $positionArray['x1'] = 0;
1605
                $positionArray['y1'] = $padding;
1606
                $positionArray['x2'] = $this->width;
1607
                $positionArray['y2'] = $thickness + $padding;
1608
                break;
1609
            case 'r':
1610
                $positionArray['x1'] = $this->width - $thickness - $padding;
1611
                $positionArray['y1'] = 0;
1612
                $positionArray['x2'] = $this->width - $padding;
1613
                $positionArray['y2'] = $this->height;
1614
                break;
1615
            case 'b':
1616
                $positionArray['x1'] = 0;
1617
                $positionArray['y1'] = $this->height - $thickness - $padding;
1618
                $positionArray['x2'] = $this->width;
1619
                $positionArray['y2'] = $this->height - $padding;
1620
                break;
1621
            case 'l':
1622
                $positionArray['x1'] = $padding;
1623
                $positionArray['y1'] = 0;
1624
                $positionArray['x2'] = $thickness + $padding;
1625
                $positionArray['y2'] = $this->height;
1626
                break;
1627
1628
            default:
1629
                break;
1630
        }
1631
1632
        return $positionArray;
1633
    }
1634
1635
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1636
        Get EXIF Data
1637
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1638
1639
    public function getExif($debug = true)
1640
        # Author:     Jarrod Oberto
1641
        # Date:       07-05-2011
1642
        # Purpose:    Get image EXIF data
1643
        # Param in:   n/a
1644
        # Param out:  An associate array of EXIF data
1645
        # Reference:
1646
        # Notes:
1647
        # 23 May 13 : added orientation flag -jco
1648
        #
1649
    {
1650
        if (!$this->debug || !$debug) {
1651
            $debug = false;
1652
        }
1653
1654
        // *** Check all is good - check the EXIF library exists and the file exists, too.
1655
        if (!$this->testEXIFInstalled()) {
1656
            if ($debug) {
1657
                die('The EXIF Library is not installed.');
1658
            } else {
1659
                return [];
1660
            }
1661
        }
1662
        if (!file_exists($this->fileName)) {
1663
            if ($debug) {
1664
                die('Image not found.');
1665
            } else {
1666
                return [];
1667
            }
1668
        }
1669
        if ('.jpg' !== $this->fileExtension) {
1670
            if ($debug) {
1671
                die('Metadata not supported for this image type.');
1672
            } else {
1673
                return [];
1674
            }
1675
        }
1676
        $exifData = exif_read_data($this->fileName, 'IFD0');
1677
1678
        // *** Format the apperture value
1679
        $ev            = isset($exifData['ApertureValue']) ? $exifData['ApertureValue'] : '';
1680
        $apPeicesArray = explode('/', $ev);
1681
        if (2 == count($apPeicesArray)) {
1682
            $apertureValue = round($apPeicesArray[0] / $apPeicesArray[1], 2, PHP_ROUND_HALF_DOWN) . ' EV';
1683
        } else {
1684
            $apertureValue = '';
1685
        }
1686
1687
        // *** Format the focal length
1688
        $focalLength   = isset($exifData['FocalLength']) ? $exifData['FocalLength'] : '';
1689
        $flPeicesArray = explode('/', $focalLength);
1690
        if (2 == count($flPeicesArray)) {
1691
            $focalLength = $flPeicesArray[0] / $flPeicesArray[1] . '.0 mm';
1692
        } else {
1693
            $focalLength = '';
1694
        }
1695
1696
        // *** Format fNumber
1697
        $fNumber       = isset($exifData['FNumber']) ? $exifData['FNumber'] : '';
1698
        $fnPeicesArray = explode('/', $fNumber);
1699
        if (2 == count($fnPeicesArray)) {
1700
            $fNumber = $fnPeicesArray[0] / $fnPeicesArray[1];
1701
        } else {
1702
            $fNumber = '';
1703
        }
1704
1705
        // *** Resolve ExposureProgram
1706
        if (isset($exifData['ExposureProgram'])) {
1707
            $ep = $exifData['ExposureProgram'];
1708
        }
1709
        if (isset($ep)) {
1710
            $ep = $this->resolveExposureProgram($ep);
1711
        }
1712
1713
        // *** Resolve MeteringMode
1714
        $mm = isset($exifData['MeteringMode']) ? $exifData['MeteringMode'] : '';
1715
        $mm = $this->resolveMeteringMode($mm);
1716
1717
        // *** Resolve Flash
1718
        $flash = isset($exifData['Flash']) ? $exifData['Flash'] : '';
1719
        $flash = $this->resolveFlash($flash);
1720
1721
        if (isset($exifData['Make'])) {
1722
            $exifDataArray['make'] = $exifData['Make'];
1723
        } else {
1724
            $exifDataArray['make'] = '';
1725
        }
1726
1727
        if (isset($exifData['Model'])) {
1728
            $exifDataArray['model'] = $exifData['Model'];
1729
        } else {
1730
            $exifDataArray['model'] = '';
1731
        }
1732
1733
        if (isset($exifData['DateTime'])) {
1734
            $exifDataArray['date'] = $exifData['DateTime'];
1735
        } else {
1736
            $exifDataArray['date'] = '';
1737
        }
1738
1739
        if (isset($exifData['ExposureTime'])) {
1740
            $exifDataArray['exposure time'] = $exifData['ExposureTime'] . ' sec.';
1741
        } else {
1742
            $exifDataArray['exposure time'] = '';
1743
        }
1744
1745
        if ('' != $apertureValue) {
1746
            $exifDataArray['aperture value'] = $apertureValue;
1747
        } else {
1748
            $exifDataArray['aperture value'] = '';
1749
        }
1750
1751
        if (isset($exifData['COMPUTED']['ApertureFNumber'])) {
1752
            $exifDataArray['f-stop'] = $exifData['COMPUTED']['ApertureFNumber'];
1753
        } else {
1754
            $exifDataArray['f-stop'] = '';
1755
        }
1756
1757
        if (isset($exifData['FNumber'])) {
1758
            $exifDataArray['fnumber'] = $exifData['FNumber'];
1759
        } else {
1760
            $exifDataArray['fnumber'] = '';
1761
        }
1762
1763
        if ('' != $fNumber) {
1764
            $exifDataArray['fnumber value'] = $fNumber;
1765
        } else {
1766
            $exifDataArray['fnumber value'] = '';
1767
        }
1768
1769
        if (isset($exifData['ISOSpeedRatings'])) {
1770
            $exifDataArray['iso'] = $exifData['ISOSpeedRatings'];
1771
        } else {
1772
            $exifDataArray['iso'] = '';
1773
        }
1774
1775
        if ('' != $focalLength) {
1776
            $exifDataArray['focal length'] = $focalLength;
1777
        } else {
1778
            $exifDataArray['focal length'] = '';
1779
        }
1780
1781
        if (isset($ep)) {
1782
            $exifDataArray['exposure program'] = $ep;
1783
        } else {
1784
            $exifDataArray['exposure program'] = '';
1785
        }
1786
1787
        if ('' != $mm) {
1788
            $exifDataArray['metering mode'] = $mm;
1789
        } else {
1790
            $exifDataArray['metering mode'] = '';
1791
        }
1792
1793
        if ('' != $flash) {
1794
            $exifDataArray['flash status'] = $flash;
1795
        } else {
1796
            $exifDataArray['flash status'] = '';
1797
        }
1798
1799
        if (isset($exifData['Artist'])) {
1800
            $exifDataArray['creator'] = $exifData['Artist'];
1801
        } else {
1802
            $exifDataArray['creator'] = '';
1803
        }
1804
1805
        if (isset($exifData['Copyright'])) {
1806
            $exifDataArray['copyright'] = $exifData['Copyright'];
1807
        } else {
1808
            $exifDataArray['copyright'] = '';
1809
        }
1810
1811
        // *** Orientation
1812
        if (isset($exifData['Orientation'])) {
1813
            $exifDataArray['orientation'] = $exifData['Orientation'];
1814
        } else {
1815
            $exifDataArray['orientation'] = '';
1816
        }
1817
1818
        return $exifDataArray;
1819
    }
1820
1821
    ## --------------------------------------------------------
1822
1823
    private function resolveExposureProgram($ep)
1824
    {
1825
        switch ($ep) {
1826
            case 0:
1827
                $ep = '';
1828
                break;
1829
            case 1:
1830
                $ep = 'manual';
1831
                break;
1832
            case 2:
1833
                $ep = 'normal program';
1834
                break;
1835
            case 3:
1836
                $ep = 'aperture priority';
1837
                break;
1838
            case 4:
1839
                $ep = 'shutter priority';
1840
                break;
1841
            case 5:
1842
                $ep = 'creative program';
1843
                break;
1844
            case 6:
1845
                $ep = 'action program';
1846
                break;
1847
            case 7:
1848
                $ep = 'portrait mode';
1849
                break;
1850
            case 8:
1851
                $ep = 'landscape mode';
1852
                break;
1853
1854
            default:
1855
                break;
1856
        }
1857
1858
        return $ep;
1859
    }
1860
1861
    ## --------------------------------------------------------
1862
1863
    private function resolveMeteringMode($mm)
1864
    {
1865
        switch ($mm) {
1866
            case 0:
1867
                $mm = 'unknown';
1868
                break;
1869
            case 1:
1870
                $mm = 'average';
1871
                break;
1872
            case 2:
1873
                $mm = 'center weighted average';
1874
                break;
1875
            case 3:
1876
                $mm = 'spot';
1877
                break;
1878
            case 4:
1879
                $mm = 'multi spot';
1880
                break;
1881
            case 5:
1882
                $mm = 'pattern';
1883
                break;
1884
            case 6:
1885
                $mm = 'partial';
1886
                break;
1887
            case 255:
1888
                $mm = 'other';
1889
                break;
1890
1891
            default:
1892
                break;
1893
        }
1894
1895
        return $mm;
1896
    }
1897
1898
    ## --------------------------------------------------------
1899
1900
    private function resolveFlash($flash)
1901
    {
1902
        switch ($flash) {
1903
            case 0:
1904
                $flash = 'flash did not fire';
1905
                break;
1906
            case 1:
1907
                $flash = 'flash fired';
1908
                break;
1909
            case 5:
1910
                $flash = 'strobe return light not detected';
1911
                break;
1912
            case 7:
1913
                $flash = 'strobe return light detected';
1914
                break;
1915
            case 9:
1916
                $flash = 'flash fired, compulsory flash mode';
1917
                break;
1918
            case 13:
1919
                $flash = 'flash fired, compulsory flash mode, return light not detected';
1920
                break;
1921
            case 15:
1922
                $flash = 'flash fired, compulsory flash mode, return light detected';
1923
                break;
1924
            case 16:
1925
                $flash = 'flash did not fire, compulsory flash mode';
1926
                break;
1927
            case 24:
1928
                $flash = 'flash did not fire, auto mode';
1929
                break;
1930
            case 25:
1931
                $flash = 'flash fired, auto mode';
1932
                break;
1933
            case 29:
1934
                $flash = 'flash fired, auto mode, return light not detected';
1935
                break;
1936
            case 31:
1937
                $flash = 'flash fired, auto mode, return light detected';
1938
                break;
1939
            case 32:
1940
                $flash = 'no flash function';
1941
                break;
1942
            case 65:
1943
                $flash = 'flash fired, red-eye reduction mode';
1944
                break;
1945
            case 69:
1946
                $flash = 'flash fired, red-eye reduction mode, return light not detected';
1947
                break;
1948
            case 71:
1949
                $flash = 'flash fired, red-eye reduction mode, return light detected';
1950
                break;
1951
            case 73:
1952
                $flash = 'flash fired, compulsory flash mode, red-eye reduction mode';
1953
                break;
1954
            case 77:
1955
                $flash = 'flash fired, compulsory flash mode, red-eye reduction mode, return light not detected';
1956
                break;
1957
            case 79:
1958
                $flash = 'flash fired, compulsory flash mode, red-eye reduction mode, return light detected';
1959
                break;
1960
            case 89:
1961
                $flash = 'flash fired, auto mode, red-eye reduction mode';
1962
                break;
1963
            case 93:
1964
                $flash = 'flash fired, auto mode, return light not detected, red-eye reduction mode';
1965
                break;
1966
            case 95:
1967
                $flash = 'flash fired, auto mode, return light detected, red-eye reduction mode';
1968
                break;
1969
1970
            default:
1971
                break;
1972
        }
1973
1974
        return $flash;
1975
    }
1976
1977
1978
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1979
        Get IPTC Data
1980
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1981
1982
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
1983
        Write IPTC Data
1984
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
1985
1986
    public function writeIPTCcaption($value)
1987
        # Caption
1988
    {
1989
        $this->writeIPTC(120, $value);
1990
    }
1991
1992
    ## --------------------------------------------------------
1993
1994
    public function writeIPTCwriter($value)
1995
    {
1996
        //$this->writeIPTC(65, $value);
1997
    }
1998
1999
    ## --------------------------------------------------------
2000
2001
    private function writeIPTC($dat, $value)
2002
    {
2003
        # LIMIT TO JPG
2004
2005
        $caption_block = $this->iptc_maketag(2, $dat, $value);
2006
        $image_string  = iptcembed($caption_block, $this->fileName);
2007
        file_put_contents('iptc.jpg', $image_string);
2008
    }
2009
2010
    ## --------------------------------------------------------
2011
2012
    private function iptc_maketag($rec, $dat, $val)
2013
        # Author:   Thies C. Arntzen
2014
        # Purpose:    Function to format the new IPTC text
2015
        # Param in:   $rec: Application record. (We’re working with #2)
2016
        #       $dat: Index. (120 for caption, 118 for contact. See the IPTC IIM
2017
        #         specification:
2018
        #         http://www.iptc.org/std/IIM/4.1/specification/IIMV4.1.pdf
2019
        #       $val: Value/data/text. Make sure this is within the length
2020
        #         constraints of the IPTC IIM specification
2021
        # Ref:      http://blog.peterhaza.no/working-with-image-meta-data-in-exif-and-iptc-headers-from-php/
2022
        #       http://php.net/manual/en/function.iptcembed.php
2023
        #
2024
    {
2025
        $len = strlen($val);
2026
        if ($len < 0x8000) {
2027
            return chr(0x1c) . chr($rec) . chr($dat) . chr($len >> 8) . chr($len & 0xff) . $val;
2028
        } else {
2029
            return chr(0x1c) . chr($rec) . chr($dat) . chr(0x80) . chr(0x04) . chr(($len >> 24) & 0xff) . chr(($len >> 16) & 0xff) . chr(($len >> 8) & 0xff) . chr(($len) & 0xff) . $val;
2030
        }
2031
    }
2032
2033
2034
2035
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
2036
        Write XMP Data
2037
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
2038
2039
    //http://xmpphptoolkit.sourceforge.net/
2040
2041
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
2042
        Add Text
2043
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
2044
2045
    public function addText($text, $pos = '20x20', $padding = 0, $fontColor = '#fff', $fontSize = 12, $angle = 0, $font = null)
2046
        # Author:     Jarrod Oberto
2047
        # Date:       18-11-09
2048
        # Purpose:    Add text to an image
2049
        # Param in:
2050
        # Param out:
2051
        # Reference:  http://php.net/manual/en/function.imagettftext.php
2052
        # Notes:      Make sure you supply the font.
2053
        #
2054
    {
2055
        // *** Convert color
2056
        $rgbArray = $this->formatColor($fontColor);
2057
        $r        = $rgbArray['r'];
2058
        $g        = $rgbArray['g'];
2059
        $b        = $rgbArray['b'];
2060
2061
        // *** Get text font
2062
        try {
2063
            $font = $this->getTextFont($font);
2064
        } catch (Exception $e) {
2065
        }
2066
2067
        // *** Get text size
2068
        $textSizeArray = $this->getTextSize($fontSize, $angle, $font, $text);
2069
        $textWidth     = $textSizeArray['width'];
2070
        $textHeight    = $textSizeArray['height'];
2071
2072
        // *** Find co-ords to place text
2073
        $posArray = $this->calculatePosition($pos, $padding, $textWidth, $textHeight, false);
2074
        $x        = $posArray['width'];
2075
        $y        = $posArray['height'];
2076
2077
        $fontColor = imagecolorallocate($this->imageResized, $r, $g, $b);
2078
2079
        // *** Add text
2080
        imagettftext($this->imageResized, $fontSize, $angle, $x, $y, $fontColor, $font, $text);
2081
    }
2082
2083
    ## --------------------------------------------------------
2084
2085
    private function getTextFont($font)
2086
    {
2087
        // *** Font path (shou
2088
        $fontPath = __DIR__ . '/' . $this->fontDir;
2089
2090
        // *** The below is/may be needed depending on your version (see ref)
2091
        putenv('GDFONTPATH=' . realpath('.'));
2092
2093
        // *** Check if the passed in font exsits...
2094
        if (null == $font || !file_exists($font)) {
2095
            // *** ...If not, default to this font.
2096
            $font = $fontPath . '/arimo.ttf';
2097
2098
            // *** Check our default font exists...
2099
            if (!file_exists($font)) {
2100
                // *** If not, return false
2101
                if ($this->debug) {
2102
                    die('Font not found');
2103
                } else {
2104
                    return false;
2105
                }
2106
            }
2107
        }
2108
2109
        return $font;
2110
    }
2111
2112
    ## --------------------------------------------------------
2113
2114
    private function getTextSize($fontSize, $angle, $font, $text)
2115
    {
2116
        // *** Define box (so we can get the width)
2117
        $box = @imagettfbbox($fontSize, $angle, $font, $text);
2118
2119
        // ***  Get width of text from dimensions
2120
        $textWidth = abs($box[4] - $box[0]);
2121
2122
        // ***  Get height of text from dimensions (should also be same as $fontSize)
2123
        $textHeight = abs($box[5] - $box[1]);
2124
2125
        return ['height' => $textHeight, 'width' => $textWidth];
2126
    }
2127
2128
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
2129
        Add Watermark
2130
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
2131
2132
    public function addWatermark($watermarkImage, $pos, $padding = 0, $opacity = 0)
2133
        # Author:     Jarrod Oberto
2134
        # Date:       18-11-09
2135
        # Purpose:    Add watermark image
2136
        # Param in:   (str) $watermark: The watermark image
2137
        #             (str) $pos: Could be a pre-determined position such as:
2138
        #                 tl = top left,
2139
        #                 t  = top (middle),
2140
        #                 tr = top right,
2141
        #                 l  = left,
2142
        #                 m  = middle,
2143
        #                 r  = right,
2144
        #                 bl = bottom left,
2145
        #                 b  = bottom (middle),
2146
        #                 br = bottom right
2147
        #               Or, it could be a co-ordinate position such as: 50x100
2148
        #
2149
        #             (int) $padding: If using a pre-determined position you can
2150
        #               adjust the padding from the edges by passing an amount
2151
        #               in pixels. If using co-ordinates, this value is ignored.
2152
        # Param out:
2153
        # Reference:  http://www.php.net/manual/en/image.examples-watermark.php
2154
        # Notes:      Based on example in reference.
2155
        #
2156
        #
2157
    {
2158
        // Load the stamp and the photo to apply the watermark to
2159
        try {
2160
            $stamp = $this->openImage($watermarkImage);
2161
        } catch (Exception $e) {
2162
        }    # stamp
2163
        $im    = $this->imageResized;            # photo
2164
2165
        // *** Get stamps width and height
2166
        $sx = imagesx($stamp);
2167
        $sy = imagesy($stamp);
2168
2169
        // *** Find co-ords to place image
2170
        $posArray = $this->calculatePosition($pos, $padding, $sx, $sy);
2171
        $x        = $posArray['width'];
2172
        $y        = $posArray['height'];
2173
2174
        // *** Set watermark opacity
2175
        if ('.png' === strtolower(strrchr($watermarkImage, '.'))) {
2176
            $opacity = $this->invertTransparency($opacity, 100);
2177
            $this->filterOpacity($stamp, $opacity);
2178
        }
2179
2180
        // Copy the watermark image onto our photo
2181
        imagecopy($im, $stamp, $x, $y, 0, 0, imagesx($stamp), imagesy($stamp));
2182
    }
2183
2184
    ## --------------------------------------------------------
2185
2186
    private function calculatePosition($pos, $padding, $assetWidth, $assetHeight, $upperLeft = true)
2187
        #
2188
        # Author:   Jarrod Oberto
2189
        # Date:   08-05-11
2190
        # Purpose:  Calculate the x, y pixel cordinates of the asset to place
2191
        # Params in:  (str) $pos: Either something like: "tl", "l", "br" or an
2192
        #         exact position like: "100x50"
2193
        #       (int) $padding: The amount of padding from the edge. Only
2194
        #         used for the predefined $pos.
2195
        #       (int) $assetWidth: The width of the asset to add to the image
2196
        #       (int) $assetHeight: The height of the asset to add to the image
2197
        #       (bol) $upperLeft: if true, the asset will be positioned based
2198
        #         on the upper left x, y coords. If false, it means you're
2199
        #         using the lower left as the basepoint and this will
2200
        #         convert it to the upper left position
2201
        # Params out:
2202
        # NOTE: this is done from the UPPER left corner!! But will convert lower
2203
        #   left basepoints to upper left if $upperleft is set to false
2204
        #
2205
        #
2206
    {
2207
        $pos = strtolower($pos);
2208
2209
        // *** If co-ords have been entered
2210
        if (strstr($pos, 'x')) {
2211
            $pos = str_replace(' ', '', $pos);
2212
2213
            $xyArray = explode('x', $pos);
2214
            list($width, $height) = $xyArray;
2215
        } else {
2216
            switch ($pos) {
2217
                case 'tl':
2218
                    $width  = 0 + $padding;
2219
                    $height = 0 + $padding;
2220
                    break;
2221
2222
                case 't':
2223
                    $width  = ($this->width / 2) - ($assetWidth / 2);
2224
                    $height = 0 + $padding;
2225
                    break;
2226
2227
                case 'tr':
2228
                    $width  = $this->width - $assetWidth - $padding;
2229
                    $height = 0 + $padding;
2230
                    break;
2231
2232
                case 'l':
2233
                    $width  = 0 + $padding;
2234
                    $height = ($this->height / 2) - ($assetHeight / 2);
2235
                    break;
2236
2237
                case 'm':
2238
                    $width  = ($this->width / 2) - ($assetWidth / 2);
2239
                    $height = ($this->height / 2) - ($assetHeight / 2);
2240
                    break;
2241
2242
                case 'r':
2243
                    $width  = $this->width - $assetWidth - $padding;
2244
                    $height = ($this->height / 2) - ($assetHeight / 2);
2245
                    break;
2246
2247
                case 'bl':
2248
                    $width  = 0 + $padding;
2249
                    $height = $this->height - $assetHeight - $padding;
2250
                    break;
2251
2252
                case 'b':
2253
                    $width  = ($this->width / 2) - ($assetWidth / 2);
2254
                    $height = $this->height - $assetHeight - $padding;
2255
                    break;
2256
2257
                case 'br':
2258
                    $width  = $this->width - $assetWidth - $padding;
2259
                    $height = $this->height - $assetHeight - $padding;
2260
                    break;
2261
2262
                default:
2263
                    $width  = 0;
2264
                    $height = 0;
2265
                    break;
2266
            }
2267
        }
2268
2269
        if (!$upperLeft) {
2270
            $height = $height + $assetHeight;
2271
        }
2272
2273
        return ['width' => $width, 'height' => $height];
2274
    }
2275
2276
    ## --------------------------------------------------------
2277
2278
    private function filterOpacity(&$img, $opacity = 75)
2279
        #
2280
        # Author:     aiden dot mail at freemail dot hu
2281
        # Author date:  29-03-08 08:16
2282
        # Date added:   08-05-11
2283
        # Purpose:    Change opacity of image
2284
        # Params in:    $img: Image resource id
2285
        #         (int) $opacity: the opacity amount: 0-100, 100 being not opaque.
2286
        # Params out:   (bool) true on success, else false
2287
        # Ref:      http://www.php.net/manual/en/function.imagefilter.php#82162
2288
        # Notes:      png only
2289
        #
2290
    {
2291
        if (!isset($opacity)) {
2292
            return false;
2293
        }
2294
2295
        if (100 == $opacity) {
2296
            return true;
2297
        }
2298
2299
        $opacity /= 100;
2300
2301
        //get image width and height
2302
        $w = imagesx($img);
2303
        $h = imagesy($img);
2304
2305
        //turn alpha blending off
2306
        imagealphablending($img, false);
2307
2308
        //find the most opaque pixel in the image (the one with the smallest alpha value)
2309
        $minalpha = 127;
2310
        for ($x = 0; $x < $w; ++$x) {
2311
            for ($y = 0; $y < $h; ++$y) {
2312
                $alpha = (imagecolorat($img, $x, $y) >> 24) & 0xFF;
2313
                if ($alpha < $minalpha) {
2314
                    $minalpha = $alpha;
2315
                }
2316
            }
2317
        }
2318
2319
        //loop through image pixels and modify alpha for each
2320
        for ($x = 0; $x < $w; ++$x) {
2321
            for ($y = 0; $y < $h; ++$y) {
2322
                //get current alpha value (represents the TANSPARENCY!)
2323
                $colorxy = imagecolorat($img, $x, $y);
2324
                $alpha   = ($colorxy >> 24) & 0xFF;
2325
                //calculate new alpha
2326
                if (127 !== $minalpha) {
2327
                    $alpha = 127 + 127 * $opacity * ($alpha - 127) / (127 - $minalpha);
2328
                } else {
2329
                    $alpha += 127 * $opacity;
2330
                }
2331
                //get the color index with new alpha
2332
                $alphacolorxy = imagecolorallocatealpha($img, ($colorxy >> 16) & 0xFF, ($colorxy >> 8) & 0xFF, $colorxy & 0xFF, $alpha);
2333
                //set pixel with the new color + opacity
2334
                if (!imagesetpixel($img, $x, $y, $alphacolorxy)) {
2335
                    return false;
2336
                }
2337
            }
2338
        }
2339
2340
        return true;
2341
    }
2342
2343
    ## --------------------------------------------------------
2344
2345
    private function openImage($file)
2346
        # Author:     Jarrod Oberto
2347
        # Date:       27-02-08
2348
        # Purpose:
2349
        # Param in:
2350
        # Param out:  n/a
2351
        # Reference:
2352
        # Notes:
2353
        #
2354
    {
2355
        if (!file_exists($file) && !$this->checkStringStartsWith('http://', $file)) {
2356
            if ($this->debug) {
2357
                die('Image not found.');
2358
            } else {
2359
                die();
2360
            }
2361
        }
2362
2363
        // *** Get extension
2364
        $extension = strrchr($file, '.');
2365
        $extension = strtolower($extension);
2366
2367
        switch ($extension) {
2368
            case '.jpg':
2369
            case '.jpeg':
2370
                $img = @imagecreatefromjpeg($file);
2371
                break;
2372
            case '.gif':
2373
                $img = @imagecreatefromgif($file);
2374
                break;
2375
            case '.png':
2376
                $img = @imagecreatefrompng($file);
2377
                break;
2378
            case '.bmp':
2379
                $img = @$this->ImageCreateFromBMP($file);
2380
                break;
2381
            case '.psd':
2382
                $img = @$this->imagecreatefrompsd($file);
2383
                break;
2384
2385
            // ... etc
2386
2387
            default:
2388
                $img = false;
2389
                break;
2390
        }
2391
2392
        return $img;
2393
    }
2394
2395
    ## --------------------------------------------------------
2396
2397
    public function reset()
2398
        #
2399
        # Author:   Jarrod Oberto
2400
        # Date:   30-08-11
2401
        # Purpose:  Reset the resource (allow further editing)
2402
        # Params in:
2403
        # Params out:
2404
        # Notes:
2405
        #
2406
    {
2407
        $this->__destruct();
2408
        $this->image        = null;
2409
        $this->imageResized = null;
2410
        gc_collect_cycles();
2411
        try {
2412
            $this->__construct($this->fileName);
2413
        } catch (Exception $e) {
2414
        }
2415
    }
2416
2417
    ## --------------------------------------------------------
2418
2419
    public function saveImage($savePath, $imageQuality = '100')
2420
        # Author:     Jarrod Oberto
2421
        # Date:       27-02-08
2422
        # Purpose:    Saves the image
2423
        # Param in:   $savePath: Where to save the image including filename:
2424
        #             $imageQuality: image quality you want the image saved at 0-100
2425
        # Param out:  n/a
2426
        # Reference:
2427
        # Notes:      * gif doesn't have a quality parameter
2428
        #             * jpg has a quality setting 0-100 (100 being the best)
2429
        #             * png has a quality setting 0-9 (0 being the best)
2430
        #
2431
        #             * bmp files have no native support for bmp files. We use a
2432
        #               third party class to save as bmp.
2433
    {
2434
        // *** Perform a check or two.
2435
        if (!is_resource($this->imageResized)) {
2436
            if ($this->debug) {
2437
                die('saveImage: This is not a resource.');
2438
            } else {
2439
                die();
2440
            }
2441
        }
2442
        $fileInfoArray = pathinfo($savePath);
2443
        clearstatcache();
2444
        if (!is_writable($fileInfoArray['dirname'])) {
2445
            if ($this->debug) {
2446
                die('The path is not writable. Please check your permissions.');
2447
            } else {
2448
                die();
2449
            }
2450
        }
2451
2452
        // *** Get extension
2453
        $extension = strrchr($savePath, '.');
2454
        $extension = strtolower($extension);
2455
2456
        $error = '';
2457
2458
        switch ($extension) {
2459
            case '.jpg':
2460
            case '.jpeg':
2461
                $this->checkInterlaceImage($this->isInterlace);
2462
                if (imagetypes() & IMG_JPG) {
2463
                    imagejpeg($this->imageResized, $savePath, $imageQuality);
2464
                } else {
2465
                    $error = 'jpg';
2466
                }
2467
                break;
2468
2469
            case '.gif':
2470
                $this->checkInterlaceImage($this->isInterlace);
2471
                if (imagetypes() & IMG_GIF) {
2472
                    imagegif($this->imageResized, $savePath);
2473
                } else {
2474
                    $error = 'gif';
2475
                }
2476
                break;
2477
2478
            case '.png':
2479
                // *** Scale quality from 0-100 to 0-9
2480
                $scaleQuality = round(($imageQuality / 100) * 9);
2481
2482
                // *** Invert qualit setting as 0 is best, not 9
2483
                $invertScaleQuality = 9 - $scaleQuality;
2484
2485
                $this->checkInterlaceImage($this->isInterlace);
2486
                if (imagetypes() & IMG_PNG) {
2487
                    imagepng($this->imageResized, $savePath, $invertScaleQuality);
2488
                } else {
2489
                    $error = 'png';
2490
                }
2491
                break;
2492
2493
            case '.bmp':
2494
                file_put_contents($savePath, $this->GD2BMPstring($this->imageResized));
2495
                break;
2496
2497
            // ... etc
2498
2499
            default:
2500
2501
                // *** No extension - No save.
2502
                $this->errorArray[] = 'This file type (' . $extension . ') is not supported. File not saved.';
2503
                break;
2504
        }
2505
2506
        //imagedestroy($this->imageResized);
2507
2508
        // *** Display error if a file type is not supported.
2509
        if ('' != $error) {
2510
            $this->errorArray[] = $error . ' support is NOT enabled. File not saved.';
2511
        }
2512
    }
2513
2514
    ## --------------------------------------------------------
2515
2516
    public function displayImage($fileType = 'jpg', $imageQuality = '100')
2517
        # Author:     Jarrod Oberto
2518
        # Date:       18-11-09
2519
        # Purpose:    Display images directly to the browser
2520
        # Param in:   The image type you want to display
2521
        # Param out:
2522
        # Reference:
2523
        # Notes:
2524
        #
2525
    {
2526
        if (!is_resource($this->imageResized)) {
2527
            if ($this->debug) {
2528
                die('saveImage: This is not a resource.');
2529
            } else {
2530
                die();
2531
            }
2532
        }
2533
2534
        switch ($fileType) {
2535
            case 'jpg':
2536
            case 'jpeg':
2537
                header('Content-type: image/jpeg');
2538
                imagejpeg($this->imageResized, '', $imageQuality);
2539
                break;
2540
            case 'gif':
2541
                header('Content-type: image/gif');
2542
                imagegif($this->imageResized);
2543
                break;
2544
            case 'png':
2545
                header('Content-type: image/png');
2546
2547
                // *** Scale quality from 0-100 to 0-9
2548
                $scaleQuality = round(($imageQuality / 100) * 9);
2549
2550
                // *** Invert qualit setting as 0 is best, not 9
2551
                $invertScaleQuality = 9 - $scaleQuality;
2552
2553
                imagepng($this->imageResized, '', $invertScaleQuality);
2554
                break;
2555
            case 'bmp':
2556
                echo 'bmp file format is not supported.';
2557
                break;
2558
2559
            // ... etc
2560
2561
            default:
2562
                // *** No extension - No save.
2563
                break;
2564
        }
2565
        //imagedestroy($this->imageResized);
2566
    }
2567
2568
    ## --------------------------------------------------------
2569
2570
    public function setTransparency($bool)
2571
        # Sep 2011
2572
    {
2573
        $this->keepTransparency = $bool;
2574
    }
2575
2576
    ## --------------------------------------------------------
2577
2578
    public function setFillColor($value)
2579
        # Sep 2011
2580
        # Param in:   (mixed) $value: (array) Could be an array of RGB
2581
        #             (str) Could be hex #ffffff or #fff, fff, ffffff
2582
        #
2583
        # If the keepTransparency is set to false, then no transparency is to be used.
2584
        # This is ideal when you want to save as jpg.
2585
        #
2586
        # this method allows you to set the background color to use instead of
2587
        # transparency.
2588
        #
2589
    {
2590
        $colorArray           = $this->formatColor($value);
2591
        $this->fillColorArray = $colorArray;
2592
    }
2593
2594
    ## --------------------------------------------------------
2595
2596
    public function setCropFromTop($value)
2597
        # Sep 2011
2598
    {
2599
        $this->cropFromTopPercent = $value;
2600
    }
2601
2602
    ## --------------------------------------------------------
2603
2604
    public function testGDInstalled()
2605
        # Author:     Jarrod Oberto
2606
        # Date:       27-02-08
2607
        # Purpose:    Test to see if GD is installed
2608
        # Param in:   n/a
2609
        # Param out:  (bool) True is gd extension loaded otherwise false
2610
        # Reference:
2611
        # Notes:
2612
        #
2613
    {
2614
        if (extension_loaded('gd') && function_exists('gd_info')) {
2615
            $gdInstalled = true;
2616
        } else {
2617
            $gdInstalled = false;
2618
        }
2619
2620
        return $gdInstalled;
2621
    }
2622
2623
    ## --------------------------------------------------------
2624
2625
    public function testEXIFInstalled()
2626
        # Author:     Jarrod Oberto
2627
        # Date:       08-05-11
2628
        # Purpose:    Test to see if EXIF is installed
2629
        # Param in:   n/a
2630
        # Param out:  (bool) True is exif extension loaded otherwise false
2631
        # Reference:
2632
        # Notes:
2633
        #
2634
    {
2635
        if (extension_loaded('exif')) {
2636
            $exifInstalled = true;
2637
        } else {
2638
            $exifInstalled = false;
2639
        }
2640
2641
        return $exifInstalled;
2642
    }
2643
2644
    ## --------------------------------------------------------
2645
2646
    public function testIsImage()
2647
        # Author:     Jarrod Oberto
2648
        # Date:       28 Nov 16
2649
        # Purpose:    Test if file is an image
2650
        # Param in:
2651
        # Param out:  n/a
2652
        # Reference:
2653
        # Notes: A simpler, less restrictive method would be to just check for
2654
        #           the 'image' part of 'image/gif', 'image/jpg', etc.
2655
        #
2656
    {
2657
        $file    = $this->fileName;
2658
        $isImage = false;
2659
2660
        $finfo    = finfo_open(FILEINFO_MIME_TYPE);
2661
        $mimeType = finfo_file($finfo, $file);
2662
        finfo_close($finfo);
2663
2664
        switch ($mimeType) {
2665
            case 'image/jpeg':
2666
            case 'image/gif':
2667
            case 'image/png':
2668
            case 'image/bmp':
2669
            case 'image/x-windows-bmp':
2670
                $isImage = true;
2671
                break;
2672
            default:
2673
                $isImage = false;
2674
        }
2675
2676
        return $isImage;
2677
    }
2678
2679
    ## --------------------------------------------------------
2680
2681
    public function getIsImage()
2682
        # Author:     Jarrod Oberto
2683
        # Date:       28 Nov 16
2684
        # Purpose:    Get testIsImage result
2685
        # Param in:
2686
        # Param out:  n/a
2687
        # Reference:
2688
        # Notes:
2689
        #
2690
    {
2691
        return $this->isImage;
2692
    }
2693
2694
    ## --------------------------------------------------------
2695
2696
    public function testFunct()
2697
        # Author:     Jarrod Oberto
2698
        # Date:       27-02-08
2699
        # Purpose:    Test Function
2700
        # Param in:   n/a
2701
        # Param out:  n/a
2702
        # Reference:
2703
        # Notes:
2704
        #
2705
    {
2706
        echo $this->height;
2707
    }
2708
2709
    ## --------------------------------------------------------
2710
2711
    public function setForceStretch($value)
2712
        # Author:     Jarrod Oberto
2713
        # Date:       23-12-10
2714
        # Purpose:
2715
        # Param in:   (bool) $value
2716
        # Param out:  n/a
2717
        # Reference:
2718
        # Notes:
2719
        #
2720
    {
2721
        $this->forceStretch = $value;
2722
    }
2723
2724
    ## --------------------------------------------------------
2725
2726
    public function setFile($fileName)
2727
        # Author:     Jarrod Oberto
2728
        # Date:       28-02-08
2729
        # Purpose:
2730
        # Param in:   n/a
2731
        # Param out:  n/a
2732
        # Reference:
2733
        # Notes:
2734
        #
2735
    {
2736
        try {
2737
            self::__construct($fileName);
2738
        } catch (Exception $e) {
2739
        }
2740
    }
2741
2742
    ## --------------------------------------------------------
2743
2744
    public function getFileName()
2745
        # Author:     Jarrod Oberto
2746
        # Date:       10-09-08
2747
        # Purpose:
2748
        # Param in:   n/a
2749
        # Param out:  n/a
2750
        # Reference:
2751
        # Notes:
2752
        #
2753
    {
2754
        return $this->fileName;
2755
    }
2756
2757
    ## --------------------------------------------------------
2758
2759
    public function getHeight()
2760
    {
2761
        return $this->height;
2762
    }
2763
2764
    ## --------------------------------------------------------
2765
2766
    public function getWidth()
2767
    {
2768
        return $this->width;
2769
    }
2770
2771
    ## --------------------------------------------------------
2772
2773
    public function getOriginalHeight()
2774
    {
2775
        return $this->heightOriginal;
2776
    }
2777
2778
    ## --------------------------------------------------------
2779
2780
    public function getOriginalWidth()
2781
    {
2782
        return $this->widthOriginal;
2783
    }
2784
2785
    ## --------------------------------------------------------
2786
2787
    public function getErrors()
2788
        # Author:     Jarrod Oberto
2789
        # Date:       19-11-09
2790
        # Purpose:    Returns the error array
2791
        # Param in:   n/a
2792
        # Param out:  Array of errors
2793
        # Reference:
2794
        # Notes:
2795
        #
2796
    {
2797
        return $this->errorArray;
2798
    }
2799
2800
    ## --------------------------------------------------------
2801
2802
    private function checkInterlaceImage($isEnabled)
2803
        # jpg will use progressive (they don't use interace)
2804
    {
2805
        if ($isEnabled) {
2806
            imageinterlace($this->imageResized, $isEnabled);
2807
        }
2808
    }
2809
2810
    ## --------------------------------------------------------
2811
2812
    protected function formatColor($value)
2813
        # Author:     Jarrod Oberto
2814
        # Date:       09-05-11
2815
        # Purpose:    Determine color method passed in and return color as RGB
2816
        # Param in:   (mixed) $value: (array) Could be an array of RGB
2817
        #             (str) Could be hex #ffffff or #fff, fff, ffffff
2818
        # Param out:
2819
        # Reference:
2820
        # Notes:
2821
        #
2822
    {
2823
        $rgbArray = [];
2824
2825
        // *** If it's an array it should be R, G, B
2826
        if (is_array($value)) {
2827
            if (0 == key($value) && 3 == count($value)) {
2828
                $rgbArray['r'] = $value[0];
2829
                $rgbArray['g'] = $value[1];
2830
                $rgbArray['b'] = $value[2];
2831
            } else {
2832
                $rgbArray = $value;
2833
            }
2834
        } elseif ('transparent' === strtolower($value)) {
2835
            $rgbArray = [
2836
                'r' => 255,
2837
                'g' => 255,
2838
                'b' => 255,
2839
                'a' => 127
2840
            ];
2841
        } else {
2842
            // *** ...Else it should be hex. Let's make it RGB
2843
            $rgbArray = $this->hex2dec($value);
2844
        }
2845
2846
        return $rgbArray;
2847
    }
2848
2849
    ## --------------------------------------------------------
2850
2851
    public function hex2dec($hex)
2852
        # Purpose:  Convert #hex color to RGB
2853
    {
2854
        $color = str_replace('#', '', $hex);
2855
2856
        if (3 == strlen($color)) {
2857
            $color = $color . $color;
2858
        }
2859
2860
        $rgb = [
2861
            'r' => hexdec(substr($color, 0, 2)),
2862
            'g' => hexdec(substr($color, 2, 2)),
2863
            'b' => hexdec(substr($color, 4, 2)),
2864
            'a' => 0
2865
        ];
2866
        return $rgb;
2867
    }
2868
2869
    ## --------------------------------------------------------
2870
2871
    private function createImageColor($colorArray)
2872
    {
2873
        $r = $colorArray['r'];
2874
        $g = $colorArray['g'];
2875
        $b = $colorArray['b'];
2876
2877
        return imagecolorallocate($this->imageResized, $r, $g, $b);
2878
    }
2879
2880
    ## --------------------------------------------------------
2881
2882
    private function testColorExists($colorArray)
2883
    {
2884
        $r = $colorArray['r'];
2885
        $g = $colorArray['g'];
2886
        $b = $colorArray['b'];
2887
2888
        if (-1 == imagecolorexact($this->imageResized, $r, $g, $b)) {
2889
            return false;
2890
        } else {
2891
            return true;
2892
        }
2893
    }
2894
2895
    ## --------------------------------------------------------
2896
2897
    private function findUnusedGreen()
2898
        # Purpose:  We find a green color suitable to use like green-screen effect.
2899
        #     Therefore, the color must not exist in the image.
2900
    {
2901
        $green = 255;
2902
2903
        do {
2904
            $greenChroma = [0, $green, 0];
2905
            $colorArray  = $this->formatColor($greenChroma);
2906
            $match       = $this->testColorExists($colorArray);
2907
            $green--;
2908
        } while (false == $match && $green > 0);
2909
2910
        // *** If no match, just bite the bullet and use green value of 255
2911
        if (!$match) {
2912
            $greenChroma = [0, $green, 0];
2913
        }
2914
2915
        return $greenChroma;
2916
    }
2917
2918
    ## --------------------------------------------------------
2919
2920
    private function findUnusedBlue()
2921
        # Purpose:  We find a green color suitable to use like green-screen effect.
2922
        #     Therefore, the color must not exist in the image.
2923
    {
2924
        $blue = 255;
2925
2926
        do {
2927
            $blueChroma = [0, 0, $blue];
2928
            $colorArray = $this->formatColor($blueChroma);
2929
            $match      = $this->testColorExists($colorArray);
2930
            $blue--;
2931
        } while (false == $match && $blue > 0);
2932
2933
        // *** If no match, just bite the bullet and use blue value of 255
2934
        if (!$match) {
2935
            $blueChroma = [0, 0, $blue];
2936
        }
2937
2938
        return $blueChroma;
2939
    }
2940
2941
    ## --------------------------------------------------------
2942
2943
    private function invertTransparency($value, $originalMax, $invert = true)
2944
        # Purpose:  This does two things:
2945
        #       1) Convert the range from 0-127 to 0-100
2946
        #       2) Inverts value to 100 is not transparent while 0 is fully
2947
        #          transparent (like Photoshop)
2948
    {
2949
        // *** Test max range
2950
        if ($value > $originalMax) {
2951
            $value = $originalMax;
2952
        }
2953
2954
        // *** Test min range
2955
        if ($value < 0) {
2956
            $value = 0;
2957
        }
2958
2959
        if ($invert) {
2960
            return $originalMax - (($value / 100) * $originalMax);
2961
        } else {
2962
            return ($value / 100) * $originalMax;
2963
        }
2964
    }
2965
2966
    ## --------------------------------------------------------
2967
2968
    private function transparentImage($src)
2969
    {
2970
        // *** making images with white bg transparent
2971
        $r1 = 0;
2972
        $g1 = 255;
2973
        $b1 = 0;
2974
        for ($x = 0; $x < imagesx($src); ++$x) {
2975
            for ($y = 0; $y < imagesy($src); ++$y) {
2976
                $color = imagecolorat($src, $x, $y);
2977
                $r     = ($color >> 16) & 0xFF;
2978
                $g     = ($color >> 8) & 0xFF;
2979
                $b     = $color & 0xFF;
2980
                for ($i = 0; $i < 270; ++$i) {
2981
                    //if ($r . $g . $b == ($r1 + $i) . ($g1 + $i) . ($b1 + $i)) {
2982
                    if (0 == $r && 255 == $g && 0 == $b) {
2983
                        //if ($g == 255) {
2984
                        $trans_colour = imagecolorallocatealpha($src, 0, 0, 0, 127);
2985
                        imagefill($src, $x, $y, $trans_colour);
2986
                    }
2987
                }
2988
            }
2989
        }
2990
2991
        return $src;
2992
    }
2993
2994
    ## --------------------------------------------------------
2995
2996
    public function checkStringStartsWith($needle, $haystack)
2997
        # Check if a string starts with a specific pattern
2998
    {
2999
        return (substr($haystack, 0, strlen($needle)) == $needle);
3000
    }
3001
3002
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
3003
        BMP SUPPORT (SAVING) - James Heinrich
3004
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
3005
3006
    private function GD2BMPstring(&$gd_image)
3007
        # Author:     James Heinrich
3008
        # Purpose:    Save file as type bmp
3009
        # Param in:   The image canvas (passed as ref)
3010
        # Param out:
3011
        # Reference:
3012
        # Notes:      This code was stripped out of two external files
3013
        #             (phpthumb.bmp.php,phpthumb.functions.php) and added below to
3014
        #             avoid dependancies.
3015
        #
3016
    {
3017
        $imageX = imagesx($gd_image);
3018
        $imageY = imagesy($gd_image);
3019
3020
        $BMP = '';
3021
        for ($y = ($imageY - 1); $y >= 0; $y--) {
3022
            $thisline = '';
3023
            for ($x = 0; $x < $imageX; ++$x) {
3024
                $argb     = $this->GetPixelColor($gd_image, $x, $y);
3025
                $thisline .= chr($argb['blue']) . chr($argb['green']) . chr($argb['red']);
3026
            }
3027
            while (strlen($thisline) % 4) {
3028
                $thisline .= "\x00";
3029
            }
3030
            $BMP .= $thisline;
3031
        }
3032
3033
        $bmpSize = strlen($BMP) + 14 + 40;
3034
        // BITMAPFILEHEADER [14 bytes] - http://msdn.microsoft.com/library/en-us/gdi/bitmaps_62uq.asp
3035
        $BITMAPFILEHEADER = 'BM';                                    // WORD    bfType;
3036
        $BITMAPFILEHEADER .= $this->LittleEndian2String($bmpSize, 4); // DWORD   bfSize;
3037
        $BITMAPFILEHEADER .= $this->LittleEndian2String(0, 2); // WORD    bfReserved1;
3038
        $BITMAPFILEHEADER .= $this->LittleEndian2String(0, 2); // WORD    bfReserved2;
3039
        $BITMAPFILEHEADER .= $this->LittleEndian2String(54, 4); // DWORD   bfOffBits;
3040
3041
        // BITMAPINFOHEADER - [40 bytes] http://msdn.microsoft.com/library/en-us/gdi/bitmaps_1rw2.asp
3042
        $BITMAPINFOHEADER = $this->LittleEndian2String(40, 4); // DWORD  biSize;
3043
        $BITMAPINFOHEADER .= $this->LittleEndian2String($imageX, 4); // LONG   biWidth;
3044
        $BITMAPINFOHEADER .= $this->LittleEndian2String($imageY, 4); // LONG   biHeight;
3045
        $BITMAPINFOHEADER .= $this->LittleEndian2String(1, 2); // WORD   biPlanes;
3046
        $BITMAPINFOHEADER .= $this->LittleEndian2String(24, 2); // WORD   biBitCount;
3047
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biCompression;
3048
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biSizeImage;
3049
        $BITMAPINFOHEADER .= $this->LittleEndian2String(2835, 4); // LONG   biXPelsPerMeter;
3050
        $BITMAPINFOHEADER .= $this->LittleEndian2String(2835, 4); // LONG   biYPelsPerMeter;
3051
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biClrUsed;
3052
        $BITMAPINFOHEADER .= $this->LittleEndian2String(0, 4); // DWORD  biClrImportant;
3053
3054
        return $BITMAPFILEHEADER . $BITMAPINFOHEADER . $BMP;
3055
    }
3056
3057
    ## --------------------------------------------------------
3058
3059
    private function GetPixelColor(&$img, $x, $y)
3060
        # Author:     James Heinrich
3061
        # Purpose:
3062
        # Param in:
3063
        # Param out:
3064
        # Reference:
3065
        # Notes:
3066
        #
3067
    {
3068
        if (!is_resource($img)) {
3069
            return false;
3070
        }
3071
        return @imagecolorsforindex($img, @imagecolorat($img, $x, $y));
3072
    }
3073
3074
    ## --------------------------------------------------------
3075
3076
    private function LittleEndian2String($number, $minbytes = 1)
3077
        # Author:     James Heinrich
3078
        # Purpose:    BMP SUPPORT (SAVING)
3079
        # Param in:
3080
        # Param out:
3081
        # Reference:
3082
        # Notes:
3083
        #
3084
    {
3085
        $intstring = '';
3086
        while ($number > 0) {
3087
            $intstring = $intstring . chr($number & 255);
3088
            $number    >>= 8;
3089
        }
3090
        return str_pad($intstring, $minbytes, "\x00", STR_PAD_RIGHT);
3091
    }
3092
3093
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
3094
        BMP SUPPORT (READING)
3095
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
3096
3097
    private function ImageCreateFromBMP($filename)
3098
        # Author:     DHKold
3099
        # Date:     The 15th of June 2005
3100
        # Version:    2.0B
3101
        # Purpose:    To create an image from a BMP file.
3102
        # Param in:   BMP file to open.
3103
        # Param out:  Return a resource like the other ImageCreateFrom functions
3104
        # Reference:  http://us3.php.net/manual/en/function.imagecreate.php#53879
3105
        # Bug fix:    Author:   domelca at terra dot es
3106
        #             Date:   06 March 2008
3107
        #             Fix:    Correct 16bit BMP support
3108
        # Notes:
3109
        #
3110
    {
3111
        //Ouverture du fichier en mode binaire
3112
        if (!$f1 = fopen($filename, 'rb')) {
3113
            return false;
3114
        }
3115
3116
        //1 : Chargement des ent�tes FICHIER
3117
        $FILE = unpack('vfile_type/Vfile_size/Vreserved/Vbitmap_offset', fread($f1, 14));
3118
        if (19778 != $FILE['file_type']) {
3119
            return false;
3120
        }
3121
3122
        //2 : Chargement des ent�tes BMP
3123
        $BMP           = unpack(
3124
            'Vheader_size/Vwidth/Vheight/vplanes/vbits_per_pixel' . '/Vcompression/Vsize_bitmap/Vhoriz_resolution' . '/Vvert_resolution/Vcolors_used/Vcolors_important',
3125
            fread($f1, 40)
3126
        );
3127
        $BMP['colors'] = pow(2, $BMP['bits_per_pixel']);
3128
3129
        if (0 == $BMP['size_bitmap']) {
3130
            $BMP['size_bitmap'] = $FILE['file_size'] - $FILE['bitmap_offset'];
3131
        }
3132
3133
        $BMP['bytes_per_pixel']  = $BMP['bits_per_pixel'] / 8;
3134
        $BMP['bytes_per_pixel2'] = ceil($BMP['bytes_per_pixel']);
3135
        $BMP['decal']            = ($BMP['width'] * $BMP['bytes_per_pixel'] / 4);
3136
        $BMP['decal']            -= floor($BMP['width'] * $BMP['bytes_per_pixel'] / 4);
3137
        $BMP['decal']            = 4 - (4 * $BMP['decal']);
3138
3139
        if (4 == $BMP['decal']) {
3140
            $BMP['decal'] = 0;
3141
        }
3142
3143
        //3 : Chargement des couleurs de la palette
3144
        $PALETTE = [];
3145
        if ($BMP['colors'] < 16777216) {
3146
            $PALETTE = unpack('V' . $BMP['colors'], fread($f1, $BMP['colors'] * 4));
3147
        }
3148
3149
        //4 : Cr�ation de l'image
3150
        $IMG  = fread($f1, $BMP['size_bitmap']);
3151
        $VIDE = chr(0);
3152
3153
        $res = imagecreatetruecolor($BMP['width'], $BMP['height']);
3154
        $P   = 0;
3155
        $Y   = $BMP['height'] - 1;
3156
        while ($Y >= 0) {
3157
            $X = 0;
3158
            while ($X < $BMP['width']) {
3159
                if (24 == $BMP['bits_per_pixel']) {
3160
                    $COLOR = unpack('V', substr($IMG, $P, 3) . $VIDE);
3161
                } elseif (16 == $BMP['bits_per_pixel']) {
3162
                    /*
3163
                     * BMP 16bit fix
3164
                     * =================
3165
                     *
3166
                     * Ref: http://us3.php.net/manual/en/function.imagecreate.php#81604
3167
                     *
3168
                     * Notes:
3169
                     * "don't work with bmp 16 bits_per_pixel. change pixel
3170
                     * generator for this."
3171
                     *
3172
                     */
3173
3174
                    // *** Original code (don't work)
3175
                    //$COLOR = unpack("n",substr($IMG,$P,2));
3176
                    //$COLOR[1] = $PALETTE[$COLOR[1]+1];
3177
3178
                    $COLOR    = unpack('v', substr($IMG, $P, 2));
3179
                    $blue     = ($COLOR[1] & 0x001f) << 3;
3180
                    $green    = ($COLOR[1] & 0x07e0) >> 3;
3181
                    $red      = ($COLOR[1] & 0xf800) >> 8;
3182
                    $COLOR[1] = $red * 65536 + $green * 256 + $blue;
3183
                } elseif (8 == $BMP['bits_per_pixel']) {
3184
                    $COLOR    = unpack('n', $VIDE . substr($IMG, $P, 1));
3185
                    $COLOR[1] = $PALETTE[$COLOR[1] + 1];
3186
                } elseif (4 == $BMP['bits_per_pixel']) {
3187
                    $COLOR = unpack('n', $VIDE . substr($IMG, floor($P), 1));
3188
                    if (0 == ($P * 2) % 2) {
3189
                        $COLOR[1] = ($COLOR[1] >> 4);
3190
                    } else {
3191
                        $COLOR[1] = ($COLOR[1] & 0x0F);
3192
                    }
3193
                    $COLOR[1] = $PALETTE[$COLOR[1] + 1];
3194
                } elseif (1 == $BMP['bits_per_pixel']) {
3195
                    $COLOR = unpack('n', $VIDE . substr($IMG, floor($P), 1));
3196
                    if (0 == ($P * 8) % 8) {
3197
                        $COLOR[1] = $COLOR[1] >> 7;
3198
                    } elseif (1 == ($P * 8) % 8) {
3199
                        $COLOR[1] = ($COLOR[1] & 0x40) >> 6;
3200
                    } elseif (2 == ($P * 8) % 8) {
3201
                        $COLOR[1] = ($COLOR[1] & 0x20) >> 5;
3202
                    } elseif (3 == ($P * 8) % 8) {
3203
                        $COLOR[1] = ($COLOR[1] & 0x10) >> 4;
3204
                    } elseif (4 == ($P * 8) % 8) {
3205
                        $COLOR[1] = ($COLOR[1] & 0x8) >> 3;
3206
                    } elseif (5 == ($P * 8) % 8) {
3207
                        $COLOR[1] = ($COLOR[1] & 0x4) >> 2;
3208
                    } elseif (6 == ($P * 8) % 8) {
3209
                        $COLOR[1] = ($COLOR[1] & 0x2) >> 1;
3210
                    } elseif (7 == ($P * 8) % 8) {
3211
                        $COLOR[1] = ($COLOR[1] & 0x1);
3212
                    }
3213
                    $COLOR[1] = $PALETTE[$COLOR[1] + 1];
3214
                } else {
3215
                    return false;
3216
                }
3217
3218
                imagesetpixel($res, $X, $Y, $COLOR[1]);
3219
                ++$X;
3220
                $P += $BMP['bytes_per_pixel'];
3221
            }
3222
3223
            $Y--;
3224
            $P += $BMP['decal'];
3225
        }
3226
3227
        //Fermeture du fichier
3228
        fclose($f1);
3229
3230
        return $res;
3231
    }
3232
3233
    /*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*-
3234
        PSD SUPPORT (READING)
3235
    *-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-*-**-*-*-*-*-*-*-*-*-*-*-*-*-*/
3236
3237
    private function imagecreatefrompsd($fileName)
3238
        # Author:     Tim de Koning
3239
        # Version:    1.3
3240
        # Purpose:    To create an image from a PSD file.
3241
        # Param in:   PSD file to open.
3242
        # Param out:  Return a resource like the other ImageCreateFrom functions
3243
        # Reference:  http://www.kingsquare.nl/phppsdreader
3244
        # Notes:
3245
        #
3246
    {
3247
        if (file_exists($this->psdReaderPath)) {
3248
            include_once($this->psdReaderPath);
3249
3250
            $psdReader = new PhpPsdReader($fileName);
3251
3252
            if (isset($psdReader->infoArray['error'])) {
3253
                return '';
3254
            } else {
3255
                return $psdReader->getImage();
3256
            }
3257
        } else {
3258
            return false;
3259
        }
3260
    }
3261
3262
    ## --------------------------------------------------------
3263
3264
    public function __destruct()
3265
    {
3266
        if (is_resource($this->imageResized)) {
3267
            imagedestroy($this->imageResized);
3268
        }
3269
    }
3270
3271
    ## --------------------------------------------------------
3272
3273
}
3274
3275
3276