How to stop PHP iMagick auto-rotating images based on EXIF 'orientation' data

前端 未结 5 1393
伪装坚强ぢ
伪装坚强ぢ 2020-12-02 23:52

Currently working with PHP and iMagick to develop a poster printing Web application.

This is the example image I am using to test upload/image editing features of th

相关标签:
5条回答
  • function thumbnailImage($imagePath,$color,$quality)
    {   
    
        $fileType = pathinfo($imagePath, PATHINFO_EXTENSION);
    
        if (!empty($fileType))
        {
            switch($fileType)
            {
                case "gif":
                    $im = imagecreatefromgif($imagePath);
                    break;
    
                case "jpg":
                    $im = imagecreatefromjpeg($imagePath);
                    break;
    
                case "jpeg":
                    $im = imagecreatefromjpeg($imagePath);
                    break;
    
                case "png":
                    $im = imagecreatefrompng($imagePath);
                    break;
            }
        }
    
        $imagick = new Imagick();
        $imagick->readImage($imagePath);
        $compression_type = Imagick::COMPRESSION_JPEG; 
        $imagick->setImageCompression($compression_type);
        $imagick->setImageCompressionQuality($quality);
    
        if ($fileType == "jpeg" || $fileType == "jpeg" )
        {
            $exif = exif_read_data($imagePath);
            if (!empty($exif['Orientation'])) {
                switch ($exif['Orientation']) {
                    case 3:
                        $imagick = imagerotate($imagick, 180, 0);
                        break;
    
                    case 6:
                        $imagick = imagerotate($imagick, -90, 0);
                        break;
    
                    case 8:
                        $imagick = imagerotate($imagick, 90, 0);
                        break;
                }
            }
        }
        $imagick->setImageBackgroundColor($color);
        $imagick->thumbnailImage(150, 150, true, true);
        header("Content-type: image/jpg");
        echo $imagick->getImageBlob();
    }
    

    Super easy thumbnailer. Call with

    thumbnailImage('file.ext','#ffffff',50); // full path, color, quality 1-100
    

    Works with jpg, gif and png but obviously only does the EXIF stuff on jpeg

    The thumbnail will load on the page and you can call it by URL if you specify your $_GETs

    enjoy

    0 讨论(0)
  • 2020-12-02 23:57

    Try Imagick::setImageOrientation. Experiment with the available constants.

    0 讨论(0)
  • 2020-12-03 00:02

    This code in orrd's excellent answer requires iMagick version 6.3+.:

    $image->setImageOrientation(imagick::ORIENTATION_TOPLEFT);

    Works perfectly and takes care of os/device orientation differences. Would not work with 6.2.

    I had coded to get device. Here in case someone needs it.

    $ua = $_SERVER['HTTP_USER_AGENT'];
    $strcut = stristr($ua, '(')."<br>";
    $textlen = strpos($strcut,";");
    $deviceos = substr($strcut,1,($textlen-1));
    echo "Device O/S: * $deviceos"."<br>";
    
    0 讨论(0)
  • 2020-12-03 00:05

    Good start -- a few additions to make the function more robust. First, case 3 occurs when the image appears upside down. There's a GREAT illustration of the different orientation codes by Calvin Hass. It's possible that orientation information may appear at a different part of the exif_read_data array (depending on camera model, I think), so I've tried to take that into account in my example code.

    Something like this:

    public function fixOrientation() {
    
        $exif = exif_read_data($this->imgSrc);
    
        if( isset($exif['Orientation']) )
            $orientation = $exif['Orientation'];
        elseif( isset($exif['IFD0']['Orientation']) )
            $orientation = $exif['IFD0']['Orientation'];
        else
            return false;
    
        switch($orientation) {
            case 3: // rotate 180 degrees
                $this->image->rotateimage("#FFF", 180);
            break;
    
            case 6: // rotate 90 degrees CW
                $this->image->rotateimage("#FFF", 90);
            break;
    
            case 8: // rotate 90 degrees CCW
                $this->image->rotateimage("#FFF", -90);
            break;
        }
    }
    

    The transformation & save leaves you without the previous EXIF information, including Orientation. The lack of Orientation in the transformed image will prevent further processing from attempting to 'correct' things by rotating again. I do wish Imagick had support for ImageMagick's -auto-orient, but oh well.

    Oh, also: the rotation is a lossy operation (unless you use jpegtran), so you should try to only do it in conjunction with a resize or other transformation.

    0 讨论(0)
  • 2020-12-03 00:11

    "However - iMagick, when __construct'ed with this image, automatically rotates it an additional 90 degrees CCW as per [Orientation] => 6 (I think!)."

    The problem is actually the opposite of that. Imagick doesn't auto rotate the image. You're only seeing it correctly in other software / your web browser because those programs do auto rotate it based on the EXIF info. Certain operations in Imagick will cause you to lose that correct EXIF info (copying the image, thumbnailImage(), stripImage(), and other manipulations). So what you need to do in that case is actually physically rotate the image.

    The answer from ajmicek is good, but it could be improved a bit by using Imagick's own built in functions rather than the PHP EXIF functions. Also that snippet seems to have been a part of a class, so it can't be used as a separate function as-is. It's also a good idea to set the correct EXIF orientation with setImageOrientation() after you rotate it.

    // Note: $image is an Imagick object, not a filename! See example use below.
    function autoRotateImage($image) {
        $orientation = $image->getImageOrientation();
    
        switch($orientation) {
            case imagick::ORIENTATION_BOTTOMRIGHT: 
                $image->rotateimage("#000", 180); // rotate 180 degrees
                break;
    
            case imagick::ORIENTATION_RIGHTTOP:
                $image->rotateimage("#000", 90); // rotate 90 degrees CW
                break;
    
            case imagick::ORIENTATION_LEFTBOTTOM: 
                $image->rotateimage("#000", -90); // rotate 90 degrees CCW
                break;
        }
    
        // Now that it's auto-rotated, make sure the EXIF data is correct in case the EXIF gets saved with the image!
        $image->setImageOrientation(imagick::ORIENTATION_TOPLEFT);
    }
    

    Example use:

    $image = new Imagick('my-image-file.jpg');
    autoRotateImage($image);
    // - Do other stuff to the image here -
    $image->writeImage('result-image.jpg');
    
    0 讨论(0)
提交回复
热议问题