Implement proper parsing of rgb / rgba CSS strings.

This commit is contained in:
Jürg Lehni 2013-11-24 00:23:32 +01:00
parent 829e878243
commit b8b02417a0

View file

@ -50,49 +50,58 @@ var Color = Base.extend(new function() {
gradient: ['gradient', 'origin', 'destination', 'highlight'] gradient: ['gradient', 'origin', 'destination', 'highlight']
}; };
var componentParsers = {}, // Parsers of values for setters, by type and property // Parsers of values for setters, by type and property
var componentParsers = {},
// Cache and canvas context for color name lookup
colorCache = {}, colorCache = {},
colorCtx; colorCtx;
function nameToRGB(name) { function fromCSS(string) {
var cached = colorCache[name]; var match = string.match(/^#(\w{1,2})(\w{1,2})(\w{1,2})$/),
components;
if (match) {
components = [0, 0, 0];
for (var i = 0; i < 3; i++) {
var value = match[i + 1];
components[i] = parseInt(value.length == 1
? value + value : value, 16) / 255;
}
} else if (match = string.match(/^rgba?\((.*)\)$/)) {
components = match[1].split(',');
for (var i = 0, l = components.length; i < l; i++) {
var value = parseFloat(components[i]);
components[i] = i < 3 ? value / 255 : value;
}
} else {
// Named
var cached = colorCache[string];
if (!cached) { if (!cached) {
// Use a canvas to draw to with the given name and then retrieve rgb // Use a canvas to draw to with the given name and then retrieve
// values from. Build a cache for all the used colors. // RGB values from. Build a cache for all the used colors.
if (!colorCtx) { if (!colorCtx) {
colorCtx = CanvasProvider.getContext(1, 1); colorCtx = CanvasProvider.getContext(1, 1);
colorCtx.globalCompositeOperation = 'copy'; colorCtx.globalCompositeOperation = 'copy';
} }
// Set the current fillStyle to transparent, so that it will be // Set the current fillStyle to transparent, so that it will be
// transparent instead of the previously set color in case the new // transparent instead of the previously set color in case the
// color can not be interpreted. // new color can not be interpreted.
colorCtx.fillStyle = 'rgba(0,0,0,0)'; colorCtx.fillStyle = 'rgba(0,0,0,0)';
// Set the fillStyle of the context to the passed name and fill the // Set the fillStyle of the context to the passed name and fill
// canvas with it, then retrieve the data for the drawn pixel: // the canvas with it, then retrieve the data for the drawn
colorCtx.fillStyle = name; // pixel:
colorCtx.fillStyle = string;
colorCtx.fillRect(0, 0, 1, 1); colorCtx.fillRect(0, 0, 1, 1);
var data = colorCtx.getImageData(0, 0, 1, 1).data; var data = colorCtx.getImageData(0, 0, 1, 1).data;
cached = colorCache[name] = [ cached = colorCache[string] = [
data[0] / 255, data[0] / 255,
data[1] / 255, data[1] / 255,
data[2] / 255 data[2] / 255
]; ];
} }
return cached.slice(); components = cached.slice();
}
function hexToRGB(string) {
var hex = string.match(/^#?(\w{1,2})(\w{1,2})(\w{1,2})$/);
if (hex.length >= 4) {
var components = [0, 0, 0];
for (var i = 0; i < 3; i++) {
var value = hex[i + 1];
components[i] = parseInt(value.length == 1
? value + value : value, 16) / 255;
} }
return components; return components;
} }
}
// For hsb-rgb conversion, used to lookup the right parameters in the // For hsb-rgb conversion, used to lookup the right parameters in the
// values array. // values array.
@ -533,9 +542,7 @@ var Color = Base.extend(new function() {
if (values.length > length) if (values.length > length)
values = slice.call(values, 0, length); values = slice.call(values, 0, length);
} else if (argType === 'string') { } else if (argType === 'string') {
components = arg.match(/^#[0-9a-f]{3,6}$/i) components = fromCSS(arg);
? hexToRGB(arg)
: nameToRGB(arg);
type = 'rgb'; type = 'rgb';
} else if (argType === 'object') { } else if (argType === 'object') {
if (arg.constructor === Color) { if (arg.constructor === Color) {