繁体   English   中英

使用PHP GD Library更改图像的“色相”?

[英]Change “HUE” of an image with PHP GD Library?

我想使用PHP图像处理功能更改图像的“色相”。 哪个是正确的过滤器?

注意:我来自Photoshop背景,因此,如果我对“色相”的解释与其他人不同,这就是我的意思。

在photoshop中,可以使用“色相”滤镜更改图像的颜色,而不会实际影响图像的设计。

我目前正在使用下面的功能为图像重新着色,但是由于该功能完全重新绘制了图像而丢失了设计,因此该功能无法满足我的需求。

    function set_theme_color_header($hex)
    {
        $info = hexToRGB($hex); //utility function that converts hex to rgb
        $token = "header.gif";
        $img = imagecreatefromgif("header-template.gif";
        $color = imagecolorallocate($img, $info["red"], $info["green"], $info["blue"]);
        imagecolorset($img, 0, $info["red"], $info["green"], $info["blue"]);
        imagegif($img, $token);
    }
    ?>

我很确定PHP中没有可以处理色调变化的函数。 但是您可以编写自己的函数来执行此操作,以下是更改图像色调的步骤:

  1. 逐像素遍历图像
  2. 使用imagecolorat()获取像素处的RGB颜色
  3. 将RGB值转换为HSL值
  4. 更改色相值,不设置饱和度和亮度(色相值为0到360)
  5. 将新的HSL值转换回RGB
  6. 更改当前像素的颜色

实际上,尝试一下似乎很有趣,因此,如果您找不到其他解决方案,我可以自己做,然后张贴在这里。

编辑

编写功能。 要在PHP中更改色相,您将需要在RGB和HSL颜色空间之间进行转换的函数以及进行图像遍历的函数。 这可能有点难看,但效果很好。 在较大的图像上相当慢。

function imagehue(&$image, $angle) {
    if($angle % 360 == 0) return;
    $width = imagesx($image);
    $height = imagesy($image);

    for($x = 0; $x < $width; $x++) {
        for($y = 0; $y < $height; $y++) {
            $rgb = imagecolorat($image, $x, $y);
            $r = ($rgb >> 16) & 0xFF;
            $g = ($rgb >> 8) & 0xFF;
            $b = $rgb & 0xFF;            
            $alpha = ($rgb & 0x7F000000) >> 24;
            list($h, $s, $l) = rgb2hsl($r, $g, $b);
            $h += $angle / 360;
            if($h > 1) $h--;
            list($r, $g, $b) = hsl2rgb($h, $s, $l);            
            imagesetpixel($image, $x, $y, imagecolorallocatealpha($image, $r, $g, $b, $alpha));
        }
    }
}

以下是转换色彩空间所需的帮助程序功能,该功能可以从http://www.actionscript.org/forums/showthread.php3?t=50746中无耻地复制,并进行一些较小的改动:

function rgb2hsl($r, $g, $b) {
   $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--;
   }

   return array($h, $s, $v);
}

function hsl2rgb($h, $s, $v) {
    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;
    }    
    return array($r, $g, $B);
}

最后是一个用法示例。 本示例打开一个图像,将其色相移动180°并输出到浏览器:

header('Content-type: image/png');
$image = imagecreatefrompng('image.png');
imagehue($image, 180);
imagepng($image);

由于色相角度以度为单位设置,因此将0、360、720或360的任意倍数设置都不会改变图像。

您通常会从RGB转换为HSL色彩空间,此时您可以直接操纵色调。 完成所需的操作后,即可转换回RGB。

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM