mirror of
https://github.com/scratchfoundation/paper.js.git
synced 2025-01-04 03:45:58 -05:00
Color: Improve CSS string parser and docs
This commit is contained in:
parent
e41ed5e723
commit
da3a36230f
2 changed files with 89 additions and 39 deletions
|
@ -56,9 +56,9 @@ var Color = Base.extend(new function() {
|
||||||
colorCache = {},
|
colorCache = {},
|
||||||
colorCtx;
|
colorCtx;
|
||||||
|
|
||||||
// TODO: Implement hsv, etc. CSS parsing!
|
|
||||||
function fromCSS(string) {
|
function fromCSS(string) {
|
||||||
var match = string.match(/^#(\w{1,2})(\w{1,2})(\w{1,2})$/),
|
var match = string.match(/^#(\w{1,2})(\w{1,2})(\w{1,2})$/),
|
||||||
|
type = 'rgb',
|
||||||
components;
|
components;
|
||||||
if (match) {
|
if (match) {
|
||||||
// Hex
|
// Hex
|
||||||
|
@ -68,12 +68,34 @@ var Color = Base.extend(new function() {
|
||||||
components[i] = parseInt(value.length == 1
|
components[i] = parseInt(value.length == 1
|
||||||
? value + value : value, 16) / 255;
|
? value + value : value, 16) / 255;
|
||||||
}
|
}
|
||||||
} else if (match = string.match(/^rgba?\((.*)\)$/)) {
|
} else if (match = string.match(/^(rgb|hsl)a?\((.*)\)$/)) {
|
||||||
// RGB / RGBA
|
// RGB / RGBA or HSL / HSLA
|
||||||
components = match[1].split(',');
|
type = match[1];
|
||||||
for (var i = 0, l = components.length; i < l; i++) {
|
components = match[2].split(/[,\s]+/g);
|
||||||
var value = +components[i];
|
var isHSL = type === 'hsl';
|
||||||
components[i] = i < 3 ? value / 255 : value;
|
for (var i = 0, l = Math.min(components.length, 4); i < l; i++) {
|
||||||
|
var component = components[i];
|
||||||
|
// Use `parseFloat()` instead of `+value` to parse '\d+%' to
|
||||||
|
// float for HSL:
|
||||||
|
var value = parseFloat(component);
|
||||||
|
if (isHSL) {
|
||||||
|
if (i === 0) {
|
||||||
|
// handle 'deg', 'turn', 'rad' 'grad':
|
||||||
|
var unit = component.match(/([a-z]*)$/)[1];
|
||||||
|
value *= ({
|
||||||
|
turn: 360,
|
||||||
|
rad: 180 / Math.PI,
|
||||||
|
grad: 0.9 // 360 / 400
|
||||||
|
}[unit] || 1);
|
||||||
|
} else if (i < 3) {
|
||||||
|
// Percentages to 0..1
|
||||||
|
value /= 100;
|
||||||
|
}
|
||||||
|
} else if (i < 3) {
|
||||||
|
// RGB color values to 0..1
|
||||||
|
value /= 255;
|
||||||
|
}
|
||||||
|
components[i] = value;
|
||||||
}
|
}
|
||||||
} else if (window) {
|
} else if (window) {
|
||||||
// Named
|
// Named
|
||||||
|
@ -106,7 +128,7 @@ var Color = Base.extend(new function() {
|
||||||
// Web-workers can't resolve CSS color names, for now.
|
// Web-workers can't resolve CSS color names, for now.
|
||||||
components = [0, 0, 0];
|
components = [0, 0, 0];
|
||||||
}
|
}
|
||||||
return components;
|
return [type, 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
|
||||||
|
@ -237,13 +259,17 @@ var Color = Base.extend(new function() {
|
||||||
// hsb and hsl. Handle this here separately, by testing for
|
// hsb and hsl. Handle this here separately, by testing for
|
||||||
// overlaps and skipping conversion if the type is /hs[bl]/
|
// overlaps and skipping conversion if the type is /hs[bl]/
|
||||||
hasOverlap = /^(hue|saturation)$/.test(name),
|
hasOverlap = /^(hue|saturation)$/.test(name),
|
||||||
// Produce value parser function for the given type / propeprty
|
// Produce value parser function for the given type / property
|
||||||
// name combination.
|
parser = componentParsers[type][index] = type === 'gradient'
|
||||||
parser = componentParsers[type][index] = name === 'gradient'
|
? name === 'gradient'
|
||||||
|
// gradient property of gradient color:
|
||||||
? function(value) {
|
? function(value) {
|
||||||
var current = this._components[0];
|
var current = this._components[0];
|
||||||
value = Gradient.read(Array.isArray(value) ? value
|
value = Gradient.read(
|
||||||
: arguments, 0, { readNull: true });
|
Array.isArray(value)
|
||||||
|
? value
|
||||||
|
: arguments, 0, { readNull: true }
|
||||||
|
);
|
||||||
if (current !== value) {
|
if (current !== value) {
|
||||||
if (current)
|
if (current)
|
||||||
current._removeOwner(this);
|
current._removeOwner(this);
|
||||||
|
@ -252,21 +278,21 @@ var Color = Base.extend(new function() {
|
||||||
}
|
}
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
: type === 'gradient'
|
// all other (point) properties of gradient color:
|
||||||
? function(/* value */) {
|
: function(/* value */) {
|
||||||
return Point.read(arguments, 0, {
|
return Point.read(arguments, 0, {
|
||||||
readNull: name === 'highlight',
|
readNull: name === 'highlight',
|
||||||
clone: true
|
clone: true
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
// Normal number component properties:
|
||||||
: function(value) {
|
: function(value) {
|
||||||
// NOTE: We don't clamp values here, they're only
|
// NOTE: We don't clamp values here, they're only
|
||||||
// clamped once the actual CSS values are produced.
|
// clamped once the actual CSS values are produced.
|
||||||
// Gotta love the fact that isNaN(null) is false,
|
// Gotta love the fact that isNaN(null) is false,
|
||||||
// while isNaN(undefined) is true.
|
// while isNaN(undefined) is true.
|
||||||
return value == null || isNaN(value) ? 0 : value;
|
return value == null || isNaN(value) ? 0 : +value;
|
||||||
};
|
};
|
||||||
|
|
||||||
this['get' + part] = function() {
|
this['get' + part] = function() {
|
||||||
return this._type === type
|
return this._type === type
|
||||||
|| hasOverlap && /^hs[bl]$/.test(this._type)
|
|| hasOverlap && /^hs[bl]$/.test(this._type)
|
||||||
|
@ -411,6 +437,25 @@ var Color = Base.extend(new function() {
|
||||||
* }
|
* }
|
||||||
* });
|
* });
|
||||||
*/
|
*/
|
||||||
|
/**
|
||||||
|
* Creates a Color object from a CSS string. All common CSS color string
|
||||||
|
* formats are supported:
|
||||||
|
* - Named colors (e.g. `'red'`, `'fuchsia'`, …)
|
||||||
|
* - Hex strings (`'#ffff00'`, `'#ff0'`, …)
|
||||||
|
* - RGB strings (`'rgb(255, 128, 0)'`, `'rgba(255, 128, 0, 0.5)'`, …)
|
||||||
|
* - HSL strings (`'hsl(180deg, 20%, 50%)'`,
|
||||||
|
* `'hsla(3.14rad, 20%, 50%, 0.5)'`, …)
|
||||||
|
*
|
||||||
|
* @name Color#initialize
|
||||||
|
* @param {String} color the color's CSS string representation
|
||||||
|
*
|
||||||
|
* @example {@paperscript}
|
||||||
|
* var circle = new Path.Circle({
|
||||||
|
* center: [80, 50],
|
||||||
|
* radius: 30,
|
||||||
|
* fillColor: new Color('rgba(255, 255, 0, 0.5)')
|
||||||
|
* });
|
||||||
|
*/
|
||||||
/**
|
/**
|
||||||
* Creates a gradient Color object.
|
* Creates a gradient Color object.
|
||||||
*
|
*
|
||||||
|
@ -544,8 +589,9 @@ var Color = Base.extend(new function() {
|
||||||
if (values.length > length)
|
if (values.length > length)
|
||||||
values = Base.slice(values, 0, length);
|
values = Base.slice(values, 0, length);
|
||||||
} else if (argType === 'string') {
|
} else if (argType === 'string') {
|
||||||
type = 'rgb';
|
var converted = fromCSS(arg);
|
||||||
components = fromCSS(arg);
|
type = converted[0];
|
||||||
|
components = converted[1];
|
||||||
if (components.length === 4) {
|
if (components.length === 4) {
|
||||||
alpha = components[3];
|
alpha = components[3];
|
||||||
components.length--;
|
components.length--;
|
||||||
|
|
|
@ -61,22 +61,26 @@ test('Creating Colors', function() {
|
||||||
'Color from name (red)');
|
'Color from name (red)');
|
||||||
|
|
||||||
equals(new Color('#ff0000'), new Color(1, 0, 0),
|
equals(new Color('#ff0000'), new Color(1, 0, 0),
|
||||||
'Color from hex code');
|
'Color from hex string');
|
||||||
|
|
||||||
equals(new Color('rgb(255, 0, 0)'), new Color(1, 0, 0),
|
equals(new Color('rgb(255, 0, 0)'), new Color(1, 0, 0),
|
||||||
'Color from RGB code');
|
'Color from rgb() string');
|
||||||
|
|
||||||
equals(new Color('rgba(255, 0, 0, 0.5)'), new Color(1, 0, 0, 0.5),
|
equals(new Color('rgba(255, 0, 0, 0.5)'), new Color(1, 0, 0, 0.5),
|
||||||
'Color from RGBA code');
|
'Color from rgba() string');
|
||||||
|
|
||||||
|
equals(new Color('hsl(180deg, 20%, 40%)'),
|
||||||
|
new Color({ hue: 180, saturation: 0.2, lightness: 0.4 }),
|
||||||
|
'Color from hsl() string');
|
||||||
|
|
||||||
equals(new Color({ red: 1, green: 0, blue: 1}),
|
equals(new Color({ red: 1, green: 0, blue: 1}),
|
||||||
new Color(1, 0, 1), 'Color from rgb object literal');
|
new Color(1, 0, 1), 'Color from RGB object literal');
|
||||||
|
|
||||||
equals(new Color({ gray: 0.2 }),
|
equals(new Color({ gray: 0.2 }),
|
||||||
new Color(0.2), 'Color from gray object literal');
|
new Color(0.2), 'Color from gray object literal');
|
||||||
|
|
||||||
equals(new Color({ hue: 0, saturation: 1, brightness: 1}),
|
equals(new Color({ hue: 0, saturation: 1, brightness: 1}),
|
||||||
new Color(1, 0, 0).convert('hsb'), 'Color from hsb object literal');
|
new Color(1, 0, 0).convert('hsb'), 'Color from HSB object literal');
|
||||||
|
|
||||||
equals(new Color([1, 0, 0]), new Color(1, 0, 0),
|
equals(new Color([1, 0, 0]), new Color(1, 0, 0),
|
||||||
'RGB Color from array');
|
'RGB Color from array');
|
||||||
|
|
Loading…
Reference in a new issue