5

PHP GD中,如何在不丢失任何颜色的情况下将真彩色图像转换为调色板。用它不起作用。我有一个函数可以遍历所有颜色并过滤它们(例如灰度)。而且它不会保留所有颜色,比如这张兰博基尼的照片-imagetruecolortopallete

图片

替代文字

这是我的代码

$im = imagecreatefrompng("lamborghini.png");
$degrees = 0;

$img = imagecreatetruecolor(imagesx($im), imagesy($im));
imagecopy($img, $im, 0, 0, 0, 0, imagesx($im), imagesy($im));
imagetruecolortopalette($img, true, 256);
$t = imagecolorstotal($img);
for ($i = 0; $i < $t; $i++) {
  $rgb = imagecolorsforindex($img, $i);
  $hsv =rgbtohsv($rgb['red'], $rgb['green'], $rgb['blue']);
  $h = $degrees;
  $s = $hsv['s'];
  $v = $hsv['v'];
  while ($h > 360) {$h -= 360;};
        $nrgb = hsvtorgb($h, $s, $v);
  imagecolorset($img, $i, $nrgb['r'], $nrgb['g'], $nrgb['b']);
}
imagecopy($im, $img, 0, 0, 0, 0, imagesx($img), imagesy($img));

header('Content-type: image/png');

imagepng($im);
imagedestroy($im);

看起来像这样

替代文字

你可以看到它失去了颜色。有什么解决办法吗?

另外我认为这与我的代码无关,而是如何imagetruecolortopalette输出它

4

3 回答 3

5

检查返回imagecolorstotal,无论您将要抖动的颜色数量设置多高,您总是得到 256 种颜色作为返回。PNG-8 和 GIF 格式仅支持多达 256 种颜色的调色板。因此,即使您可以在调色板中使用超过 256 种颜色,您也必须将其保存为真彩色以供任何人使用,从而使整个转换过程浪费时间。换句话说imagetruecolortopallete,上限是 256 色,你不能再高了。

以下是如何以真彩色实现它,尽管它需要大量资源。如果您想更有效地执行此操作,也许可以查看 imagemagick。

$im = imagecreatefrompng("lamb.png");
$img = imagecreatetruecolor(imagesx($im), imagesy($im));
$degrees = 0;
if ($degrees > 360) {$degrees = $degrees % 360 ;}

foreach (range(0, imagesx($im) - 1) as $x ) {
    foreach (range(0, imagesy($im) - 1) as $y) {
        $rgb = imagecolorat($im, $x, $y);
        $r = ($rgb >> 16) & 0xFF;
        $g = ($rgb >> 8) & 0xFF;
        $b = $rgb & 0xFF;
        $hsv = rgbtohsv($r,$g,$b);
        $rgb = hsvtorgb($degrees, $hsv['s'], $hsv['v']);
        imagesetpixel($img, $x,$y,imagecolorallocate($img, $rgb['r'], $rgb['g'], $rgb['b']));
    }
}

imagepng($img, 'lamb2.png');

编辑:添加 rgbtohsv & hsvtorgb 功能。这些函数不是我写的。

function rgbtohsv ($R, $G, $B) {
 // HSV Results:Number 0-1
$HSL = array();

$var_R = ($R / 255);
$var_G = ($G / 255);
$var_B = ($B / 255);

$var_Min = min($var_R, $var_G, $var_B);
$var_Max = max($var_R, $var_G, $var_B);
$del_Max = $var_Max - $var_Min;

$V = $var_Max;

if ($del_Max == 0)
{
$H = 0;
$S = 0;
}
else
{
$S = $del_Max / $var_Max;

$del_R = ( ( ( $var_Max  - $var_R ) / 6 ) + ( $del_Max / 2 ) ) / $del_Max;
$del_G = ( ( ( $var_Max  - $var_G ) / 6 ) + ( $del_Max / 2 ) ) / $del_Max;
$del_B = ( ( ( $var_Max  - $var_B ) / 6 ) + ( $del_Max / 2 ) ) / $del_Max;

if ($var_R == $var_Max) $H = $del_B - $del_G;
else if ($var_G == $var_Max) $H = ( 1 / 3 ) + $del_R - $del_B;
else if ($var_B == $var_Max) $H = ( 2 / 3 ) + $del_G - $del_R;

if ($H<0) $H++;
if ($H>1) $H--;
}

$HSL['h'] = $H;
$HSL['s'] = $S;
$HSL['v'] = $V;

return $HSL;
}

function hsvtorgb ($H, $S, $V) 
{
$RGB = array();

if($S == 0)
{
$R = $G = $B = $V * 255;
}
else
{
$var_H = $H * 6;
$var_i = floor( $var_H );
$var_1 = $V * ( 1 - $S );
$var_2 = $V * ( 1 - $S * ( $var_H - $var_i ) );
$var_3 = $V * ( 1 - $S * (1 - ( $var_H - $var_i ) ) );

if ($var_i == 0) { $var_R = $V ; $var_G = $var_3 ; $var_B = $var_1 ; }
else if ($var_i == 1) { $var_R = $var_2 ; $var_G = $V ; $var_B = $var_1 ; }
else if ($var_i == 2) { $var_R = $var_1 ; $var_G = $V ; $var_B = $var_3 ; }
else if ($var_i == 3) { $var_R = $var_1 ; $var_G = $var_2 ; $var_B = $V ; }
else if ($var_i == 4) { $var_R = $var_3 ; $var_G = $var_1 ; $var_B = $V ; }
else { $var_R = $V ; $var_G = $var_1 ; $var_B = $var_2 ; }

$R = $var_R * 255;
$G = $var_G * 255;
$B = $var_B * 255;
}

$RGB['r'] = $R;
$RGB['g'] = $G;
$RGB['b'] = $B;

return $RGB;
}
于 2010-09-17T03:43:08.187 回答
2

将真彩色转换为调色板总是会涉及丢失颜色,除非您的调色板表足够大以容纳图像中的每种独特颜色。兰博基尼很可能有超过(比如说)255 种黄色,然后你仍然需要调色板槽来放置图像的其余部分。这就是为什么会有真彩色图像。没有调色板,但每个像素都可以有自己的专用 RGB 三元组,以更准确地表示原始场景。

话虽这么说,我看不出转换真彩色图片会如何将黄色变为红色,就像在您的示例图像中一样。您究竟是如何进行此调色板转换的?对每个像素进行采样并对其进行某种转换?

于 2010-09-16T23:01:39.463 回答
0

这个功能怎么样:http ://www.php.net/manual/en/function.imagetruecolortopalette.php 。不过,可能需要弄乱颜色的数量才能使其正确。

于 2010-09-16T22:57:29.880 回答