/*! * Canvas - Context2d * Copyright (c) 2010 LearnBoost * MIT Licensed */ /** * Module dependencies. */ var canvas = require('../build/default/canvas') , Context2d = canvas.CanvasRenderingContext2d , CanvasGradient = canvas.CanvasGradient , ImageData = canvas.ImageData , PixelArray = canvas.CanvasPixelArray , colors = require('./colors'); /** * Export `Context2d` as the module. */ var Context2d = exports = module.exports = Context2d; /** * Cache color string RGBA values. */ var cache = {}; /** * Text baselines. */ var baselines = ['alphabetic', 'top', 'bottom', 'middle', 'ideographic', 'hanging']; /** * Font RegExp helpers. */ var weights = 'normal|bold|bolder|lighter|[1-9]00' , styles = 'normal|italic|oblique' , units = 'px|pt|pc|in|cm|mm|%' , string = '\'([^\']+)\'|"([^"]+)"|[\\w-]+'; /** * Font parser RegExp; */ var fontre = new RegExp('^ *' + '(?:(' + weights + ') *)?' + '(?:(' + styles + ') *)?' + '([\\d\\.]+)(' + units + ') *' + '((?:' + string + ')( *, *(?:' + string + '))*)' ); /** * Return a function used to normalize an RGBA color `prop` * or the pattern previously set. * * @param {String} prop * @return {Function} * @api private */ function getter(prop) { return function(){ var val = this[prop]; if (val instanceof CanvasGradient) return val; if (1 == val[3]) { return '#' + val[0].toString(16) + val[1].toString(16) + val[2].toString(16); } else { return 'rgba(' + val[0] + ', ' + val[1] + ', ' + val[2] + ', ' + val[3] + ')'; } } } /** * Parse font `str`. * * @param {String} str * @return {Object} * @api private */ var parseFont = exports.parseFont = function(str){ var font = {} , captures = fontre.exec(str); // Invalid if (!captures) return; // Populate font object font.weight = captures[1] || 'normal'; font.style = captures[2] || 'normal'; font.size = parseFloat(captures[3]); font.unit = captures[4]; font.family = captures[5]; return font; }; /** * Parse the given color `str`. * * Current supports: * * - #nnn * - #nnnnnn * - rgb(r,g,b) * - rgba(r,g,b,a) * - color * * Examples * * - #fff * - #FFF * - #FFFFFF * - rgb(255,255,5) * - rgba(255,255,5,.8) * - rgba(255,255,5,0.8) * - hsl(300,100%,50%) * - hsla(300,100%,50%,0.5) * - white * - red * * @param {String} str * @return {Array} * @api private */ var parseColor = exports.parseColor = function(str){ if (cache[str]) return cache[str]; str = colors[str] || String(str); // RGBA if (0 == str.indexOf('rgba')) { var captures = /rgba\((\d{1,3}) *, *(\d{1,3}) *, *(\d{1,3}) *, *(\d+\.\d+|\.\d+|\d+) *\)/.exec(str); if (!captures) return; return cache[str] = [ parseInt(captures[1], 10) , parseInt(captures[2], 10) , parseInt(captures[3], 10) , parseFloat(captures[4], 10) ]; // RGB } else if (0 == str.indexOf('rgb')) { var captures = /rgb\((\d{1,3}) *, *(\d{1,3}) *, *(\d{1,3}) *\)/.exec(str); if (!captures) return; return cache[str] = [ parseInt(captures[1], 10) , parseInt(captures[2], 10) , parseInt(captures[3], 10) , 1 ]; // #RRGGBB } else if ('#' == str[0] && str.length > 4) { var captures = /#([a-fA-F\d]{2})([a-fA-F\d]{2})([a-fA-F\d]{2})/.exec(str); if (!captures) return; return cache[str] = [ parseInt(captures[1], 16) , parseInt(captures[2], 16) , parseInt(captures[3], 16) , 1 ]; // #RGB } else if ('#' == str[0]) { var captures = /#([a-fA-F\d])([a-fA-F\d])([a-fA-F\d])/.exec(str); if (!captures) return; return cache[str] = [ parseInt(captures[1] + captures[1], 16) , parseInt(captures[2] + captures[2], 16) , parseInt(captures[3] + captures[3], 16) , 1 ]; // HSLA } else if (0 == str.indexOf('hsla')) { var captures = /hsla\((\d+\.\d+|\.\d+|\d+) *, *(\d+\.\d+|\.\d+|\d+)%? *, *(\d+\.\d+|\.\d+|\d+)%? *, *(\d+\.\d+|\.\d+|\d+) *\)/.exec(str); if (!captures) return; return cache[str] = hsl2rgb( (((parseFloat(captures[1], 10) % 360) + 360) % 360) / 360 , parseFloat(captures[2], 10) / 100.0 , parseFloat(captures[3], 10) / 100.0 , parseFloat(captures[4], 10) ); // HSL } else if (0 == str.indexOf('hsl')) { var captures = /hsl\((\d+\.\d+|\.\d+|\d+) *, *(\d+\.\d+|\.\d+|\d+)%? *, *(\d+\.\d+|\.\d+|\d+)%? *\)/.exec(str); if (!captures) return; return cache[str] = hsl2rgb( (((parseFloat(captures[1], 10) % 360) + 360) % 360) / 360 , parseFloat(captures[2], 10) / 100.0 , parseFloat(captures[3], 10) / 100.0 , 1 ); } }; /** * HSLA -> RGB. * * @param {Number} h * @param {Number} s * @param {Number} l * @param {Number} a * @return {Array} * @api private */ function hsl2rgb(h,s,l,a) { var m2 = l <= 0.5 ? l*(s+1) : l+s-l*s , m1 = l*2-m2; return [ Math.round(hue2rgb(m1, m2, h+1/3) * 255) , Math.round(hue2rgb(m1, m2, h) * 255) , Math.round(hue2rgb(m1, m2, h-1/3) * 255) , a ]; } /** * Hue to RGB * * @param {Number} m1 * @param {Number} m2 * @param {Number} h * @return {Number} * @api private */ function hue2rgb(m1,m2,h) { if (h<0) h = h+1; if (h>1) h = h-1; if (h*6<1) return m1+(m2-m1)*h*6; if (h*2<1) return m2; if (h*3<2) return m1+(m2-m1)*(2/3-h)*6; return m1; } /** * Add `color` stop at the given `offset`. * * @param {Number} offset * @param {String} color * @api public */ CanvasGradient.prototype.addColorStop = function(offset, color){ var rgba; if (rgba = parseColor(color)) { this.addColorStopRGBA( offset , rgba[0] , rgba[1] , rgba[2] , rgba[3]); } }; /** * Create a linear gradient at the given point `(x0, y0)` and `(x1, y1)`. * * @param {Number} x0 * @param {Number} y0 * @param {Number} x1 * @param {Number} y1 * @return {CanvasGradient} * @api public */ Context2d.prototype.createLinearGradient = function(x0, y0, x1, y1){ return new CanvasGradient(x0, y0, x1, y1); }; /** * Create a radial gradient at the given point `(x0, y0)` and `(x1, y1)` * and radius `r0` and `r1`. * * @param {Number} x0 * @param {Number} y0 * @param {Number} r0 * @param {Number} x1 * @param {Number} y1 * @param {Number} r1 * @return {CanvasGradient} * @api public */ Context2d.prototype.createRadialGradient = function(x0, y0, r0, x1, y1, r1){ return new CanvasGradient(x0, y0, r0, x1, y1, r1); }; /** * Reset transform matrix to identity, then apply the given args. * * @param {...} * @api public */ Context2d.prototype.setTransform = function(){ this.resetTransform(); this.transform.apply(this, arguments); }; /** * Set the fill style with the given css color string. * * @see exports.parseColor() * @api public */ Context2d.prototype.__defineSetter__('fillStyle', function(val){ if (val instanceof CanvasGradient) { this.lastFillStyle = val; this.setFillPattern(val); } else if ('string' == typeof val) { var rgba; if (rgba = parseColor(val)) { this.lastFillStyle = rgba; this.setFillRGBA( rgba[0] , rgba[1] , rgba[2] , rgba[3]); } } }); /** * Get the current fill style string. * * @api public */ Context2d.prototype.__defineGetter__('fillStyle', getter('lastFillStyle')); /** * Set the stroke style with the given css color string. * * @see exports.parseColor() * @api public */ Context2d.prototype.__defineSetter__('strokeStyle', function(val){ if (val instanceof CanvasGradient) { this.lastStrokeStyle = val; this.setStrokePattern(val); } else if ('string' == typeof val) { var rgba; if (rgba = parseColor(val)) { this.lastStrokeStyle = rgba; this.setStrokeRGBA( rgba[0] , rgba[1] , rgba[2] , rgba[3]); } } }); /** * Get the current stroke style string. * * @api public */ Context2d.prototype.__defineGetter__('strokeStyle', getter('lastStrokeStyle')); /** * Set the shadow color with the given css color string. * * @see exports.parseColor() * @api public */ Context2d.prototype.__defineSetter__('shadowColor', function(val){ if ('string' == typeof val) { var rgba; if (rgba = parseColor(val)) { this.lastShadowColor = rgba; this.setShadowRGBA( rgba[0] , rgba[1] , rgba[2] , rgba[3]); } } }); /** * Get the current shadow color string. * * @api public */ Context2d.prototype.__defineGetter__('shadowColor', getter('lastShadowColor')); /** * Set font. * * @see exports.parseFont() * @api public */ Context2d.prototype.__defineSetter__('font', function(val){ if ('string' == typeof val) { var font; if (font = cache[val] || parseFont(val)) { this.lastFontString = val; // TODO: dpi // TODO: remaining unit conversion switch (font.unit) { case 'pt': font.size /= .75; break; case 'in': font.size *= 96; break; case 'mm': font.size *= 96.0 / 25.4; break; case 'cm': font.size *= 96.0 / 2.54; break; } // Cache font object cache[val] = font; // Set font this.setFont( font.weight , font.style , font.size , font.unit , font.family); } } }); /** * Get the current font. * * @api public */ Context2d.prototype.__defineGetter__('font', function(){ return this.lastFontString || '10px sans-serif'; }); /** * Set text baseline. * * @api public */ Context2d.prototype.__defineSetter__('textBaseline', function(val){ var n = baselines.indexOf(val); if (~n) { this.lastBaseline = val; this.setTextBaseline(n); } }); /** * Get the current baseline setting. * * @api public */ Context2d.prototype.__defineGetter__('textAlign', function(){ return this.lastBaseline || 'alphabetic'; }); /** * Set text alignment. * * @api public */ Context2d.prototype.__defineSetter__('textAlign', function(val){ switch (val) { case 'center': this.setTextAlignment(0); this.lastTextAlignment = val; break; case 'left': case 'start': this.setTextAlignment(-1); this.lastTextAlignment = val; break; case 'right': case 'end': this.setTextAlignment(1); this.lastTextAlignment = val; break; } }); /** * Get the current font. * * @see exports.parseFont() * @api public */ Context2d.prototype.__defineGetter__('textAlign', function(){ return this.lastTextAlignment || 'start'; }); /** * Get `ImageData` with the given rect. * * @param {Number} x * @param {Number} y * @param {Number} width * @param {Number} height * @return {ImageData} * @api public */ Context2d.prototype.getImageData = function(x, y, width, height){ var arr = new PixelArray(this.canvas, x, y, width, height); return new ImageData(arr); }; /** * Create `ImageData` with the given dimensions or * `ImageData` instance for dimensions. * * @param {Number|ImageData} width * @param {Number} height * @return {ImageData} * @api public */ Context2d.prototype.createImageData = function(width, height){ if (width instanceof ImageData) { height = width.height; width = width.width; } return new ImageData(new PixelArray(width, height)); };