/* <copyright> This file contains proprietary software owned by Motorola Mobility, Inc.<br/> No rights, expressed or implied, whatsoever to this software are provided by Motorola Mobility, Inc. hereunder.<br/> (c) Copyright 2011 Motorola Mobility, Inc. All Rights Reserved. </copyright> */ //////////////////////////////////////////////////////////////////////// // var Montage = require("montage/core/core").Montage, Component = require("montage/ui/component").Component; //////////////////////////////////////////////////////////////////////// // exports.ColorModel = Montage.create(Component, { //////////////////////////////////////////////////////////////////// // hasTemplate: { enumerable: false, value: false }, //////////////////////////////////////////////////////////////////// //HSV Value of current color selected _gradient: { enumerable: false, value: null }, //////////////////////////////////////////////////////////////////// //HSV Value of current color selected gradient: { enumerable: true, get: function() { return this._gradient; }, set: function(value) { this._gradient = value; //Updating color selected (converting to all modes) this.updateColorSelected('gradient', value); } }, //////////////////////////////////////////////////////////////////// //HSV Value of current color selected _hsv: { enumerable: false, value: null }, //////////////////////////////////////////////////////////////////// //HSV Value of current color selected hsv: { enumerable: true, get: function() { return this._hsv; }, set: function(value) { this._hsv = value; //Updating color selected (converting to all modes) this.updateColorSelected('hsv', value); } }, //////////////////////////////////////////////////////////////////// //RGB Value of current color selected _rgb: { enumerable: false, value: null }, //////////////////////////////////////////////////////////////////// //RGB Value of current color selected rgb: { enumerable: true, get: function() { return this._rgb; }, set: function(value) { this._rgb = value; //Updating color selected (converting to all modes) this.updateColorSelected('rgb', value); } }, //////////////////////////////////////////////////////////////////// //HSL Value of current color selected _hsl: { enumerable: false, value: null }, //////////////////////////////////////////////////////////////////// //HSL Value of current color selected hsl: { enumerable: true, get: function() { return this._hsl; }, set: function(value) { this._hsl = value; //Updating color selected (converting to all modes) this.updateColorSelected('hsl', value); } }, //////////////////////////////////////////////////////////////////// //HEX Value of current color selected _hex: { numerable: false, value: null }, //////////////////////////////////////////////////////////////////// //HEX Value of current color selected hex: { enumerable: true, get: function() { return this._hex; }, set: function(value) { this._hex = value; //Updating color selected (converting to all modes) this.updateColorSelected('hex', value); } }, //////////////////////////////////////////////////////////////////// //ALPHA Value of current color selected _alpha: { enumerable: false, value: {value: 1, type: 'change', wasSetByCode: true} }, //////////////////////////////////////////////////////////////////// //ALPHA Value of current color selected alpha: { enumerable: true, get: function() { return this._alpha; }, set: function(value) { value.value = Math.ceil(value.value*100)/100; this._alpha = value; // if (this.rgb || this.hsl) { this._dispatchChangeEvent('alpha', value); } } }, //////////////////////////////////////////////////////////////////// //Input (fill or stroke) Value of current color selected _input: { enumerable: false, value: 'fill' }, //////////////////////////////////////////////////////////////////// //Input Value of current color selected input: { enumerable: true, get: function() { return this._input; }, set: function(value) { this._input = value; //Dispatching change event this._dispatchChangeEvent('input', value); } }, //////////////////////////////////////////////////////////////////// //Color mode of current color selected _mode: { enumerable: false, value: null }, //////////////////////////////////////////////////////////////////// //Color mode of current color selected mode: { enumerable: true, get: function() { return this._mode; }, set: function(value) { this._mode = value; //Dispatching change event this._dispatchChangeEvent('mode', value); } }, //////////////////////////////////////////////////////////////////// //Stroke Color Value of current color selected _stroke: { enumerable: false, value: null }, //////////////////////////////////////////////////////////////////// //Stroke Color Value of current color selected stroke: { enumerable: true, get: function() { return this._stroke; }, set: function(value) { this._stroke = value; } }, //////////////////////////////////////////////////////////////////// //Fill Color Value of current color selected _fill: { enumerable: false, value: null }, //////////////////////////////////////////////////////////////////// //Fill Color Value of current color selected fill: { enumerable: true, get: function() { return this._fill; }, set: function(value) { this._fill = value; } }, //////////////////////////////////////////////////////////////////// //History Value array of current color selected colorHistory: { enumerable: false, value: {stroke: [{m: 'rgb', c: {r: 0, g: 0, b: 0}, a: 1}, {m: 'rgb', c: {r: 0, g: 0, b: 0}, a: 1}], fill: [{m: 'rgb', c: {r: 0, g: 0, b: 0}, a: 1}, {m: 'rgb', c: {r: 0, g: 0, b: 0}, a: 1}]} }, //////////////////////////////////////////////////////////////////// //History Value array of current color selected _addColorHistory: { enumerable: true, value: function(input, mode, color, alpha) { //TODO: Add limit if (this.colorHistory[input.toLowerCase()].length > 1) { if (this.colorHistory[input.toLowerCase()][this.colorHistory[input.toLowerCase()].length-1].c !== color || this.colorHistory[input.toLowerCase()][this.colorHistory[input.toLowerCase()].length-1].a !== alpha.value) { this.colorHistory[input.toLowerCase()].push({m: mode, c: color, a: alpha.value}); } } else { this.colorHistory[input.toLowerCase()].push({m: mode, c: color, a: alpha.value}); } } }, //////////////////////////////////////////////////////////////////// // applyNoColor: { enumerable: true, value: function () { // var nocolor = {}; nocolor.wasSetByCode = true; nocolor.type = 'change'; this.updateColorSelected('nocolor', nocolor); } }, //////////////////////////////////////////////////////////////////// //Method to update color selected and convert to other modes updateColorSelected: { enumerable: true, //value: function (input, mode, color, alpha) { value: function (mode, color) { //////////////////////////////////////////////////////////// //Checking for color mode to convert colors switch (mode.toLocaleLowerCase()) { //////////////////////////////////////////////////////// case 'gradient': //Checking for match of previous gradient if (color !== this.gradient) { //Setting value and breaking out of function this.gradient = color; return; } // this._hex = '------'; this._rgb = null; this._hsv = null; this._hsl = null; break; //////////////////////////////////////////////////////// case 'rgb': //Checking for match of previous (RGB) if (color.r !== this.rgb.r && color.g !== this.rgb.g && color.b !== this.rgb.b) { //Setting value and breaking out of function this.rgb = color; return; } //Setting other color mode values this._hsv = this.rgbToHsv(color.r, color.g, color.b); this._hsl = this.rgbToHsl(color.r, color.g, color.b); this._hex = this.rgbToHex(color.r, color.g, color.b); this._gradient = null; break; //////////////////////////////////////////////////////// case 'hsv': //Checking for match of previous (HSV) if (color.h !== this.hsv.h && color.s !== this.hsv.s && color.v !== this.hsv.v) { //Setting value and breaking out of function this.hsv = color; return; } //Setting other color mode values this._rgb = this.hsvToRgb(color.h/(2*Math.PI), color.s, color.v); this._hsl = this.rgbToHsl(this.rgb.r, this.rgb.g, this.rgb.b); this._hex = this.rgbToHex(this.rgb.r, this.rgb.g, this.rgb.b); this._gradient = null; break; //////////////////////////////////////////////////////// case 'hsl': //Checking for match of previous (HSV) if (color.h !== this.hsl.h && color.s !== this.hsl.s && color.l !== this.hsl.l) { //Setting value and breaking out of function this.hsl = color; return; } //Setting other color mode values this._rgb = this.hslToRgb(color.h/360, color.s/100, color.l/100); //This is a hack to keep the values of color spectrum the same on limits (B/W) var hsvTemp = this.rgbToHsv(this.rgb.r, this.rgb.g, this.rgb.b); this._hsv = {h: (this._hsl.h/360)*(2*Math.PI), s: hsvTemp.s, v: hsvTemp.v}; this._hex = this.rgbToHex(this.rgb.r, this.rgb.g, this.rgb.b); this._gradient = null; break; //////////////////////////////////////////////////////// case 'hex': switch (color.length) { case 1: this.applyNoColor(); return; break; case 2: this.applyNoColor(); return; break; case 3: color = color[0]+color[0]+color[1]+color[1]+color[2]+color[2]; break; case 4: this.applyNoColor(); return; break; case 5: this.applyNoColor(); return; break; case 6: //Nothing break; default: this.applyNoColor(); return; break; } //Checking for match of previous (HEX) if (color !== this.hex) { //Setting value and breaking out of function this.hex = color; return; } //Setting other color mode values this._rgb = this.hexToRgb(color); this._hsv = this.rgbToHsv(this.rgb.r, this.rgb.g, this.rgb.b); this._hsl = this.rgbToHsl(this.rgb.r, this.rgb.g, this.rgb.b); this._gradient = null; break; //////////////////////////////////////////////////////// case 'nocolor': // this._hex = '------'; this._rgb = null; this._hsv = null; this._hsl = null; this._gradient = null; break; //////////////////////////////////////////////////////// default: console.log('ERROR: An error occurred in ColorManager. '+mode+' was not identified in updateColorSelected'); break; //////////////////////////////////////////////////////// } //Dispatching change event this._dispatchChangeEvent(mode, color); } }, //////////////////////////////////////////////////////////////////// //Dispatching "Change" event _dispatchChangeEvent: { //Modes: rgb, hsl, hsv, hex, alpha, input, mode value: function(mode, value) { //Creating custom event to dispatch and include values var colorEvent = document.createEvent("CustomEvent"); // if (value && value.type) { if (value.type === "change") { // colorEvent.initEvent("change", true, true); //Storing the selected color in the history array if ((mode === 'rgb' || mode === 'hsv' || mode === 'hsl' || mode === 'hex' || mode === 'nocolor' || mode === 'gradient') && this.input !== 'chip') { this._addColorHistory(this.input, mode, value, this.alpha); } else if (mode === 'alpha' && this.input !== 'chip') { var c = this.colorHistory[this.input][this.colorHistory[this.input].length-1]; this._addColorHistory(this.input, c.m, c.c, this.alpha); } } else if (value.type === "changing") { // colorEvent.initEvent("changing", true, true); } else { // colorEvent.initEvent("error", true, true); } } else { // colorEvent.initEvent("error", true, true); } //Checking for event to be color, if so, adding color data if (mode === 'rgb' || mode === 'hsv' || mode === 'hsl' || mode === 'hex' || mode === 'alpha' || mode === 'nocolor' || mode === 'gradient') { //Returning color history of input if (this.input === 'stroke') { colorEvent.history = this.colorHistory.stroke; } else if (this.input === 'fill') { colorEvent.history = this.colorHistory.fill; } else { colorEvent.history = null; } } //Also returning other color mode values if (this.input) colorEvent.input = this.input; if (this.alpha) colorEvent.alpha = this.alpha.value; if (this.hsl) colorEvent.hsla = {h: this.hsl.h, s: this.hsl.s, l: this.hsl.l, a: this.alpha.value, css: 'hsla('+this.hsl.h+', '+this.hsl.s+'%, '+this.hsl.l+'%, '+this.alpha.value+')'}; if (this.rgb) { colorEvent.rgba = {r: this.rgb.r, g: this.rgb.g, b: this.rgb.b, a: this.alpha.value, css: 'rgba('+ this.rgb.r +', '+this.rgb.g+', '+this.rgb.b+', '+this.alpha.value+')'}; colorEvent.webGlColor = [this.rgb.r/255, this.rgb.g/255, this.rgb.b/255, this.alpha.value]; } if (this.hex) colorEvent.hex = this.hex; //Standard values that apply to any event colorEvent.value = value; colorEvent.mode = mode; if (value && value.wasSetByCode) { colorEvent.wasSetByCode = value.wasSetByCode; } this.dispatchEvent(colorEvent); } }, //////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////// // COLOR CONVERTION METHODS (public) //////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////// //Convert RGB to HEX (RGB = 0 to 255) rgbToHex: { enumerable: true, value: function (r, g, b) { //Coverting color channel to Hex function getHex (c) { var filter = "0123456789ABCDEF"; c = parseInt (c, 10); if (isNaN(c)) return "00"; c = Math.max(0,Math.min(c,255)); return filter.charAt((c-c%16)/16) + filter.charAt(c%16); } //Returning HEX string return (getHex(r)+getHex(g)+getHex(b)); } }, //////////////////////////////////////////////////////////////////// //Convert HEX to RGB (no # symbol) hexToRgb: { enumerable: true, value: function (value) { //Spliting string converting to values var r = parseInt(value.substring(0,2), 16), g = parseInt(value.substring(2,4), 16), b = parseInt(value.substring(4,6), 16); //Checking for valid values if (isNaN(r)) return null; if (isNaN(g)) return null; if (isNaN(b)) return null; return {r: r, g: g, b: b}; } }, //////////////////////////////////////////////////////////////////// //Convert HSV to RGB (HSV = 0 to 1) hsvToRgb: { enumerable: true, value: function (h, s, v) { var r, g, b, i = Math.floor(h * 6), f = h * 6 - i, p = v * (1 - s), q = v * (1 - f * s), t = v * (1 - (1 - f) * s); // switch (i % 6){ case 0: r = v, g = t, b = p; break; case 1: r = q, g = v, b = p; break; case 2: r = p, g = v, b = t; break; case 3: r = p, g = q, b = v; break; case 4: r = t, g = p, b = v; break; case 5: r = v, g = p, b = q; break; } //Returning RGB object return {r: Math.round(r * 255), g: Math.round(g * 255), b: Math.round(b * 255)}; } }, //////////////////////////////////////////////////////////////////// //Convert RGB to HSV (RGB = 0 to 255) rgbToHsv: { enumerable: true, value: function (r, g, b) { //RGB covertion to percentage r = r/255, g = g/255, b = b/255; var max = Math.max(r, g, b), min = Math.min(r, g, b), h, s, v, d; h = s = v = max; d = max - min; s = max == 0 ? 0 : d / max; //TODO: Check for Hue not to change if S or V = 0 if (max == min) { h = 0; } else { 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; } //Returing HSV object return {h: h*Math.PI*2, s: s, v: v}; } }, //////////////////////////////////////////////////////////////////// //Convert RGB TO HSL (RGB = 0 to 255) rgbToHsl: { enumerable: true, value: function (r, g, b) { //RGB covertion to percentage r /= 255, g /= 255, b /= 255; var max = Math.max(r, g, b), min = Math.min(r, g, b), h, s, l, d; h = s = l = (max + min) / 2; //TODO: Check for Hue not to change if S or L = 0 if (max == min) { h = s = 0; } else { 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; } //Returing HSL object return {h: h*360, s: s*100, l: l*100}; } }, //////////////////////////////////////////////////////////////////// //Convert HSL to RGB (HSL = 0 to 1) hslToRgb: { enumerable: true, value: function (h, s, l) { var r, g, b; if (s == 0) { r = g = b = l; } else { function getRgb (p, q, t) { if(t < 0) t += 1; if(t > 1) t -= 1; if(t < 1/6) return p + (q - p) * 6 * t; if(t < 1/2) return q; if(t < 2/3) return p + (q - p) * (2/3 - t) * 6; return p; } var q = l < 0.5 ? l * (1 + s) : l + s - l * s; var p = 2 * l - q; r = getRgb(p, q, h + 1/3); g = getRgb(p, q, h); b = getRgb(p, q, h - 1/3); } //Returning RGB object return {r: Math.round(r * 255), g: Math.round(g * 255), b: Math.round(b * 255)}; } }, //////////////////////////////////////////////////////////////////// //Returns WebGL color array in [r, g, b, a] format where the values are [0,1] given a color object colorToWebGl: { enumerable: true, value: function (color) { var temp; if (color) { if(color.l !== undefined) { temp = this.hslToRgb(color.h/360, color.s/100, color.l/100); } else if (color.r !== undefined) { temp = color; } else if (color.gradientMode) { // TODO - Need to handle gradients at some point return null; } temp.a = color.a; } //WebGL uses array if (temp) { return [temp.r/255, temp.g/255, temp.b/255, temp.a]; } else { return null; } } }, //////////////////////////////////////////////////////////////////// //Returns a color object given a WebGL color array/object with gradient stops webGlToColor: { enumerable: true, value: function (c) { if(c) { if(c.gradientMode) { // Gradient var i = 0, len, css, stops = c.color, gradient; // Create the CSS string if (c.gradientMode === 'radial') { css = '-webkit-radial-gradient(center, ellipse cover'; } else { css = '-webkit-gradient(linear, left top, right top'; } //Sorting array (must be sorted for radial gradients, at least in Chrome stops.sort(function(a,b){return a.position - b.position}); //Looping through stops in gradient to create CSS len = stops.length; for (i=0; i < len; i++) { //Adding to CSS String if (c.gradientMode === 'radial' && stops[i].value) { css += ', '+stops[i].value.css+' '+stops[i].position+'% '; } else if (stops[i].value){ css += ', color-stop('+stops[i].position+'%,'+stops[i].value.css+')'; } } //Closing the CSS strings css += ')'; gradient = {stops: c.color, mode: c.gradientMode, gradientMode: c.gradientMode, css: css}; return {mode: 'gradient', value: gradient, color: gradient}; } else if(c.length === 4) { // CSS return this.application.ninja.colorController.getColorObjFromCss('rgba(' + c[0]*255 + ', ' + c[1]*255 + ', ' + c[2]*255 + ', ' + c[3] +')'); } } return null; } } //////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////// });