有没有办法使用 PHP-GD 对图像进行 FishEye(或桶形变换)效果?我用一些代码找到了这个,但我很难将它移植到 PHP。
2 回答
带有GD的PHP不能以可接受的方式做这样的事情,逐像素处理图像会非常慢......
Imagick 确实支持使您能够编写自己的表达式 ( fximage ) 的函数,之后所有内容都将在 Imagick 内部处理。
因此,我找到了一种方法来满足您在 Imagick 中的要求,我采用了“Scott builds Software”博客中的表达方式 - imagick 中的鱼眼效果。您可以在他的博客中阅读该表达式的完整解释。此函数的更多文档可在ImageMagick官方网站上找到,您可以在那里了解如何构建自己的表达式。
请注意,关于返回值的 PHP 文档不正确,我也在那里发表了评论。该函数返回实际的 Imagick 对象。
所以这是你的代码:
<?php
/* Create new object */
$im = new Imagick();
/* Create new checkerboard pattern */
$im->newPseudoImage(100, 100, "pattern:checkerboard");
/* Set the image format to png */
$im->setImageFormat('png');
/* Fill background area with transparent */
$trans = Imagick::VIRTUALPIXELMETHOD_TRANSPARENT;
$im->setImageVirtualPixelMethod($trans);
/* Activate matte */
$im->setImageMatte(true);
/* This is the expression that define how to do the fisheye effect */
$distort_expression =
'kk=w*0.5;
ll=h*0.5;
dx=(i-kk);
dy=(j-ll);
aa=atan2(dy,dx);
rr=hypot(dy,dx);
rs=rr*rr/hypot(kk,ll);
px=kk+rs*cos(aa);
py=ll+rs*sin(aa);
p{px,py}';
/* Perform the distortion */
$im = $im->fxImage($distort_expression);
/* Ouput the image */
header("Content-Type: image/png");
echo $im;
?>
无论如何,请记住这仍然很慢,无论你做什么都要小心......
但是 -使用 GD 和快速是可能的!与 ImageMagick 相比,
创建一个大小为(2*SourceWidth)/PI的新图像。
遍历新图像的每个像素并找到与中心的距离。
d source =hypot(x-centerX,y-centerY)通过d dest
在源图像中找到对应的距离。=2*r*asin(d source /r)/2 r是目标图像的半宽。
查看带有基准的示例:http: //meindesign.net/picture2bubble/picture2bubble.php
function fisheye($infilename,$outfilename){
$im=imagecreatefrompng($infilename);
$ux=imagesx($im);//Source imgage width(x)
$uy=imagesy($im);//Source imgage height(y)
$umx=$ux/2;//Source middle
$umy=$uy/2;
if($ux>$uy)$ow=2*$uy/pi();//Width for the destionation image
else $ow=2*$ux/pi();
$out=imagecreatetruecolor($ow+1,$ow+1);
$trans=imagecolortransparent($out,ImageColorAllocate($out,0,0,0));
imagefill($im,1,1,$trans);
for($c=0;$c<imagecolorstotal($im);$c++){//Copy palette
$col=imagecolorsforindex($im,$c);
imagecolorset($out,$c,$col[red],$col[green],$col[blue]);
}
$om=$ow/2;//destination middle
for($x=0;$x<=$ow;++$x){//Loop X in destination image
for($y=0;$y<=$ow;++$y){//Loop y in destination image
$otx=$x-$om;//X in relation to the middle
$oty=$y-$om;//Y in relation to the middle
$oh=hypot($otx,$oty);//distance
$arc=(2*$om*asin($oh/$om))/(2);
$factor=$arc/$oh;
if($oh<=$om){//if pixle inside radius
$color=imagecolorat($im,round($otx*$factor+$umx),round($oty*$factor+$umy));
$r = ($color >> 16) & 0xFF;
$g = ($color >> 8) & 0xFF;
$b = $color & 0xFF;
$temp=imagecolorexact($out, $r, $g, $b);
imagesetpixel($out,$x,$y,$temp);
}
}
}
imagepng($out,$outfilename);
}