使用 PHP 的 GDlib imagecopyresampled 时可以保留 PNG 图像透明度吗?
-
09-06-2019 - |
题
以下 PHP 代码片段使用 GD 将浏览器上传的 PNG 大小调整为 128x128。它效果很好,除了原始图像中的透明区域被替换为纯色(在我的例子中为黑色)。
虽然 imagesavealpha
已定,有些事情不太对劲。
保持重新采样图像透明度的最佳方法是什么?
$uploadTempFile = $myField[ 'tmp_name' ]
list( $uploadWidth, $uploadHeight, $uploadType )
= getimagesize( $uploadTempFile );
$srcImage = imagecreatefrompng( $uploadTempFile );
imagesavealpha( $targetImage, true );
$targetImage = imagecreatetruecolor( 128, 128 );
imagecopyresampled( $targetImage, $srcImage,
0, 0,
0, 0,
128, 128,
$uploadWidth, $uploadHeight );
imagepng( $targetImage, 'out.png', 9 );
解决方案
imagealphablending( $targetImage, false );
imagesavealpha( $targetImage, true );
为我做的。谢谢塞贾约兹。
请注意,目标图像需要 Alpha 设置,而不是源图像。
编辑:完整的替换代码。另请参阅下面的答案及其评论。这不能保证在任何方面都是完美的,但确实达到了我当时的需求。
$uploadTempFile = $myField[ 'tmp_name' ]
list( $uploadWidth, $uploadHeight, $uploadType )
= getimagesize( $uploadTempFile );
$srcImage = imagecreatefrompng( $uploadTempFile );
$targetImage = imagecreatetruecolor( 128, 128 );
imagealphablending( $targetImage, false );
imagesavealpha( $targetImage, true );
imagecopyresampled( $targetImage, $srcImage,
0, 0,
0, 0,
128, 128,
$uploadWidth, $uploadHeight );
imagepng( $targetImage, 'out.png', 9 );
其他提示
你为什么把事情搞得这么复杂?以下是我使用的,到目前为止它已经为我完成了工作。
$im = ImageCreateFromPNG($source);
$new_im = imagecreatetruecolor($new_size[0],$new_size[1]);
imagecolortransparent($new_im, imagecolorallocate($new_im, 0, 0, 0));
imagecopyresampled($new_im,$im,0,0,0,0,$new_size[0],$new_size[1],$size[0],$size[1]);
我相信这应该可以解决问题:
$srcImage = imagecreatefrompng($uploadTempFile);
imagealphablending($srcImage, false);
imagesavealpha($srcImage, true);
编辑: PHP 文档中有人声称 imagealphablending
应该是真的,而不是假的。YMMV。
可能对某些人有帮助的补充:
构建图像时可以切换 imagealphablending。我需要这个的具体情况,我想在透明背景上组合一些半透明的PNG。
首先,将 imagealphablending 设置为 false,并用透明颜色填充新创建的真彩色图像。如果 imagealphablending 为 true,则不会发生任何事情,因为透明填充将与黑色默认背景合并并产生黑色。
然后将 imagealphablending 切换为 true 并将一些 PNG 图像添加到画布中,留下一些背景可见(即没有填满整个图像)。
结果是一个具有透明背景的图像和多个组合的 PNG 图像。
我制作了一个调整 JPEG/GIF/PNG 等图像大小的功能 copyimageresample
PNG 图像仍然保持透明度:
$myfile=$_FILES["youimage"];
function ismyimage($myfile) {
if((($myfile["type"] == "image/gif") || ($myfile["type"] == "image/jpg") || ($myfile["type"] == "image/jpeg") || ($myfile["type"] == "image/png")) && ($myfile["size"] <= 2097152 /*2mb*/) ) return true;
else return false;
}
function upload_file($myfile) {
if(ismyimage($myfile)) {
$information=getimagesize($myfile["tmp_name"]);
$mywidth=$information[0];
$myheight=$information[1];
$newwidth=$mywidth;
$newheight=$myheight;
while(($newwidth > 600) || ($newheight > 400 )) {
$newwidth = $newwidth-ceil($newwidth/100);
$newheight = $newheight-ceil($newheight/100);
}
$files=$myfile["name"];
if($myfile["type"] == "image/gif") {
$tmp=imagecreatetruecolor($newwidth,$newheight);
$src=imagecreatefromgif($myfile["tmp_name"]);
imagecopyresampled($tmp, $src, 0, 0, 0, 0, $newwidth, $newheight, $mywidth, $myheight);
$con=imagegif($tmp, $files);
imagedestroy($tmp);
imagedestroy($src);
if($con){
return true;
} else {
return false;
}
} else if(($myfile["type"] == "image/jpg") || ($myfile["type"] == "image/jpeg") ) {
$tmp=imagecreatetruecolor($newwidth,$newheight);
$src=imagecreatefromjpeg($myfile["tmp_name"]);
imagecopyresampled($tmp, $src, 0, 0, 0, 0, $newwidth, $newheight, $mywidth, $myheight);
$con=imagejpeg($tmp, $files);
imagedestroy($tmp);
imagedestroy($src);
if($con) {
return true;
} else {
return false;
}
} else if($myfile["type"] == "image/png") {
$tmp=imagecreatetruecolor($newwidth,$newheight);
$src=imagecreatefrompng($myfile["tmp_name"]);
imagealphablending($tmp, false);
imagesavealpha($tmp,true);
$transparent = imagecolorallocatealpha($tmp, 255, 255, 255, 127);
imagefilledrectangle($tmp, 0, 0, $newwidth, $newheight, $transparent);
imagecopyresampled($tmp, $src, 0, 0, 0, 0, $newwidth, $newheight, $mywidth, $myheight);
$con=imagepng($tmp, $files);
imagedestroy($tmp);
imagedestroy($src);
if($con) {
return true;
} else {
return false;
}
}
} else
return false;
}
我想这可能会起作用:
$uploadTempFile = $myField[ 'tmp_name' ]
list( $uploadWidth, $uploadHeight, $uploadType )
= getimagesize( $uploadTempFile );
$srcImage = imagecreatefrompng( $uploadTempFile );
$targetImage = imagecreatetruecolor( 128, 128 );
$transparent = imagecolorallocate($targetImage,0,255,0);
imagecolortransparent($targetImage,$transparent);
imagefilledrectangle($targetImage,0,0,127,127,$transparent);
imagecopyresampled( $targetImage, $srcImage,
0, 0,
0, 0,
128, 128,
$uploadWidth, $uploadHeight );
imagepng( $targetImage, 'out.png', 9 );
缺点是图像中每 100% 的绿色像素都会被剥离。无论如何,希望它有帮助:)
重新分级保留透明度,然后是的,就像其他帖子中所述,imagesavealpha() 必须设置为 true,要使用 alpha 标志 imagealphablending() 必须设置为 false,否则它不起作用。
我还在你的代码中发现了两件小事:
- 你不需要打电话
getimagesize()
获取宽度/高度imagecopyresmapled()
- 这
$uploadWidth
和$uploadHeight
应该-1
值,因为坐标开始于0
并不是1
, ,所以它会将它们复制到一个空像素中。将其替换为:imagesx($targetImage) - 1
和imagesy($targetImage) - 1
, ,相对应该做:)
这是我的总测试代码。这个对我有用
$imageFileType = pathinfo($_FILES["image"]["name"], PATHINFO_EXTENSION);
$filename = 'test.' . $imageFileType;
move_uploaded_file($_FILES["image"]["tmp_name"], $filename);
$source_image = imagecreatefromjpeg($filename);
$source_imagex = imagesx($source_image);
$source_imagey = imagesy($source_image);
$dest_imagex = 400;
$dest_imagey = 600;
$dest_image = imagecreatetruecolor($dest_imagex, $dest_imagey);
imagecopyresampled($dest_image, $source_image, 0, 0, 0, 0, $dest_imagex, $dest_imagey, $source_imagex, $source_imagey);
imagesavealpha($dest_image, true);
$trans_colour = imagecolorallocatealpha($dest_image, 0, 0, 0, 127);
imagefill($dest_image, 0, 0, $trans_colour);
imagepng($dest_image,"test1.png",1);
注意源图像的 width
和 height
传递给的值 imagecopyresampled
功能。如果它们大于实际的源图像尺寸,则图像区域的其余部分将填充黑色。
我结合了 ceejayoz 和 Cheekysoft 的答案,这给了我最好的结果。没有 imagealphablending() 和 imagesavealpha() 图像不清晰:
$img3 = imagecreatetruecolor(128, 128);
imagecolortransparent($img3, imagecolorallocate($img3, 0, 0, 0));
imagealphablending( $img3, false );
imagesavealpha( $img3, true );
imagecopyresampled($img3, $srcImage, 0, 0, 0, 0, 128, 128, $uploadWidth, $uploadHeight);
imagepng($img3, 'filename.png', 9);