/* * Paper.js - The Swiss Army Knife of Vector Graphics Scripting. * http://paperjs.org/ * * Copyright (c) 2011 - 2013, Juerg Lehni & Jonathan Puckey * http://lehni.org/ & http://jonathanpuckey.com/ * * Distributed under the MIT license. See LICENSE file for details. * * All rights reserved. */ /** * @name Shape * * @class * * @extends Item */ var Shape = Item.extend(/** @lends Shape# */{ _class: 'Shape', _transformContent: false, initialize: function Shape(type, point, size) { Item.call(this, point); this._type = type; this._size = size; }, _draw: function(ctx, param) { var style = this._style, size = this._size, width = size.width, height = size.height, fillColor = style.getFillColor(), strokeColor = style.getStrokeColor(); if (fillColor || strokeColor || param.clip) { ctx.beginPath(); switch (this._type) { case 'rect': ctx.rect(-width / 2, -height / 2, width, height); break; case 'circle': // Average half of width & height for radius... ctx.arc(0, 0, (width + height) / 4, 0, Math.PI * 2, true); break; case 'ellipse': // Use four bezier curves and KAPPA value to aproximate ellipse var mx = width / 2, my = height / 2, kappa = Numerical.KAPPA, cx = mx * kappa, cy = my * kappa; ctx.moveTo(-mx, 0); ctx.bezierCurveTo(-mx, -cy, -cx, -my, 0, -my); ctx.bezierCurveTo(cx, -my, mx, -cy, mx, 0); ctx.bezierCurveTo(mx, cy, cx, my, 0, my); ctx.bezierCurveTo(-cx, my, -mx, cy, -mx, 0); break; } } if (!param.clip && (fillColor || strokeColor)) { this._setStyles(ctx); if (fillColor) ctx.fill(); if (strokeColor) ctx.stroke(); } }, _canComposite: function() { // A path with only a fill or a stroke can be directly blended, but if // it has both, it needs to be drawn into a separate canvas first. return !(this.hasFill() && this.hasStroke()); }, _getBounds: function(getter, matrix) { var rect = new Rectangle(this._size).setCenter(0, 0); if (getter !== 'getBounds' && this.hasStroke()) rect = rect.expand(this.getStrokeWidth()); return matrix ? matrix._transformBounds(rect) : rect; }, _contains: function _contains(point) { switch (this._type) { case 'rect': return _contains.base.call(this, point); case 'circle': case 'ellipse': return point.divide(this._size).getLength() <= 0.5; } }, _hitTest: function _hitTest(point, options) { if (this.hasStroke()) { var type = this._type, strokeWidth = this.getStrokeWidth(); switch (type) { case 'rect': var rect = new Rectangle(this._size).setCenter(0, 0), outer = rect.expand(strokeWidth), inner = rect.expand(-strokeWidth); if (outer._containsPoint(point) && !inner._containsPoint(point)) return new HitResult('stroke', this); break; case 'circle': case 'ellipse': var size = this._size, width = size.width, height = size.height, radius; if (type === 'ellipse') { // Calculate ellipse radius at angle var angle = point.getAngleInRadians(), x = width * Math.sin(angle), y = height * Math.cos(angle); radius = width * height / (2 * Math.sqrt(x * x + y * y)); } else { // Average half of width & height for radius... radius = (width + height) / 4; } if (2 * Math.abs(point.getLength() - radius) <= strokeWidth) return new HitResult('stroke', this); break; } } return _hitTest.base.apply(this, arguments); }, statics: new function() { function createShape(type, point, size, args) { var shape = new Shape(type, point, size), named = Base.getNamed(args); if (named) shape._set(named); return shape; } return { Circle: function(/* center, radius */) { var center = Point.readNamed(arguments, 'center'), radius = Base.readNamed(arguments, 'radius'); return createShape('circle', center, new Size(radius * 2), arguments); }, Rectangle: function(/* rectangle */) { var rect = Rectangle.readNamed(arguments, 'rectangle'); return createShape('rect', rect.getCenter(true), rect.getSize(true), arguments); }, Ellipse: function(/* rectangle */) { var rect = Rectangle.readNamed(arguments, 'rectangle'); return createShape('ellipse', rect.getCenter(true), rect.getSize(true), arguments); } }; } });