9

使用 javascript/jquery,我想将 rgba 值数组排序为可见光谱的颜色。通过这样做,类似的阴影应该聚集在一起。有没有插件可以做到这一点,或者我将如何去做?

光谱图像:http ://www.gamonline.com/catalog/colortheory/images/spectrum.gif

4

2 回答 2

14

披露:我是下面推荐的库的作者。

如果您不介意使用库,这里是 Oriol 详细回复的更简洁版本。它使用sc-color 库

var sorted = colorArray.sort(function(colorA, colorB) {
    return sc_color(colorA).hue() - sc_color(colorB).hue();
});
于 2012-08-13T14:16:22.423 回答
11

如果您的颜色数组是这样的:

var rgbArr = [c1, c2, c3, ...]

其中每种颜色ci都是 0 到 255 之间的三个数字的数组

ci = [red, green, blue]

然后,您可以使用此函数将颜色转换为 HSL

function rgbToHsl(c) {
  var r = c[0]/255, g = c[1]/255, b = c[2]/255;
  var max = Math.max(r, g, b), min = Math.min(r, g, b);
  var h, s, l = (max + min) / 2;

  if(max == min) {
    h = s = 0; // achromatic
  } else {
    var d = max - min;
    s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
    switch(max){
      case r: h = (g - b) / d + (g < b ? 6 : 0); break;
      case g: h = (b - r) / d + 2; break;
      case b: h = (r - g) / d + 4; break;
    }
    h /= 6;
  }
  return new Array(h * 360, s * 100, l * 100);
}

并按色调排序

var sortedRgbArr = rgbArr.map(function(c, i) {
  // Convert to HSL and keep track of original indices
  return {color: rgbToHsl(c), index: i};
}).sort(function(c1, c2) {
  // Sort by hue
  return c1.color[0] - c2.color[0];
}).map(function(data) {
  // Retrieve original RGB color
  return rgbArr[data.index];
});

这是一个可运行的示例(感谢Ionică Bizău):

function display(container, arr) {
  container = document.querySelector(container);
  arr.forEach(function(c) {
    var el = document.createElement("div");
    el.style.backgroundColor = "rgb(" + c.join(", ") + ")";
    container.appendChild(el);
  })
}
function rgbToHsl(c) {
  var r = c[0] / 255,
      g = c[1] / 255,
      b = c[2] / 255;
  var max = Math.max(r, g, b),
      min = Math.min(r, g, b);
  var h, s, l = (max + min) / 2;

  if (max == min) {
    h = s = 0; // achromatic
  } else {
    var d = max - min;
    s = l > 0.5 ? d / (2 - max - min) : d / (max + min);
    switch (max) {
      case r:
        h = (g - b) / d + (g < b ? 6 : 0);
        break;
      case g:
        h = (b - r) / d + 2;
        break;
      case b:
        h = (r - g) / d + 4;
        break;
    }
    h /= 6;
  }
  return new Array(h * 360, s * 100, l * 100);
}

var rgbArr = [];
for (var i = 0; i < 100; ++i) {
  rgbArr.push([
    Math.floor(Math.random() * 256),
    Math.floor(Math.random() * 256),
    Math.floor(Math.random() * 256)
  ]);
}
display("#before", rgbArr);

var sortedRgbArr = rgbArr.map(function(c, i) {
  // Convert to HSL and keep track of original indices
  return {color: rgbToHsl(c), index: i};
}).sort(function(c1, c2) {
  // Sort by hue
  return c1.color[0] - c2.color[0];
}).map(function(data) {
  // Retrieve original RGB color
  return rgbArr[data.index];
});
display("#after", sortedRgbArr);
#before > div,
#after > div {
  width: 1%;
  height: 20px;
  display: inline-block;
}
Random colors: <div id="before"></div>
Same colors, sorted by hue: <div id="after"></div>

sortedRgbArr将包含rgbArr或多或少像可见光谱的颜色排序的 rgb 颜色。

问题是 HSL 频谱如下所示:

HSL光谱

您的光谱很奇怪,因为它没有所有颜色,例如粉红色。

我想那是因为自然界中不存在粉红色,它是光谱中相反极端颜色的组合。但是我们有 rgb 格式,所以你必须决定你想要它在哪里。

此外,您的光谱似乎是从较低的波长到较高的波长,而不是频率。但是,您的频谱与 HSL 的频谱相反。

如果您希望它像您的光谱一样,请替换c1.color[0] - c2.color[0]为。c2.color[0] - c1.color[0]

于 2012-08-12T16:51:21.977 回答