PHP GD图像透视

4

你好,是否有可能转换图像的透视,使其新形状为等腰梯形? 我看到了使用Imagick的解决方案,但这可能涉及重写整个图像操作脚本...(更不用说学习了,我对此过敏)

2个回答

16
我已经改进了James发布的函数。
新增内容如下:
  • 支持垂直/水平透视
  • 透明度
  • 现在看起来更像是在3D空间中旋转物体(越近的物体越大,越远的物体越小)
以下是我的函数:
define("TOP",0);
define("BOTTOM",1);
define("LEFT",2);
define("RIGHT",3);

function perspective($i,$gradient=0.85,$rightdown=TOP,$background=0xFFFFFF, $alpha=0) {
        $w=imagesx($i);
        $h=imagesy($i);
        $col=imagecolorallocatealpha($i,($background>>16)&0xFF,($background>>8)&0xFF,$background&0xFF,$alpha);

        $mult=5;
        $li=imagecreatetruecolor($w*$mult,$h*$mult);
        imagealphablending($li,false);
        imagefilledrectangle($li,0,0,$w*$mult,$h*$mult,$col);
        imagesavealpha($li,true);

        imagecopyresized($li,$i,0,0,0,0,$w*$mult,$h*$mult,$w,$h);
        imagedestroy($i);
        $w*=$mult;
        $h*=$mult;


        $image=imagecreatetruecolor($w,$h);
        imagealphablending($image,false);
        imagefilledrectangle($image,0,0,$w,$h,$col);
        imagealphablending($image,true);

        imageantialias($image,true);
        $test=$h*$gradient;

        $rdmod=$rightdown%2;
        $min=1;
        if($rightdown<2){
            for($y=0;$y<$h;$y++){
                $ny=$rdmod? $y : $h-$y;
                $off=round((1-$gradient)*$w*($ny/$h));
                $t=((1-pow(1-pow(($ny/$h),2),0.5))*(1-$gradient)+($ny/$h)*$gradient);
                $nt=$rdmod? $t : 1-$t;
                if(abs(0.5-$nt)<$min){
                    $min=abs(0.5-$nt);
                    $naty=$off;
                }
                imagecopyresampled($image,$li,
                                    round($off/2),$y,
                                    0,abs($nt*$h),
                                    $w-$off,1,
                                    $w,1);
            }
        } else {
            for($x=0;$x<$w;$x++){
                $nx=$rdmod? $x : $w-$x;
                $off=round((1-$gradient)*$h*($nx/$w));
                $t=((1-pow(1-pow(($nx/$w),2),0.5))*(1-$gradient)+($nx/$w)*$gradient);
                $nt=$rdmod? $t : 1-$t;
                if(abs(0.5-$nt)<$min){
                    $min=abs(0.5-$nt);
                    $natx=$off;
                }
                imagecopyresampled($image,$li,
                                    $x,round($off/2),
                                    abs($nt*$w),0,
                                    1,$h-$off,
                                    1,$h);
            }
        }
        imagedestroy($li);

        imageantialias($image,false);
        imagealphablending($image,false);
        imagesavealpha($image,true);

        $i=imagecreatetruecolor(($w+$naty)/$mult,($h+$natx)/$mult);
        imagealphablending($i,false);
        imagefilledrectangle($i,0,0,($w+$naty)/$mult,($h+$natx)/$mult,$col);
        imagealphablending($i,true);
        imageantialias($i,true);
        imagecopyresampled($i,$image,0,0,0,0,($w+$naty)/$mult,($h+$natx)/$mult,$w,$h);
        imagedestroy($image);
        imagealphablending($i,false);
        imageantialias($i,false);
        imagesavealpha($i,true);
        return $i;
    }

例子:

Original image: http://imgur.com/iX5Aski

使用函数:

$img=perspective($img,0.5,TOP,0xFFFFFF,127);

结果:

$img=perspective($img,0.2,RIGHT,0xFFFFFF,127);

Result:


这太棒了,感谢@Daniel。 - gkd
@Daniel 如果您能添加一些注释,那就太好了。 - Alauddin Ahmed

4

链接已失效,但这是最近的缓存版本: https://web.archive.org/web/20130424232913/http://valokuva.org/?p=112 - Andy Gee

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接