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