Skip to content

Instantly share code, notes, and snippets.

@danielwestendorf
Last active December 10, 2015 14:19
Show Gist options
  • Save danielwestendorf/4446741 to your computer and use it in GitHub Desktop.
Save danielwestendorf/4446741 to your computer and use it in GitHub Desktop.
Obtain the dominant colors in an image that are drastically different than each other using client side HTML and JS. A la iTunes 11 Albums view.
<!DOCTYPE html>
<html>
<head>
<title>Dominant Color</title>
<style type="text/css">
html, body {
font-family: Gotham, Helvetica, Arial, sans-serif;
width: 100%;
height: 100%;
background-image: url();
color: #eee;
font-size: 22px;
}
#source a {color: #eee;}
#browser-not-supported, #source {
color: #eee;
text-align: center;
}
#drop-zone {
width: 90%;
margin: 40px auto;
/*padding: 10% 0;*/
display: none;
position: relative;
z-index: 0;
border: 2px dashed #eee;
}
#image-view {
margin-top: 20px;
width: 90%;
margin: auto;
}
#results .wrapper {text-align: center; border: 2px solid black; margin-bottom: 20px; font-size: 16px; padding: 20px 0; }
#drop-zone p {
position: relative;
text-align: center;
z-index: 0;
padding: 10% 0;
}
#drop-zone input {
position: absolute;
z-index: 1;
top: 0;
text-align: center;
width: 100%;
height: 100%;
opacity: 0;
}
</style>
</head>
<body>
<div id="drop-zone">
<p>Click or Drop image here</p>
<input type="file" id="file-input" multiple=true>
</div>
<div id="browser-not-supported">
<h2>Oh No!</h2>
<p>Your browser doesn't support this fancy new stuff. Try upgrading to a modern browser and trying again.</p>
</div>
<div id="image-view">
<div id='results'>
</div>
</div>
<div id="source"><a href="https://gist.github.com/4446741">View Source</a></div>
<script type="text/javascript">
function compareColor(a, b) {
if (a.count > b.count) {
return -1;
} else {
return 1;
}
};
//Define pixel
//Pixel represents a single pixel from an image
function Pixel(r, g, b) {
this.r = parseInt(r);
this.g = parseInt(g);
this.b = parseInt(b);
this.count = 1; //If created, there is at least one instance
};
//css format RGB
Pixel.prototype.cssRGB = function() {
return 'rgb('+this.r+', '+this.g+', '+this.b+')';
};
//convert to a ColorLChab object
Pixel.prototype.toLChab = function() {
return new ColorRGB(this.r / 255, this.g / 255, this.b / 255).toLinearRGB().toXYZ().toLab().toLChab()
};
//Define SelectedImage
function SelectedImage(file) {
this.file = file;
this.imageData;
this.image;
this.primaryColor;
this.secondaryColor;
this.delta;
};
//read the file
SelectedImage.prototype.readFile = function() {
var reader = new FileReader();
reader.onload = (function(selectedImage) {
return function(e) {
selectedImage.imageData = e.target.result;
selectedImage.findDominantColors();
};
})(this);
reader.readAsDataURL(this.file);
};
//find the dominant colors
SelectedImage.prototype.findDominantColors = function() {
var image = new Image();
image.src = this.imageData;
image.style.maxWidth = "100%";
image.onload = (function(selectedImage) {
//we need to wait for the image to load to get the dimensions
return function(e) {
selectedImage.image = e.target
var canvasObject = document.createElement('canvas');
canvasObject.width = selectedImage.image.width;
canvasObject.height = selectedImage.image.height;
var ctx = canvasObject.getContext('2d');
ctx.drawImage(selectedImage.image, 0, 0);
var imagePixels = ctx.getImageData(0,0, selectedImage.image.width, selectedImage.image.height);
var dominatColorTable = new Object();
for(var i = 0; i < imagePixels.data.length; i+=4) {
if (imagePixels.data[i+3] != 0) {
//the pixel isn't transparent
var pixel = new Pixel(imagePixels.data[i], imagePixels.data[i+1], imagePixels.data[i+2]);
var pixelString = ''+imagePixels.data[i]+','+imagePixels.data[i+1]+','+imagePixels.data[i+2];
if (dominatColorTable[pixelString]) {dominatColorTable[pixelString].count++} else {dominatColorTable[pixelString] = pixel};
} else {
//pixel is transparent
};
};
colorTableArray = new Array();
for(color in dominatColorTable) {colorTableArray.push(dominatColorTable[color])};
colorTableArray.sort(compareColor);
//Need to group colors
var primaryColor = colorTableArray[0];
var secondaryColor;
var secondaryLChab;
var delta;
var primaryLChab = primaryColor.toLChab();
//Find the next primary color that is substantially different parseInt()
for (var i = 1; i < colorTableArray.length; i++) {
secondaryColor = colorTableArray[i];
secondaryLChab = secondaryColor.toLChab();
delta = deltaE2000(primaryLChab, secondaryLChab).toFixed(2);
if (delta > 17) break; //This determines how big the difference in colors needs to be (lower is more similar)
};
selectedImage.primaryColor = primaryColor;
selectedImage.secondaryColor = secondaryColor;
selectedImage.delta = delta;
showDominantColor(selectedImage);
}
})(this);
};
function showDominantColor(selectedImage) {
var wrapper = document.createElement('div');
wrapper.className = "wrapper";
wrapper.appendChild(selectedImage.image);
var p = document.createElement('p');
// console.log(selectedImage);
p.innerHTML = 'Filename: ' + selectedImage.file.name + '<br>Filesize: ' + selectedImage.file.size + '<br>Type: ' + selectedImage.file.type + '<br>Primary Color: ' + selectedImage.primaryColor.cssRGB() + '<br>Secondary Color: ' + selectedImage.secondaryColor.cssRGB() + '<br>Distance Between Colors: ' + selectedImage.delta;
p.style.color = selectedImage.primaryColor.cssRGB();
wrapper.style.backgroundColor = selectedImage.secondaryColor.cssRGB();
wrapper.style.borderColor = selectedImage.primaryColor.cssRGB();
wrapper.appendChild(p);
document.getElementById('results').appendChild(wrapper);
}
function handleDragOver(event) {
event.stopPropagation();
event.preventDefault();
event.dataTransfer.dropEffect = 'copy';
};
function handleFileSelect(event) {
event.stopPropagation();
event.preventDefault();
var files;
if (event.type == "drop") {
//The file was dropped
console.log("File was dropped");
files = event.dataTransfer.files;
} else if (event.type == "change") {
//The input was selected
console.log("File was selected via the input");
files = event.target.files;
};
for (var i = 0; files.length > i; i++) {
var file = files[i];
if (file.type.match('image.*')) {
//the file is an image, so let's continue processing
var selectedImage = new SelectedImage(file)
selectedImage.readFile();
}
};
};
function listenForFiles() {
var dropZone = document.getElementById('drop-zone');
var fileInput = document.getElementById('file-input');
//listen for the drag and drop events
dropZone.addEventListener('dragover', handleDragOver, false);
dropZone.addEventListener('drop', handleFileSelect, false);
//listen for the input change event
fileInput.addEventListener('change', handleFileSelect, false);
};
if (window.File && window.FileReader && window.FileList && window.Blob) {
//The demo is supported by the browser, show the drop zone and continue with magic
document.getElementById('browser-not-supported').style.display = "none";
document.getElementById('drop-zone').style.display = "block";
listenForFiles();
};
/*
Color conversions
Copyright (c) 2011, Cory Nelson ([email protected])
All rights reserved.
Redistribution and use in source and binary forms, with or without
modification, are permitted provided that the following conditions are met:
* Redistributions of source code must retain the above copyright
notice, this list of conditions and the following disclaimer.
* Redistributions in binary form must reproduce the above copyright
notice, this list of conditions and the following disclaimer in the
documentation and/or other materials provided with the distribution.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
DISCLAIMED. IN NO EVENT SHALL <COPYRIGHT HOLDER> BE LIABLE FOR ANY
DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
(INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
When possible, constants are given as accurate pre-computed rationals. When not,
they are given at double precision with a comment on how to compute them.
*/
// http://svn.int64.org/viewvc/int64/colors/colors.js
function clamp(x, min, max)
{
if(x < min) return min;
if(x > max) return max;
return x;
}
function clamphue(hue)
{
hue %= Math.PI * 2;
if(hue < 0)
{
hue += Math.PI * 2;
}
return hue;
}
function ColorLShuv(L, S, h, clamped)
{
this.clamped = clamped || L < 0 || L > 100; // TODO: what is S min/max?
this.L = clamp(L, 0, 100);
this.S = S;
this.h = clamphue(h);
this.toLChuv = function()
{
return new ColorLChuv(this.L, this.S * this.L, this.h);
};
}
function ColorLChuv(L, C, h, clamped)
{
this.clamped = clamped || L < 0 || L > 100 || C < 0 || C > 7.40066582332174237e2;
this.L = clamp(L, 0, 100);
this.C = clamp(C, 0, 7.40066582332174237e2); // 240/316141*sqrt(950343809713)
this.h = clamphue(h);
this.toLShuv = function()
{
return new ColorLShuv(this.L, this.C / this.L, this.h, this.clamped);
};
this.toLuv = function()
{
var h = clamphue(this.h);
return new ColorLuv(this.L, Math.cos(h) * this.C, Math.sin(h) * this.C, this.clamped);
};
}
function ColorLuv(L, u, v, clamped)
{
this.clamped = clamped || L < 0 || L > 100 || u < -81304600/316141 || v > 54113280/316141; // TODO: what is u max, and v min??
this.L = clamp(L, 0, 100);
this.u = (u < -81304600/316141) ? -81304600/316141 : u;
this.v = (v > 54113280/316141) ? 54113280/316141 : v;
this.toLChuv = function()
{
return new ColorLChuv(
this.L,
Math.sqrt(this.u * this.u + this.v * this.v),
Math.atan2(this.v, this.u),
this.clamped);
};
this.toXYZ = function()
{
var rdiv = Color.refX + Color.refY * 15 + Color.refZ * 3;
var ur = Color.refX * 4 / rdiv;
var vr = Color.refY * 9 / rdiv;
if(L > 8)
{
var Y = (this.L + 16) / 116;
Y = Y * Y * Y;
}
else
{
var Y = this.L * (27/24389);
}
var a = (this.L * 52 / (this.u + this.L * 13 * ur) - 1) / 3;
var b = -5 * Y;
var d = (this.L * 39 / (this.v + this.L * 13 * vr) - 5) * Y;
var X = (d - b) / (a + 1/3);
var Z = X * a + b;
return new ColorXYZ(X, Y, Z, this.clamped);
};
}
function ColorLChab(L, C, h, clamped)
{
this.clamped = clamped || L < 0 || L > 100 || C < 0 || C > 4.64238345442629658e2;
this.L = clamp(L, 0, 100);
this.C = clamp(C, 0, 4.64238345442629658e2); // 2500*sqrt(1/29)
this.h = clamphue(h);
this.toLab = function()
{
var h = clamphue(this.h);
return new ColorLab(this.L, Math.cos(h) * this.C, Math.sin(h) * this.C, this.clamped);
};
}
function ColorLab(L, a, b, clamped)
{
this.clamped = clamped || L < 0 || L > 100 || a < -12500/29 || a > 12500/29 || b < -5000/29 || b > 5000/29;
this.L = clamp(L, 0, 100);
this.a = clamp(a, -12500/29, 12500/29);
this.b = clamp(b, -5000/29, 5000/29);
this.toXYZ = function()
{
function toXYZc(c)
{
var c3 = c * c * c;
if(c3 > 216 / 24389) return c3;
return c * (108/841) - (432/24389);
}
var Y = (this.L + 16) / 116;
return new ColorXYZ(
toXYZc(Y + this.a / 500) * Color.refX,
toXYZc(Y) * Color.refY,
toXYZc(Y - this.b / 200) * Color.refZ,
this.clamped);
};
this.toLChab = function()
{
return new ColorLChab(
this.L,
Math.sqrt(this.a * this.a + this.b * this.b),
Math.atan2(this.b, this.a),
this.clamped);
};
}
function ColorxyY(x, y, Y, clamped)
{
this.clamped = clamped || x < 0 || x > 1 || y < 0 || y > 1 || Y < 0 || Y > 1;
this.x = clamp(x, 0, 1);
this.y = clamp(y, 0, 1);
this.Y = clamp(Y, 0, 1);
this.toXYZ = function()
{
if(Math.abs(this.y) != 0)
{
var mul = this.Y / this.y;
return new ColorXYZ(this.x * mul, this.Y, (1 - this.x - this.y) * mul);
}
else
{
return new ColorXYZ(0, 0, 0);
}
};
}
function ColorXYZ(X, Y, Z, clamped)
{
this.clamped = clamped || X < 0 || X > Color.refX || Y < 0 || Y > Color.refY || Z < 0 || Z > Color.refZ;
this.X = clamp(X, 0, Color.refX);
this.Y = clamp(Y, 0, Color.refY);
this.Z = clamp(Z, 0, Color.refZ);
this.toLinearRGB = function()
{
return new ColorLinearRGB(
this.X * (641589/197960) + this.Y * (-608687/395920) + this.Z * (-49353/98980),
this.X * (-42591639/43944050) + this.Y * (82435961/43944050) + this.Z * (1826061/43944050),
this.X * (49353/887015) + this.Y * (-180961/887015) + this.Z * (49353/46685),
this.clamped);
};
this.toxyY = function()
{
var div = this.X + this.Y + this.Z;
if(Math.abs(div) == 0)
{
div = 1;
}
return new ColorxyY(this.X / div, this.Y / div, this.Y, this.clamped);
};
this.toLab = function()
{
function toLabc(c)
{
if (c > 216 / 24389) return Math.pow(c, 1 / 3);
return c * (841 / 108) + (4 / 49);
}
var X = toLabc(this.X / Color.refX);
var Y = toLabc(this.Y / Color.refY);
var Z = toLabc(this.Z / Color.refZ);
return new ColorLab(116 * Y - 16, 500 * (X - Y), 200 * (Y - Z), this.clamped);
};
this.toLuv = function()
{
var rdiv = Color.refX + Color.refY * 15 + Color.refZ * 3;
var ur = Color.refX * 4 / rdiv;
var vr = Color.refY * 9 / rdiv;
var div = this.X + this.Y * 15 + this.Z * 3;
if(Math.abs(div) == 0)
{
div = 1;
}
var u = this.X * 4 / div;
var v = this.Y * 9 / div;
var yr = this.Y / Color.refY;
if(yr > 216/24389)
{
var L = Math.pow(yr, 1 / 3) * 116 - 16;
}
else
{
var L = yr * (24389/27);
}
return new ColorLuv(L, L * 13 * (u - ur), L * 13 * (v - vr), this.clamped);
};
}
function ColorLinearRGB(R, G, B, clamped)
{
this.clamped = clamped || R < 0 || R > 1 || G < 0 || G > 1 || B < 0 || B > 1;
this.R = clamp(R, 0, 1);
this.G = clamp(G, 0, 1);
this.B = clamp(B, 0, 1);
this.toRGB = function()
{
function toRGBc(c)
{
if(c > 0.0031308) return Math.pow(c, 1 / 2.4) * 1.055 - 0.055;
return c * 12.92;
}
return new ColorRGB(toRGBc(this.R), toRGBc(this.G), toRGBc(this.B), this.clamped);
};
this.toXYZ = function()
{
return new ColorXYZ(
this.R * (5067776/12288897) + this.G * (4394405/12288897) + this.B * (4435075/24577794),
this.R * (871024/4096299) + this.G * (8788810/12288897) + this.B * (887015/12288897),
this.R * (79184/4096299) + this.G * (4394405/36866691) + this.B * (70074185/73733382),
this.clamped);
};
}
function ColorHSV(H, S, V, clamped)
{
this.clamped = clamped || S < 0 || S > 1 || V < 0 || V > 1;
this.H = clamphue(H);
this.S = clamp(S, 0, 1);
this.V = clamp(V, 0, 1);
this.toRGB = function()
{
if(this.S <= 0)
{
return new ColorRGB(this.V, this.V, this.V, this.clamped);
}
var H = clamphue(this.H) / 60;
var C = this.V * this.S;
var m = this.V - C;
var X = C * (1 - Math.abs(H % 2 - 1)) + m;
C += m;
if(H >= 5) return new ColorRGB(C, m, X, this.clamped);
if(H >= 4) return new ColorRGB(X, m, C, this.clamped);
if(H >= 3) return new ColorRGB(m, X, C, this.clamped);
if(H >= 2) return new ColorRGB(m, C, X, this.clamped);
if(H >= 1) return new ColorRGB(X, C, m, this.clamped);
return new ColorRGB(C, X, m, this.clamped);
};
}
function ColorHSL(H, S, L, clamped)
{
this.clamped = clamped || S < 0 || S > 1 || L < 0 || L > 1;
this.H = clamphue(H);
this.S = clamp(S, 0, 1);
this.L = clamp(L, 0, 1);
this.toRGB = function()
{
if(this.S <= 0)
{
return new ColorRGB(this.S, this.S, this.S, this.clamped);
}
var H = clamphue(this.H) / 60;
var C = (1 - Math.abs(this.L * 2 - 1)) * this.S;
var m = this.L - C * 0.5;
var X = C * (1 - Math.abs(H % 2 - 1)) + m;
C += m;
if(H >= 5) return new ColorRGB(C, m, X, this.clamped);
if(H >= 4) return new ColorRGB(X, m, C, this.clamped);
if(H >= 3) return new ColorRGB(m, X, C, this.clamped);
if(H >= 2) return new ColorRGB(m, C, X, this.clamped);
if(H >= 1) return new ColorRGB(X, C, m, this.clamped);
return new ColorRGB(C, X, m, this.clamped);
};
}
function ColorRGB(R, G, B, clamped)
{
this.clamped = clamped || R < 0 || R > 1 || G < 0 || G > 1 || B < 0 || B > 1;
this.R = clamp(R, 0, 1);
this.G = clamp(G, 0, 1);
this.B = clamp(B, 0, 1);
this.toHSV = function()
{
var min = Math.min(this.R, this.G, this.B);
var max = Math.max(this.R, this.G, this.B);
var delta = max - min;
if(Math.abs(delta) != 0)
{
var S = delta / max;
if(max == this.R) var H = (this.G - this.B) / delta;
else if(max == this.G) var H = (this.B - this.R) / delta + 2;
else var H = (this.R - this.G) / delta + 4;
}
else
{
var H = 0;
var S = 0;
}
return new ColorHSV(H * 60, S, max, this.clamped);
};
this.toHSL = function()
{
var min = Math.min(this.R, this.G, this.B);
var max = Math.max(this.R, this.G, this.B);
var delta = max - min;
var L = (max + min) * 0.5;
if(Math.abs(delta) != 0)
{
if(L < 0.5) var S = delta / (max + min);
else var S = delta / (2 - max - min);
if(max == this.R) var H = (this.G - this.B) / delta;
else if(max == this.G) var H = (this.B - this.R) / delta + 2;
else var H = (this.R - this.G) / delta + 4;
}
else
{
var H = 0;
var S = 0;
}
return new ColorHSL(H * 60, S, L, this.clamped);
};
this.toLinearRGB = function()
{
function toLinearRGBc(c)
{
if(c > 0.04045) return Math.pow((c + 0.055) / 1.055, 2.4);
return c / 12.92;
}
return new ColorLinearRGB(toLinearRGBc(this.R), toLinearRGBc(this.G), toLinearRGBc(this.B), this.clamped);
};
this.toYUV = function()
{
var mat = Color.yuvmatrix;
var y = this.R * mat.rScale + this.G * mat.gScale + this.B * mat.bScale;
var u = (this.B - y) / (1 - mat.bScale) * 0.5 + 0.5;
var v = (this.R - y) / (1 - mat.rScale) * 0.5 + 0.5;
return new ColorYUV(y, u, v, this.clamped);
};
this.toYIQ = function()
{
return new ColorYIQ(
this.R * 0.299 + this.G * 0.587 + this.B * 0.114,
this.R * 0.5 + this.G * -0.23038159508364756 + this.B * -0.26961840491635244 + 0.5,
this.R * -0.202349432337541121 + this.G * 0.5 + this.B * -0.297650567662458879 + 0.5,
this.clamped);
};
}
function ColorYUV(Y, U, V, clamped)
{
this.clamped = clamped || Y < 0 || Y > 1 || U < 0 || V > 1 || V < 0 || V > 1;
this.Y = clamp(Y, 0, 1);
this.U = clamp(U, 0, 1);
this.V = clamp(V, 0, 1);
this.toRGB = function()
{
var mat = Color.yuvmatrix;
var u = (this.U - 0.5) / 0.5 * (1 - mat.bScale);
var v = (this.V - 0.5) / 0.5 * (1 - mat.rScale);
var r = v + this.Y;
var b = u + this.Y;
var g = (this.Y - r * mat.rScale - b * mat.bScale) / mat.gScale;
return new ColorRGB(r, g, b, this.clamped);
};
}
function ColorYIQ(Y, I, Q, clamped)
{
this.clamped = clamped || Y < 0 || Y > 1 || I < 0 || I > 1 || Q < 0 || Q > 1;
this.Y = clamp(Y, 0, 1);
this.I = clamp(I, 0, 1);
this.Q = clamp(Q, 0, 1);
this.toRGB = function()
{
var i = this.I - 0.5;
var q = this.Q - 0.5;
return new ColorRGB(
this.Y + i * 1.13933588212202582 - q * 0.649035964281386078,
this.Y - i * 0.32416610079155499 + q * 0.676636193255190191,
this.Y - i * 1.31908708412142932 - q * 1.78178677298826495,
this.clamped);
};
}
// CIE Delta E 1976
// JND: ~2.3
function deltaE1976(lab1, lab2)
{
var delta_L = lab1.L - lab2.L;
var delta_a = lab1.a - lab2.a;
var delta_b = lab1.b - lab2.b;
return Math.sqrt(delta_L * delta_L + delta_a * delta_a + delta_b * delta_b);
}
// CIE Delta E 1994
function deltaE1994(lab1, lab2, type)
{
var C1 = Math.sqrt(lab1.a * lab1.a + lab1.b * lab1.b);
var C2 = Math.sqrt(lab2.a * lab2.a + lab2.b * lab2.b);
var delta_L = lab1.L - lab2.L;
var delta_C = C1 - C2;
var delta_a = lab1.a - lab2.a;
var delta_b = lab1.b - lab2.b;
var delta_H = Math.sqrt(delta_a * delta_a + delta_b * delta_b - delta_C * delta_C);
if(type == 'graphic arts')
{
delta_C /= C1 * 0.045 + 1;
delta_H /= C1 * 0.015 + 1;
}
else if(type == 'textiles')
{
delta_L *= 0.5;
delta_C /= C1 * 0.048 + 1;
delta_H /= C1 * 0.014 + 1;
}
return Math.sqrt(delta_L * delta_L + delta_C * delta_C + delta_H * delta_H);
}
// CIE Delta E 2000
// Note: maximum is about 158 for colors in the sRGB gamut.
function deltaE2000(lch1, lch2)
{
var avg_L = (lch1.L + lch2.L) * 0.5;
var delta_L = lch2.L - lch1.L;
var avg_C = (lch1.C + lch2.C) * 0.5;
var delta_C = lch1.C - lch2.C;
var avg_H = (lch1.h + lch2.h) * 0.5;
if(Math.abs(lch1.h - lch2.h) > Math.PI)
{
avg_H += Math.PI;
}
var delta_H = lch2.h - lch1.h;
if(Math.abs(delta_H) > Math.PI)
{
if(lch2.h <= lch1.h) delta_H += Math.PI * 2;
else delta_H -= Math.PI * 2;
}
delta_H = Math.sqrt(lch1.C * lch2.C) * Math.sin(delta_H) * 2;
var T = 1
- 0.17 * Math.cos(avg_H - Math.PI / 6)
+ 0.24 * Math.cos(avg_H * 2)
+ 0.32 * Math.cos(avg_H * 3 + Math.PI / 30)
- 0.20 * Math.cos(avg_H * 4 - Math.PI * 7/20);
var SL = avg_L - 50;
SL *= SL;
SL = SL * 0.015 / Math.sqrt(SL + 20) + 1;
var SC = avg_C * 0.045 + 1;
var SH = avg_C * T * 0.015 + 1;
var delta_Theta = avg_H / 25 - Math.PI * 11/180;
delta_Theta = Math.exp(delta_Theta * -delta_Theta) * (Math.PI / 6);
var RT = Math.pow(avg_C, 7);
RT = Math.sqrt(RT / (RT + 6103515625)) * Math.sin(delta_Theta) * -2; // 6103515625 = 25^7
delta_L /= SL;
delta_C /= SC;
delta_H /= SH;
return Math.sqrt(delta_L * delta_L + delta_C * delta_C + delta_H * delta_H + RT * delta_C * delta_H);
}
var Color =
{
refX: 31271/32902, // normalized standard observer D65.
refY: 1,
refZ: 35827/32902,
yuvmatrices:
{
'bt601':
{
name: 'ITU-R BT.601 (DVD, JPEG, Youtube)',
rScale: 0.299,
gScale: 0.587,
bScale: 0.114
},
'bt709':
{
name: 'ITU-R BT.709 (HDTV)',
rScale: 0.2125,
gScale: 0.7154,
bScale: 0.0721
},
'smpte240m':
{
name: 'SMPTE 240M (very old HDTV)',
rScale: 0.212,
gScale: 0.701,
bScale: 0.087
},
'fcc':
{
name: 'FCC',
rScale: 0.3,
gScale: 0.59,
bScale: 0.11
}
},
colorspaces:
{
'rgb':
{
name: 'RGB',
components: ['Red', 'Green', 'Blue'],
componentInfo:
[
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 }
],
toColor: function(x) { return new ColorRGB(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.R, x.G, x.B ]; },
conversions:
{
'hsl': function(x) { return x.toHSL(); },
'hsv': function(x) { return x.toHSV(); },
'yuv': function(x) { return x.toYUV(); },
'yiq': function(x) { return x.toYIQ(); },
'linear_rgb': function(x) { return x.toLinearRGB(); }
}
},
'linear_rgb':
{
name: 'Linear RGB',
components: ['Red', 'Green', 'Blue'],
componentInfo:
[
{ minimum: 0, maximum: 1, scale: 1023 },
{ minimum: 0, maximum: 1, scale: 1023 },
{ minimum: 0, maximum: 1, scale: 1023 }
],
toColor: function(x) { return new ColorLinearRGB(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.R, x.G, x.B ]; },
conversions:
{
'rgb': function(x) { return x.toRGB(); },
'xyz': function(x) { return x.toXYZ(); },
}
},
'hsl':
{
name: 'HSL',
components: ['Hue', 'Saturation', 'Lightness'],
componentInfo:
[
{ minimum: 0, maximum: 359, scale: 1 },
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 }
],
toColor: function(x) { return new ColorHSL(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.H, x.S, x.L ]; },
conversions:
{
'rgb': function(x) { return x.toRGB(); }
}
},
'hsv':
{
name: 'HSV',
components: ['Hue', 'Saturation', 'Value'],
componentInfo:
[
{ minimum: 0, maximum: 359, scale: 1 },
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 }
],
toColor: function(x) { return new ColorHSV(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.H, x.S, x.V ]; },
conversions:
{
'rgb': function(x) { return x.toRGB(); }
}
},
'yuv':
{
name: 'Y′UV',
components: ['Luma', 'Chroma U', 'Chroma V'],
componentInfo:
[
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 }
],
toColor: function(x) { return new ColorYUV(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.Y, x.U, x.V ]; },
conversions:
{
'rgb': function(x) { return x.toRGB(); }
}
},
'yiq':
{
name: 'Y′IQ',
components: ['Luma', 'Chroma I', 'Chroma Q'],
componentInfo:
[
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 },
{ minimum: 0, maximum: 1, scale: 255 }
],
toColor: function(x) { return new ColorYIQ(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.Y, x.I, x.Q ]; },
conversions:
{
'rgb': function(x) { return x.toRGB(); }
}
},
'xyy':
{
name: 'CIE xyY',
components: ['x', 'y', 'Y'],
componentInfo:
[
{ minimum: 0, maximum: 1, scale: 100 },
{ minimum: 0, maximum: 1, scale: 100 },
{ minimum: 0, maximum: 1, scale: 100 }
],
toColor: function(x) { return new ColorxyY(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.x, x.y, x.Y ]; },
conversions:
{
'xyz': function(x) { return x.toXYZ(); }
}
},
'xyz':
{
name: 'CIE XYZ',
components: ['X', 'Y', 'Z'],
componentInfo:
[
{ minimum: 0, maximum: 0.9505, scale: 100 },
{ minimum: 0, maximum: 1, scale: 100 },
{ minimum: 0, maximum: 1.089, scale: 100 }
],
toColor: function(x) { return new ColorXYZ(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.X, x.Y, x.Z ]; },
conversions:
{
'linear_rgb': function(x) { return x.toLinearRGB(); },
'lab': function(x) { return x.toLab(); },
'luv': function(x) { return x.toLuv(); },
'xyy': function(x) { return x.toxyY(); }
}
},
'lab':
{
name: 'CIE L*a*b*',
components: ['Lightness', 'a', 'b'],
componentInfo:
[
{ minimum: 0, maximum: 100, scale: 1 },
{ minimum: -12500/29, maximum: 12500/29, scale: 1 },
{ minimum: -5000/29, maximum: 5000/29, scale: 1 }
],
toColor: function(x) { return new ColorLab(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.L, x.a, x.b ]; },
conversions:
{
'xyz': function(x) { return x.toXYZ(); },
'LChab': function(x) { return x.toLChab(); }
}
},
'LChab':
{
name: function(c)
{
c.appendChild(document.createTextNode('CIE L*C*h'));
var sub = document.createElement('sub');
sub.appendChild(document.createTextNode('ab'));
c.appendChild(sub);
},
components: ['Lightness', 'Chroma', 'Hue'],
componentInfo:
[
{ minimum: 0, maximum: 100, scale: 1 },
{ minimum: 0, maximum: 4.64238345442629658e2, scale: 1 },
{ minimum: 0, maximum: Math.PI*2, scale: 180/Math.PI }
],
toColor: function(x) { return new ColorLChab(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.L, x.C, x.h ]; },
conversions:
{
'lab': function(x) { return x.toLab(); }
}
},
'luv':
{
name: 'CIE L*u*v*',
components: ['Lightness', 'u', 'v'],
componentInfo:
[
{ minimum: 0, maximum: 100, scale: 1 },
{ minimum: -81304600/316141, maximum: 720, scale: 1 },
{ minimum: -160, maximum: 54113280/316141, scale: 1 }
],
toColor: function(x) { return new ColorLuv(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.L, x.u, x.v ]; },
conversions:
{
'xyz': function(x) { return x.toXYZ(); },
'LChuv': function(x) { return x.toLChuv(); }
}
},
'LChuv':
{
name: function(c)
{
c.appendChild(document.createTextNode('CIE L*C*h'));
var sub = document.createElement('sub');
sub.appendChild(document.createTextNode('uv'));
c.appendChild(sub);
},
components: ['Lightness', 'Chroma', 'Hue'],
componentInfo:
[
{ minimum: 0, maximum: 100, scale: 1 },
{ minimum: 0, maximum: 7.40066582332174237e2, scale: 1 },
{ minimum: 0, maximum: Math.PI*2, scale: 180/Math.PI }
],
toColor: function(x) { return new ColorLChuv(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.L, x.C, x.h ]; },
conversions:
{
'luv': function(x) { return x.toLuv(); },
'lshuv': function(x) { return x.toLShuv(); }
}
},
'lshuv':
{
name: function(c)
{
c.appendChild(document.createTextNode('CIE L*Sh'));
var sub = document.createElement('sub');
sub.appendChild(document.createTextNode('uv'));
c.appendChild(sub);
},
components: ['Lightness', 'Saturation', 'Hue'],
componentInfo:
[
{ minimum: 0, maximum: 100, scale: 1 },
{ minimum: 0, maximum: 4.5, scale: 25 },
{ minimum: 0, maximum: Math.PI*2, scale: 180/Math.PI }
],
toColor: function(x) { return new ColorLShuv(x[0], x[1], x[2]); },
toGeneric: function(x) { return [ x.L, x.S, x.h ]; },
conversions:
{
'LChuv': function(x) { return x.toLChuv(); }
}
}
}
};
</script>
</body>
</html>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment