5

我有一组图像。在此图像中存在绿色 alpha 颜色。我需要将透明颜色替换为另一种颜色(我想替换为白色 alpha 颜色)。

它是我的代码:

$img = imagecreatefrompng($path . $file); 
$white_color_transparent = imagecolorallocatealpha($img, 255, 255, 255, 127);
for ($y = 0; $y < imagesy($img); $y++) {
    for ($x = 0; $x < imagesx($img); $x++) {
        $rgb = imagecolorat($img, $x, $y);
        $pixel_color = imagecolorsforindex($img, $rgb);
        if ($pixel_color['alpha'] != 0 && $pixel_color['alpha'] != 127){
            imagesetpixel($img, $x, $y, $white_color_transparent);
        }
    }
}

imagealphablending($img, false);
imagesavealpha($img, true);            
imagepng($img, $path .  $file);

当我之前添加时,我的结果颜色与 id 源图像相同

imagesetpixel($img, $x, $y, $white_color_transparent);

那行:

imagesetpixel($img, $x, $y, $white_color);

我只得到没有透明度的白色。

4

2 回答 2

18

没有看到您的图像,我不确定是否有一种简单的方法可以做到这一点。

编辑2:

我从第一次编辑中注意到的一件事是,它不适用于具有透明度且后面没有不透明层的图像。

这是我尝试处理具有一定透明度的图像。正如您在示例中看到的那样。结果并不完美。

<?php
//These colors here are the rgba values of the green transparency I used in my test image. You will need to know this to reverse the rgb blending.
$red = 28;
$green = 198;
$blue = 72;
$alpha = .48;

$img = imagecreatefrompng('test.png');

//Adding in the image blending that Zombaya brought up. Need this so it overwrites the pixel instead of blending it
imagealphablending($img, false);
for ($y = 0; $y < imagesy($img); $y++) {
    for ($x = 0; $x < imagesx($img); $x++) {
        $rgb = imagecolorat($img, $x, $y);
        $pixel_color = imagecolorsforindex($img, $rgb);

        //If pixel color matches our alpha layer color, we change it white transparent
        //Not sure how accurate the rounding for the alpha conversion is. Maybe the reason for the green edges in this example:
        if($pixel_color['red'] == $red && 
            $pixel_color['green'] == $green &&
            $pixel_color['blue'] == $blue &&
            $pixel_color['alpha'] == round(127 - ($alpha * 127))){
            $color = imagecolorallocatealpha($img, 255, 255, 255, 127);
        } else {
            //This is the algorithm from the link. But reordered to get the old color before the transparent color went over it.
            $oldR = ($pixel_color['red'] - $alpha * $red) /  (1 - $alpha);
            $oldG = ($pixel_color['green'] - $alpha * $green ) / (1 - $alpha);
            $oldB = ($pixel_color['blue'] - $alpha * $blue) / (1 - $alpha);

            $color = imagecolorallocatealpha($img, $oldR, $oldG, $oldB, $pixel_color['alpha']);
        }
        imagesetpixel($img, $x, $y, $color);
    }
}
imagesavealpha($img, true);            
imagepng($img, 'test_result.png');

在此处输入图像描述-> 没有透明度的原始

在此处输入图像描述-> 具有透明度的起始图像

在此处输入图像描述-> 生成的图像。注意绿色边缘。我可能很幸运对红色和绿色使用了类似的不透明度。白色部分是透明的

编辑:

我想了更多,我发现了这个链接:如何通过指定 alpha 混合量来计算 RGB 颜色?

这有助于我了解如何解决您的问题。尽管这取决于您是否只有在图像上透明的纯色。如果是渐变,那就更难了。

<?php
//These colors here are the rgba values of the green transparency I used in my test image. You will need to know this to reverse the rgb blending.
$red = 44;
$green = 215;
$blue = 56;
$alpha = .45;

$img = imagecreatefrompng('test2.png');

for ($y = 0; $y < imagesy($img); $y++) {
    for ($x = 0; $x < imagesx($img); $x++) {
        $rgb = imagecolorat($img, $x, $y);
        $pixel_color = imagecolorsforindex($img, $rgb);
            //This is the algorithm from the link. But reordered to get the old color before the transparent color went over it.
        $oldR = ($pixel_color['red'] - $alpha * $red) /  (1 - $alpha);
        $oldG = ($pixel_color['green'] - $alpha * $green ) / (1 - $alpha);
        $oldB = ($pixel_color['blue'] - $alpha * $blue) / (1 - $alpha);

        $color = imagecolorallocate($img, $oldR, $oldG, $oldB);
        imagesetpixel($img, $x, $y, $color);
    }
}
imagesavealpha($img, true);            
imagepng($img, 'test_result.png');

在此处输入图像描述- 起始图像

在此处输入图像描述- 后像

这是上面代码中使用的 rgb 计算的解释。该链接显示了基本公式:

out = alpha * new + (1 - alpha) * old

out是将原始颜色与新颜色混合后的结果颜色。

所以我们需要重新排列公式以获得旧颜色,从而产生:

old = (out - alpha * new) / (1 - alpha)

旧答案:

为什么这条线不起作用:

$white_color_transparent = imagecolorallocatealpha($img, 255, 255, 255, 127);

127 表示完全透明,也就是说它是不可见的。这就是为什么结果图像与起始图像相同的原因。

为什么这条线会创建一个全白的图像:

$white_color = imagecolorallocatealpha($img, 255, 255, 255, 0); 

或者

$white_color = imagecolorallocate($img, 255, 255, 255);

这基本上只是在所选像素上放置一个白色像素(0 表示没有透明度),这就是所选像素变为全白的原因。

现在,如果您执行以下操作:

$white_color_semi_transparent = imagecolorallocatealpha($img, 255, 255, 255, 40); 

与其用透明的白色代替透明的绿色,不如用更浅的绿色代替半透明的白色。

请参阅我的测试示例:

起始图像- 起始图像

在此处输入图像描述- 生成颜色为 255,255,255,40 的图像

所以这告诉我们两件事:

  1. 首先是颜色基本上应用在现有像素的颜色之上,因此更浅的透明绿色而不是透明的白色。
  2. 其次,注意棕色部分不受影响。棕色部分实际上是绿色不透明层下方的不透明红色。这说明图像没有考虑像 Photoshop 这样的图层,基本上将棕色读取为棕色,而不是绿色透明色后面的红色。这是有道理的,因为 PNG 文件不附带图层信息。

我不确定你会如何解决你的问题。这将取决于你的形象。我认为可以得到你想要的颜色,但显然你的图像越复杂就越难。例如,我的两个重叠的正方形有两种颜色,而不是一张照片。

于 2012-05-25T18:18:15.293 回答
0

我使用 Gohn67 的图像作为工作宽度。

诀窍是在尝试转换图像之前关闭 alpha 混合。这样,您可以使用imagesetpixel()将像素设置为您想要的确切值。

从手册上看imagealphablending

在非混合模式下,绘图颜色将使用其 Alpha 通道信息逐字复制,替换目标像素。

这导致了以下代码:

$img = imagecreatefrompng($path.$file);
imagealphablending($img, false); // Turn off blending
$white_color_transparent = imagecolorallocatealpha($img, 255, 255, 255, 127);
for ($y = 0; $y < imagesy($img); $y++) {
    for ($x = 0; $x < imagesx($img); $x++) {
        $rgb = imagecolorat($img, $x, $y);
        $pixel_color = imagecolorsforindex($img, $rgb);
        if ($pixel_color['alpha'] != 0 && $pixel_color['alpha'] != 127){
            imagesetpixel($img, $x, $y, $white_color_transparent);
        }
    }
}
imagesavealpha($img, true);            
imagepng($img, $path.$file);

原始图像

原始图像

生成的图像

结果图像
(背景其实是透明的,但是这里看不到)

我想说的是,它现在用这种完全透明的白色代替了所有透明的颜色。如果您希望将所有彩色透明度替换为具有相同 alpha 值的透明白色,您可以执行以下操作:

imagesetpixel($img, $x, $y, imagecolorallocatealpha($img, 255, 255, 255, $pixel_color['alpha']));

为了只选择绿色透明像素,我会计算每个像素的色调并检查它是否在指定范围内。

于 2012-05-25T20:49:30.663 回答