按颜色排序

use*_*442 23 php sorting colors

我有一个很长的列表(1000+)的十六进制颜色在一般颜色类别(红色,橙色,蓝色等)中分解.当我显示每个类别中的颜色列表时,我需要按阴影顺序显示它们.即先是浅红色,最后是深红色.

算法会做什么?(谷歌搜索失败了我)

Get*_*etz 11

我知道这个问题已经过时了,但是我没有找到解决这个问题的方法,所以我稍微调查了一下,并想分享我为假设的未来google做的事情.

首先,转换为HSL是个好主意.但是当你的颜色没有"分类"时,只能通过色调或光线排序并不能完全解决问题.

给定一个看起来像这样的数组:

$colors = [
            [ 'color' => '#FDD4CD'],
            [ 'color' => '#AE3B3B'],
            [ 'color' => '#DB62A0'],
            ...
          ]
Run Code Online (Sandbox Code Playgroud)

首先,我们将所有Hex颜色转换为HSL

foreach ($colors as &$color) {
       $color['hsl'] = hexToHsl($color['color']);
}


/**
 * Convert a hexadecimal color in RGB
 * @param string $hex
 * @return array
 */
function hexToHsl($hex){
    list($r, $g, $b) = sscanf($hex, "#%02x%02x%02x");
    return rgbToHsl($r, $g, $b);
}

/**
 * Convert a RGB color in its HSL value
 * @param int $r red
 * @param int $g green
 * @param int $b blue
 * @return array
 */
function rgbToHsl($r, $g, $b)
{
    $r /= 255;
    $g /= 255;
    $b /= 255;

    $max = max($r, $g, $b);
    $min = min($r, $g, $b);

    $h = 0;
    $l = ($max + $min) / 2;
    $d = $max - $min;

    if ($d == 0) {
        $h = $s = 0; // achromatic
    } else {
        $s = $d / (1 - abs(2 * $l - 1));

        switch ($max) {
            case $r:
                $h = 60 * fmod((($g - $b) / $d), 6);
                if ($b > $g) {
                    $h += 360;
                }
                break;

            case $g:
                $h = 60 * (($b - $r) / $d + 2);
                break;

            case $b:
                $h = 60 * (($r - $g) / $d + 4);
                break;
        }
    }
    return array('h' => round($h, 2), 's' => round($s, 2), 'l' => round($l, 2));
}
Run Code Online (Sandbox Code Playgroud)

然后排序颜色

我们比较:

  • 如果它们处于相同的"间隔"(这有助于理解为什么我选择30°)它们的色调.所以我们只在两者都在[0-30],[30-60],[60-90],...时才比较色调...
  • 如果不是在相同的间隔中,则按亮度排序,然后按饱和度排序,如果两者共享相同的亮度.

所以:

usort($colors, function ($a, $b) {
    //Compare the hue when they are in the same "range"
    if(!huesAreinSameInterval($a['hsl']['h'],$b['hsl']['h'])){
       if ($a['hsl']['h'] < $b['hsl']['h'])
           return -1;
       if ($a['hsl']['h'] > $b['hsl']['h'])
           return 1;
    }
    if ($a['hsl']['l'] < $b['hsl']['l'])
        return 1;
    if ($a['hsl']['l'] > $b['hsl']['l'])
        return -1;
    if ($a['hsl']['s'] < $b['hsl']['s'])
         return -1;
    if ($a['hsl']['s'] > $b['hsl']['s'])
          return 1;
    return 0;
 });

/**
 * Check if two hues are in the same given interval
 * @param float $hue1
 * @param float $hue2
 * @param int $interval
 * @return bool
 */
function huesAreinSameInterval($hue1, $hue2, $interval = 30){
    return (round(($hue1 / $interval), 0, PHP_ROUND_HALF_DOWN) === round(($hue2 / $interval), 0, PHP_ROUND_HALF_DOWN));
}
Run Code Online (Sandbox Code Playgroud)

rgbToHsl在www.brandonheyer.com上找到

stackoverflow上找到hexToRgb


Jam*_*hek 8

将颜色从RGB转换为HSV或HSL比例,然后按值或亮度对它们进行排序.亮度可能会更好,因为它可以更好地捕捉"褪色"的颜色,例如粉红色 - >红色 - >深红色.

  • 这里有一个PHP实现(小心DaniWeb弹出窗口)我不能保证它的质量. (2认同)
  • 感谢您的链接,我发现这个链接非常有效:http://www.phpclasses.org/browse/package/4598.html现在处理排序,将报告结果.谢谢您的帮助! (2认同)
  • 好的,最后的.使用此类将Hex转换为其他版本.非常有用:http://www.phpclasses.org/browse/package/4598.html我尝试在HSV上进行排序,但这似乎不如对RGB值进行排序那么正确.然而,这完全符合我的要求.非常感谢大家的帮助! (2认同)