/* * Paper.js * * This file is part of Paper.js, a JavaScript Vector Graphics Library, * based on Scriptographer.org and designed to be largely API compatible. * http://paperjs.org/ * http://scriptographer.org/ * * Distributed under the MIT license. See LICENSE file for details. * * Copyright (c) 2011, Juerg Lehni & Jonathan Puckey * http://lehni.org/ & http://jonathanpuckey.com/ * * All rights reserved. */ var Color = this.Color = Base.extend(new function() { var nameToRGBColor = new function() { var cache = {}, context; return function(name) { var color = cache[name]; if (!color) { // Use a 1x1 Canvas to draw to with the given name and then // retrieve the rgb values from. Build a cache for all these // colors on a per use basis. if (!context) { var canvas = CanvasProvider.getCanvas(Size.create(1, 1)); context = canvas.getContext('2d'); context.globalCompositeOperation = 'copy'; } // Set the fillStyle of the context to the passed name // and fill the canvas with it. Then retrieve the first pixel: context.fillStyle = name; context.fillRect(0, 0, 1, 1); var data = context.getImageData(0, 0, 1, 1).data, rgb = []; for (var i = 0; i < 3; i++) rgb[i] = data[i] / 255; // If the name wasn't found, rgb is [0, 0, 0] if (rgb.join('') == '000' && name != 'black') { throw new Error('The named color "' + name + '" does not exist.'); } else { color = cache[name] = RGBColor.read(rgb); } } return color; } } function hexToRGBColor(string) { var hex = string.match(/^#?(\w{1,2})(\w{1,2})(\w{1,2})$/); if (hex.length >= 4) { var rgb = new Array(3); for (var i = 0; i < 3; i++) { var channel = hex[i + 1]; rgb[i] = parseInt(channel.length == 1 ? channel + channel : channel, 16) / 255; } return RGBColor.read(rgb); } }; return { beans: true, initialize: function(arg) { var isArray = Array.isArray(arg); if (typeof arg == 'object' && !isArray) { if (!this._colorType) { // Called on the abstract Color class. Guess color type // from arg if (arg.red !== undefined) { return new RGBColor(arg.red, arg.green, arg.blue, arg.alpha); } else if (arg.gray !== undefined) { return new GrayColor(arg.gray, arg.alpha); } else if (arg.hue !== undefined) { return new HSBColor(arg.hue, arg.saturation, arg.brightness, arg.alpha); } } else { // Called on a subclass instance. Return the converted // color. var color = Color.read(arguments, 0, 1); return this._colorType ? color.convert(this._colorType) : color; } } else if (typeof arg == 'string') { var rgbColor = arg.match(/^#[0-9a-f]{3,6}$/i) ? hexToRGBColor(arg) : nameToRGBColor(arg); return this._colorType ? rgbColor.convert(this._colorType) : rgbColor; } else { var components = isArray ? arg : arguments; if (!this._colorType) { // Called on the abstract Color class. Guess color type // from arg //if (components.length >= 4) // return new CMYKColor(components); if (components.length >= 3) return new RGBColor(components); return new GrayColor(components); } else { // Called on a subclass instance. Just copy over // components. Base.each(this._components, function(name, i) { var value = components[i]; // TODO: should we call the setter? // this['set' + name.capitalize()] this[name] = value !== undefined ? value // TODO: is this correct? // Shouldn't alpha be set to -1? : name == 'alpha' ? 1 : null; }, this); } } }, getType: function() { return this._colorType; }, getComponents: function() { var l = this._components.length; var components = new Array(l); for (var i = 0; i < l; i++) { components[i] = this['_' + this._components[i]]; } return components; }, /** * A value between 0 and 1 that specifies the color's alpha value. * All colors of the different subclasses support alpha values. */ getAlpha: function() { return this._alpha != null ? this._alpha : 1; }, setAlpha: function(alpha) { this._alpha = Math.min(Math.max(alpha, 0), 1); this._cssString = null; }, /** * Checks if the color has an alpha value. * * @return {@true if the color has an alpha value} */ hasAlpha: function() { return this._alpha != null; }, /** * Checks if the component color values of the color are the * same as those of the supplied one. * * @param obj the GrayColor to compare with * @return {@true if the GrayColor is the same} */ equals: function(color) { if (color._colorType == this._colorType) { for (var i = 0, l = this._components; i < l; i++) { var component = '_' + this._components[i]; if (this[component] !== color[component]) return false; } return true; } return false; }, toString: function() { var string = ''; for (var i = 0, l = this._components.length; i < l; i++) { var component = this._components[i]; var value = this['_' + component]; if (component == 'alpha' && value == null) value = 1; string += (i > 0 ? ', ' : '') + component + ': ' + value; } return '{ ' + string + ' }'; }, toCssString: function() { if (!this._cssString) { var color = this._colorType == 'rgb' ? this : this.convert('rgb'); var alpha = color.getAlpha(); var components = [ Math.round(color.getRed() * 255), Math.round(color.getGreen() * 255), Math.round(color.getBlue() * 255), alpha != null ? this.alpha : 1 ]; this._cssString = 'rgba(' + components.join(', ') + ')'; } return this._cssString; }, getCanvasStyle: function() { return this.toCssString(); } }; }, new function() { var converters = { 'rgb-hsb': function(color) { var r = color._red, g = color._green, b = color._blue, alpha = color._alpha, max = Math.max(r, g, b), min = Math.min(r, g, b), delta = max - min, hue, saturation = (max != 0) ? delta / max : 0, brightness = max; if (saturation == 0) { hue = 0; } else { var rr = (max - r) / delta, gr = (max - g) / delta, br = (max - b) / delta; hue = r == max ? br - gr : g == max ? 2 + rr - br : 4 + gr - rr; hue /= 6; if (hue < 0) hue++; } return new HSBColor(hue * 360, saturation, brightness, alpha); }, 'hsb-rgb': function(color) { var h = color._hue, s = color._saturation, b = color._brightness, a = color._alpha, f = h % 60, p = (b * (1 - s)) / 1, q = (b * (60 - s * f)) / 60, t = (b * (60 - s * (60 - f))) / 60; switch (Math.floor(h / 60)) { case 0: return new RGBColor(b, t, p, a); case 1: return new RGBColor(q, b, p, a); case 2: return new RGBColor(p, b, t, a); case 3: return new RGBColor(p, q, b, a); case 4: return new RGBColor(t, p, b, a); case 5: return new RGBColor(b, p, q, a); } }, 'rgb-gray': function(color) { // Using the standard NTSC conversion formula that is used for // calculating the effective luminance of an RGB color: // http://www.mathworks.com/support/solutions/en/data/1-1ASCU/index.html?solution=1-1ASCU return new GrayColor(1 - (color._red * 0.2989 + color._green * 0.587 + color._blue * 0.114), color._alpha ); }, 'gray-rgb': function(color) { var component = 1 - color.getGray(); return new RGBColor(component, component, component, color._alpha); }, 'hsb-gray': function(color) { var rgbColor = converters['hsb-rgb'](color); return converters['rgb-gray'](rgbColor); }, 'gray-hsb': function(color) { return new HSBColor(0, 0, 1 - color._gray, color._alpha); } }; return { convert: function(type) { return this._colorType == type ? this : converters[this._colorType + '-' + type](this); } }; }, new function() { return Base.each({ rgb: ['red', 'green', 'blue'], hsb: ['hue', 'saturation', 'brightness'], gray: ['gray'] }, function(components, colorType) { Base.each(components, function(component) { this['get' + Base.capitalize(component)] = function() { return this.convert(colorType)[component]; }; this['set' + Base.capitalize(component)] = function(value) { var color = this.convert(colorType); color[component] = value; color = color.convert(this._colorType); for (var i = 0, l = this._components.length; i < l; i++) { this[this._components[i]] = color[this._components[i]]; } }; }, this); }, { beans: true }); });