diff --git a/dist/pixi.js b/dist/pixi.js deleted file mode 100644 index 2e6b3f4..0000000 --- a/dist/pixi.js +++ /dev/null @@ -1,19896 +0,0 @@ -!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.PIXI=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o", - "Richard Davey " - ], - "main": "./src/index.js", - "homepage": "http://goodboydigital.com/", - "bugs": "https://github.com/GoodBoyDigital/pixi.js/issues", - "license": "MIT", - "repository": { - "type": "git", - "url": "https://github.com/GoodBoyDigital/pixi.js.git" - }, - "scripts": { - "test": "gulp test", - "docs": "./node_modules/.bin/jsdoc -c ./gulp/util/jsdoc.conf.json" - }, - "devDependencies": { - "browserify": "^8.0.2", - "chai": "^1.10.0", - "del": "^1.1.0", - "gulp": "^3.8.10", - "gulp-jshint": "^1.9.0", - "gulp-plumber": "^0.6.6", - "gulp-rename": "^1.2.0", - "gulp-uglify": "^1.0.2", - "gulp-util": "^3.0.1", - "ink-docstrap": "^0.4.12", - "jsdoc": "^3.3.0-alpha13", - "jshint-summary": "^0.4.0", - "karma": "^0.12.28", - "karma-firefox-launcher": "^0.1.0", - "karma-mocha": "^0.1.10", - "karma-spec-reporter": "^0.0.16", - "mocha": "^2.1.0", - "require-dir": "^0.1.0", - "run-sequence": "^1.0.2", - "vinyl-buffer": "^1.0.0", - "vinyl-source-stream": "^1.0.0", - "watchify": "^2.2.1" - }, - "dependencies": { - "webgl-enabled": "^1.0.2" - } -} - -},{}],4:[function(require,module,exports){ -/** - * Constant values used in pixi - * - * @mixin const - */ -module.exports = { - /** - * Constant to identify the WEBGL Renderer Type - * - * @static - * @constant - * @property {number} WEBGL_RENDERER - */ - WEBGL_RENDERER: 1, - - /** - * Constant to identify the CANVAS Renderer Type - * - * @static - * @constant - * @property {number} CANVAS_RENDERER - */ - CANVAS_RENDERER: 2, - - /** - * String of the current PIXI version - * - * @static - * @constant - * @property {string} VERSION - */ - VERSION: require('../../package.json').version, - - /** - * Various blend modes supported by PIXI. IMPORTANT - The WebGL renderer only supports - * the NORMAL, ADD, MULTIPLY and SCREEN blend modes. Anything else will silently act like - * NORMAL. - * - * @static - * @constant - * @property {object} blendModes - * @property {number} blendModes.NORMAL - * @property {number} blendModes.ADD - * @property {number} blendModes.MULTIPLY - * @property {number} blendModes.SCREEN - * @property {number} blendModes.OVERLAY - * @property {number} blendModes.DARKEN - * @property {number} blendModes.LIGHTEN - * @property {number} blendModes.COLOR_DODGE - * @property {number} blendModes.COLOR_BURN - * @property {number} blendModes.HARD_LIGHT - * @property {number} blendModes.SOFT_LIGHT - * @property {number} blendModes.DIFFERENCE - * @property {number} blendModes.EXCLUSION - * @property {number} blendModes.HUE - * @property {number} blendModes.SATURATION - * @property {number} blendModes.COLOR - * @property {number} blendModes.LUMINOSITY - */ - blendModes: { - NORMAL: 0, - ADD: 1, - MULTIPLY: 2, - SCREEN: 3, - OVERLAY: 4, - DARKEN: 5, - LIGHTEN: 6, - COLOR_DODGE: 7, - COLOR_BURN: 8, - HARD_LIGHT: 9, - SOFT_LIGHT: 10, - DIFFERENCE: 11, - EXCLUSION: 12, - HUE: 13, - SATURATION: 14, - COLOR: 15, - LUMINOSITY: 16 - }, - - /** - * The scale modes that are supported by pixi. - * - * The DEFAULT scale mode affects the default scaling mode of future operations. - * It can be re-assigned to either LINEAR or NEAREST, depending upon suitability. - * - * @static - * @constant - * @property {object} scaleModes - * @property {number} scaleModes.DEFAULT=LINEAR - * @property {number} scaleModes.LINEAR Smooth scaling - * @property {number} scaleModes.NEAREST Pixelating scaling - */ - scaleModes: { - DEFAULT: 0, - LINEAR: 0, - NEAREST: 1 - }, - - /** - * The prefix that denotes a URL is for a retina asset - * - * @static - * @constant - * @property {string} RETINA_PREFIX - */ - RETINA_PREFIX: '@2x', - - /** - * The default render options if none are supplied to {@link PIXI.WebGLRenderer} - * or {@link PIXI.CanvasRenderer}. - * - * @static - * @constant - * @property {object} defaultRenderOptions - * @property {HTMLCanvasElement} defaultRenderOptions.view=null - * @property {boolean} defaultRenderOptions.transparent=false - * @property {boolean} defaultRenderOptions.antialias=false - * @property {boolean} defaultRenderOptions.preserveDrawingBuffer=false - * @property {number} defaultRenderOptions.resolution=1 - * @property {number} defaultRenderOptions.backgroundColor=0x000000 - * @property {boolean} defaultRenderOptions.clearBeforeRender=true - * @property {boolean} defaultRenderOptions.autoResize=false - */ - defaultRenderOptions: { - view: null, - resolution: 1, - antialias: false, - autoResize: false, - transparent: false, - backgroundColor: 0x000000, - clearBeforeRender: true, - preserveDrawingBuffer: false - }, - - /** - * Constants that identify shapes, mainly to prevent `instanceof` calls. - * - * @static - * @constant - * @property {object} SHAPES - * @property {object} SHAPES.POLY=0 - * @property {object} SHAPES.RECT=1 - * @property {object} SHAPES.CIRC=2 - * @property {object} SHAPES.ELIP=3 - * @property {object} SHAPES.RREC=4 - */ - SHAPES: { - POLY: 0, - RECT: 1, - CIRC: 2, - ELIP: 3, - RREC: 4 - } -}; - -},{"../../package.json":3}],5:[function(require,module,exports){ -var math = require('../math'); - -/** - * The base class for all objects that are rendered on the screen. - * This is an abstract class and should not be used on its own rather it should be extended. - * - * @class - * @namespace PIXI - */ -function DisplayObject() { - /** - * The coordinate of the object relative to the local coordinates of the parent. - * - * @member {Point} - */ - this.position = new math.Point(); - - /** - * The scale factor of the object. - * - * @member {Point} - */ - this.scale = new math.Point(1, 1); - - /** - * The pivot point of the displayObject that it rotates around - * - * @member {Point} - */ - this.pivot = new math.Point(0, 0); - - /** - * The rotation of the object in radians. - * - * @member {number} - */ - this.rotation = 0; - - /** - * The opacity of the object. - * - * @member {number} - */ - this.alpha = 1; - - /** - * The visibility of the object. If false the object will not be drawn, and - * the updateTransform function will not be called. - * - * @member {boolean} - */ - this.visible = true; - - /** - * Can this object be rendered, if false the object will not be drawn but the updateTransform - * methods will still be called. - * - * @member {boolean} - */ - this.renderable = false; - - /** - * The display object container that contains this display object. - * - * @member {DisplayObjectContainer} - * @readOnly - */ - this.parent = null; - - /** - * The multiplied alpha of the displayObject - * - * @member {number} - * @readOnly - */ - this.worldAlpha = 1; - - /** - * Current transform of the object based on world (parent) factors - * - * @member {Matrix} - * @readOnly - */ - this.worldTransform = new math.Matrix(); - - /** - * The area the filter is applied to. This is used as more of an optimisation - * rather than figuring out the dimensions of the displayObject each frame you can set this rectangle - * - * @member {Rectangle} - */ - this.filterArea = null; - - /** - * cached sin rotation - * - * @member {number} - * @private - */ - this._sr = 0; - - /** - * cached cos rotation - * - * @member {number} - * @private - */ - this._cr = 1; - - /** - * The original, cached bounds of the object - * - * @member {Rectangle} - * @private - */ - this._bounds = new math.Rectangle(0, 0, 1, 1); - - /** - * The most up-to-date bounds of the object - * - * @member {Rectangle} - * @private - */ - this._currentBounds = null; - - /** - * The original, cached mask of the object - * - * @member {Rectangle} - * @private - */ - this._mask = null; - - /** - * Cached internal flag. - * - * @member {boolean} - * @private - */ - this._cacheIsDirty = false; -} - -// constructor -DisplayObject.prototype.constructor = DisplayObject; -module.exports = DisplayObject; - -Object.defineProperties(DisplayObject.prototype, { - /** - * The position of the displayObject on the x axis relative to the local coordinates of the parent. - * - * @member {number} - * @memberof DisplayObject# - */ - x: { - get: function () { - return this.position.x; - }, - set: function (value) { - this.position.x = value; - } - }, - - /** - * The position of the displayObject on the y axis relative to the local coordinates of the parent. - * - * @member {number} - * @memberof DisplayObject# - */ - y: { - get: function () { - return this.position.y; - }, - set: function (value) { - this.position.y = value; - } - }, - - /** - * Indicates if the sprite is globally visible. - * - * @member {boolean} - * @memberof DisplayObject# - * @readonly - */ - worldVisible: { - get: function () { - var item = this; - - do { - if (!item.visible) { - return false; - } - - item = item.parent; - } while(item); - - return true; - } - }, - - /** - * Sets a mask for the displayObject. A mask is an object that limits the visibility of an object to the shape of the mask applied to it. - * In PIXI a regular mask must be a PIXI.Graphics object. This allows for much faster masking in canvas as it utilises shape clipping. - * To remove a mask, set this property to null. - * - * @member {Graphics} - * @memberof DisplayObject# - */ - mask: { - get: function () { - return this._mask; - }, - set: function (value) { - if (this._mask) { - this._mask.isMask = false; - } - - this._mask = value; - - if (this._mask) { - this._mask.isMask = true; - } - } - }, - - /** - * Sets the filters for the displayObject. - * * IMPORTANT: This is a webGL only feature and will be ignored by the canvas renderer. - * To remove filters simply set this property to 'null' - * - * @member {Filter[]} - * @memberof DisplayObject# - */ - filters: { - get: function () { - return this._filters; - }, - set: function (value) { - if (value) { - // now put all the passes in one place.. - var passes = []; - - for (var i = 0; i < value.length; i++) { - var filterPasses = value[i].passes; - - for (var j = 0; j < filterPasses.length; j++) { - passes.push(filterPasses[j]); - } - } - - // TODO change this as it is legacy - this._filterBlock = { target: this, filterPasses: passes }; - } - - this._filters = value; - } - } -}); - -/* - * Updates the object transform for rendering - * - * TODO - Optimization pass! - * - * @private - */ -DisplayObject.prototype.updateTransform = function () { - if (!this.parent) { - return; - } - - // create some matrix refs for easy access - var pt = this.parent.worldTransform; - var wt = this.worldTransform; - - // temporary matrix variables - var a, b, c, d, tx, ty; - - // so if rotation is between 0 then we can simplify the multiplication process.. - if (this.rotation % math.PI_2) { - // check to see if the rotation is the same as the previous render. This means we only need to use sin and cos when rotation actually changes - if (this.rotation !== this.rotationCache) { - this.rotationCache = this.rotation; - this._sr = Math.sin(this.rotation); - this._cr = Math.cos(this.rotation); - } - - // get the matrix values of the displayobject based on its transform properties.. - a = this._cr * this.scale.x; - b = this._sr * this.scale.x; - c = -this._sr * this.scale.y; - d = this._cr * this.scale.y; - tx = this.position.x; - ty = this.position.y; - - // check for pivot.. not often used so geared towards that fact! - if (this.pivot.x || this.pivot.y) { - tx -= this.pivot.x * a + this.pivot.y * c; - ty -= this.pivot.x * b + this.pivot.y * d; - } - - // concat the parent matrix with the objects transform. - wt.a = a * pt.a + b * pt.c; - wt.b = a * pt.b + b * pt.d; - wt.c = c * pt.a + d * pt.c; - wt.d = c * pt.b + d * pt.d; - wt.tx = tx * pt.a + ty * pt.c + pt.tx; - wt.ty = tx * pt.b + ty * pt.d + pt.ty; - } - else { - // lets do the fast version as we know there is no rotation.. - a = this.scale.x; - d = this.scale.y; - - tx = this.position.x - this.pivot.x * a; - ty = this.position.y - this.pivot.y * d; - - wt.a = a * pt.a; - wt.b = a * pt.b; - wt.c = d * pt.c; - wt.d = d * pt.d; - wt.tx = tx * pt.a + ty * pt.c + pt.tx; - wt.ty = tx * pt.b + ty * pt.d + pt.ty; - } - - // multiply the alphas.. - this.worldAlpha = this.alpha * this.parent.worldAlpha; -}; - -// performance increase to avoid using call.. (10x faster) -DisplayObject.prototype.displayObjectUpdateTransform = DisplayObject.prototype.updateTransform; - -/** - * Retrieves the bounds of the displayObject as a rectangle object - * - * @param matrix {Matrix} - * @return {Rectangle} the rectangular bounding area - */ -DisplayObject.prototype.getBounds = function (/* matrix */) { - return math.Rectangle.EMPTY; -}; - -/** - * Retrieves the local bounds of the displayObject as a rectangle object - * - * @return {Rectangle} the rectangular bounding area - */ -DisplayObject.prototype.getLocalBounds = function () { - return this.getBounds(math.Matrix.IDENTITY); -}; - -/** - * Calculates the global position of the display object - * - * @param position {Point} The world origin to calculate from - * @return {Point} A point object representing the position of this object - */ -DisplayObject.prototype.toGlobal = function (position) { - // don't need to u[date the lot - this.displayObjectUpdateTransform(); - return this.worldTransform.apply(position); -}; - -/** - * Calculates the local position of the display object relative to another point - * - * @param position {Point} The world origin to calculate from - * @param [from] {DisplayObject} The DisplayObject to calculate the global position from - * @return {Point} A point object representing the position of this object - */ -DisplayObject.prototype.toLocal = function (position, from) { - if (from) { - position = from.toGlobal(position); - } - - // don't need to update the lot - this.displayObjectUpdateTransform(); - return this.worldTransform.applyInverse(position); -}; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} The renderer - * @private - */ -DisplayObject.prototype.renderWebGL = function (/* renderer */) { - // OVERWRITE; -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} The renderer - * @private - */ -DisplayObject.prototype.renderCanvas = function (/* renderer */) { - // OVERWRITE; -}; - -},{"../math":12}],6:[function(require,module,exports){ -var math = require('../math'), - DisplayObject = require('./DisplayObject'), - RenderTexture = require('../textures/RenderTexture'), - // Sprite = require('./Sprite'), - _tempMatrix = new math.Matrix(); - -/** - * A DisplayObjectContainer represents a collection of display objects. - * It is the base class of all display objects that act as a container for other objects. - * - * @class - * @extends DisplayObject - * @namespace PIXI - */ -function DisplayObjectContainer() { - DisplayObject.call(this); - - /** - * The array of children of this container. - * - * @member {DisplayObject[]} - * @readonly - */ - this.children = []; - - /** - * Cached internal flag. - * - * @member {boolean} - * @private - */ - this._cacheAsBitmap = false; - - this._cachedSprite = null; -} - -// constructor -DisplayObjectContainer.prototype = Object.create(DisplayObject.prototype); -DisplayObjectContainer.prototype.constructor = DisplayObjectContainer; -module.exports = DisplayObjectContainer; - -Object.defineProperties(DisplayObjectContainer.prototype, { - /** - * The width of the displayObjectContainer, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof DisplayObjectContainer# - */ - width: { - get: function () { - return this.scale.x * this.getLocalBounds().width; - }, - set: function (value) { - - var width = this.getLocalBounds().width; - - if(width !== 0) { - this.scale.x = value / width; - } - else { - this.scale.x = 1; - } - - - this._width = value; - } - }, - - /** - * The height of the displayObjectContainer, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof DisplayObjectContainer# - */ - height: { - get: function () { - return this.scale.y * this.getLocalBounds().height; - }, - set: function (value) { - - var height = this.getLocalBounds().height; - - if (height !== 0) { - this.scale.y = value / height ; - } - else { - this.scale.y = 1; - } - - this._height = value; - } - }, - - /** - * Set if this display object is cached as a bitmap. - * This basically takes a snap shot of the display object as it is at that moment. It can provide a performance benefit for complex static displayObjects. - * To remove simply set this property to 'null' - * - * @member {boolean} - * @memberof DisplayObject# - */ - cacheAsBitmap: { - get: function () { - return this._cacheAsBitmap; - }, - set: function (value) { - if (this._cacheAsBitmap === value) { - return; - } - - if (value) { - this._generateCachedSprite(); - } - else { - this._destroyCachedSprite(); - } - - this._cacheAsBitmap = value; - } - } -}); - -/** - * Adds a child to the container. - * - * @param child {DisplayObject} The DisplayObject to add to the container - * @return {DisplayObject} The child that was added. - */ -DisplayObjectContainer.prototype.addChild = function (child) { - return this.addChildAt(child, this.children.length); -}; - -/** - * Adds a child to the container at a specified index. If the index is out of bounds an error will be thrown - * - * @param child {DisplayObject} The child to add - * @param index {Number} The index to place the child in - * @return {DisplayObject} The child that was added. - */ -DisplayObjectContainer.prototype.addChildAt = function (child, index) { - // prevent adding self as child - if (child === this) { - return; - } - - if (index >= 0 && index <= this.children.length) { - if (child.parent) { - child.parent.removeChild(child); - } - - child.parent = this; - - this.children.splice(index, 0, child); - - return child; - } - else { - throw new Error(child + 'addChildAt: The index '+ index +' supplied is out of bounds ' + this.children.length); - } -}; - -/** - * Swaps the position of 2 Display Objects within this container. - * - * @param child {DisplayObject} - * @param child2 {DisplayObject} - */ -DisplayObjectContainer.prototype.swapChildren = function (child, child2) { - if (child === child2) { - return; - } - - var index1 = this.getChildIndex(child); - var index2 = this.getChildIndex(child2); - - if (index1 < 0 || index2 < 0) { - throw new Error('swapChildren: Both the supplied DisplayObjects must be a child of the caller.'); - } - - this.children[index1] = child2; - this.children[index2] = child; -}; - -/** - * Returns the index position of a child DisplayObject instance - * - * @param child {DisplayObject} The DisplayObject instance to identify - * @return {Number} The index position of the child display object to identify - */ -DisplayObjectContainer.prototype.getChildIndex = function (child) { - var index = this.children.indexOf(child); - - if (index === -1) { - throw new Error('The supplied DisplayObject must be a child of the caller'); - } - - return index; -}; - -/** - * Changes the position of an existing child in the display object container - * - * @param child {DisplayObject} The child DisplayObject instance for which you want to change the index number - * @param index {Number} The resulting index number for the child display object - */ -DisplayObjectContainer.prototype.setChildIndex = function (child, index) { - if (index < 0 || index >= this.children.length) { - throw new Error('The supplied index is out of bounds'); - } - - var currentIndex = this.getChildIndex(child); - - this.children.splice(currentIndex, 1); //remove from old position - this.children.splice(index, 0, child); //add at new position -}; - -/** - * Returns the child at the specified index - * - * @param index {Number} The index to get the child from - * @return {DisplayObject} The child at the given index, if any. - */ -DisplayObjectContainer.prototype.getChildAt = function (index) { - if (index < 0 || index >= this.children.length) { - throw new Error('getChildAt: Supplied index ' + index + ' does not exist in the child list, or the supplied DisplayObject must be a child of the caller'); - } - - return this.children[index]; -}; - -/** - * Removes a child from the container. - * - * @param child {DisplayObject} The DisplayObject to remove - * @return {DisplayObject} The child that was removed. - */ -DisplayObjectContainer.prototype.removeChild = function (child) { - var index = this.children.indexOf(child); - - if (index === -1) { - return; - } - - return this.removeChildAt(index); -}; - -/** - * Removes a child from the specified index position. - * - * @param index {Number} The index to get the child from - * @return {DisplayObject} The child that was removed. - */ -DisplayObjectContainer.prototype.removeChildAt = function (index) { - var child = this.getChildAt(index); - - child.parent = null; - this.children.splice(index, 1); - - return child; -}; - -/** - * Removes all children from this container that are within the begin and end indexes. - * - * @param beginIndex {Number} The beginning position. Default value is 0. - * @param endIndex {Number} The ending position. Default value is size of the container. - */ -DisplayObjectContainer.prototype.removeChildren = function (beginIndex, endIndex) { - var begin = beginIndex || 0; - var end = typeof endIndex === 'number' ? endIndex : this.children.length; - var range = end - begin; - - if (range > 0 && range <= end) { - var removed = this.children.splice(begin, range); - - for (var i = 0; i < removed.length; ++i) { - removed[i].parent = null; - } - - return removed; - } - else if (range === 0 && this.children.length === 0) { - return []; - } - else { - throw new RangeError('removeChildren: numeric values are outside the acceptable range.'); - } -}; - -/** - * Generates and updates the cached sprite for this object. - * - */ -DisplayObjectContainer.prototype.updateCachedSprite = function () { - this._generateCachedSprite(); -}; - -/** - * Useful function that returns a texture of the displayObject object that can then be used to create sprites - * This can be quite useful if your displayObject is static / complicated and needs to be reused multiple times. - * - * @param resolution {Number} The resolution of the texture being generated - * @param scaleMode {Number} See {{#crossLink "PIXI/scaleModes:property"}}PIXI.scaleModes{{/crossLink}} for possible values - * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used to generate the texture. - * @return {Texture} a texture of the graphics object - */ -DisplayObjectContainer.prototype.generateTexture = function (resolution, scaleMode, renderer) { - var bounds = this.getLocalBounds(); - - var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0, renderer, scaleMode, resolution); - - _tempMatrix.tx = -bounds.x; - _tempMatrix.ty = -bounds.y; - - renderTexture.render(this, _tempMatrix); - - return renderTexture; -}; - -/* - * Updates the transform on all children of this container for rendering - * - * @private - */ -DisplayObjectContainer.prototype.updateTransform = function () { - if (!this.visible) { - return; - } - - this.displayObjectUpdateTransform(); - - if (this._cacheAsBitmap) { - return; - } - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].updateTransform(); - } -}; - -// performance increase to avoid using call.. (10x faster) -DisplayObjectContainer.prototype.displayObjectContainerUpdateTransform = DisplayObjectContainer.prototype.updateTransform; - -/** - * Retrieves the bounds of the displayObjectContainer as a rectangle. The bounds calculation takes all visible children into consideration. - * - * @return {Rectangle} The rectangular bounding area - */ -DisplayObjectContainer.prototype.getBounds = function () { - if (this.children.length === 0) { - return math.Rectangle.EMPTY; - } - - // TODO the bounds have already been calculated this render session so return what we have - - var minX = Infinity; - var minY = Infinity; - - var maxX = -Infinity; - var maxY = -Infinity; - - var childBounds; - var childMaxX; - var childMaxY; - - var childVisible = false; - - for (var i = 0, j = this.children.length; i < j; ++i) { - var child = this.children[i]; - - if (!child.visible) { - continue; - } - - childVisible = true; - - childBounds = this.children[i].getBounds(); - - minX = minX < childBounds.x ? minX : childBounds.x; - minY = minY < childBounds.y ? minY : childBounds.y; - - childMaxX = childBounds.width + childBounds.x; - childMaxY = childBounds.height + childBounds.y; - - maxX = maxX > childMaxX ? maxX : childMaxX; - maxY = maxY > childMaxY ? maxY : childMaxY; - } - - if (!childVisible) { - return math.Rectangle.EMPTY; - } - - this._bounds.x = minX; - this._bounds.y = minY; - this._bounds.width = maxX - minX; - this._bounds.height = maxY - minY; - - // TODO: store a reference so that if this function gets called again in the render cycle we do not have to recalculate - //this._currentBounds = bounds; - - return this._bounds; -}; - -/** - * Retrieves the non-global local bounds of the displayObjectContainer as a rectangle. - * The calculation takes all visible children into consideration. - * - * @return {Rectangle} The rectangular bounding area - */ -DisplayObjectContainer.prototype.getLocalBounds = function () { - var matrixCache = this.worldTransform; - - this.worldTransform = math.Matrix.IDENTITY; - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].updateTransform(); - } - - this.worldTransform = matrixCache; - - return this.getBounds(); -}; - -/** - * Renders the object using the WebGL renderer - * - * TODO - Optimization pass! - * - * @param renderer {WebGLRenderer} The renderer - */ -DisplayObjectContainer.prototype.renderWebGL = function (renderer) { - // if the object is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0) { - return; - } - - if (this._cacheAsBitmap) { - this._renderCachedSprite(renderer); - return; - } - - var i, j; - - // do a quick check to see if this element has a mask or a filter. - if (this._mask || this._filters) { - // push filter first as we need to ensure the stencil buffer is correct for any masking - if (this._filters) { - renderer.spriteBatch.flush(); - renderer.filterManager.pushFilter(this._filterBlock); - } - - if (this._mask) { - renderer.spriteBatch.stop(); - renderer.maskManager.pushMask(this.mask, renderer); - renderer.spriteBatch.start(); - } - - // add this object to the batch, only rendered if it has a texture. - if (this.texture) { - renderer.spriteBatch.render(this); - } - - // now loop through the children and make sure they get rendered - for (i = 0, j = this.children.length; i < j; i++) { - this.children[i].renderWebGL(renderer); - } - - // time to stop the sprite batch as either a mask element or a filter draw will happen next - renderer.spriteBatch.stop(); - - if (this._mask) { - renderer.maskManager.popMask(this._mask, renderer); - } - - if (this._filters) { - renderer.filterManager.popFilter(); - } - - renderer.spriteBatch.start(); - } - else { - if (this.texture) { - renderer.spriteBatch.render(this); - } - - // simple render children! - for (i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderWebGL(renderer); - } - - } -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} The renderer - */ -DisplayObjectContainer.prototype.renderCanvas = function (renderer) { - if (!this.visible || this.alpha <= 0) { - return; - } - - if (this._cacheAsBitmap) { - this._renderCachedSprite(renderer); - return; - } - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderCanvas(renderer); - } - - if (this._mask) { - renderer.maskManager.popMask(renderer); - } -}; - -/** - * Internal method. - * - * @param renderer {WebGLRenderer|CanvasRenderer} The renderer - * @private - */ -DisplayObjectContainer.prototype._renderCachedSprite = function (renderer) { - this._cachedSprite.worldAlpha = this.worldAlpha; - - if (renderer.gl) { - this._cachedSprite.renderWebGL(renderer); - } - else { - this._cachedSprite.renderCanvas(renderer); - } -}; - -/** - * Internal method. - * - * @private - */ -DisplayObjectContainer.prototype._generateCachedSprite = function () { - var bounds = this.getLocalBounds(); - - if (!this._cachedSprite) { - // TODO - RenderTexture now *requires* a renderer instance, so this is like broken - // because `renderer` isn't actually in scope here :P - var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0); - - this._cachedSprite = new Sprite(renderTexture); - this._cachedSprite.worldTransform = this.worldTransform; - } - else { - this._cachedSprite.texture.resize(bounds.width | 0, bounds.height | 0); - } - - var tempFilters = this._filters; - this._filters = null; - - this._cachedSprite.filters = tempFilters; - - _tempMatrix.tx = -bounds.x; - _tempMatrix.ty = -bounds.y; - - this._cachedSprite.texture.render(this, _tempMatrix, true); - - this._cachedSprite.anchor.x = -(bounds.x / bounds.width); - this._cachedSprite.anchor.y = -(bounds.y / bounds.height); - - this._filters = tempFilters; -}; - -/** - * Destroys the cached sprite. - * - * @private - */ -DisplayObjectContainer.prototype._destroyCachedSprite = function () { - if (!this._cachedSprite) { - return; - } - - // TODO: Pool this sprite - this._cachedSprite.destroy(true, true); - this._cachedSprite = null; -}; - -},{"../math":12,"../textures/RenderTexture":43,"./DisplayObject":5}],7:[function(require,module,exports){ -var math = require('../math'), - Texture = require('../textures/Texture'), - DisplayObjectContainer = require('./DisplayObjectContainer'), - CanvasTinter = require('../renderers/canvas/utils/CanvasTinter'), - utils = require('../utils'), - CONST = require('../const'); - -/** - * The Sprite object is the base for all textured objects that are rendered to the screen - * - * A sprite can be created directly from an image like this: - * - * ```js - * var sprite = new Sprite.fromImage('assets/image.png'); - * ``` - * - * @class Sprite - * @extends DisplayObjectContainer - * @namespace PIXI - * @param texture {Texture} The texture for this sprite - */ -function Sprite(texture) { - DisplayObjectContainer.call(this); - - /** - * The anchor sets the origin point of the texture. - * The default is 0,0 this means the texture's origin is the top left - * Setting than anchor to 0.5,0.5 means the textures origin is centered - * Setting the anchor to 1,1 would mean the textures origin points will be the bottom right corner - * - * @member {Point} - */ - this.anchor = new math.Point(); - - /** - * The texture that the sprite is using - * - * @member {Texture} - * @private - */ - this._texture = null; - - /** - * The width of the sprite (this is initially set by the texture) - * - * @member {number} - * @private - */ - this._width = 0; - - /** - * The height of the sprite (this is initially set by the texture) - * - * @member {number} - * @private - */ - this._height = 0; - - /** - * The tint applied to the sprite. This is a hex value. A value of 0xFFFFFF will remove any tint effect. - * - * @member {number} - * @default 0xFFFFFF - */ - this.tint = 0xFFFFFF; - - /** - * The blend mode to be applied to the sprite. Set to CONST.blendModes.NORMAL to remove any blend mode. - * - * @member {number} - * @default CONST.blendModes.NORMAL; - */ - this.blendMode = CONST.blendModes.NORMAL; - - /** - * The shader that will be used to render the sprite. Set to null to remove a current shader. - * - * @member {AbstractFilter} - */ - this.shader = null; - - this.renderable = true; - - // call texture setter - this.texture = texture || Texture.EMPTY; -} - -Sprite.prototype.destroy = function (destroyTexture, destroyBaseTexture) { - DisplayObjectContainer.prototype.destroy.call(this); - - this.anchor = null; - - if (destroyTexture) { - this._texture.destroy(destroyBaseTexture); - } - - this._texture = null; - this.shader = null; -}; - -// constructor -Sprite.prototype = Object.create(DisplayObjectContainer.prototype); -Sprite.prototype.constructor = Sprite; -module.exports = Sprite; - -Object.defineProperties(Sprite.prototype, { - /** - * The width of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member - * @memberof Sprite# - */ - width: { - get: function () { - return this.scale.x * this.texture.frame.width; - }, - set: function (value) { - this.scale.x = value / this.texture.frame.width; - this._width = value; - } - }, - - /** - * The height of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member - * @memberof Sprite# - */ - height: { - get: function () { - return this.scale.y * this.texture.frame.height; - }, - set: function (value) { - this.scale.y = value / this.texture.frame.height; - this._height = value; - } - }, - - /** - * The height of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member - * @memberof Sprite# - */ - texture: { - get: function () { - return this._texture; - }, - set: function (value) { - if (this._texture === value) { - return; - } - - this._texture = value; - this.cachedTint = 0xFFFFFF; - - if (value) { - // wait for the texture to load - if (value.baseTexture.hasLoaded) { - this._onTextureUpdate(); - } - else { - value.once('update', this._onTextureUpdate.bind(this)); - } - } - } - }, -}); - -/** - * When the texture is updated, this event will fire to update the scale and frame - * - * @private - */ -Sprite.prototype._onTextureUpdate = function () { - // so if _width is 0 then width was not set.. - if (this._width) { - this.scale.x = this._width / this.texture.frame.width; - } - - if (this._height) { - this.scale.y = this._height / this.texture.frame.height; - } -}; - -/** - * Returns the bounds of the Sprite as a rectangle. The bounds calculation takes the worldTransform into account. - * - * @param matrix {Matrix} the transformation matrix of the sprite - * @return {Rectangle} the framing rectangle - */ -Sprite.prototype.getBounds = function (matrix) { - var width = this.texture.frame.width; - var height = this.texture.frame.height; - - var w0 = width * (1-this.anchor.x); - var w1 = width * -this.anchor.x; - - var h0 = height * (1-this.anchor.y); - var h1 = height * -this.anchor.y; - - var worldTransform = matrix || this.worldTransform ; - - var a = worldTransform.a; - var b = worldTransform.b; - var c = worldTransform.c; - var d = worldTransform.d; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var minX, - maxX, - minY, - maxY; - - if(b === 0 && c === 0) - { - // scale may be negative! - if (a < 0) { - a *= -1; - } - - if (d < 0) { - d *= -1; - } - - // this means there is no rotation going on right? RIGHT? - // if thats the case then we can avoid checking the bound values! yay - minX = a * w1 + tx; - maxX = a * w0 + tx; - minY = d * h1 + ty; - maxY = d * h0 + ty; - } - else - { - var x1 = a * w1 + c * h1 + tx; - var y1 = d * h1 + b * w1 + ty; - - var x2 = a * w0 + c * h1 + tx; - var y2 = d * h1 + b * w0 + ty; - - var x3 = a * w0 + c * h0 + tx; - var y3 = d * h0 + b * w0 + ty; - - var x4 = a * w1 + c * h0 + tx; - var y4 = d * h0 + b * w1 + ty; - - minX = x1; - minX = x2 < minX ? x2 : minX; - minX = x3 < minX ? x3 : minX; - minX = x4 < minX ? x4 : minX; - - minY = y1; - minY = y2 < minY ? y2 : minY; - minY = y3 < minY ? y3 : minY; - minY = y4 < minY ? y4 : minY; - - maxX = x1; - maxX = x2 > maxX ? x2 : maxX; - maxX = x3 > maxX ? x3 : maxX; - maxX = x4 > maxX ? x4 : maxX; - - maxY = y1; - maxY = y2 > maxY ? y2 : maxY; - maxY = y3 > maxY ? y3 : maxY; - maxY = y4 > maxY ? y4 : maxY; - } - - var bounds = this._bounds; - - bounds.x = minX; - bounds.width = maxX - minX; - - bounds.y = minY; - bounds.height = maxY - minY; - - // store a reference so that if this function gets called again in the render cycle we do not have to recalculate - this._currentBounds = bounds; - - return bounds; -}; - -/** -* Renders the object using the Canvas renderer -* -* @param renderer {CanvasRenderer} The renderer -*/ -Sprite.prototype.renderCanvas = function (renderer) { - if (!this.visible || this.alpha <= 0 || this.texture.crop.width <= 0 || this.texture.crop.height <= 0) { - return; - } - - if (this.blendMode !== renderer.currentBlendMode) { - renderer.currentBlendMode = this.blendMode; - renderer.context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode]; - } - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - // Ignore null sources - if (this.texture.valid) { - var resolution = this.texture.baseTexture.resolution / renderer.resolution; - - renderer.context.globalAlpha = this.worldAlpha; - - // If smoothingEnabled is supported and we need to change the smoothing property for this texture - if (renderer.smoothProperty && renderer.scaleMode !== this.texture.baseTexture.scaleMode) { - renderer.scaleMode = this.texture.baseTexture.scaleMode; - renderer.context[renderer.smoothProperty] = (renderer.scaleMode === CONST.scaleModes.LINEAR); - } - - // If the texture is trimmed we offset by the trim x/y, otherwise we use the frame dimensions - var dx = (this.texture.trim ? this.texture.trim.x : 0) - (this.anchor.x * this.texture.trim.width); - var dy = (this.texture.trim ? this.texture.trim.y : 0) - (this.anchor.y * this.texture.trim.height); - - // Allow for pixel rounding - if (renderer.roundPixels) { - renderer.context.setTransform( - this.worldTransform.a, - this.worldTransform.b, - this.worldTransform.c, - this.worldTransform.d, - (this.worldTransform.tx * renderer.resolution) | 0, - (this.worldTransform.ty * renderer.resolution) | 0 - ); - - dx = dx | 0; - dy = dy | 0; - } - else { - renderer.context.setTransform( - this.worldTransform.a, - this.worldTransform.b, - this.worldTransform.c, - this.worldTransform.d, - this.worldTransform.tx * renderer.resolution, - this.worldTransform.ty * renderer.resolution - ); - } - - if (this.tint !== 0xFFFFFF) { - if (this.cachedTint !== this.tint) { - this.cachedTint = this.tint; - - // TODO clean up caching - how to clean up the caches? - this.tintedTexture = CanvasTinter.getTintedTexture(this, this.tint); - } - - renderer.context.drawImage( - this.tintedTexture, - 0, - 0, - this.texture.crop.width, - this.texture.crop.height, - dx / resolution, - dy / resolution, - this.texture.crop.width / resolution, - this.texture.crop.height / resolution - ); - } - else { - renderer.context.drawImage( - this.texture.baseTexture.source, - this.texture.crop.x, - this.texture.crop.y, - this.texture.crop.width, - this.texture.crop.height, - dx / resolution, - dy / resolution, - this.texture.crop.width / resolution, - this.texture.crop.height / resolution - ); - } - } - - for (var i = 0, j = this.children.length; i < j; i++) { - this.children[i].renderCanvas(renderer); - } - - if (this._mask) { - renderer.maskManager.popMask(renderer); - } -}; - -// some helper functions.. - -/** - * Helper function that creates a sprite that will contain a texture from the TextureCache based on the frameId - * The frame ids are created when a Texture packer file has been loaded - * - * @static - * @param frameId {String} The frame Id of the texture in the cache - * @return {Sprite} A new Sprite using a texture from the texture cache matching the frameId - */ -Sprite.fromFrame = function (frameId) { - var texture = utils.TextureCache[frameId]; - - if (!texture) { - throw new Error('The frameId "' + frameId + '" does not exist in the texture cache' + this); - } - - return new Sprite(texture); -}; - -/** - * Helper function that creates a sprite that will contain a texture based on an image url - * If the image is not in the texture cache it will be loaded - * - * @static - * @param imageId {String} The image url of the texture - * @return {Sprite} A new Sprite using a texture from the texture cache matching the image id - */ -Sprite.fromImage = function (imageId, crossorigin, scaleMode) { - return new Sprite(Texture.fromImage(imageId, crossorigin, scaleMode)); -}; - -},{"../const":4,"../math":12,"../renderers/canvas/utils/CanvasTinter":24,"../textures/Texture":44,"../utils":50,"./DisplayObjectContainer":6}],8:[function(require,module,exports){ -var DisplayObjectContainer = require('./DisplayObjectContainer'), - WebGLFastSpriteBatch = require('../renderers/webgl/utils/WebGLFastSpriteBatch'); - -/** - * The SpriteBatch class is a really fast version of the DisplayObjectContainer built solely for speed, - * so use when you need a lot of sprites or particles. The tradeoff of the SpriteBatch is that advanced - * functionality will not work. SpriteBatch implements only the basic object transform (position, scale, rotation). - * Any other functionality like tinting, masking, etc will not work on sprites in this batch. - * - * It's extremely easy to use : - * - * ```js - * var container = new SpriteBatch(); - * - * for(var i = 0; i < 100; ++i) { - * var sprite = new PIXI.Sprite.fromImage("myImage.png"); - * container.addChild(sprite); - * } - * ``` - * - * And here you have a hundred sprites that will be renderer at the speed of light. - * - * @class - * @namespace PIXI - */ - -//TODO RENAME to PARTICLE CONTAINER? -function SpriteBatch() { - DisplayObjectContainer.call(this); -} - -SpriteBatch.prototype = Object.create(DisplayObjectContainer.prototype); -SpriteBatch.prototype.constructor = SpriteBatch; -module.exports = SpriteBatch; - -/** - * Updates the object transform for rendering - * - * @private - */ -SpriteBatch.prototype.updateTransform = function () { - // TODO don't need to! - this.displayObjectUpdateTransform(); - // PIXI.DisplayObjectContainer.prototype.updateTransform.call( this ); -}; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} The webgl renderer - * @private - */ -SpriteBatch.prototype.renderWebGL = function (renderer) { - if (!this.visible || this.alpha <= 0 || !this.children.length) { - return; - } - - renderer.spriteBatch.stop(); - - renderer.shaderManager.setShader(renderer.shaderManager.fastShader); - - renderer.fastSpriteBatch.begin(this); - renderer.fastSpriteBatch.render(this); - - renderer.spriteBatch.start(); -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} The canvas renderer - * @private - */ -SpriteBatch.prototype.renderCanvas = function (renderer) { - if (!this.visible || this.alpha <= 0 || !this.children.length) { - return; - } - - var context = renderer.context; - var transform = this.worldTransform; - var isRotated = true; - - context.globalAlpha = this.worldAlpha; - - this.displayObjectUpdateTransform(); - - for (var i = 0; i < this.children.length; ++i) { - var child = this.children[i]; - - if (!child.visible) { - continue; - } - - var frame = child.texture.frame; - - context.globalAlpha = this.worldAlpha * child.alpha; - - if (child.rotation % (Math.PI * 2) === 0) { - // this is the fastest way to optimise! - if rotation is 0 then we can avoid any kind of setTransform call - if (isRotated) { - context.setTransform( - transform.a, - transform.b, - transform.c, - transform.d, - transform.tx, - transform.ty - ); - - isRotated = false; - } - - context.drawImage( - child.texture.baseTexture.source, - frame.x, - frame.y, - frame.width, - frame.height, - ((child.anchor.x) * (-frame.width * child.scale.x) + child.position.x + 0.5) | 0, - ((child.anchor.y) * (-frame.height * child.scale.y) + child.position.y + 0.5) | 0, - frame.width * child.scale.x, - frame.height * child.scale.y - ); - } - else { - if (!isRotated) { - isRotated = true; - } - - child.displayObjectUpdateTransform(); - - var childTransform = child.worldTransform; - - if (renderer.roundPixels) { - context.setTransform( - childTransform.a, - childTransform.b, - childTransform.c, - childTransform.d, - childTransform.tx | 0, - childTransform.ty | 0 - ); - } - else { - context.setTransform( - childTransform.a, - childTransform.b, - childTransform.c, - childTransform.d, - childTransform.tx, - childTransform.ty - ); - } - - context.drawImage( - child.texture.baseTexture.source, - frame.x, - frame.y, - frame.width, - frame.height, - ((child.anchor.x) * (-frame.width) + 0.5) | 0, - ((child.anchor.y) * (-frame.height) + 0.5) | 0, - frame.width, - frame.height - ); - } - } -}; - -},{"../renderers/webgl/utils/WebGLFastSpriteBatch":38,"./DisplayObjectContainer":6}],9:[function(require,module,exports){ -/** - * @file Main export of the PIXI core library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -var core = module.exports = { - CONST: require('./const'), - - // utils - utils: require('./utils'), - math: require('./math'), - - // display - DisplayObject: require('./display/DisplayObject'), - DisplayObjectContainer: require('./display/DisplayObjectContainer'), - Sprite: require('./display/Sprite'), - SpriteBatch: require('./display/SpriteBatch'), - - // primitives - Graphics: require('./primitives/Graphics'), - GraphicsData: require('./primitives/GraphicsData'), - - // textures - Texture: require('./textures/Texture'), - BaseTexture: require('./textures/BaseTexture'), - RenderTexture: require('./textures/RenderTexture'), - VideoBaseTexture: require('./textures/VideoBaseTexture'), - - // renderers - canvas - CanvasRenderer: require('./renderers/canvas/CanvasRenderer'), - CanvasGraphics: require('./renderers/canvas/utils/CanvasGraphics'), - CanvasBuffer: require('./renderers/canvas/utils/CanvasBuffer'), - - // renderers - webgl - WebGLRenderer: require('./renderers/webgl/WebGLRenderer'), - WebGLGraphics: require('./renderers/webgl/utils/WebGLGraphics'), - - /** - * This helper function will automatically detect which renderer you should be using. - * WebGL is the preferred renderer as it is a lot faster. If webGL is not supported by - * the browser then this function will return a canvas renderer - * - * @param width=800 {number} the width of the renderers view - * @param height=600 {number} the height of the renderers view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment) - * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you - * need to call toDataUrl on the webgl context - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - * @param [noWebGL=false] {Boolean} prevents selection of WebGL renderer, even if such is present - * - * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer - */ - autoDetectRenderer: function (width, height, options, noWebGL) { - width = width || 800; - height = height || 600; - - if (!noWebGL && require('webgl-enabled')()) { - return new core.WebGLRenderer(width, height, options); - } - - return new core.CanvasRenderer(width, height, options); - }, - - /** - * This helper function will automatically detect which renderer you should be using. This function is very - * similar to the autoDetectRenderer function except that is will return a canvas renderer for android. - * Even thought both android chrome supports webGL the canvas implementation perform better at the time of writing. - * This function will likely change and update as webGL performance improves on these devices. - * - * @param width=800 {number} the width of the renderers view - * @param height=600 {number} the height of the renderers view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment) - * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you - * need to call toDataUrl on the webgl context - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - * - * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer - */ - autoDetectRecommendedRenderer: function (width, height, options) { - var isAndroid = /Android/i.test(navigator.userAgent); - - return core.autoDetectRenderer(width, height, options, isAndroid); - } -}; - -},{"./const":4,"./display/DisplayObject":5,"./display/DisplayObjectContainer":6,"./display/Sprite":7,"./display/SpriteBatch":8,"./math":12,"./primitives/Graphics":18,"./primitives/GraphicsData":19,"./renderers/canvas/CanvasRenderer":20,"./renderers/canvas/utils/CanvasBuffer":21,"./renderers/canvas/utils/CanvasGraphics":22,"./renderers/webgl/WebGLRenderer":25,"./renderers/webgl/utils/WebGLGraphics":39,"./textures/BaseTexture":42,"./textures/RenderTexture":43,"./textures/Texture":44,"./textures/VideoBaseTexture":46,"./utils":50,"webgl-enabled":2}],10:[function(require,module,exports){ -var Point = require('./Point'); - -/** - * The Matrix class is now an object, which makes it a lot faster, - * here is a representation of it : - * | a | b | tx| - * | c | d | ty| - * | 0 | 0 | 1 | - * - * @class - * @namespace PIXI - */ -function Matrix() { - /** - * @member {number} - * @default 1 - */ - this.a = 1; - - /** - * @member {number} - * @default 0 - */ - this.b = 0; - - /** - * @member {number} - * @default 0 - */ - this.c = 0; - - /** - * @member {number} - * @default 1 - */ - this.d = 1; - - /** - * @member {number} - * @default 0 - */ - this.tx = 0; - - /** - * @member {number} - * @default 0 - */ - this.ty = 0; -} - -Matrix.prototype.constructor = Matrix; -module.exports = Matrix; - -/** - * Creates a Matrix object based on the given array. The Element to Matrix mapping order is as follows: - * - * a = array[0] - * b = array[1] - * c = array[3] - * d = array[4] - * tx = array[2] - * ty = array[5] - * - * @param array {number[]} The array that the matrix will be populated from. - */ -Matrix.prototype.fromArray = function (array) { - this.a = array[0]; - this.b = array[1]; - this.c = array[3]; - this.d = array[4]; - this.tx = array[2]; - this.ty = array[5]; -}; - -/** - * Creates an array from the current Matrix object. - * - * @param transpose {boolean} Whether we need to transpose the matrix or not - * @return {number[]} the newly created array which contains the matrix - */ -Matrix.prototype.toArray = function (transpose) { - if (!this.array) { - this.array = new Float32Array(9); - } - - var array = this.array; - - if (transpose) { - array[0] = this.a; - array[1] = this.b; - array[2] = 0; - array[3] = this.c; - array[4] = this.d; - array[5] = 0; - array[6] = this.tx; - array[7] = this.ty; - array[8] = 1; - } - else { - array[0] = this.a; - array[1] = this.c; - array[2] = this.tx; - array[3] = this.b; - array[4] = this.d; - array[5] = this.ty; - array[6] = 0; - array[7] = 0; - array[8] = 1; - } - - return array; -}; - -/** - * Get a new position with the current transformation applied. - * Can be used to go from a child's coordinate space to the world coordinate space. (e.g. rendering) - * - * @param pos {Point} The origin - * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input) - * @return {Point} The new point, transformed through this matrix - */ -Matrix.prototype.apply = function (pos, newPos) { - newPos = newPos || new Point(); - - newPos.x = this.a * pos.x + this.c * pos.y + this.tx; - newPos.y = this.b * pos.x + this.d * pos.y + this.ty; - - return newPos; -}; - -/** - * Get a new position with the inverse of the current transformation applied. - * Can be used to go from the world coordinate space to a child's coordinate space. (e.g. input) - * - * @param pos {Point} The origin - * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input) - * @return {Point} The new point, inverse-transformed through this matrix - */ -Matrix.prototype.applyInverse = function (pos, newPos) { - newPos = newPos || new Point(); - - var id = 1 / (this.a * this.d + this.c * -this.b); - - newPos.x = this.d * id * pos.x + -this.c * id * pos.y + (this.ty * this.c - this.tx * this.d) * id; - newPos.y = this.a * id * pos.y + -this.b * id * pos.x + (-this.ty * this.a + this.tx * this.b) * id; - - return newPos; -}; - -/** - * Translates the matrix on the x and y. - * - * @param {number} x - * @param {number} y - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.translate = function (x, y) { - this.tx += x; - this.ty += y; - - return this; -}; - -/** - * Applies a scale transformation to the matrix. - * - * @param {number} x The amount to scale horizontally - * @param {number} y The amount to scale vertically - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.scale = function (x, y) { - this.a *= x; - this.d *= y; - this.c *= x; - this.b *= y; - this.tx *= x; - this.ty *= y; - - return this; -}; - - -/** - * Applies a rotation transformation to the matrix. - * - * @param {number} angle - The angle in radians. - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.rotate = function (angle) { - var cos = Math.cos( angle ); - var sin = Math.sin( angle ); - - var a1 = this.a; - var c1 = this.c; - var tx1 = this.tx; - - this.a = a1 * cos-this.b * sin; - this.b = a1 * sin+this.b * cos; - this.c = c1 * cos-this.d * sin; - this.d = c1 * sin+this.d * cos; - this.tx = tx1 * cos - this.ty * sin; - this.ty = tx1 * sin + this.ty * cos; - - return this; -}; - -/** - * Appends the given Matrix to this Matrix. - * - * @param {Matrix} matrix - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.append = function (matrix) { - var a1 = this.a; - var b1 = this.b; - var c1 = this.c; - var d1 = this.d; - - this.a = matrix.a * a1 + matrix.b * c1; - this.b = matrix.a * b1 + matrix.b * d1; - this.c = matrix.c * a1 + matrix.d * c1; - this.d = matrix.c * b1 + matrix.d * d1; - - this.tx = matrix.tx * a1 + matrix.ty * c1 + this.tx; - this.ty = matrix.tx * b1 + matrix.ty * d1 + this.ty; - - return this; -}; - -/** - * Resets this Matix to an identity (default) matrix. - * - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.identity = function () { - this.a = 1; - this.b = 0; - this.c = 0; - this.d = 1; - this.tx = 0; - this.ty = 0; - - return this; -}; - -Matrix.IDENTITY = new Matrix(); - -},{"./Point":11}],11:[function(require,module,exports){ -/** - * The Point object represents a location in a two-dimensional coordinate system, where x represents - * the horizontal axis and y represents the vertical axis. - * - * @class - * @namespace PIXI - * @param [x=0] {number} position of the point on the x axis - * @param [y=0] {number} position of the point on the y axis - */ -function Point(x, y) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; -} - -Point.prototype.constructor = Point; -module.exports = Point; - -/** - * Creates a clone of this point - * - * @return {Point} a copy of the point - */ -Point.prototype.clone = function () { - return new Point(this.x, this.y); -}; - -/** - * Sets the point to a new x and y position. - * If y is omitted, both x and y will be set to x. - * - * @param [x=0] {number} position of the point on the x axis - * @param [y=0] {number} position of the point on the y axis - */ -Point.prototype.set = function (x, y) { - this.x = x || 0; - this.y = y || ( (y !== 0) ? this.x : 0 ) ; -}; - -},{}],12:[function(require,module,exports){ -/** - * @namespace PIXI.math - */ -module.exports = { - /** - * @property {number} PI_2 - Math.PI x 2 - * @constant - * @static - */ - PI_2: Math.PI * 2, - - /** - * @property {number} RAD_TO_DEG - Constant conversion factor for converting radians to degrees - * @constant - * @static - */ - RAD_TO_DEG: 180 / Math.PI, - - /** - * @property {Number} DEG_TO_RAD - Constant conversion factor for converting degrees to radians - * @constant - * @static - */ - DEG_TO_RAD: Math.PI / 180, - - Point: require('./Point'), - Matrix: require('./Matrix'), - - Circle: require('./shapes/Circle'), - Ellipse: require('./shapes/Ellipse'), - Polygon: require('./shapes/Polygon'), - Rectangle: require('./shapes/Rectangle'), - RoundedRectangle: require('./shapes/RoundedRectangle') -}; - -},{"./Matrix":10,"./Point":11,"./shapes/Circle":13,"./shapes/Ellipse":14,"./shapes/Polygon":15,"./shapes/Rectangle":16,"./shapes/RoundedRectangle":17}],13:[function(require,module,exports){ -var Rectangle = require('./Rectangle'), - CONST = require('../../const'); - -/** - * The Circle object can be used to specify a hit area for displayObjects - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the center of this circle - * @param y {number} The Y coordinate of the center of this circle - * @param radius {number} The radius of the circle - */ -function Circle(x, y, radius) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.radius = radius || 0; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.CIRC; -} - -Circle.prototype.constructor = Circle; -module.exports = Circle; - -/** - * Creates a clone of this Circle instance - * - * @method clone - * @return {Circle} a copy of the Circle - */ -Circle.prototype.clone = function () { - return new Circle(this.x, this.y, this.radius); -}; - -/** - * Checks whether the x and y coordinates given are contained within this circle - * - * @method contains - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this Circle - */ -Circle.prototype.contains = function (x, y) { - if (this.radius <= 0) { - return false; - } - - var dx = (this.x - x), - dy = (this.y - y), - r2 = this.radius * this.radius; - - dx *= dx; - dy *= dy; - - return (dx + dy <= r2); -}; - -/** -* Returns the framing rectangle of the circle as a Rectangle object -* -* @method getBounds -* @return {Rectangle} the framing rectangle -*/ -Circle.prototype.getBounds = function () { - return new Rectangle(this.x - this.radius, this.y - this.radius, this.radius * 2, this.radius * 2); -}; - -},{"../../const":4,"./Rectangle":16}],14:[function(require,module,exports){ -var Rectangle = require('./Rectangle'), - CONST = require('../../const'); - -/** - * The Ellipse object can be used to specify a hit area for displayObjects - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the center of the ellipse - * @param y {number} The Y coordinate of the center of the ellipse - * @param width {number} The half width of this ellipse - * @param height {number} The half height of this ellipse - */ -function Ellipse(x, y, width, height) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.width = width || 0; - - /** - * @member {number} - * @default 0 - */ - this.height = height || 0; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.ELIP; -} - -Ellipse.prototype.constructor = Ellipse; -module.exports = Ellipse; - -/** - * Creates a clone of this Ellipse instance - * - * @method clone - * @return {Ellipse} a copy of the ellipse - */ -Ellipse.prototype.clone = function () { - return new Ellipse(this.x, this.y, this.width, this.height); -}; - -/** - * Checks whether the x and y coordinates given are contained within this ellipse - * - * @method contains - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coords are within this ellipse - */ -Ellipse.prototype.contains = function (x, y) { - if (this.width <= 0 || this.height <= 0) { - return false; - } - - //normalize the coords to an ellipse with center 0,0 - var normx = ((x - this.x) / this.width), - normy = ((y - this.y) / this.height); - - normx *= normx; - normy *= normy; - - return (normx + normy <= 1); -}; - -/** -* Returns the framing rectangle of the ellipse as a Rectangle object -* -* @method getBounds -* @return {Rectangle} the framing rectangle -*/ -Ellipse.prototype.getBounds = function () { - return new Rectangle(this.x - this.width, this.y - this.height, this.width, this.height); -}; - -},{"../../const":4,"./Rectangle":16}],15:[function(require,module,exports){ -var Point = require('../Point'), - CONST = require('../../const'); - -/** - * @class - * @namespace PIXI - * @param points* {Point[]|number[]|Point...|number...} This can be an array of Points that form the polygon, - * a flat array of numbers that will be interpreted as [x,y, x,y, ...], or the arguments passed can be - * all the points of the polygon e.g. `new Polygon(new Point(), new Point(), ...)`, or the - * arguments passed can be flat x,y values e.g. `new Polygon(x,y, x,y, x,y, ...)` where `x` and `y` are - * Numbers. - */ -function Polygon(points) { - //if points isn't an array, use arguments as the array - if (!(points instanceof Array)) { - points = Array.prototype.slice.call(arguments); - } - - //if this is a flat array of numbers, convert it to points - if (points[0] instanceof Point) { - var p = []; - for (var i = 0, il = points.length; i < il; i++) { - p.push(points[i].x, points[i].y); - } - - points = p; - } - - this.closed = true; - - /** - * An array of the points of this polygon - * - * @member {Point[]} - */ - this.points = points; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.POLY; -} - -Polygon.prototype.constructor = Polygon; -module.exports = Polygon; - -/** - * Creates a clone of this polygon - * - * @return {Polygon} a copy of the polygon - */ -Polygon.prototype.clone = function () { - return new Polygon(this.points.slice()); -}; - -/** - * Checks whether the x and y coordinates passed to this function are contained within this polygon - * - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this polygon - */ -Polygon.prototype.contains = function (x, y) { - var inside = false; - - // use some raycasting to test hits - // https://github.com/substack/point-in-polygon/blob/master/index.js - var length = this.points.length / 2; - - for (var i = 0, j = length - 1; i < length; j = i++) { - var xi = this.points[i * 2], yi = this.points[i * 2 + 1], - xj = this.points[j * 2], yj = this.points[j * 2 + 1], - intersect = ((yi > y) !== (yj > y)) && (x < (xj - xi) * (y - yi) / (yj - yi) + xi); - - if (intersect) { - inside = !inside; - } - } - - return inside; -}; - -},{"../../const":4,"../Point":11}],16:[function(require,module,exports){ -var CONST = require('../../const'); - -/** - * the Rectangle object is an area defined by its position, as indicated by its top-left corner point (x, y) and by its width and its height. - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the upper-left corner of the rectangle - * @param y {number} The Y coordinate of the upper-left corner of the rectangle - * @param width {number} The overall width of this rectangle - * @param height {number} The overall height of this rectangle - */ -function Rectangle(x, y, width, height) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.width = width || 0; - - /** - * @member {number} - * @default 0 - */ - this.height = height || 0; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.RECT; -} - -Rectangle.prototype.constructor = Rectangle; -module.exports = Rectangle; - -/** - * A constant empty rectangle. - * - * @static - * @constant - */ -Rectangle.EMPTY = new Rectangle(0, 0, 0, 0); - - -/** - * Creates a clone of this Rectangle - * - * @return {Rectangle} a copy of the rectangle - */ -Rectangle.prototype.clone = function () { - return new Rectangle(this.x, this.y, this.width, this.height); -}; - -/** - * Checks whether the x and y coordinates given are contained within this Rectangle - * - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this Rectangle - */ -Rectangle.prototype.contains = function (x, y) { - if (this.width <= 0 || this.height <= 0) { - return false; - } - - if (x >= this.x && x <= this.x + this.width) { - if (y >= this.y && y <= this.y + this.height) { - return true; - } - } - - return false; -}; - -},{"../../const":4}],17:[function(require,module,exports){ -var CONST = require('../../const'); - -/** - * The Rounded Rectangle object is an area defined by its position and has nice rounded corners, as indicated by its top-left corner point (x, y) and by its width and its height. - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the upper-left corner of the rounded rectangle - * @param y {number} The Y coordinate of the upper-left corner of the rounded rectangle - * @param width {number} The overall width of this rounded rectangle - * @param height {number} The overall height of this rounded rectangle - * @param radius {number} Controls the radius of the rounded corners - */ -function RoundedRectangle(x, y, width, height, radius) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.width = width || 0; - - /** - * @member {number} - * @default 0 - */ - this.height = height || 0; - - /** - * @member {number} - * @default 20 - */ - this.radius = radius || 20; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.RREC; -} - -RoundedRectangle.prototype.constructor = RoundedRectangle; -module.exports = RoundedRectangle; - -/** - * Creates a clone of this Rounded Rectangle - * - * @return {RoundedRectangle} a copy of the rounded rectangle - */ -RoundedRectangle.prototype.clone = function () { - return new RoundedRectangle(this.x, this.y, this.width, this.height, this.radius); -}; - -/** - * Checks whether the x and y coordinates given are contained within this Rounded Rectangle - * - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this Rounded Rectangle - */ -RoundedRectangle.prototype.contains = function (x, y) { - if (this.width <= 0 || this.height <= 0) { - return false; - } - - if (x >= this.x && x <= this.x + this.width) { - if (y >= this.y && y <= this.y + this.height) { - return true; - } - } - - return false; -}; - -},{"../../const":4}],18:[function(require,module,exports){ -var DisplayObjectContainer = require('../display/DisplayObjectContainer'), - Sprite = require('../display/Sprite'), - Texture = require('../textures/Texture'), - CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'), - CanvasGraphics = require('../renderers/canvas/utils/CanvasGraphics'), - WebGLGraphics = require('../renderers/webgl/utils/WebGLGraphics'), - GraphicsData = require('./GraphicsData'), - math = require('../math'), - CONST = require('../const'); - -/** - * The Graphics class contains methods used to draw primitive shapes such as lines, circles and - * rectangles to the display, and color and fill them. - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - */ -function Graphics() { - DisplayObjectContainer.call(this); - - this.renderable = true; - - /** - * The alpha value used when filling the Graphics object. - * - * @member {number} - * @default 1 - */ - this.fillAlpha = 1; - - /** - * The width (thickness) of any lines drawn. - * - * @member {number} - * @default 0 - */ - this.lineWidth = 0; - - /** - * The color of any lines drawn. - * - * @member {string} - * @default 0 - */ - this.lineColor = 0; - - /** - * Graphics data - * - * @member {GraphicsData[]} - * @private - */ - this.graphicsData = []; - - /** - * The tint applied to the graphic shape. This is a hex value. Apply a value of 0xFFFFFF to reset the tint. - * - * @member {number} - * @default 0xFFFFFF - */ - this.tint = 0xFFFFFF; - - /** - * The blend mode to be applied to the graphic shape. Apply a value of blendModes.NORMAL to reset the blend mode. - * - * @member {number} - * @default CONST.blendModes.NORMAL; - */ - this.blendMode = CONST.blendModes.NORMAL; - - /** - * Current path - * - * @member {GraphicsData} - * @private - */ - this.currentPath = null; - - /** - * Array containing some WebGL-related properties used by the WebGL renderer. - * - * @member {object} - * @private - */ - // TODO - _webgl should use a prototype object, not a random undocumented object... - this._webGL = {}; - - /** - * Whether this shape is being used as a mask. - * - * @member {boolean} - */ - this.isMask = false; - - /** - * The bounds' padding used for bounds calculation. - * - * @member {number} - */ - this.boundsPadding = 0; - - /** - * A cache of the local bounds to prevent recalculation. - * - * @member {Rectangle} - * @private - */ - this._localBounds = new math.Rectangle(0,0,1,1); - - /** - * Used to detect if the graphics object has changed. If this is set to true then the graphics - * object will be recalculated. - * - * @member {boolean} - * @private - */ - this.dirty = true; - - /** - * Used to detect if the WebGL graphics object has changed. If this is set to true then the - * graphics object will be recalculated. - * - * @member {boolean} - * @private - */ - this.glDirty = false; - - /** - * Used to detect if the cached sprite object needs to be updated. - * - * @member {boolean} - * @private - */ - this.cachedSpriteDirty = false; -} - -// constructor -Graphics.prototype = Object.create(DisplayObjectContainer.prototype); -Graphics.prototype.constructor = Graphics; -module.exports = Graphics; - -Object.defineProperties(Graphics.prototype, { - /** - * When cacheAsBitmap is set to true the graphics object will be rendered as if it was a sprite. - * This is useful if your graphics element does not change often, as it will speed up the rendering - * of the object in exchange for taking up texture memory. It is also useful if you need the graphics - * object to be anti-aliased, because it will be rendered using canvas. This is not recommended if - * you are constantly redrawing the graphics element. - * - * @member {boolean} - * @memberof Graphics# - * @default false - * @private - */ - cacheAsBitmap: { - get: function () { - return this._cacheAsBitmap; - }, - set: function (value) { - this._cacheAsBitmap = value; - - if (this._cacheAsBitmap) { - this._generateCachedSprite(); - } - else { - this.destroyCachedSprite(); - this.dirty = true; - } - } - } -}); - -/** - * Creates a new Graphics object with the same values as this one. - * - * @return {Graphics} - */ -GraphicsData.prototype.clone = function () { - var clone = new Graphics(); - - clone.renderable = this.renderable; - clone.fillAlpha = this.fillAlpha; - clone.lineWidth = this.lineWidth; - clone.lineColor = this.lineColor; - clone.tint = this.tint; - clone.blendMode = this.blendMode; - clone.isMask = this.isMask; - clone.boundsPadding = this.boundsPadding; - clone.dirty = this.dirty; - clone.glDirty = this.glDirty; - clone.cachedSpriteDirty = this.cachedSpriteDirty; - - // copy graphics data - for (var i = 0; i < this.graphicsData.length; ++i) { - clone.graphicsData.push(this.graphicsData.clone()); - } - - clone.currentPath = clone.graphicsData[clone.graphicsData.length - 1]; - - clone.updateLocalBounds(); - - return clone; -}; - -/** - * Specifies the line style used for subsequent calls to Graphics methods such as the lineTo() method or the drawCircle() method. - * - * @param lineWidth {number} width of the line to draw, will update the objects stored style - * @param color {number} color of the line to draw, will update the objects stored style - * @param alpha {number} alpha of the line to draw, will update the objects stored style - * @return {Graphics} - */ -Graphics.prototype.lineStyle = function (lineWidth, color, alpha) { - this.lineWidth = lineWidth || 0; - this.lineColor = color || 0; - this.lineAlpha = (arguments.length < 3) ? 1 : alpha; - - if (this.currentPath) { - if (this.currentPath.shape.points.length) { - // halfway through a line? start a new one! - this.drawShape( new math.Polygon( this.currentPath.shape.points.slice(-2) )); - } - else { - // otherwise its empty so lets just set the line properties - this.currentPath.lineWidth = this.lineWidth; - this.currentPath.lineColor = this.lineColor; - this.currentPath.lineAlpha = this.lineAlpha; - } - } - - return this; -}; - -/** - * Moves the current drawing position to x, y. - * - * @param x {number} the X coordinate to move to - * @param y {number} the Y coordinate to move to - * @return {Graphics} - */ -Graphics.prototype.moveTo = function (x, y) { - this.drawShape(new math.Polygon([x,y])); - - return this; -}; - -/** - * Draws a line using the current line style from the current drawing position to (x, y); - * The current drawing position is then set to (x, y). - * - * @param x {number} the X coordinate to draw to - * @param y {number} the Y coordinate to draw to - * @return {Graphics} - */ -Graphics.prototype.lineTo = function (x, y) { - this.currentPath.shape.points.push(x, y); - this.dirty = true; - - return this; -}; - -/** - * Calculate the points for a quadratic bezier curve and then draws it. - * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c - * - * @param cpX {number} Control point x - * @param cpY {number} Control point y - * @param toX {number} Destination point x - * @param toY {number} Destination point y - * @return {Graphics} - */ -Graphics.prototype.quadraticCurveTo = function (cpX, cpY, toX, toY) { - if (this.currentPath) { - if (this.currentPath.shape.points.length === 0) { - this.currentPath.shape.points = [0, 0]; - } - } - else { - this.moveTo(0,0); - } - - var xa, - ya, - n = 20, - points = this.currentPath.shape.points; - - if (points.length === 0) { - this.moveTo(0, 0); - } - - var fromX = points[points.length-2]; - var fromY = points[points.length-1]; - - var j = 0; - for (var i = 1; i <= n; ++i) { - j = i / n; - - xa = fromX + ( (cpX - fromX) * j ); - ya = fromY + ( (cpY - fromY) * j ); - - points.push( xa + ( ((cpX + ( (toX - cpX) * j )) - xa) * j ), - ya + ( ((cpY + ( (toY - cpY) * j )) - ya) * j ) ); - } - - this.dirty = true; - - return this; -}; - -/** - * Calculate the points for a bezier curve and then draws it. - * - * @param cpX {number} Control point x - * @param cpY {number} Control point y - * @param cpX2 {number} Second Control point x - * @param cpY2 {number} Second Control point y - * @param toX {number} Destination point x - * @param toY {number} Destination point y - * @return {Graphics} - */ -Graphics.prototype.bezierCurveTo = function (cpX, cpY, cpX2, cpY2, toX, toY) { - if (this.currentPath) { - if (this.currentPath.shape.points.length === 0) { - this.currentPath.shape.points = [0, 0]; - } - } - else { - this.moveTo(0,0); - } - - var n = 20, - dt, - dt2, - dt3, - t2, - t3, - points = this.currentPath.shape.points; - - var fromX = points[points.length-2]; - var fromY = points[points.length-1]; - - var j = 0; - - for (var i = 1; i <= n; ++i) { - j = i / n; - - dt = (1 - j); - dt2 = dt * dt; - dt3 = dt2 * dt; - - t2 = j * j; - t3 = t2 * j; - - points.push( dt3 * fromX + 3 * dt2 * j * cpX + 3 * dt * t2 * cpX2 + t3 * toX, - dt3 * fromY + 3 * dt2 * j * cpY + 3 * dt * t2 * cpY2 + t3 * toY); - } - - this.dirty = true; - - return this; -}; - -/** - * The arcTo() method creates an arc/curve between two tangents on the canvas. - * - * "borrowed" from https://code.google.com/p/fxcanvas/ - thanks google! - * - * @param x1 {number} The x-coordinate of the beginning of the arc - * @param y1 {number} The y-coordinate of the beginning of the arc - * @param x2 {number} The x-coordinate of the end of the arc - * @param y2 {number} The y-coordinate of the end of the arc - * @param radius {number} The radius of the arc - * @return {Graphics} - */ -Graphics.prototype.arcTo = function (x1, y1, x2, y2, radius) { - if (this.currentPath) { - if (this.currentPath.shape.points.length === 0) { - this.currentPath.shape.points.push(x1, y1); - } - } - else { - this.moveTo(x1, y1); - } - - var points = this.currentPath.shape.points, - fromX = points[points.length-2], - fromY = points[points.length-1], - a1 = fromY - y1, - b1 = fromX - x1, - a2 = y2 - y1, - b2 = x2 - x1, - mm = Math.abs(a1 * b2 - b1 * a2); - - if (mm < 1.0e-8 || radius === 0) { - if (points[points.length-2] !== x1 || points[points.length-1] !== y1) { - points.push(x1, y1); - } - } - else { - var dd = a1 * a1 + b1 * b1, - cc = a2 * a2 + b2 * b2, - tt = a1 * a2 + b1 * b2, - k1 = radius * Math.sqrt(dd) / mm, - k2 = radius * Math.sqrt(cc) / mm, - j1 = k1 * tt / dd, - j2 = k2 * tt / cc, - cx = k1 * b2 + k2 * b1, - cy = k1 * a2 + k2 * a1, - px = b1 * (k2 + j1), - py = a1 * (k2 + j1), - qx = b2 * (k1 + j2), - qy = a2 * (k1 + j2), - startAngle = Math.atan2(py - cy, px - cx), - endAngle = Math.atan2(qy - cy, qx - cx); - - this.arc(cx + x1, cy + y1, radius, startAngle, endAngle, b1 * a2 > b2 * a1); - } - - this.dirty = true; - - return this; -}; - -/** - * The arc method creates an arc/curve (used to create circles, or parts of circles). - * - * @param cx {number} The x-coordinate of the center of the circle - * @param cy {number} The y-coordinate of the center of the circle - * @param radius {number} The radius of the circle - * @param startAngle {number} The starting angle, in radians (0 is at the 3 o'clock position of the arc's circle) - * @param endAngle {number} The ending angle, in radians - * @param anticlockwise {boolean} Optional. Specifies whether the drawing should be counterclockwise or clockwise. False is default, and indicates clockwise, while true indicates counter-clockwise. - * @return {Graphics} - */ -Graphics.prototype.arc = function (cx, cy, radius, startAngle, endAngle, anticlockwise) { - var startX = cx + Math.cos(startAngle) * radius; - var startY = cy + Math.sin(startAngle) * radius; - var points; - - // TODO - This if-else makes no sense. It uses currentPath in the else where it doesn't exist... - if (this.currentPath) { - points = this.currentPath.shape.points; - - if (points.length === 0) { - points.push(startX, startY); - } - else if (points[points.length-2] !== startX || points[points.length-1] !== startY) { - points.push(startX, startY); - } - } - else { - this.moveTo(startX, startY); - points = this.currentPath.shape.points; - } - - if (startAngle === endAngle) { - return this; - } - - if (!anticlockwise && endAngle <= startAngle) { - endAngle += Math.PI * 2; - } - else if (anticlockwise && startAngle <= endAngle) { - startAngle += Math.PI * 2; - } - - var sweep = anticlockwise ? (startAngle - endAngle) *-1 : (endAngle - startAngle); - var segs = (Math.abs(sweep)/ (Math.PI * 2)) * 40; - - if (sweep === 0) { - return this; - } - - var theta = sweep/(segs*2); - var theta2 = theta*2; - - var cTheta = Math.cos(theta); - var sTheta = Math.sin(theta); - - var segMinus = segs - 1; - - var remainder = ( segMinus % 1 ) / segMinus; - - for (var i = 0; i <= segMinus; ++i) { - var real = i + remainder * i; - var angle = ((theta) + startAngle + (theta2 * real)); - - var c = Math.cos(angle); - var s = -Math.sin(angle); - - points.push(( (cTheta * c) + (sTheta * s) ) * radius + cx, - ( (cTheta * -s) + (sTheta * c) ) * radius + cy); - } - - this.dirty = true; - - return this; -}; - -/** - * Specifies a simple one-color fill that subsequent calls to other Graphics methods - * (such as lineTo() or drawCircle()) use when drawing. - * - * @param color {number} the color of the fill - * @param alpha {number} the alpha of the fill - * @return {Graphics} - */ -Graphics.prototype.beginFill = function (color, alpha) { - this.filling = true; - this.fillColor = color || 0; - this.fillAlpha = (alpha === undefined) ? 1 : alpha; - - if (this.currentPath) { - if (this.currentPath.shape.points.length <= 2) { - this.currentPath.fill = this.filling; - this.currentPath.fillColor = this.fillColor; - this.currentPath.fillAlpha = this.fillAlpha; - } - } - return this; -}; - -/** - * Applies a fill to the lines and shapes that were added since the last call to the beginFill() method. - * - * @return {Graphics} - */ -Graphics.prototype.endFill = function () { - this.filling = false; - this.fillColor = null; - this.fillAlpha = 1; - - return this; -}; - -/** - * - * @param x {number} The X coord of the top-left of the rectangle - * @param y {number} The Y coord of the top-left of the rectangle - * @param width {number} The width of the rectangle - * @param height {number} The height of the rectangle - * @return {Graphics} - */ -Graphics.prototype.drawRect = function ( x, y, width, height ) { - this.drawShape(new math.Rectangle(x,y, width, height)); - - return this; -}; - -/** - * - * @param x {number} The X coord of the top-left of the rectangle - * @param y {number} The Y coord of the top-left of the rectangle - * @param width {number} The width of the rectangle - * @param height {number} The height of the rectangle - * @param radius {number} Radius of the rectangle corners - */ -Graphics.prototype.drawRoundedRect = function ( x, y, width, height, radius ) { - this.drawShape(new math.RoundedRectangle(x, y, width, height, radius)); - - return this; -}; - -/** - * Draws a circle. - * - * @param x {number} The X coordinate of the center of the circle - * @param y {number} The Y coordinate of the center of the circle - * @param radius {number} The radius of the circle - * @return {Graphics} - */ -Graphics.prototype.drawCircle = function (x, y, radius) { - this.drawShape(new math.Circle(x,y, radius)); - - return this; -}; - -/** - * Draws an ellipse. - * - * @param x {number} The X coordinate of the center of the ellipse - * @param y {number} The Y coordinate of the center of the ellipse - * @param width {number} The half width of the ellipse - * @param height {number} The half height of the ellipse - * @return {Graphics} - */ -Graphics.prototype.drawEllipse = function (x, y, width, height) { - this.drawShape(new math.Ellipse(x, y, width, height)); - - return this; -}; - -/** - * Draws a polygon using the given path. - * - * @param path {Array} The path data used to construct the polygon. - * @return {Graphics} - */ -Graphics.prototype.drawPolygon = function (path) { - if (!(path instanceof Array)) { - path = Array.prototype.slice.call(arguments); - } - - this.drawShape(new math.Polygon(path)); - - return this; -}; - -/** - * Clears the graphics that were drawn to this Graphics object, and resets fill and line style settings. - * - * @return {Graphics} - */ -Graphics.prototype.clear = function () { - this.lineWidth = 0; - this.filling = false; - - this.dirty = true; - this.clearDirty = true; - this.graphicsData = []; - - return this; -}; - -/** - * Useful function that returns a texture of the graphics object that can then be used to create sprites - * This can be quite useful if your geometry is complicated and needs to be reused multiple times. - * - * @param resolution {number} The resolution of the texture being generated - * @param scaleMode {number} Should be one of the scaleMode consts - * @return {Texture} a texture of the graphics object - */ -Graphics.prototype.generateTexture = function (resolution, scaleMode) { - resolution = resolution || 1; - - var bounds = this.getBounds(); - - var canvasBuffer = new CanvasBuffer(bounds.width * resolution, bounds.height * resolution); - - var texture = Texture.fromCanvas(canvasBuffer.canvas, scaleMode); - texture.baseTexture.resolution = resolution; - - canvasBuffer.context.scale(resolution, resolution); - - canvasBuffer.context.translate(-bounds.x,-bounds.y); - - CanvasGraphics.renderGraphics(this, canvasBuffer.context); - - return texture; -}; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} - */ -Graphics.prototype.renderWebGL = function (renderer) { - // if the sprite is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0 || this.isMask === true) { - return; - } - - if (this._cacheAsBitmap) { - if (this.dirty || this.cachedSpriteDirty) { - this._generateCachedSprite(); - - // we will also need to update the texture on the gpu too! - this.updateCachedSpriteTexture(); - - this.cachedSpriteDirty = false; - this.dirty = false; - } - - this._cachedSprite.worldAlpha = this.worldAlpha; - - Sprite.prototype.renderWebGL.call(this._cachedSprite, renderer); - - return; - } - else { - renderer.spriteBatch.stop(); - renderer.blendModeManager.setBlendMode(this.blendMode); - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - if (this._filters) { - renderer.filterManager.pushFilter(this._filterBlock); - } - - // check blend mode - if (this.blendMode !== renderer.spriteBatch.currentBlendMode) { - renderer.spriteBatch.currentBlendMode = this.blendMode; - - var blendModeWebGL = renderer.blendModes[renderer.spriteBatch.currentBlendMode]; - - renderer.spriteBatch.gl.blendFunc(blendModeWebGL[0], blendModeWebGL[1]); - } - - // check if the webgl graphic needs to be updated - if (this.glDirty) { - this.dirty = true; - this.glDirty = false; - } - - WebGLGraphics.renderGraphics(this, renderer); - - // only render if it has children! - if (this.children.length) { - renderer.spriteBatch.start(); - - // simple render children! - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderWebGL(renderer); - } - - renderer.spriteBatch.stop(); - } - - if (this._filters) { - renderer.filterManager.popFilter(); - } - - if (this._mask) { - renderer.maskManager.popMask(this.mask, renderer); - } - - renderer.drawCount++; - - renderer.spriteBatch.start(); - } -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} - * @private - */ -Graphics.prototype.renderCanvas = function (renderer) { - // if the sprite is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0 || this.isMask === true) { - return; - } - - if (this._cacheAsBitmap) { - if (this.dirty || this.cachedSpriteDirty) { - this._generateCachedSprite(); - - // we will also need to update the texture - this.updateCachedSpriteTexture(); - - this.cachedSpriteDirty = false; - this.dirty = false; - } - - this._cachedSprite.alpha = this.alpha; - - Sprite.prototype.renderCanvas.call(this._cachedSprite, renderer); - - return; - } - else { - var context = renderer.context; - var transform = this.worldTransform; - - if (this.blendMode !== renderer.currentBlendMode) { - renderer.currentBlendMode = this.blendMode; - context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode]; - } - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - var resolution = renderer.resolution; - context.setTransform( - transform.a * resolution, - transform.b * resolution, - transform.c * resolution, - transform.d * resolution, - transform.tx * resolution, - transform.ty * resolution - ); - - CanvasGraphics.renderGraphics(this, context); - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderCanvas(renderer); - } - - if (this._mask) { - renderer.maskManager.popMask(renderer); - } - } -}; - -/** - * Retrieves the bounds of the graphic shape as a rectangle object - * - * @return {Rectangle} the rectangular bounding area - */ -Graphics.prototype.getBounds = function (matrix) { - // return an empty object if the item is a mask! - if (this.isMask) { - return math.Rectangle.EMPTY; - } - - if (this.dirty) { - this.updateLocalBounds(); - - this.glDirty = true; - this.cachedSpriteDirty = true; - this.dirty = false; - } - - var bounds = this._localBounds; - - var w0 = bounds.x; - var w1 = bounds.width + bounds.x; - - var h0 = bounds.y; - var h1 = bounds.height + bounds.y; - - var worldTransform = matrix || this.worldTransform; - - var a = worldTransform.a; - var b = worldTransform.b; - var c = worldTransform.c; - var d = worldTransform.d; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var x1 = a * w1 + c * h1 + tx; - var y1 = d * h1 + b * w1 + ty; - - var x2 = a * w0 + c * h1 + tx; - var y2 = d * h1 + b * w0 + ty; - - var x3 = a * w0 + c * h0 + tx; - var y3 = d * h0 + b * w0 + ty; - - var x4 = a * w1 + c * h0 + tx; - var y4 = d * h0 + b * w1 + ty; - - var maxX = x1; - var maxY = y1; - - var minX = x1; - var minY = y1; - - minX = x2 < minX ? x2 : minX; - minX = x3 < minX ? x3 : minX; - minX = x4 < minX ? x4 : minX; - - minY = y2 < minY ? y2 : minY; - minY = y3 < minY ? y3 : minY; - minY = y4 < minY ? y4 : minY; - - maxX = x2 > maxX ? x2 : maxX; - maxX = x3 > maxX ? x3 : maxX; - maxX = x4 > maxX ? x4 : maxX; - - maxY = y2 > maxY ? y2 : maxY; - maxY = y3 > maxY ? y3 : maxY; - maxY = y4 > maxY ? y4 : maxY; - - this._bounds.x = minX; - this._bounds.width = maxX - minX; - - this._bounds.y = minY; - this._bounds.height = maxY - minY; - - return this._bounds; -}; - -/** - * Update the bounds of the object - * - */ -Graphics.prototype.updateLocalBounds = function () { - var minX = Infinity; - var maxX = -Infinity; - - var minY = Infinity; - var maxY = -Infinity; - - if (this.graphicsData.length) { - var shape, points, x, y, w, h; - - for (var i = 0; i < this.graphicsData.length; i++) { - var data = this.graphicsData[i]; - var type = data.type; - var lineWidth = data.lineWidth; - shape = data.shape; - - if (type === CONST.SHAPES.RECT || type === CONST.SHAPES.RREC) { - x = shape.x - lineWidth/2; - y = shape.y - lineWidth/2; - w = shape.width + lineWidth; - h = shape.height + lineWidth; - - minX = x < minX ? x : minX; - maxX = x + w > maxX ? x + w : maxX; - - minY = y < minY ? y : minY; - maxY = y + h > maxY ? y + h : maxY; - } - else if (type === CONST.SHAPES.CIRC) { - x = shape.x; - y = shape.y; - w = shape.radius + lineWidth/2; - h = shape.radius + lineWidth/2; - - minX = x - w < minX ? x - w : minX; - maxX = x + w > maxX ? x + w : maxX; - - minY = y - h < minY ? y - h : minY; - maxY = y + h > maxY ? y + h : maxY; - } - else if (type === CONST.SHAPES.ELIP) { - x = shape.x; - y = shape.y; - w = shape.width + lineWidth/2; - h = shape.height + lineWidth/2; - - minX = x - w < minX ? x - w : minX; - maxX = x + w > maxX ? x + w : maxX; - - minY = y - h < minY ? y - h : minY; - maxY = y + h > maxY ? y + h : maxY; - } - else { - // POLY - points = shape.points; - - for (var j = 0; j < points.length; j += 2) { - x = points[j]; - y = points[j+1]; - - minX = x-lineWidth < minX ? x-lineWidth : minX; - maxX = x+lineWidth > maxX ? x+lineWidth : maxX; - - minY = y-lineWidth < minY ? y-lineWidth : minY; - maxY = y+lineWidth > maxY ? y+lineWidth : maxY; - } - } - } - } - else { - minX = 0; - maxX = 0; - minY = 0; - maxY = 0; - } - - var padding = this.boundsPadding; - - this._localBounds.x = minX - padding; - this._localBounds.width = (maxX - minX) + padding * 2; - - this._localBounds.y = minY - padding; - this._localBounds.height = (maxY - minY) + padding * 2; -}; - -/** - * Generates the cached sprite when the sprite has cacheAsBitmap = true - * - * @private - */ -Graphics.prototype._generateCachedSprite = function () { - var bounds = this.getLocalBounds(); - - if (!this._cachedSprite) { - var canvasBuffer = new CanvasBuffer(bounds.width, bounds.height); - var texture = Texture.fromCanvas(canvasBuffer.canvas); - - this._cachedSprite = new Sprite(texture); - this._cachedSprite.buffer = canvasBuffer; - - this._cachedSprite.worldTransform = this.worldTransform; - } - else { - this._cachedSprite.buffer.resize(bounds.width, bounds.height); - } - - // leverage the anchor to account for the offset of the element - this._cachedSprite.anchor.x = -( bounds.x / bounds.width ); - this._cachedSprite.anchor.y = -( bounds.y / bounds.height ); - - // this._cachedSprite.buffer.context.save(); - this._cachedSprite.buffer.context.translate(-bounds.x,-bounds.y); - - // make sure we set the alpha of the graphics to 1 for the render.. - this.worldAlpha = 1; - - // now render the graphic.. - CanvasGraphics.renderGraphics(this, this._cachedSprite.buffer.context); - - this._cachedSprite.alpha = this.alpha; -}; - -/** - * Updates texture size based on canvas size - * - * @private - */ -Graphics.prototype.updateCachedSpriteTexture = function () { - var cachedSprite = this._cachedSprite; - var texture = cachedSprite.texture; - var canvas = cachedSprite.buffer.canvas; - - texture.baseTexture.width = canvas.width; - texture.baseTexture.height = canvas.height; - texture.crop.width = texture.frame.width = canvas.width; - texture.crop.height = texture.frame.height = canvas.height; - - cachedSprite._width = canvas.width; - cachedSprite._height = canvas.height; - - // update the dirty base textures - texture.baseTexture.dirty(); -}; - -/** - * Destroys a previous cached sprite. - * - */ -Graphics.prototype.destroyCachedSprite = function () { - this._cachedSprite.texture.destroy(true); - - // let the gc collect the unused sprite - // TODO could be object pooled! - this._cachedSprite = null; -}; - -/** - * Draws the given shape to this Graphics object. Can be any of Circle, Rectangle, Ellipse, Line or Polygon. - * - * @param {Circle|Rectangle|Ellipse|Line|Polygon} shape The Shape object to draw. - * @return {GraphicsData} The generated GraphicsData object. - */ -Graphics.prototype.drawShape = function (shape) { - if (this.currentPath) { - // check current path! - if (this.currentPath.shape.points.length <= 2) { - this.graphicsData.pop(); - } - } - - this.currentPath = null; - - var data = new GraphicsData(this.lineWidth, this.lineColor, this.lineAlpha, this.fillColor, this.fillAlpha, this.filling, shape); - - this.graphicsData.push(data); - - if (data.type === CONST.SHAPES.POLY) { - data.shape.closed = this.filling; - this.currentPath = data; - } - - this.dirty = true; - - return data; -}; - -},{"../const":4,"../display/DisplayObjectContainer":6,"../display/Sprite":7,"../math":12,"../renderers/canvas/utils/CanvasBuffer":21,"../renderers/canvas/utils/CanvasGraphics":22,"../renderers/webgl/utils/WebGLGraphics":39,"../textures/Texture":44,"./GraphicsData":19}],19:[function(require,module,exports){ -/** - * A GraphicsData object. - * - * @class - * @namespace PIXI - */ -function GraphicsData(lineWidth, lineColor, lineAlpha, fillColor, fillAlpha, fill, shape) { - this.lineWidth = lineWidth; - this.lineColor = lineColor; - this.lineAlpha = lineAlpha; - this._lineTint = lineColor; - - this.fillColor = fillColor; - this.fillAlpha = fillAlpha; - this._fillTint = fillColor; - this.fill = fill; - - this.shape = shape; - this.type = shape.type; -} - -GraphicsData.prototype.constructor = GraphicsData; -module.exports = GraphicsData; - -/** - * Creates a new GraphicsData object with the same values as this one. - * - * @return {GraphicsData} - */ -GraphicsData.prototype.clone = function () { - return new GraphicsData( - this.lineWidth, - this.lineColor, - this.lineAlpha, - this.fillColor, - this.fillAlpha, - this.fill, - this.shape - ); -}; - -},{}],20:[function(require,module,exports){ -var CanvasMaskManager = require('./utils/CanvasMaskManager'), - utils = require('../../utils'), - CONST = require('../../const'); - -/** - * The CanvasRenderer draws the scene and all its content onto a 2d canvas. This renderer should be used for browsers that do not support webGL. - * Don't forget to add the CanvasRenderer.view to your DOM or you will not see anything :) - * - * @class - * @namespace PIXI - * @param [width=800] {number} the width of the canvas view - * @param [height=600] {number} the height of the canvas view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - * @param [options.clearBeforeRender=true] {boolean} This sets if the CanvasRenderer will clear the canvas or not before the new render pass. - */ -function CanvasRenderer(width, height, options) { - utils.sayHello('Canvas'); - - if (options) { - for (var i in CONST.defaultRenderOptions) { - if (typeof options[i] === 'undefined') { - options[i] = CONST.defaultRenderOptions[i]; - } - } - } - else { - options = CONST.defaultRenderOptions; - } - - /** - * The renderer type. - * - * @member {number} - */ - this.type = CONST.CANVAS_RENDERER; - - /** - * The resolution of the canvas. - * - * @member {number} - */ - this.resolution = options.resolution; - - /** - * This sets if the CanvasRenderer will clear the canvas or not before the new render pass. - * If the scene is NOT transparent Pixi will use a canvas sized fillRect operation every frame to set the canvas background color. - * If the scene is transparent Pixi will use clearRect to clear the canvas every frame. - * Disable this by setting this to false. For example if your game has a canvas filling background image you often don't need this set. - * - * @member {boolean} - * @default - */ - this.clearBeforeRender = options.clearBeforeRender; - - /** - * The background color as a number. - * - * @member {number} - * @private - */ - this._backgroundColor = 0x000000; - - /** - * The background color as a string. - * - * @member {string} - * @private - */ - this._backgroundColorString = '#000000'; - - this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter - - /** - * Whether the render view is transparent - * - * @member {boolean} - */ - this.transparent = options.transparent; - - /** - * Whether the render view should be resized automatically - * - * @member {boolean} - */ - this.autoResize = options.autoResize || false; - - - /** - * The width of the canvas view - * - * @member {number} - * @default 800 - */ - this.width = width || 800; - - /** - * The height of the canvas view - * - * @member {number} - * @default 600 - */ - this.height = height || 600; - - this.width *= this.resolution; - this.height *= this.resolution; - - /** - * The canvas element that everything is drawn to. - * - * @member {HTMLCanvasElement} - */ - this.view = options.view || document.createElement('canvas'); - - /** - * The canvas 2d context that everything is drawn with - * @member {CanvasRenderingContext2D} - */ - this.context = this.view.getContext('2d', { alpha: this.transparent }); - - /** - * Boolean flag controlling canvas refresh. - * - * @member {boolean} - */ - this.refresh = true; - - this.view.width = this.width * this.resolution; - this.view.height = this.height * this.resolution; - - /** - * Internal var. - * - * @member {number} - */ - this.count = 0; - - /** - * Instance of a CanvasMaskManager, handles masking when using the canvas renderer - * @member {CanvasMaskManager} - */ - this.maskManager = new CanvasMaskManager(); - - /** - * If true Pixi will Math.floor() x/y values when rendering, stopping pixel interpolation. - * Handy for crisp pixel art and speed on legacy devices. - * - * @member {boolean} - */ - this.roundPixels = false; - - this.scaleMode = null; - - this.smoothProperty = null; - - if (this.context.imageSmoothingEnabled) { - this.smoothProperty = 'imageSmoothingEnabled'; - } - else if (this.context.webkitImageSmoothingEnabled) { - this.smoothProperty = 'webkitImageSmoothingEnabled'; - } - else if (this.context.mozImageSmoothingEnabled) { - this.smoothProperty = 'mozImageSmoothingEnabled'; - } - else if (this.context.oImageSmoothingEnabled) { - this.smoothProperty = 'oImageSmoothingEnabled'; - } - else if (this.context.msImageSmoothingEnabled) { - this.smoothProperty = 'msImageSmoothingEnabled'; - } - - this.currentBlendMode = CONST.blendModes.NORMAL; - - this.blendModes = null; - - this._mapBlendModes(); - - this.resize(width, height); -} - -// constructor -CanvasRenderer.prototype.constructor = CanvasRenderer; -module.exports = CanvasRenderer; - -Object.defineProperties(CanvasRenderer.prototype, { - /** - * The background color to fill if not transparent - * - * @member {number} - * @memberof CanvasRenderer# - */ - backgroundColor: { - get: function () { - return this._backgroundColor; - }, - set: function (val) { - this._backgroundColor = val; - this._backgroundColorString = utils.hex2string(val); - } - } -}); - -/** - * Renders the object to this canvas view - * - * @param object {DisplayObject} the object to be rendered - */ -CanvasRenderer.prototype.render = function (object) { - object.updateTransform(); - - this.context.setTransform(1,0,0,1,0,0); - - this.context.globalAlpha = 1; - - this.currentBlendMode = CONST.blendModes.NORMAL; - this.context.globalCompositeOperation = this.blendModes[CONST.blendModes.NORMAL]; - - if (navigator.isCocoonJS && this.view.screencanvas) { - this.context.fillStyle = 'black'; - this.context.clear(); - } - - if (this.clearBeforeRender) { - if (this.transparent) { - this.context.clearRect(0, 0, this.width, this.height); - } - else { - this.context.fillStyle = this._backgroundColorString; - this.context.fillRect(0, 0, this.width , this.height); - } - } - - this.renderDisplayObject(object); -}; - -/** - * Removes everything from the renderer and optionally removes the Canvas DOM element. - * - * @param [removeView=false] {boolean} Removes the Canvas element from the DOM. - */ -CanvasRenderer.prototype.destroy = function (removeView) { - if (removeView && this.view.parent) { - this.view.parent.removeChild(this.view); - } - - this.view = null; - this.context = null; - this.maskManager = null; -}; - -/** - * Resizes the canvas view to the specified width and height - * - * @param width {number} the new width of the canvas view - * @param height {number} the new height of the canvas view - */ -CanvasRenderer.prototype.resize = function (width, height) { - this.width = width * this.resolution; - this.height = height * this.resolution; - - this.view.width = this.width; - this.view.height = this.height; - - if (this.autoResize) { - this.view.style.width = this.width / this.resolution + 'px'; - this.view.style.height = this.height / this.resolution + 'px'; - } -}; - -/** - * Renders a display object - * - * @param displayObject {DisplayObject} The displayObject to render - * @private - */ -CanvasRenderer.prototype.renderDisplayObject = function (displayObject) { - displayObject.renderCanvas(this); -}; - -/** - * Maps Pixi blend modes to canvas blend modes. - * - * @private - */ -CanvasRenderer.prototype._mapBlendModes = function () { - if (!this.blendModes) { - this.blendModes = {}; - - if (utils.canUseNewCanvasBlendModes()) { - this.blendModes[CONST.blendModes.NORMAL] = 'source-over'; - this.blendModes[CONST.blendModes.ADD] = 'lighter'; //IS THIS OK??? - this.blendModes[CONST.blendModes.MULTIPLY] = 'multiply'; - this.blendModes[CONST.blendModes.SCREEN] = 'screen'; - this.blendModes[CONST.blendModes.OVERLAY] = 'overlay'; - this.blendModes[CONST.blendModes.DARKEN] = 'darken'; - this.blendModes[CONST.blendModes.LIGHTEN] = 'lighten'; - this.blendModes[CONST.blendModes.COLOR_DODGE] = 'color-dodge'; - this.blendModes[CONST.blendModes.COLOR_BURN] = 'color-burn'; - this.blendModes[CONST.blendModes.HARD_LIGHT] = 'hard-light'; - this.blendModes[CONST.blendModes.SOFT_LIGHT] = 'soft-light'; - this.blendModes[CONST.blendModes.DIFFERENCE] = 'difference'; - this.blendModes[CONST.blendModes.EXCLUSION] = 'exclusion'; - this.blendModes[CONST.blendModes.HUE] = 'hue'; - this.blendModes[CONST.blendModes.SATURATION] = 'saturation'; - this.blendModes[CONST.blendModes.COLOR] = 'color'; - this.blendModes[CONST.blendModes.LUMINOSITY] = 'luminosity'; - } - else { - // this means that the browser does not support the cool new blend modes in canvas 'cough' ie 'cough' - this.blendModes[CONST.blendModes.NORMAL] = 'source-over'; - this.blendModes[CONST.blendModes.ADD] = 'lighter'; //IS THIS OK??? - this.blendModes[CONST.blendModes.MULTIPLY] = 'source-over'; - this.blendModes[CONST.blendModes.SCREEN] = 'source-over'; - this.blendModes[CONST.blendModes.OVERLAY] = 'source-over'; - this.blendModes[CONST.blendModes.DARKEN] = 'source-over'; - this.blendModes[CONST.blendModes.LIGHTEN] = 'source-over'; - this.blendModes[CONST.blendModes.COLOR_DODGE] = 'source-over'; - this.blendModes[CONST.blendModes.COLOR_BURN] = 'source-over'; - this.blendModes[CONST.blendModes.HARD_LIGHT] = 'source-over'; - this.blendModes[CONST.blendModes.SOFT_LIGHT] = 'source-over'; - this.blendModes[CONST.blendModes.DIFFERENCE] = 'source-over'; - this.blendModes[CONST.blendModes.EXCLUSION] = 'source-over'; - this.blendModes[CONST.blendModes.HUE] = 'source-over'; - this.blendModes[CONST.blendModes.SATURATION] = 'source-over'; - this.blendModes[CONST.blendModes.COLOR] = 'source-over'; - this.blendModes[CONST.blendModes.LUMINOSITY] = 'source-over'; - } - } -}; - -},{"../../const":4,"../../utils":50,"./utils/CanvasMaskManager":23}],21:[function(require,module,exports){ -/** - * Creates a Canvas element of the given size. - * - * @class - * @namespace PIXI - * @param width {number} the width for the newly created canvas - * @param height {number} the height for the newly created canvas - */ -function CanvasBuffer(width, height) { - /** - * The Canvas object that belongs to this CanvasBuffer. - * - * @member {HTMLCanvasElement} - */ - this.canvas = document.createElement('canvas'); - - /** - * A CanvasRenderingContext2D object representing a two-dimensional rendering context. - * - * @member {CanvasRenderingContext2D} - */ - this.context = this.canvas.getContext('2d'); - - this.canvas.width = width; - this.canvas.height = height; -} - -CanvasBuffer.prototype.constructor = CanvasBuffer; -module.exports = CanvasBuffer; - -Object.defineProperties(CanvasBuffer.prototype, { - /** - * The width of the canvas buffer in pixels. - * - * @member {number} - * @memberof CanvasBuffer# - */ - width: { - get: function () { - return this.canvas.width; - }, - set: function (val) { - this.canvas.width = val; - } - }, - /** - * The height of the canvas buffer in pixels. - * - * @member {number} - * @memberof CanvasBuffer# - */ - height: { - get: function () { - return this.canvas.height; - }, - set: function (val) { - this.canvas.height = val; - } - } -}); - -/** - * Clears the canvas that was created by the CanvasBuffer class. - * - * @private - */ -CanvasBuffer.prototype.clear = function () { - this.context.setTransform(1, 0, 0, 1, 0, 0); - this.context.clearRect(0,0, this.canvas.width, this.canvas.height); -}; - -/** - * Resizes the canvas to the specified width and height. - * - * @param width {number} the new width of the canvas - * @param height {number} the new height of the canvas - */ -CanvasBuffer.prototype.resize = function (width, height) { - this.canvas.width = width; - this.canvas.height = height; -}; - -},{}],22:[function(require,module,exports){ -var CONST = require('../../../const'); - -/** - * A set of functions used by the canvas renderer to draw the primitive graphics data. - * - * @namespace PIXI - */ -var CanvasGraphics = module.exports = {}; - -/* - * Renders a Graphics object to a canvas. - * - * @param graphics {Graphics} the actual graphics object to render - * @param context {CanvasRenderingContext2D} the 2d drawing method of the canvas - */ -CanvasGraphics.renderGraphics = function (graphics, context) { - var worldAlpha = graphics.worldAlpha; - - if (graphics.dirty) { - this.updateGraphicsTint(graphics); - graphics.dirty = false; - } - - for (var i = 0; i < graphics.graphicsData.length; i++) { - var data = graphics.graphicsData[i]; - var shape = data.shape; - - var fillColor = data._fillTint; - var lineColor = data._lineTint; - - context.lineWidth = data.lineWidth; - - if (data.type === CONST.SHAPES.POLY) { - context.beginPath(); - - var points = shape.points; - - context.moveTo(points[0], points[1]); - - for (var j=1; j < points.length/2; j++) { - context.lineTo(points[j * 2], points[j * 2 + 1]); - } - - if (shape.closed) { - context.lineTo(points[0], points[1]); - } - - // if the first and last point are the same close the path - much neater :) - if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) { - context.closePath(); - } - - if (data.fill) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - else if (data.type === CONST.SHAPES.RECT) { - - if (data.fillColor || data.fillColor === 0) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fillRect(shape.x, shape.y, shape.width, shape.height); - - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.strokeRect(shape.x, shape.y, shape.width, shape.height); - } - } - else if (data.type === CONST.SHAPES.CIRC) { - // TODO - need to be Undefined! - context.beginPath(); - context.arc(shape.x, shape.y, shape.radius,0,2*Math.PI); - context.closePath(); - - if (data.fill) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - else if (data.type === CONST.SHAPES.ELIP) { - // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas - - var w = shape.width * 2; - var h = shape.height * 2; - - var x = shape.x - w/2; - var y = shape.y - h/2; - - context.beginPath(); - - var kappa = 0.5522848, - ox = (w / 2) * kappa, // control point offset horizontal - oy = (h / 2) * kappa, // control point offset vertical - xe = x + w, // x-end - ye = y + h, // y-end - xm = x + w / 2, // x-middle - ym = y + h / 2; // y-middle - - context.moveTo(x, ym); - context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y); - context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym); - context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye); - context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym); - - context.closePath(); - - if (data.fill) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - else if (data.type === CONST.SHAPES.RREC) { - var rx = shape.x; - var ry = shape.y; - var width = shape.width; - var height = shape.height; - var radius = shape.radius; - - var maxRadius = Math.min(width, height) / 2 | 0; - radius = radius > maxRadius ? maxRadius : radius; - - context.beginPath(); - context.moveTo(rx, ry + radius); - context.lineTo(rx, ry + height - radius); - context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height); - context.lineTo(rx + width - radius, ry + height); - context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius); - context.lineTo(rx + width, ry + radius); - context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry); - context.lineTo(rx + radius, ry); - context.quadraticCurveTo(rx, ry, rx, ry + radius); - context.closePath(); - - if (data.fillColor || data.fillColor === 0) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - } -}; - -/* - * Renders a graphics mask - * - * @private - * @param graphics {Graphics} the graphics which will be used as a mask - * @param context {CanvasRenderingContext2D} the context 2d method of the canvas - */ -CanvasGraphics.renderGraphicsMask = function (graphics, context) { - var len = graphics.graphicsData.length; - - if (len === 0) { - return; - } - - context.beginPath(); - - for (var i = 0; i < len; i++) { - var data = graphics.graphicsData[i]; - var shape = data.shape; - - if (data.type === CONST.SHAPES.POLY) { - - var points = shape.points; - - context.moveTo(points[0], points[1]); - - for (var j=1; j < points.length/2; j++) { - context.lineTo(points[j * 2], points[j * 2 + 1]); - } - - // if the first and last point are the same close the path - much neater :) - if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) { - context.closePath(); - } - - } - else if (data.type === CONST.SHAPES.RECT) { - context.rect(shape.x, shape.y, shape.width, shape.height); - context.closePath(); - } - else if (data.type === CONST.SHAPES.CIRC) { - // TODO - need to be Undefined! - context.arc(shape.x, shape.y, shape.radius, 0, 2 * Math.PI); - context.closePath(); - } - else if (data.type === CONST.SHAPES.ELIP) { - - // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas - - var w = shape.width * 2; - var h = shape.height * 2; - - var x = shape.x - w/2; - var y = shape.y - h/2; - - var kappa = 0.5522848, - ox = (w / 2) * kappa, // control point offset horizontal - oy = (h / 2) * kappa, // control point offset vertical - xe = x + w, // x-end - ye = y + h, // y-end - xm = x + w / 2, // x-middle - ym = y + h / 2; // y-middle - - context.moveTo(x, ym); - context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y); - context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym); - context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye); - context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym); - context.closePath(); - } - else if (data.type === CONST.SHAPES.RREC) { - - var rx = shape.x; - var ry = shape.y; - var width = shape.width; - var height = shape.height; - var radius = shape.radius; - - var maxRadius = Math.min(width, height) / 2 | 0; - radius = radius > maxRadius ? maxRadius : radius; - - context.moveTo(rx, ry + radius); - context.lineTo(rx, ry + height - radius); - context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height); - context.lineTo(rx + width - radius, ry + height); - context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius); - context.lineTo(rx + width, ry + radius); - context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry); - context.lineTo(rx + radius, ry); - context.quadraticCurveTo(rx, ry, rx, ry + radius); - context.closePath(); - } - } -}; - -CanvasGraphics.updateGraphicsTint = function (graphics) { - if (graphics.tint === 0xFFFFFF) { - return; - } - - var tintR = (graphics.tint >> 16 & 0xFF) / 255; - var tintG = (graphics.tint >> 8 & 0xFF) / 255; - var tintB = (graphics.tint & 0xFF)/ 255; - - for (var i = 0; i < graphics.graphicsData.length; i++) { - var data = graphics.graphicsData[i]; - - var fillColor = data.fillColor | 0; - var lineColor = data.lineColor | 0; - - /* - var colorR = (fillColor >> 16 & 0xFF) / 255; - var colorG = (fillColor >> 8 & 0xFF) / 255; - var colorB = (fillColor & 0xFF) / 255; - - colorR *= tintR; - colorG *= tintG; - colorB *= tintB; - - fillColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255); - - colorR = (lineColor >> 16 & 0xFF) / 255; - colorG = (lineColor >> 8 & 0xFF) / 255; - colorB = (lineColor & 0xFF) / 255; - - colorR *= tintR; - colorG *= tintG; - colorB *= tintB; - - lineColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255); - */ - - // super inline cos im an optimization NAZI :) - data._fillTint = (((fillColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((fillColor >> 8 & 0xFF) / 255 * tintG*255 << 8) + (fillColor & 0xFF) / 255 * tintB*255); - data._lineTint = (((lineColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((lineColor >> 8 & 0xFF) / 255 * tintG*255 << 8) + (lineColor & 0xFF) / 255 * tintB*255); - - } -}; - - -},{"../../../const":4}],23:[function(require,module,exports){ -var CanvasGraphics = require('./CanvasGraphics'); - -/** - * A set of functions used to handle masking. - * - * @class - * @namespace PIXI - */ -function CanvasMaskManager() {} - -CanvasMaskManager.prototype.constructor = CanvasMaskManager; -module.exports = CanvasMaskManager; - -/** - * This method adds it to the current stack of masks. - * - * @param maskData {object} the maskData that will be pushed - * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use. - */ -CanvasMaskManager.prototype.pushMask = function (maskData, renderer) { - renderer.context.save(); - - var cacheAlpha = maskData.alpha; - var transform = maskData.worldTransform; - var resolution = renderer.resolution; - - renderer.context.setTransform( - transform.a * resolution, - transform.b * resolution, - transform.c * resolution, - transform.d * resolution, - transform.tx * resolution, - transform.ty * resolution - ); - - CanvasGraphics.renderGraphicsMask(maskData, renderer.context); - - renderer.context.clip(); - - maskData.worldAlpha = cacheAlpha; -}; - -/** - * Restores the current drawing context to the state it was before the mask was applied. - * - * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use. - */ -CanvasMaskManager.prototype.popMask = function (renderer) { - renderer.context.restore(); -}; - -},{"./CanvasGraphics":22}],24:[function(require,module,exports){ -var utils = require('../../../utils'); - -/** - * Utility methods for Sprite/Texture tinting. - * - * @namespace PIXI - */ -var CanvasTinter = module.exports = {}; - -/** - * Basically this method just needs a sprite and a color and tints the sprite with the given color. - * - * @param sprite {Sprite} the sprite to tint - * @param color {number} the color to use to tint the sprite with - * @return {HTMLCanvasElement} The tinted canvas - */ -CanvasTinter.getTintedTexture = function (sprite, color) { - var texture = sprite.texture; - - color = CanvasTinter.roundColor(color); - - var stringColor = '#' + ('00000' + ( color | 0).toString(16)).substr(-6); - - texture.tintCache = texture.tintCache || {}; - - if (texture.tintCache[stringColor]) { - return texture.tintCache[stringColor]; - } - - // clone texture.. - var canvas = CanvasTinter.canvas || document.createElement('canvas'); - - //CanvasTinter.tintWithPerPixel(texture, stringColor, canvas); - CanvasTinter.tintMethod(texture, color, canvas); - - if (CanvasTinter.convertTintToImage) { - // is this better? - var tintImage = new Image(); - tintImage.src = canvas.toDataURL(); - - texture.tintCache[stringColor] = tintImage; - } - else { - texture.tintCache[stringColor] = canvas; - // if we are not converting the texture to an image then we need to lose the reference to the canvas - CanvasTinter.canvas = null; - } - - return canvas; -}; - -/** - * Tint a texture using the 'multiply' operation. - * - * @param texture {Texture} the texture to tint - * @param color {number} the color to use to tint the sprite with - * @param canvas {HTMLCanvasElement} the current canvas - */ -CanvasTinter.tintWithMultiply = function (texture, color, canvas) { - var context = canvas.getContext( '2d' ); - - var crop = texture.crop; - - canvas.width = crop.width; - canvas.height = crop.height; - - context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6); - - context.fillRect(0, 0, crop.width, crop.height); - - context.globalCompositeOperation = 'multiply'; - - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); - - context.globalCompositeOperation = 'destination-atop'; - - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); -}; - -/** - * Tint a texture using the 'overlay' operation. - * - * @param texture {Texture} the texture to tint - * @param color {number} the color to use to tint the sprite with - * @param canvas {HTMLCanvasElement} the current canvas - */ -CanvasTinter.tintWithOverlay = function (texture, color, canvas) { - var context = canvas.getContext( '2d' ); - - var crop = texture.crop; - - canvas.width = crop.width; - canvas.height = crop.height; - - context.globalCompositeOperation = 'copy'; - context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6); - context.fillRect(0, 0, crop.width, crop.height); - - context.globalCompositeOperation = 'destination-atop'; - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); - - // context.globalCompositeOperation = 'copy'; -}; - -/** - * Tint a texture pixel per pixel. - * - * @param texture {Texture} the texture to tint - * @param color {number} the color to use to tint the sprite with - * @param canvas {HTMLCanvasElement} the current canvas - */ -CanvasTinter.tintWithPerPixel = function (texture, color, canvas) { - var context = canvas.getContext( '2d' ); - - var crop = texture.crop; - - canvas.width = crop.width; - canvas.height = crop.height; - - context.globalCompositeOperation = 'copy'; - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); - - var rgbValues = utils.hex2rgb(color); - var r = rgbValues[0], g = rgbValues[1], b = rgbValues[2]; - - var pixelData = context.getImageData(0, 0, crop.width, crop.height); - - var pixels = pixelData.data; - - for (var i = 0; i < pixels.length; i += 4) { - pixels[i+0] *= r; - pixels[i+1] *= g; - pixels[i+2] *= b; - } - - context.putImageData(pixelData, 0, 0); -}; - -/** - * Rounds the specified color according to the CanvasTinter.cacheStepsPerColorChannel. - * - * @param color {number} the color to round, should be a hex color - */ -CanvasTinter.roundColor = function (color) { - var step = CanvasTinter.cacheStepsPerColorChannel; - - var rgbValues = utils.hex2rgb(color); - - rgbValues[0] = Math.min(255, (rgbValues[0] / step) * step); - rgbValues[1] = Math.min(255, (rgbValues[1] / step) * step); - rgbValues[2] = Math.min(255, (rgbValues[2] / step) * step); - - return utils.rgb2hex(rgbValues); -}; - -/** - * Number of steps which will be used as a cap when rounding colors. - * - * @member - */ -CanvasTinter.cacheStepsPerColorChannel = 8; - -/** - * Tint cache boolean flag. - * - * @member - */ -CanvasTinter.convertTintToImage = false; - -/** - * Whether or not the Canvas BlendModes are supported, consequently the ability to tint using the multiply method. - * - * @member - */ -CanvasTinter.canUseMultiply = utils.canUseNewCanvasBlendModes(); - -/** - * The tinting method that will be used. - * - */ -CanvasTinter.tintMethod = CanvasTinter.canUseMultiply ? CanvasTinter.tintWithMultiply : CanvasTinter.tintWithPerPixel; - -},{"../../../utils":50}],25:[function(require,module,exports){ -var WebGLSpriteBatch = require('./utils/WebGLSpriteBatch'), - WebGLFastSpriteBatch = require('./utils/WebGLFastSpriteBatch'), - WebGLShaderManager = require('./managers/WebGLShaderManager'), - WebGLMaskManager = require('./managers/WebGLMaskManager'), - WebGLFilterManager = require('./managers/WebGLFilterManager'), - WebGLStencilManager = require('./managers/WebGLStencilManager'), - WebGLBlendModeManager = require('./managers/WebGLBlendModeManager'), - math = require('../../math'), - utils = require('../../utils'), - CONST = require('../../const'); - -/** - * The WebGLRenderer draws the scene and all its content onto a webGL enabled canvas. This renderer - * should be used for browsers that support webGL. This Render works by automatically managing webGLBatchs. - * So no need for Sprite Batches or Sprite Clouds. - * Don't forget to add the view to your DOM or you will not see anything :) - * - * @class - * @namespace PIXI - * @param [width=0] {number} the width of the canvas view - * @param [height=0] {number} the height of the canvas view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false - * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment) - * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you need to call toDataUrl on the webgl context - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - */ -function WebGLRenderer(width, height, options) { - utils.sayHello('webGL'); - - if (options) { - for (var i in CONST.defaultRenderOptions) { - if (typeof options[i] === 'undefined') { - options[i] = CONST.defaultRenderOptions[i]; - } - } - } - else { - options = CONST.defaultRenderOptions; - } - - this.uuid = utils.uuid(); - - /** - * @member {number} - */ - this.type = CONST.WEBGL_RENDERER; - - /** - * The resolution of the renderer - * - * @member {number} - * @default 1 - */ - this.resolution = options.resolution; - - // do a catch.. only 1 webGL renderer.. - - /** - * Whether the render view is transparent - * - * @member {boolean} - */ - this.transparent = options.transparent; - - /** - * The background color as a number. - * - * @member {number} - * @private - */ - this._backgroundColor = 0x000000; - - /** - * The background color as an [R, G, B] array. - * - * @member {number[]} - * @private - */ - this._backgroundColorRgb = [0, 0, 0]; - - this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter - - /** - * Whether the render view should be resized automatically - * - * @member {boolean} - */ - this.autoResize = options.autoResize || false; - - /** - * The value of the preserveDrawingBuffer flag affects whether or not the contents of the stencil buffer is retained after rendering. - * - * @member {boolean} - */ - this.preserveDrawingBuffer = options.preserveDrawingBuffer; - - /** - * This sets if the WebGLRenderer will clear the context texture or not before the new render pass. If true: - * If the renderer is NOT transparent, Pixi will clear to alpha (0, 0, 0, 0). - * If the renderer is transparent, Pixi will clear to the target Stage's background color. - * Disable this by setting this to false. For example: if your game has a canvas filling background image, you often don't need this set. - * - * @member {boolean} - * @default - */ - this.clearBeforeRender = options.clearBeforeRender; - - /** - * The width of the canvas view - * - * @member {number} - * @default 800 - */ - this.width = width || 800; - - /** - * The height of the canvas view - * - * @member {number} - * @default 600 - */ - this.height = height || 600; - - /** - * The canvas element that everything is drawn to - * - * @member {HTMLCanvasElement} - */ - this.view = options.view || document.createElement( 'canvas' ); - - // deal with losing context.. - - /** - * @member {Function} - */ - this.contextLostBound = this.handleContextLost.bind(this); - - /** - * @member {Function} - */ - this.contextRestoredBound = this.handleContextRestored.bind(this); - - this.view.addEventListener('webglcontextlost', this.contextLostBound, false); - this.view.addEventListener('webglcontextrestored', this.contextRestoredBound, false); - - /** - * @member {object} - * @private - */ - this._contextOptions = { - alpha: this.transparent, - antialias: options.antialias, // SPEED UP?? - premultipliedAlpha:this.transparent && this.transparent !== 'notMultiplied', - stencil:true, - preserveDrawingBuffer: options.preserveDrawingBuffer - }; - - /** - * @member {Point} - */ - this.projection = new math.Point(); - - /** - * @member {Point} - */ - this.offset = new math.Point(0, 0); - - /** - * Counter for the number of draws made each frame - * - * @member {number} - */ - this.drawCount = 0; - - // time to create the render managers! each one focuses on managing a state in webGL - - /** - * Deals with managing the shader programs and their attribs - * @member {WebGLShaderManager} - */ - this.shaderManager = new WebGLShaderManager(this); - - /** - * Manages the rendering of sprites - * @member {WebGLSpriteBatch} - */ - this.spriteBatch = new WebGLSpriteBatch(this); - - /** - * Manages the rendering of sprites - * @member {WebGLFastSpriteBatch} - */ - this.fastSpriteBatch = new WebGLFastSpriteBatch(this); - - /** - * Manages the masks using the stencil buffer - * @member {WebGLMaskManager} - */ - this.maskManager = new WebGLMaskManager(this); - - /** - * Manages the filters - * @member {WebGLFilterManager} - */ - this.filterManager = new WebGLFilterManager(this); - - /** - * Manages the stencil buffer - * @member {WebGLStencilManager} - */ - this.stencilManager = new WebGLStencilManager(this); - - /** - * Manages the blendModes - * @member {WebGLBlendModeManager} - */ - this.blendModeManager = new WebGLBlendModeManager(this); - - this.blendModes = null; - - this._boundUpdateTexture = this.updateTexture.bind(this); - this._boundDestroyTexture = this.destroyTexture.bind(this); - - // time init the context.. - this._initContext(); - - // map some webGL blend modes.. - this._mapBlendModes(); -} - -// constructor -WebGLRenderer.prototype.constructor = WebGLRenderer; -module.exports = WebGLRenderer; - -utils.eventTarget.mixin(WebGLRenderer.prototype); - -Object.defineProperties(WebGLRenderer.prototype, { - /** - * The background color to fill if not transparent - * - * @member {number} - * @memberof WebGLRenderer# - */ - backgroundColor: { - get: function () { - return this._backgroundColor; - }, - set: function (val) { - this._backgroundColor = val; - utils.hex2rgb(val, this._backgroundColorRgb); - } - } -}); - -/** - * - * @private - */ -WebGLRenderer.prototype._initContext = function () { - var gl = this.view.getContext('webgl', this._contextOptions) || this.view.getContext('experimental-webgl', this._contextOptions); - this.gl = gl; - - if (!gl) { - // fail, not able to get a context - throw new Error('This browser does not support webGL. Try using the canvas renderer'); - } - - this.glContextId = WebGLRenderer.glContextId++; - gl.id = this.glContextId; - gl.renderer = this; - - // set up the default pixi settings.. - gl.disable(gl.DEPTH_TEST); - gl.disable(gl.CULL_FACE); - gl.enable(gl.BLEND); - - this.emit('context', gl); - - // now resize and we are good to go! - this.resize(this.width, this.height); -}; - -/** - * Renders the object to its webGL view - * - * @param object {DisplayObject} the object to be rendered - */ -WebGLRenderer.prototype.render = function (object) { - // no point rendering if our context has been blown up! - if (this.gl.isContextLost()) { - return; - } - - // update the scene graph - object.updateTransform(); - - var gl = this.gl; - - // -- Does this need to be set every frame? -- // - gl.viewport(0, 0, this.width, this.height); - - // make sure we are bound to the main frame buffer - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - - if (this.clearBeforeRender) { - if (this.transparent) { - gl.clearColor(0, 0, 0, 0); - } - else { - gl.clearColor(this._backgroundColorRgb[0], this._backgroundColorRgb[1], this._backgroundColorRgb[2], 1); - } - - gl.clear(gl.COLOR_BUFFER_BIT); - } - - this.renderDisplayObject(object, this.projection); -}; - -/** - * Renders a Display Object. - * - * @param displayObject {DisplayObject} The DisplayObject to render - * @param projection {Point} The projection - * @param buffer {Array} a standard WebGL buffer - */ -WebGLRenderer.prototype.renderDisplayObject = function (displayObject, projection, buffer) { - this.blendModeManager.setBlendMode(CONST.blendModes.NORMAL); - - // reset the render session data.. - this.drawCount = 0; - - // make sure to flip the Y if using a render texture.. - this.flipY = buffer ? -1 : 1; - - // set the default projection - this.projection = projection; - - //set the default offset - this.offset = this.offset; - - // start the sprite batch - this.spriteBatch.begin(); - - // start the filter manager - this.filterManager.begin(buffer); - - // render the scene! - displayObject.renderWebGL(this); - - // finish the sprite batch - this.spriteBatch.end(); -}; - -/** - * Resizes the webGL view to the specified width and height. - * - * @param width {number} the new width of the webGL view - * @param height {number} the new height of the webGL view - */ -WebGLRenderer.prototype.resize = function (width, height) { - this.width = width * this.resolution; - this.height = height * this.resolution; - - this.view.width = this.width; - this.view.height = this.height; - - if (this.autoResize) { - this.view.style.width = this.width / this.resolution + 'px'; - this.view.style.height = this.height / this.resolution + 'px'; - } - - this.gl.viewport(0, 0, this.width, this.height); - - this.projection.x = this.width / 2 / this.resolution; - this.projection.y = -this.height / 2 / this.resolution; -}; - -/** - * Updates and/or Creates a WebGL texture for the renderer's context. - * - * @param texture {BaseTexture|Texture} the texture to update - */ -WebGLRenderer.prototype.updateTexture = function (texture) { - texture = texture.baseTexture || texture; - - if (!texture.hasLoaded) { - return; - } - - var gl = this.gl; - - if (!texture._glTextures[gl.id]) { - texture._glTextures[gl.id] = gl.createTexture(); - texture.on('update', this._boundUpdateTexture); - texture.on('dispose', this._boundDestroyTexture); - } - - gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]); - - gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, texture.premultipliedAlpha); - gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, texture.source); - - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - - - if (texture.mipmap && utils.isPowerOfTwo(texture.width, texture.height)) { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR_MIPMAP_LINEAR : gl.NEAREST_MIPMAP_NEAREST); - gl.generateMipmap(gl.TEXTURE_2D); - } - else { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - } - - if (!texture._powerOf2) { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); - } - else { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT); - } - - return texture._glTextures[gl.id]; -}; - -WebGLRenderer.prototype.destroyTexture = function (texture) { - texture = texture.baseTexture || texture; - - if (!texture.hasLoaded) { - return; - } - - if (texture._glTextures[this.gl.id]) { - this.gl.deleteTexture(texture._glTextures[this.gl.id]); - } -}; - -/** - * Handles a lost webgl context - * - * @param event {Event} - * @private - */ -WebGLRenderer.prototype.handleContextLost = function (event) { - event.preventDefault(); -}; - -/** - * Handles a restored webgl context - * - * @param event {Event} - * @private - */ -WebGLRenderer.prototype.handleContextRestored = function () { - this._initContext(); - - // empty all the ol gl textures as they are useless now - for (var key in utils.TextureCache) { - var texture = utils.TextureCache[key].baseTexture; - texture._glTextures = []; - } -}; - -/** - * Removes everything from the renderer (event listeners, spritebatch, etc...) - * - * @param [removeView=false] {boolean} Removes the Canvas element from the DOM. - */ -WebGLRenderer.prototype.destroy = function (removeView) { - if (removeView && this.view.parent) { - this.view.parent.removeChild(this.view); - } - - // remove listeners - this.view.removeEventListener('webglcontextlost', this.contextLostBound); - this.view.removeEventListener('webglcontextrestored', this.contextRestoredBound); - - // time to create the render managers! each one focuses on managine a state in webGL - this.shaderManager.destroy(); - this.spriteBatch.destroy(); - this.maskManager.destroy(); - this.filterManager.destroy(); - - - // this.uuid = utils.uuid(); - // this.type = CONST.WEBGL_RENDERER; - - // this.resolution = options.resolution; - // this.transparent = options.transparent; - - this._backgroundColor = 0x000000; - this._backgroundColorRgb = null; - - // this.backgroundColor = null; - // this.autoResize = options.autoResize || false; - // this.preserveDrawingBuffer = options.preserveDrawingBuffer; - // this.clearBeforeRender = options.clearBeforeRender; - // this.width = width || 800; - // this.height = height || 600; - - this.view = null; - - this.contextLostBound = null; - this.contextRestoredBound = null; - - this._contextOptions = null; - - this.projection = null; - this.offset = null; - this.drawCount = 0; - - this.shaderManager = null; - this.spriteBatch = null; - this.maskManager = null; - this.filterManager = null; - this.stencilManager = null; - this.blendModeManager = null; - - this.blendModes = null; - - this.gl = null; - this.blendModes = null; -}; - -/** - * Maps Pixi blend modes to WebGL blend modes. - * - * @private - */ -WebGLRenderer.prototype._mapBlendModes = function () { - var gl = this.gl; - - if (!this.blendModes) { - this.blendModes = {}; - - this.blendModes[CONST.blendModes.NORMAL] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.ADD] = [gl.SRC_ALPHA, gl.DST_ALPHA]; - this.blendModes[CONST.blendModes.MULTIPLY] = [gl.DST_COLOR, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.SCREEN] = [gl.SRC_ALPHA, gl.ONE]; - this.blendModes[CONST.blendModes.OVERLAY] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.DARKEN] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.LIGHTEN] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.COLOR_DODGE] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.COLOR_BURN] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.HARD_LIGHT] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.SOFT_LIGHT] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.DIFFERENCE] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.EXCLUSION] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.HUE] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.SATURATION] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.COLOR] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.LUMINOSITY] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - } -}; - -WebGLRenderer.glContextId = 0; - -},{"../../const":4,"../../math":12,"../../utils":50,"./managers/WebGLBlendModeManager":26,"./managers/WebGLFilterManager":27,"./managers/WebGLMaskManager":29,"./managers/WebGLShaderManager":30,"./managers/WebGLStencilManager":31,"./utils/WebGLFastSpriteBatch":38,"./utils/WebGLSpriteBatch":41}],26:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLBlendModeManager(renderer) { - WebGLManager.call(this, renderer); - - /** - * @member {number} - */ - this.currentBlendMode = 99999; -} - -WebGLBlendModeManager.prototype = Object.create(WebGLManager.prototype); -WebGLBlendModeManager.prototype.constructor = WebGLBlendModeManager; -module.exports = WebGLBlendModeManager; - -/** - * Sets-up the given blendMode from WebGL's point of view. - * - * @param blendMode {number} the blendMode, should be a Pixi const, such as BlendModes.ADD - */ -WebGLBlendModeManager.prototype.setBlendMode = function (blendMode) { - if (this.currentBlendMode === blendMode) { - return false; - } - - this.currentBlendMode = blendMode; - - var mode = this.renderer.blendModes[this.currentBlendMode]; - this.renderer.gl.blendFunc(mode[0], mode[1]); - - return true; -}; - -},{"./WebGLManager":28}],27:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - FilterTexture = require('../utils/FilterTexture'), - Shader = require('../shaders/Shader'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLFilterManager(renderer) { - WebGLManager.call(this, renderer); - - /** - * @member {any[]} - */ - this.filterStack = []; - - /** - * @member {any[]]} - */ - this.texturePool = []; - - /** - * @member {number} - */ - this.offsetX = 0; - - /** - * @member {number} - */ - this.offsetY = 0; - - // listen for context and update necessary buffers - var self = this; - this.renderer.on('context', function () { - self.texturePool.length = 0; - self.initShaderBuffers(); - }); -} - -WebGLFilterManager.prototype = Object.create(WebGLManager.prototype); -WebGLFilterManager.prototype.constructor = WebGLFilterManager; -module.exports = WebGLFilterManager; - -/** - * @param renderer {WebGLRenderer} - * @param buffer {ArrayBuffer} - */ -WebGLFilterManager.prototype.begin = function (buffer) { - this.defaultShader = this.renderer.shaderManager.defaultShader; - - this.width = this.renderer.projection.x * 2; - this.height = -this.renderer.projection.y * 2; - - this.buffer = buffer; -}; - -/** - * Applies the filter and adds it to the current filter stack. - * - * @param filterBlock {object} the filter that will be pushed to the current filter stack - */ -WebGLFilterManager.prototype.pushFilter = function (filterBlock) { - var gl = this.renderer.gl; - - var projection = this.renderer.projection; - var offset = this.renderer.offset; - - filterBlock._filterArea = filterBlock.target.filterArea || filterBlock.target.getBounds(); - - // filter program - // OPTIMISATION - the first filter is free if its a simple color change? - this.filterStack.push(filterBlock); - - var filter = filterBlock.filterPasses[0]; - - this.offsetX += filterBlock._filterArea.x; - this.offsetY += filterBlock._filterArea.y; - - var texture = this.texturePool.pop(); - if (!texture) { - texture = new FilterTexture(this.renderer.gl, this.width, this.height); - } - else { - texture.resize(this.width, this.height); - } - - gl.bindTexture(gl.TEXTURE_2D, texture.texture); - - var filterArea = filterBlock._filterArea;// filterBlock.target.getBounds();///filterBlock.target.filterArea; - - var padding = filter.padding; - filterArea.x -= padding; - filterArea.y -= padding; - filterArea.width += padding * 2; - filterArea.height += padding * 2; - - var localX = filterArea.x, - localY = filterArea.y; - - if (filterArea.x < 0) { - filterArea.width += filterArea.x; - filterArea.x = 0; - } - - if (filterArea.y < 0) { - filterArea.height += filterArea.y; - filterArea.y = 0; - } - - if (localX + filterArea.width > this.width) { - filterArea.width = this.width - localX; - } - - if (localY + filterArea.height > this.height) { - filterArea.height = this.height - localY; - } - - if (filterArea.width < 0) { - filterArea.width = 0; - } - - if (filterArea.height < 0) { - filterArea.height = 0; - } - - //gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, filterArea.width, filterArea.height, 0, gl.RGBA, gl.UNSIGNED_BYTE, null); - gl.bindFramebuffer(gl.FRAMEBUFFER, texture.frameBuffer); - - // set view port - gl.viewport(0, 0, filterArea.width, filterArea.height); - - projection.x = filterArea.width/2; - projection.y = -filterArea.height/2; - - offset.x = -filterArea.x; - offset.y = -filterArea.y; - - // update projection - // now restore the regular shader.. - // this.renderer.shaderManager.setShader(this.defaultShader); - //gl.uniform2f(this.defaultShader.projectionVector, filterArea.width/2, -filterArea.height/2); - //gl.uniform2f(this.defaultShader.offsetVector, -filterArea.x, -filterArea.y); - - gl.colorMask(true, true, true, true); - gl.clearColor(0,0,0, 0); - gl.clear(gl.COLOR_BUFFER_BIT); - - filterBlock._glFilterTexture = texture; - -}; - -/** - * Removes the last filter from the filter stack and doesn't return it. - * - */ -WebGLFilterManager.prototype.popFilter = function () { - var gl = this.renderer.gl; - - var filterBlock = this.filterStack.pop(); - var filterArea = filterBlock._filterArea; - var texture = filterBlock._glFilterTexture; - var projection = this.renderer.projection; - var offset = this.renderer.offset; - - if (filterBlock.filterPasses.length > 1) { - gl.viewport(0, 0, filterArea.width, filterArea.height); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - - this.vertexArray[0] = 0; - this.vertexArray[1] = filterArea.height; - - this.vertexArray[2] = filterArea.width; - this.vertexArray[3] = filterArea.height; - - this.vertexArray[4] = 0; - this.vertexArray[5] = 0; - - this.vertexArray[6] = filterArea.width; - this.vertexArray[7] = 0; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - // now set the uvs.. - this.uvArray[2] = filterArea.width/this.width; - this.uvArray[5] = filterArea.height/this.height; - this.uvArray[6] = filterArea.width/this.width; - this.uvArray[7] = filterArea.height/this.height; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray); - - var inputTexture = texture; - var outputTexture = this.texturePool.pop(); - if (!outputTexture) { - outputTexture = new FilterTexture(this.renderer.gl, this.width, this.height); - } - outputTexture.resize(this.width, this.height); - - // need to clear this FBO as it may have some left over elements from a previous filter. - gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer ); - gl.clear(gl.COLOR_BUFFER_BIT); - - gl.disable(gl.BLEND); - - for (var i = 0; i < filterBlock.filterPasses.length-1; i++) { - var filterPass = filterBlock.filterPasses[i]; - - gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer ); - - // set texture - gl.activeTexture(gl.TEXTURE0); - gl.bindTexture(gl.TEXTURE_2D, inputTexture.texture); - - // draw texture.. - //filterPass.applyFilterPass(filterArea.width, filterArea.height); - this.applyFilterPass(filterPass, filterArea, filterArea.width, filterArea.height); - - // swap the textures.. - var temp = inputTexture; - inputTexture = outputTexture; - outputTexture = temp; - } - - gl.enable(gl.BLEND); - - texture = inputTexture; - this.texturePool.push(outputTexture); - } - - var filter = filterBlock.filterPasses[filterBlock.filterPasses.length-1]; - - this.offsetX -= filterArea.x; - this.offsetY -= filterArea.y; - - var sizeX = this.width; - var sizeY = this.height; - - var offsetX = 0; - var offsetY = 0; - - var buffer = this.buffer; - - // time to render the filters texture to the previous scene - if (this.filterStack.length === 0) { - gl.colorMask(true, true, true, true);//this.transparent); - } - else { - var currentFilter = this.filterStack[this.filterStack.length-1]; - filterArea = currentFilter._filterArea; - - sizeX = filterArea.width; - sizeY = filterArea.height; - - offsetX = filterArea.x; - offsetY = filterArea.y; - - buffer = currentFilter._glFilterTexture.frameBuffer; - } - - // TODO need to remove these global elements.. - projection.x = sizeX/2; - projection.y = -sizeY/2; - - offset.x = offsetX; - offset.y = offsetY; - - filterArea = filterBlock._filterArea; - - var x = filterArea.x-offsetX; - var y = filterArea.y-offsetY; - - // update the buffers.. - // make sure to flip the y! - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - - this.vertexArray[0] = x; - this.vertexArray[1] = y + filterArea.height; - - this.vertexArray[2] = x + filterArea.width; - this.vertexArray[3] = y + filterArea.height; - - this.vertexArray[4] = x; - this.vertexArray[5] = y; - - this.vertexArray[6] = x + filterArea.width; - this.vertexArray[7] = y; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - - this.uvArray[2] = filterArea.width/this.width; - this.uvArray[5] = filterArea.height/this.height; - this.uvArray[6] = filterArea.width/this.width; - this.uvArray[7] = filterArea.height/this.height; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray); - - gl.viewport(0, 0, sizeX, sizeY); - - // bind the buffer - gl.bindFramebuffer(gl.FRAMEBUFFER, buffer ); - - // set the blend mode! - //gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA) - - // set texture - gl.activeTexture(gl.TEXTURE0); - gl.bindTexture(gl.TEXTURE_2D, texture.texture); - - // apply! - this.applyFilterPass(filter, filterArea, sizeX, sizeY); - - // now restore the regular shader.. should happen automatically now.. - // this.renderer.shaderManager.setShader(this.defaultShader); - // gl.uniform2f(this.defaultShader.projectionVector, sizeX/2, -sizeY/2); - // gl.uniform2f(this.defaultShader.offsetVector, -offsetX, -offsetY); - - // return the texture to the pool - this.texturePool.push(texture); - filterBlock._glFilterTexture = null; -}; - - -/** - * Applies the filter to the specified area. - * - * @param filter {AbstractFilter} the filter that needs to be applied - * @param filterArea {Texture} TODO - might need an update - * @param width {number} the horizontal range of the filter - * @param height {number} the vertical range of the filter - */ -WebGLFilterManager.prototype.applyFilterPass = function (filter, filterArea, width, height) { - // use program - var gl = this.renderer.gl; - - var shader = filter.shaders[gl.id]; - - if (!shader) { - shader = new Shader(gl); - - shader.fragmentSrc = filter.fragmentSrc; - shader.uniforms = filter.uniforms; - shader.init(); - - filter.shaders[gl.id] = shader; - } - - // set the shader - this.renderer.shaderManager.setShader(shader); - -// gl.useProgram(shader.program); - - gl.uniform2f(shader.projectionVector, width/2, -height/2); - gl.uniform2f(shader.offsetVector, 0,0); - - if (filter.uniforms.dimensions) { - filter.uniforms.dimensions.value[0] = this.width;//width; - filter.uniforms.dimensions.value[1] = this.height;//height; - filter.uniforms.dimensions.value[2] = this.vertexArray[0]; - filter.uniforms.dimensions.value[3] = this.vertexArray[5];//filterArea.height; - } - - shader.syncUniforms(); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer); - gl.vertexAttribPointer(shader.aColor, 2, gl.FLOAT, false, 0, 0); - - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - - // draw the filter... - gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0 ); - - this.renderer.drawCount++; -}; - -/** - * Initialises the shader buffers. - * - */ -WebGLFilterManager.prototype.initShaderBuffers = function () { - var gl = this.renderer.gl; - - // create some buffers - this.vertexBuffer = gl.createBuffer(); - this.uvBuffer = gl.createBuffer(); - this.colorBuffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - - // bind and upload the vertexs.. - // keep a reference to the vertexFloatData.. - this.vertexArray = new Float32Array([0.0, 0.0, - 1.0, 0.0, - 0.0, 1.0, - 1.0, 1.0]); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertexArray, gl.STATIC_DRAW); - - // bind and upload the uv buffer - this.uvArray = new Float32Array([0.0, 0.0, - 1.0, 0.0, - 0.0, 1.0, - 1.0, 1.0]); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.uvArray, gl.STATIC_DRAW); - - this.colorArray = new Float32Array([1.0, 0xFFFFFF, - 1.0, 0xFFFFFF, - 1.0, 0xFFFFFF, - 1.0, 0xFFFFFF]); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.colorArray, gl.STATIC_DRAW); - - // bind and upload the index - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array([0, 1, 2, 1, 3, 2]), gl.STATIC_DRAW); - -}; - -/** - * Destroys the filter and removes it from the filter stack. - * - */ -WebGLFilterManager.prototype.destroy = function () { - var gl = this.renderer.gl; - - this.filterStack = null; - - this.offsetX = 0; - this.offsetY = 0; - - // destroy textures - for (var i = 0; i < this.texturePool.length; i++) { - this.texturePool[i].destroy(); - } - - this.texturePool = null; - - //destroy buffers.. - gl.deleteBuffer(this.vertexBuffer); - gl.deleteBuffer(this.uvBuffer); - gl.deleteBuffer(this.colorBuffer); - gl.deleteBuffer(this.indexBuffer); - - this.renderer = null; -}; - -},{"../shaders/Shader":35,"../utils/FilterTexture":37,"./WebGLManager":28}],28:[function(require,module,exports){ -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLManager(renderer) { - /** - * The renderer this manager works for. - * - * @member {WebGLRenderer} - */ - this.renderer = renderer; -} - -WebGLManager.prototype.constructor = WebGLManager; -module.exports = WebGLManager; - -WebGLManager.prototype.destroy = function () { - this.renderer = null; -}; - -},{}],29:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - WebGLGraphics = require('../utils/WebGLGraphics'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLMaskManager(renderer) { - WebGLManager.call(this, renderer); -} - -WebGLMaskManager.prototype = Object.create(WebGLManager.prototype); -WebGLMaskManager.prototype.constructor = WebGLMaskManager; -module.exports = WebGLMaskManager; - -/** - * Applies the Mask and adds it to the current filter stack. - * - * @param maskData {any[]} - */ -WebGLMaskManager.prototype.pushMask = function (maskData) { - if (maskData.dirty) { - WebGLGraphics.updateGraphics(maskData, this.renderer.gl); - } - - if (!maskData._webGL[this.renderer.gl.id].data.length) { - return; - } - - this.renderer.stencilManager.pushStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer); -}; - -/** - * Removes the last filter from the filter stack and doesn't return it. - * - * @param maskData {any[]} - */ -WebGLMaskManager.prototype.popMask = function (maskData) { - this.renderer.stencilManager.popStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer); -}; - -},{"../utils/WebGLGraphics":39,"./WebGLManager":28}],30:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - PrimitiveShader = require('../shaders/PrimitiveShader'), - ComplexPrimitiveShader = require('../shaders/ComplexPrimitiveShader'), - Shader = require('../shaders/Shader'), - FastShader = require('../shaders/FastShader'), - StripShader = require('../shaders/StripShader'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLShaderManager(renderer) { - WebGLManager.call(this, renderer); - - /** - * @member {number} - */ - this.maxAttibs = 10; - - /** - * @member {any[]} - */ - this.attribState = []; - - /** - * @member {any[]} - */ - this.tempAttribState = []; - - for (var i = 0; i < this.maxAttibs; i++) { - this.attribState[i] = false; - } - - /** - * @member {any[]} - */ - this.stack = []; - - /** - * @member {number} - * @private - */ - this._currentId = -1; - - /** - * @member {Shader} - * @private - */ - this.currentShader = null; - - // this shader is used for rendering primitives - this.primitiveShader = null; - - // this shader is used for rendering triangle strips - this.complexPrimitiveShader = null; - - // this shader is used for the default sprite rendering - this.defaultShader = null; - - // this shader is used for the fast sprite rendering - this.fastShader = null; - - // the next one is used for rendering triangle strips - this.stripShader = null; - - // listen for context and update necessary shaders - var self = this; - this.renderer.on('context', function (event) { - var gl = event.data; - - // this shader is used for rendering primitives - self.primitiveShader = new PrimitiveShader(gl); - - // this shader is used for rendering triangle strips - self.complexPrimitiveShader = new ComplexPrimitiveShader(gl); - - // this shader is used for the default sprite rendering - self.defaultShader = new Shader(gl); - - // this shader is used for the fast sprite rendering - self.fastShader = new FastShader(gl); - - // the next one is used for rendering triangle strips - self.stripShader = new StripShader(gl); - - self.setShader(self.defaultShader); - }); -} - -WebGLShaderManager.prototype = Object.create(WebGLManager.prototype); -WebGLShaderManager.prototype.constructor = WebGLShaderManager; -module.exports = WebGLShaderManager; - -/** - * Takes the attributes given in parameters. - * - * @param attribs {Array} attribs - */ -WebGLShaderManager.prototype.setAttribs = function (attribs) { - // reset temp state - var i; - - for (i = 0; i < this.tempAttribState.length; i++) { - this.tempAttribState[i] = false; - } - - // set the new attribs - for (var a in attribs) { - this.tempAttribState[attribs[a]] = true; - } - - var gl = this.renderer.gl; - - for (i = 0; i < this.attribState.length; i++) { - if (this.attribState[i] !== this.tempAttribState[i]) { - this.attribState[i] = this.tempAttribState[i]; - - if (this.attribState[i]) { - gl.enableVertexAttribArray(i); - } - else { - gl.disableVertexAttribArray(i); - } - } - } -}; - -/** - * Sets the current shader. - * - * @param shader {Any} - */ -WebGLShaderManager.prototype.setShader = function (shader) { - if (this._currentId === shader.uuid) { - return false; - } - - this._currentId = shader.uuid; - - this.currentShader = shader; - - this.renderer.gl.useProgram(shader.program); - this.setAttribs(shader.attributes); - - return true; -}; - -/** - * Destroys this object. - * - */ -WebGLShaderManager.prototype.destroy = function () { - this.attribState = null; - - this.tempAttribState = null; - - this.primitiveShader.destroy(); - this.primitiveShader = null; - - this.complexPrimitiveShader.destroy(); - this.complexPrimitiveShader = null; - - this.defaultShader.destroy(); - this.defaultShader = null; - - this.fastShader.destroy(); - this.fastShader = null; - - this.stripShader.destroy(); - this.stripShader = null; - - this.renderer = null; -}; - -},{"../shaders/ComplexPrimitiveShader":32,"../shaders/FastShader":33,"../shaders/PrimitiveShader":34,"../shaders/Shader":35,"../shaders/StripShader":36,"./WebGLManager":28}],31:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - utils = require('../../../utils'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLStencilManager(renderer) { - WebGLManager.call(this, renderer); - - this.stencilStack = []; - this.reverse = true; - this.count = 0; -} - -WebGLStencilManager.prototype = Object.create(WebGLManager.prototype); -WebGLStencilManager.prototype.constructor = WebGLStencilManager; -module.exports = WebGLStencilManager; - -/** - * Applies the Mask and adds it to the current filter stack. - * - * @param graphics {Graphics} - * @param webGLData {any[]} - */ -WebGLStencilManager.prototype.pushStencil = function (graphics, webGLData) { - var gl = this.renderer.gl; - - this.bindGraphics(graphics, webGLData, this.renderer); - - if (this.stencilStack.length === 0) { - gl.enable(gl.STENCIL_TEST); - gl.clear(gl.STENCIL_BUFFER_BIT); - this.reverse = true; - this.count = 0; - } - - this.stencilStack.push(webGLData); - - var level = this.count; - - gl.colorMask(false, false, false, false); - - gl.stencilFunc(gl.ALWAYS,0,0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT); - - // draw the triangle strip! - - if (webGLData.mode === 1) { - gl.drawElements(gl.TRIANGLE_FAN, webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 ); - - if (this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - - // draw a quad to increment.. - gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 ); - - if (this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - } - - this.reverse = !this.reverse; - } - else { - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - - gl.drawElements(gl.TRIANGLE_STRIP, webGLData.indices.length, gl.UNSIGNED_SHORT, 0 ); - - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - } - } - - gl.colorMask(true, true, true, true); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP); - - this.count++; -}; - -/** - * TODO this does not belong here! - * - * @param graphics {Graphics} - * @param webGLData {Array} - */ -WebGLStencilManager.prototype.bindGraphics = function (graphics, webGLData) { - //if (this._currentGraphics === graphics)return; - this._currentGraphics = graphics; - - var gl = this.renderer.gl; - - // bind the graphics object.. - var projection = this.renderer.projection, - offset = this.renderer.offset, - shader;// = this.renderer.shaderManager.primitiveShader; - - if (webGLData.mode === 1) { - shader = this.renderer.shaderManager.complexPrimitiveShader; - - this.renderer.shaderManager.setShader(shader); - - gl.uniform1f(shader.flipY, this.renderer.flipY); - - gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true)); - - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - - gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint)); - gl.uniform3fv(shader.color, webGLData.color); - - gl.uniform1f(shader.alpha, graphics.worldAlpha * webGLData.alpha); - - gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer); - - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 2, 0); - - - // now do the rest.. - // set the index buffer! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer); - } - else { - //this.renderer.shaderManager.activatePrimitiveShader(); - shader = this.renderer.shaderManager.primitiveShader; - this.renderer.shaderManager.setShader( shader ); - - gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true)); - - gl.uniform1f(shader.flipY, this.renderer.flipY); - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - - gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint)); - - gl.uniform1f(shader.alpha, graphics.worldAlpha); - - gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer); - - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0); - gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4); - - // set the index buffer! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer); - } -}; - -/** - * @param graphics {Graphics} - * @param webGLData {Array} - */ -WebGLStencilManager.prototype.popStencil = function (graphics, webGLData) { - var gl = this.renderer.gl; - - this.stencilStack.pop(); - - this.count--; - - if (this.stencilStack.length === 0) { - // the stack is empty! - gl.disable(gl.STENCIL_TEST); - - } - else { - - var level = this.count; - - this.bindGraphics(graphics, webGLData, this.renderer); - - gl.colorMask(false, false, false, false); - - if (webGLData.mode === 1) { - this.reverse = !this.reverse; - - if (this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - - // draw a quad to increment.. - gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 ); - - gl.stencilFunc(gl.ALWAYS,0,0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT); - - // draw the triangle strip! - gl.drawElements(gl.TRIANGLE_FAN, webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 ); - - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - } - - } - else { - // console.log("<<>>") - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - - gl.drawElements(gl.TRIANGLE_STRIP, webGLData.indices.length, gl.UNSIGNED_SHORT, 0 ); - - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - } - } - - gl.colorMask(true, true, true, true); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP); - - - } -}; - -/** - * Destroys the mask stack. - * - */ -WebGLStencilManager.prototype.destroy = function () { - this.renderer = null; - this.stencilStack = null; -}; - -},{"../../../utils":50,"./WebGLManager":28}],32:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function ComplexPrimitiveShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - // 'attribute vec2 aTextureCoord;', - // 'attribute vec4 aColor;', - - 'uniform mat3 translationMatrix;', - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - - 'uniform vec3 tint;', - 'uniform float alpha;', - 'uniform vec3 color;', - 'uniform float flipY;', - 'varying vec4 vColor;', - - 'void main(void) {', - ' vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);', - ' v -= offsetVector.xyx;', - ' gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);', - ' vColor = vec4(color * alpha * tint, alpha);',//" * vec4(tint * alpha, alpha);', - '}' - ].join('\n'), - // fragment shader - [ - 'precision mediump float;', - - 'varying vec4 vColor;', - - 'void main(void) {', - ' gl_FragColor = vColor;', - '}' - ].join('\n'), - // custom uniforms - { - tint: { type: '3f', value: [0, 0, 0] }, - flipY: { type: '1f', value: 0 }, - alpha: { type: '1f', value: 0 }, - translationMatrix: { type: 'mat3', value: new Float32Array(9) } - } - ); -} - -ComplexPrimitiveShader.prototype = Object.create(Shader.prototype); -ComplexPrimitiveShader.prototype.constructor = ComplexPrimitiveShader; -module.exports = ComplexPrimitiveShader; - -},{"./Shader":35}],33:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @extends Shader - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function FastShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - 'attribute vec2 aTextureCoord;', - 'attribute vec4 aColor;', - - 'attribute vec2 aPositionCoord;', - 'attribute vec2 aScale;', - 'attribute float aRotation;', - - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - 'uniform mat3 uMatrix;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'const vec2 center = vec2(-1.0, 1.0);', - - 'void main(void) {', - ' vec2 v;', - ' vec2 sv = aVertexPosition * aScale;', - ' v.x = (sv.x) * cos(aRotation) - (sv.y) * sin(aRotation);', - ' v.y = (sv.x) * sin(aRotation) + (sv.y) * cos(aRotation);', - ' v = ( uMatrix * vec3(v + aPositionCoord , 1.0) ).xy ;', - ' gl_Position = vec4( ( v / projectionVector) + center , 0.0, 1.0);', - ' vTextureCoord = aTextureCoord;', - // ' vec3 color = mod(vec3(aColor.y/65536.0, aColor.y/256.0, aColor.y), 256.0) / 256.0;', - ' vColor = aColor;', - '}' - ].join('\n'), - // fragment shader, use default - null, - // custom uniforms - { - uMatrix: { type: 'mat3', value: new Float32Array(9) } - }, - // custom attributes - { - aPositionCoord: 0, - aRotation: 0, - aScale: 0 - } - ); -} - -FastShader.prototype = Object.create(Shader.prototype); -FastShader.prototype.constructor = FastShader; -module.exports = FastShader; - -},{"./Shader":35}],34:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function PrimitiveShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - // 'attribute vec2 aTextureCoord;', - 'attribute vec4 aColor;', - - 'uniform mat3 translationMatrix;', - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - 'uniform float alpha;', - 'uniform float flipY;', - 'uniform vec3 tint;', - - 'varying vec4 vColor;', - - 'void main(void) {', - ' vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);', - ' v -= offsetVector.xyx;', - ' gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);', - ' vColor = aColor * vec4(tint * alpha, alpha);', - '}' - ].join('\n'), - // fragment shader - [ - 'precision mediump float;', - - 'varying vec4 vColor;', - - 'void main(void) {', - ' gl_FragColor = vColor;', - '}' - ].join('\n'), - // custom uniforms - { - tint: { type: '3f', value: [0, 0, 0] }, - flipY: { type: '1f', value: 0 }, - alpha: { type: '1f', value: 0 }, - translationMatrix: { type: 'mat3', value: new Float32Array(9) } - } - ); -} - -PrimitiveShader.prototype = Object.create(Shader.prototype); -PrimitiveShader.prototype.constructor = PrimitiveShader; -module.exports = PrimitiveShader; - -},{"./Shader":35}],35:[function(require,module,exports){ -var utils = require('../../../utils'); - -/** - * @class - * @namespace PIXI - * @param [fragmentSrc] {string} The source of the fragment shader. - * @param [vertexSrc] {string} The source of the vertex shader. - */ -function Shader(gl, vertexSrc, fragmentSrc, customUniforms, customAttributes) { - /** - * @member {number} - * @readonly - */ - this.uuid = utils.uuid(); - - /** - * @member {WebGLContext} - * @readonly - */ - this.gl = gl; - - /** - * The WebGL program. - * @member {WebGLProgram} - * @readonly - */ - this.program = null; - - this.uniforms = { - uSampler: { type: 'sampler2D', value: 0 }, - projectionVector: { type: '2f', value: { x: 0, y: 0 } }, - offsetVector: { type: '2f', value: { x: 0, y: 0 } }, - dimensions: { type: '4f', value: new Float32Array(4) } - }; - - for (var u in customUniforms) { - this.uniforms[u] = customUniforms[u]; - } - - this.attributes = { - aVertexPosition: 0, - aTextureCoord: 0, - aColor: 0 - }; - - for (var a in customAttributes) { - this.attributes[a] = customAttributes[a]; - } - - this.textureCount = 0; - - /** - * The vertex shader. - * @member {Array} - */ - this.vertexSrc = vertexSrc || [ - 'attribute vec2 aVertexPosition;', - 'attribute vec2 aTextureCoord;', - 'attribute vec4 aColor;', - - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'const vec2 center = vec2(-1.0, 1.0);', - - 'void main(void) {', - ' gl_Position = vec4( ((aVertexPosition + offsetVector) / projectionVector) + center , 0.0, 1.0);', - ' vTextureCoord = aTextureCoord;', - ' vColor = vec4(aColor.rgb * aColor.a, aColor.a);', - '}' - ].join('\n'); - - /** - * The fragment shader. - * @member {Array} - */ - this.fragmentSrc = fragmentSrc || [ - 'precision lowp float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord) * vColor ;', - '}' - ].join('\n'); - - this.init(); -} - -Shader.prototype.constructor = Shader; -module.exports = Shader; - -Shader.prototype.init = function () { - this.compile(); - - this.gl.useProgram(this.program); - - this.cacheUniformLocations(Object.keys(this.uniforms)); - this.cacheAttributeLocations(Object.keys(this.attributes)); -}; - -Shader.prototype.cacheUniformLocations = function (keys) { - for (var i = 0; i < keys.length; ++i) { - this.uniforms[keys[i]]._location = this.gl.getUniformLocation(this.program, keys[i]); - } -}; - -Shader.prototype.cacheAttributeLocations = function (keys) { - for (var i = 0; i < keys.length; ++i) { - this.attributes[keys[i]] = this.gl.getAttribLocation(this.program, keys[i]); - } - - // TODO: Check if this is needed anymore... - - // Begin worst hack eva // - - // WHY??? ONLY on my chrome pixel the line above returns -1 when using filters? - // maybe its something to do with the current state of the gl context. - // I'm convinced this is a bug in the chrome browser as there is NO reason why this should be returning -1 especially as it only manifests on my chrome pixel - // If theres any webGL people that know why could happen please help :) - // if (this.attributes.aColor === -1) { - // this.attributes.aColor = 2; - // } - - // End worst hack eva // -}; - -Shader.prototype.compile = function () { - var gl = this.gl; - - var glVertShader = this._glCompile(gl.VERTEX_SHADER, this.vertexSrc); - var glFragShader = this._glCompile(gl.FRAGMENT_SHADER, this.fragmentSrc); - - var program = gl.createProgram(); - - gl.attachShader(program, glVertShader); - gl.attachShader(program, glFragShader); - gl.linkProgram(program); - - // if linking fails, then log and cleanup - if (!gl.getProgramParameter(program, gl.LINK_STATUS)) { - window.console.error('Pixi.js Error: Could not initialize shader.'); - window.console.error('gl.VALIDATE_STATUS', gl.getProgramParameter(program, gl.VALIDATE_STATUS)); - window.console.error('gl.getError()', gl.getError()); - - // if there is a program info log, log it - if (gl.getProgramInfoLog(program) !== '') { - window.console.warn('Pixi.js Warning: gl.getProgramInfoLog()', gl.getProgramInfoLog(program)); - } - - gl.deleteProgram(program); - program = null; - } - - // clean up some shaders - gl.deleteShader(glVertShader); - gl.deleteShader(glFragShader); - - return (this.program = program); -}; - -Shader.prototype.syncUniforms = function () { - var gl = this.gl; - - this.textureCount = 1; - - for (var key in this.uniforms) { - var uniform = this.uniforms[key], - location = uniform._location, - value = uniform.value, - i, il; - - switch (uniform.type) { - case 'i': - case '1i': - gl.uniform1i(location, value); - break; - - case 'f': - case '1f': - gl.uniform1f(location, value); - break; - - case '2f': - gl.uniform2f(location, value[0], value[1]); - break; - - case '3f': - gl.uniform3f(location, value[0], value[1], value[2]); - break; - - case '4f': - gl.uniform4f(location, value[0], value[1], value[2], value[3]); - break; - - // a 2D Point object - case 'v2': - gl.uniform2f(location, value.x, value.y); - break; - - // a 3D Point object - case 'v3': - gl.uniform3f(location, value.x, value.y, value.z); - break; - - // a 4D Point object - case 'v4': - gl.uniform4f(location, value.x, value.y, value.z, value.w); - break; - - case '1iv': - gl.uniform1iv(location, value); - break; - - case '3iv': - gl.uniform3iv(location, value); - break; - - case '1fv': - gl.uniform1fv(location, value); - break; - - case '2fv': - gl.uniform2fv(location, value); - break; - - case '3fv': - gl.uniform3fv(location, value); - break; - - case '4fv': - gl.uniform4fv(location, value); - break; - - case 'm2': - case 'mat2': - case 'Matrix2fv': - gl.uniformMatrix2fv(location, uniform.transpose, value); - break; - - case 'm3': - case 'mat3': - case 'Matrix3fv': - gl.uniformMatrix3fv(location, uniform.transpose, value); - break; - - case 'm4': - case 'mat4': - case 'Matrix4fv': - gl.uniformMatrix4fv(location, uniform.transpose, value); - break; - - // a Color Value - case 'c': - if (typeof value === 'number') { - value = utils.hex2rgb(value); - } - - gl.uniform3f(location, value[0], value[1], value[2]); - break; - - // flat array of integers (JS or typed array) - case 'iv1': - gl.uniform1iv(location, value); - break; - - // flat array of integers with 3 x N size (JS or typed array) - case 'iv': - gl.uniform3iv(location, value); - break; - - // flat array of floats (JS or typed array) - case 'fv1': - gl.uniform1fv(location, value); - break; - - // flat array of floats with 3 x N size (JS or typed array) - case 'fv': - gl.uniform3fv(location, value); - break; - - // array of 2D Point objects - case 'v2v': - if (!uniform._array) { - uniform._array = new Float32Array(2 * value.length); - } - - for (i = 0, il = value.length; i < il; ++i) { - uniform._array[i * 2] = value[i].x; - uniform._array[i * 2 + 1] = value[i].y; - } - - gl.uniform2fv(location, uniform._array); - break; - - // array of 3D Point objects - case 'v3v': - if (!uniform._array) { - uniform._array = new Float32Array(3 * value.length); - } - - for (i = 0, il = value.length; i < il; ++i) { - uniform._array[i * 3] = value[i].x; - uniform._array[i * 3 + 1] = value[i].y; - uniform._array[i * 3 + 2] = value[i].z; - - } - - gl.uniform3fv(location, uniform._array); - break; - - // array of 4D Point objects - case 'v4v': - if (!uniform._array) { - uniform._array = new Float32Array(4 * value.length); - } - - for (i = 0, il = value.length; i < il; ++i) { - uniform._array[i * 4] = value[i].x; - uniform._array[i * 4 + 1] = value[i].y; - uniform._array[i * 4 + 2] = value[i].z; - uniform._array[i * 4 + 3] = value[i].w; - - } - - gl.uniform4fv(location, uniform._array); - break; - - case 't': - case 'sampler2D': - if (!uniform.value || !uniform.value.baseTexture || !uniform.value.baseTexture.hasLoaded) { - break; - } - - // activate this texture - gl.activeTexture(gl['TEXTURE' + this.textureCount]); - - // bind the texture - gl.bindTexture(gl.TEXTURE_2D, uniform.value.baseTexture._glTextures[gl.id]); - - // set uniform to texture index - gl.uniform1i(uniform._location, this.textureCount); - - // increment next texture id - this.textureCount++; - - // initialize the texture if we haven't yet - if (!uniform._init) { - this.initSampler2D(uniform); - - uniform._init = true; - } - break; - - default: - window.console.warn('Pixi.js Shader Warning: Unknown uniform type: ' + uniform.type); - } - } -}; - - -/** - * Initialises a Sampler2D uniform (which may only be available later on after initUniforms once the texture has loaded) - * - */ -Shader.prototype.initSampler2D = function (uniform) { - var gl = this.gl; - - // Extended texture data - if (uniform.textureData) { - var data = uniform.textureData; - - // GLTexture = mag linear, min linear_mipmap_linear, wrap repeat + gl.generateMipmap(gl.TEXTURE_2D); - // GLTextureLinear = mag/min linear, wrap clamp - // GLTextureNearestRepeat = mag/min NEAREST, wrap repeat - // GLTextureNearest = mag/min nearest, wrap clamp - // AudioTexture = whatever + luminance + width 512, height 2, border 0 - // KeyTexture = whatever + luminance + width 256, height 2, border 0 - - // magFilter can be: gl.LINEAR, gl.LINEAR_MIPMAP_LINEAR or gl.NEAREST - // wrapS/T can be: gl.CLAMP_TO_EDGE or gl.REPEAT - - var magFilter = (data.magFilter) ? data.magFilter : gl.LINEAR; - var minFilter = (data.minFilter) ? data.minFilter : gl.LINEAR; - var wrapS = (data.wrapS) ? data.wrapS : gl.CLAMP_TO_EDGE; - var wrapT = (data.wrapT) ? data.wrapT : gl.CLAMP_TO_EDGE; - var format = (data.luminance) ? gl.LUMINANCE : gl.RGBA; - - if (data.repeat) { - wrapS = gl.REPEAT; - wrapT = gl.REPEAT; - } - - gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, !!data.flipY); - - if (data.width) { - var width = (data.width) ? data.width : 512; - var height = (data.height) ? data.height : 2; - var border = (data.border) ? data.border : 0; - - // void texImage2D(GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, ArrayBufferView? pixels); - gl.texImage2D(gl.TEXTURE_2D, 0, format, width, height, border, format, gl.UNSIGNED_BYTE, null); - } - else { - // void texImage2D(GLenum target, GLint level, GLenum internalformat, GLenum format, GLenum type, ImageData? pixels); - gl.texImage2D(gl.TEXTURE_2D, 0, format, gl.RGBA, gl.UNSIGNED_BYTE, uniform.value.baseTexture.source); - } - - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, magFilter); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrapS); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrapT); - } -}; - -/** - * Destroys the shader. - * - */ -Shader.prototype.destroy = function () { - this.gl.deleteProgram(this.program); - - this.gl = null; - this.uniforms = null; - this.attributes = null; - - this.vertexSrc = null; - this.fragmentSrc = null; -}; - -Shader.prototype._glCompile = function (type, src) { - var shader = this.gl.createShader(type); - - if (Array.isArray(src)) { - debugger; - } - - this.gl.shaderSource(shader, src); - this.gl.compileShader(shader); - - if (!this.gl.getShaderParameter(shader, this.gl.COMPILE_STATUS)) { - window.console.log(this.gl.getShaderInfoLog(shader)); - return null; - } - - return shader; -}; - -},{"../../../utils":50}],36:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function StripShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - 'attribute vec2 aTextureCoord;', - // 'attribute vec4 aColor;', - - 'uniform mat3 translationMatrix;', - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - - 'varying vec2 vTextureCoord;', - - 'void main(void) {', - ' vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);', - ' v -= offsetVector.xyx;', - ' gl_Position = vec4( v.x / projectionVector.x -1.0, v.y / -projectionVector.y + 1.0 , 0.0, 1.0);', - ' vTextureCoord = aTextureCoord;', - '}' - ].join('\n'), - // fragment shader - [ - 'precision mediump float;', - - 'uniform float alpha;', - 'uniform sampler2D uSampler;', - - 'varying vec2 vTextureCoord;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * alpha;', - '}' - ].join('\n'), - // custom uniforms - { - alpha: { type: '1f', value: 0 }, - translationMatrix: { type: 'mat3', value: new Float32Array(9) } - } - ); -} - -StripShader.prototype = Object.create(Shader.prototype); -StripShader.prototype.constructor = StripShader; -module.exports = StripShader; - -},{"./Shader":35}],37:[function(require,module,exports){ -var CONST = require('../../../const'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - * @param width {number} the horizontal range of the filter - * @param height {number} the vertical range of the filter - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - */ -function FilterTexture(gl, width, height, scaleMode) { - /** - * @member {WebGLContext} - */ - this.gl = gl; - - // next time to create a frame buffer and texture - - /** - * @member {Any} - */ - this.frameBuffer = gl.createFramebuffer(); - - /** - * @member {Any} - */ - this.texture = gl.createTexture(); - - /** - * @member {number} - */ - scaleMode = scaleMode || CONST.scaleModes.DEFAULT; - - gl.bindTexture(gl.TEXTURE_2D, this.texture); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); - gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer ); - - gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer ); - gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0); - - // required for masking a mask?? - this.renderBuffer = gl.createRenderbuffer(); - gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer); - gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, this.renderBuffer); - - // reset render buffer - gl.bindRenderbuffer(gl.RENDERBUFFER, null); - - this.resize(width, height); -} - -FilterTexture.prototype.constructor = FilterTexture; -module.exports = FilterTexture; - -/** - * Clears the filter texture. - * - */ -FilterTexture.prototype.clear = function () { - var gl = this.gl; - - gl.clearColor(0,0,0, 0); - gl.clear(gl.COLOR_BUFFER_BIT); -}; - -/** - * Resizes the texture to the specified width and height - * - * @param width {number} the new width of the texture - * @param height {number} the new height of the texture - */ -FilterTexture.prototype.resize = function (width, height) { - if (this.width === width && this.height === height) { - return; - } - - this.width = width; - this.height = height; - - var gl = this.gl; - - gl.bindTexture(gl.TEXTURE_2D, this.texture); - gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, width , height , 0, gl.RGBA, gl.UNSIGNED_BYTE, null); - // update the stencil buffer width and height - gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer); - gl.renderbufferStorage(gl.RENDERBUFFER, gl.DEPTH_STENCIL, width , height); - - // reset render buffer - gl.bindRenderbuffer(gl.RENDERBUFFER, null); -}; - -/** - * Destroys the filter texture. - * - */ -FilterTexture.prototype.destroy = function () { - var gl = this.gl; - gl.deleteFramebuffer( this.frameBuffer ); - gl.deleteTexture( this.texture ); - - this.frameBuffer = null; - this.texture = null; -}; - -},{"../../../const":4}],38:[function(require,module,exports){ -/** - * @author Mat Groves - * - * Big thanks to the very clever Matt DesLauriers https://github.com/mattdesl/ - * for creating the original pixi version! - * - * Heavily inspired by LibGDX's WebGLSpriteBatch: - * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java - */ - -/** - * @class - * @private - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this sprite batch works for. - */ -function WebGLFastSpriteBatch(renderer) { - /** - * The renderer instance this sprite batch operates on. - * - * @member {WebGLRenderer} - */ - this.renderer = renderer; - - /** - * - * - * @member {number} - */ - this.vertSize = 10; - - /** - * - * - * @member {number} - */ - this.maxSize = 6000;//Math.pow(2, 16) / this.vertSize; - - /** - * - * - * @member {number} - */ - this.size = this.maxSize; - - //the total number of floats in our batch - var numVerts = this.size * 4 * this.vertSize; - - //the total number of indices in our batch - var numIndices = this.maxSize * 6; - - /** - * Vertex data - * - * @member {Float32Array} - */ - this.vertices = new Float32Array(numVerts); - - /** - * Index data - * - * @member {Uint16Array} - */ - this.indices = new Uint16Array(numIndices); - - /** - * - * - * @member {object} - */ - this.vertexBuffer = null; - - /** - * - * - * @member {object} - */ - this.indexBuffer = null; - - /** - * - * - * @member {number} - */ - this.lastIndexCount = 0; - - for (var i=0, j=0; i < numIndices; i += 6, j += 4) { - this.indices[i + 0] = j + 0; - this.indices[i + 1] = j + 1; - this.indices[i + 2] = j + 2; - this.indices[i + 3] = j + 0; - this.indices[i + 4] = j + 2; - this.indices[i + 5] = j + 3; - } - - /** - * - * - * @member {boolean} - */ - this.drawing = false; - - /** - * - * - * @member {number} - */ - this.currentBatchSize = 0; - - /** - * - * - * @member {BaseTexture} - */ - this.currentBaseTexture = null; - - /** - * - * - * @member {number} - */ - this.currentBlendMode = 0; - - /** - * - * - * @member {object} - */ - this.shader = null; - - /** - * - * - * @member {Matrix} - */ - this.matrix = null; - - // listen for context and update necessary buffers - var self = this; - this.renderer.on('context', function () { - self.setupContext(); - }); -} - -WebGLFastSpriteBatch.prototype.constructor = WebGLFastSpriteBatch; -module.exports = WebGLFastSpriteBatch; - -/** - * Sets the WebGL Context. - * - * @param gl {WebGLContext} the current WebGL drawing context - */ -WebGLFastSpriteBatch.prototype.setupContext = function () { - var gl = this.renderer.gl; - - // create a couple of buffers - this.vertexBuffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - - // 65535 is max index, so 65535 / 6 = 10922. - - //upload the index data - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW); -}; - -/** - * @param spriteBatch {SpriteBatch} The SpriteBatch container to prepare for. - */ -WebGLFastSpriteBatch.prototype.begin = function (spriteBatch) { - this.shader = this.renderer.shaderManager.fastShader; - - this.matrix = spriteBatch.worldTransform.toArray(true); - - this.start(); -}; - -/** - */ -WebGLFastSpriteBatch.prototype.end = function () { - this.flush(); -}; - -/** - * @param spriteBatch {SpriteBatch} The SpriteBatch container to render. - */ -WebGLFastSpriteBatch.prototype.render = function (spriteBatch) { - var children = spriteBatch.children; - var sprite = children[0]; - - // if the uvs have not updated then no point rendering just yet! - - // check texture. - if (!sprite.texture._uvs) { - return; - } - - this.currentBaseTexture = sprite.texture.baseTexture; - - // check blend mode - if (sprite.blendMode !== this.renderer.blendModeManager.currentBlendMode) { - this.flush(); - this.renderer.blendModeManager.setBlendMode(sprite.blendMode); - } - - for (var i=0,j= children.length; i= this.size) { - this.flush(); - } -}; - -/** - * - */ -WebGLFastSpriteBatch.prototype.flush = function () { - // If the batch is length 0 then return as there is nothing to draw - if (this.currentBatchSize === 0) { - return; - } - - var gl = this.renderer.gl; - - // bind the current texture - if (!this.currentBaseTexture._glTextures[gl.id]) { - this.renderer.updateTexture(this.currentBaseTexture, gl); - } - //TODO-SHOUD THIS BE ELSE??!?!?! - else { - gl.bindTexture(gl.TEXTURE_2D, this.currentBaseTexture._glTextures[gl.id]); - } - - // upload the verts to the buffer - - if (this.currentBatchSize > ( this.size * 0.5 ) ) { - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices); - } - else { - var view = this.vertices.subarray(0, this.currentBatchSize * 4 * this.vertSize); - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, view); - } - - // now draw those suckas! - gl.drawElements(gl.TRIANGLES, this.currentBatchSize * 6, gl.UNSIGNED_SHORT, 0); - - // then reset the batch! - this.currentBatchSize = 0; - - // increment the draw count - this.renderer.drawCount++; -}; - - -/** - * Ends the batch and flushes - * - */ -WebGLFastSpriteBatch.prototype.stop = function () { - this.flush(); -}; - -/** - * - */ -WebGLFastSpriteBatch.prototype.start = function () { - var gl = this.renderer.gl; - - // bind the main texture - gl.activeTexture(gl.TEXTURE0); - - // bind the buffers - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - - // set the projection - var projection = this.renderer.projection; - gl.uniform2f(this.shader.projectionVector, projection.x, projection.y); - - // set the matrix - gl.uniformMatrix3fv(this.shader.uMatrix, false, this.matrix); - - // set the pointers - var stride = this.vertSize * 4; - - gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0); - gl.vertexAttribPointer(this.shader.attributes.aPositionCoord, 2, gl.FLOAT, false, stride, 2 * 4); - gl.vertexAttribPointer(this.shader.attributes.aScale, 2, gl.FLOAT, false, stride, 4 * 4); - gl.vertexAttribPointer(this.shader.attributes.aRotation, 1, gl.FLOAT, false, stride, 6 * 4); - gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 7 * 4); - gl.vertexAttribPointer(this.shader.attributes.aColor, 1, gl.FLOAT, false, stride, 9 * 4); -}; - -},{}],39:[function(require,module,exports){ -var utils = require('../../../utils'), - math = require('../../../math'), - CONST = require('../../../const'), - WebGLGraphicsData = require('./WebGLGraphicsData'); - -/** - * A set of functions used by the webGL renderer to draw the primitive graphics data - * - * @namespace PIXI - * @private - */ -var WebGLGraphics = module.exports = {}; - -/** - * Renders the graphics object - * - * @static - * @private - * @param graphics {Graphics} - * @param renderer {WebGLRenderer} - */ -WebGLGraphics.renderGraphics = function (graphics, renderer) {//projection, offset) { - var gl = renderer.gl; - - var projection = renderer.projection, - offset = renderer.offset, - shader = renderer.shaderManager.primitiveShader, - webGLData; - - if (graphics.dirty) { - WebGLGraphics.updateGraphics(graphics, gl); - } - - var webGL = graphics._webGL[gl.id]; - - // This could be speeded up for sure! - - for (var i = 0; i < webGL.data.length; i++) { - if (webGL.data[i].mode === 1) { - webGLData = webGL.data[i]; - - renderer.stencilManager.pushStencil(graphics, webGLData, renderer); - - // render quad.. - gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 ); - - renderer.stencilManager.popStencil(graphics, webGLData, renderer); - } - else { - webGLData = webGL.data[i]; - - - renderer.shaderManager.setShader( shader );//activatePrimitiveShader(); - shader = renderer.shaderManager.primitiveShader; - gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true)); - - gl.uniform1f(shader.flipY, 1); - - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - - gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint)); - - gl.uniform1f(shader.alpha, graphics.worldAlpha); - - - gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer); - - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0); - gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4); - - // set the index buffer! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer); - gl.drawElements(gl.TRIANGLE_STRIP, webGLData.indices.length, gl.UNSIGNED_SHORT, 0 ); - } - } -}; - -/** - * Updates the graphics object - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object to update - * @param gl {WebGLContext} the current WebGL drawing context - */ -WebGLGraphics.updateGraphics = function (graphics, gl) { - // get the contexts graphics object - var webGL = graphics._webGL[gl.id]; - - // if the graphics object does not exist in the webGL context time to create it! - if (!webGL) { - webGL = graphics._webGL[gl.id] = {lastIndex:0, data:[], gl:gl}; - } - - // flag the graphics as not dirty as we are about to update it... - graphics.dirty = false; - - var i; - - // if the user cleared the graphics object we will need to clear every object - if (graphics.clearDirty) { - graphics.clearDirty = false; - - // lop through and return all the webGLDatas to the object pool so than can be reused later on - for (i = 0; i < webGL.data.length; i++) { - var graphicsData = webGL.data[i]; - graphicsData.reset(); - WebGLGraphics.graphicsDataPool.push( graphicsData ); - } - - // clear the array and reset the index.. - webGL.data = []; - webGL.lastIndex = 0; - } - - var webGLData; - - // loop through the graphics datas and construct each one.. - // if the object is a complex fill then the new stencil buffer technique will be used - // other wise graphics objects will be pushed into a batch.. - for (i = webGL.lastIndex; i < graphics.graphicsData.length; i++) { - var data = graphics.graphicsData[i]; - - if (data.type === CONST.SHAPES.POLY) { - // need to add the points the the graphics object.. - data.points = data.shape.points.slice(); - if (data.shape.closed) { - // close the poly if the value is true! - if (data.points[0] !== data.points[data.points.length-2] || data.points[1] !== data.points[data.points.length-1]) { - data.points.push(data.points[0], data.points[1]); - } - } - - // MAKE SURE WE HAVE THE CORRECT TYPE.. - if (data.fill) { - if (data.points.length >= 6) { - if (data.points.length < 6 * 2) { - webGLData = WebGLGraphics.switchMode(webGL, 0); - - var canDrawUsingSimple = WebGLGraphics.buildPoly(data, webGLData); - // console.log(canDrawUsingSimple); - - if (!canDrawUsingSimple) { - // console.log("<>>>") - webGLData = WebGLGraphics.switchMode(webGL, 1); - WebGLGraphics.buildComplexPoly(data, webGLData); - } - - } - else { - webGLData = WebGLGraphics.switchMode(webGL, 1); - WebGLGraphics.buildComplexPoly(data, webGLData); - } - } - } - - if (data.lineWidth > 0) { - webGLData = WebGLGraphics.switchMode(webGL, 0); - WebGLGraphics.buildLine(data, webGLData); - - } - } - else { - webGLData = WebGLGraphics.switchMode(webGL, 0); - - if (data.type === CONST.SHAPES.RECT) { - WebGLGraphics.buildRectangle(data, webGLData); - } - else if (data.type === CONST.SHAPES.CIRC || data.type === CONST.SHAPES.ELIP) { - WebGLGraphics.buildCircle(data, webGLData); - } - else if (data.type === CONST.SHAPES.RREC) { - WebGLGraphics.buildRoundedRectangle(data, webGLData); - } - } - - webGL.lastIndex++; - } - - // upload all the dirty data... - for (i = 0; i < webGL.data.length; i++) { - webGLData = webGL.data[i]; - - if (webGLData.dirty) { - webGLData.upload(); - } - } -}; - -/** - * @static - * @private - * @param webGL {WebGLContext} - * @param type {number} - */ -WebGLGraphics.switchMode = function (webGL, type) { - var webGLData; - - if (!webGL.data.length) { - webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl); - webGLData.mode = type; - webGL.data.push(webGLData); - } - else { - webGLData = webGL.data[webGL.data.length-1]; - - if (webGLData.mode !== type || type === 1) { - webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl); - webGLData.mode = type; - webGL.data.push(webGLData); - } - } - - webGLData.dirty = true; - - return webGLData; -}; - -/** - * Builds a rectangle to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildRectangle = function (graphicsData, webGLData) { - // --- // - // need to convert points to a nice regular data - // - var rectData = graphicsData.shape; - var x = rectData.x; - var y = rectData.y; - var width = rectData.width; - var height = rectData.height; - - if (graphicsData.fill) { - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var verts = webGLData.points; - var indices = webGLData.indices; - - var vertPos = verts.length/6; - - // start - verts.push(x, y); - verts.push(r, g, b, alpha); - - verts.push(x + width, y); - verts.push(r, g, b, alpha); - - verts.push(x , y + height); - verts.push(r, g, b, alpha); - - verts.push(x + width, y + height); - verts.push(r, g, b, alpha); - - // insert 2 dead triangles.. - indices.push(vertPos, vertPos, vertPos+1, vertPos+2, vertPos+3, vertPos+3); - } - - if (graphicsData.lineWidth) { - var tempPoints = graphicsData.points; - - graphicsData.points = [x, y, - x + width, y, - x + width, y + height, - x, y + height, - x, y]; - - - WebGLGraphics.buildLine(graphicsData, webGLData); - - graphicsData.points = tempPoints; - } -}; - -/** - * Builds a rounded rectangle to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildRoundedRectangle = function (graphicsData, webGLData) { - var rrectData = graphicsData.shape; - var x = rrectData.x; - var y = rrectData.y; - var width = rrectData.width; - var height = rrectData.height; - - var radius = rrectData.radius; - - var recPoints = []; - recPoints.push(x, y + radius); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x, y + height - radius, x, y + height, x + radius, y + height)); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width - radius, y + height, x + width, y + height, x + width, y + height - radius)); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width, y + radius, x + width, y, x + width - radius, y)); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + radius, y, x, y, x, y + radius)); - - if (graphicsData.fill) { - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var verts = webGLData.points; - var indices = webGLData.indices; - - var vecPos = verts.length/6; - - var triangles = utils.PolyK.Triangulate(recPoints); - - // - - var i = 0; - for (i = 0; i < triangles.length; i+=3) { - indices.push(triangles[i] + vecPos); - indices.push(triangles[i] + vecPos); - indices.push(triangles[i+1] + vecPos); - indices.push(triangles[i+2] + vecPos); - indices.push(triangles[i+2] + vecPos); - } - - - for (i = 0; i < recPoints.length; i++) { - verts.push(recPoints[i], recPoints[++i], r, g, b, alpha); - } - } - - if (graphicsData.lineWidth) { - var tempPoints = graphicsData.points; - - graphicsData.points = recPoints; - - WebGLGraphics.buildLine(graphicsData, webGLData); - - graphicsData.points = tempPoints; - } -}; - -/** - * Calculate the points for a quadratic bezier curve. (helper function..) - * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c - * - * @static - * @private - * @param fromX {number} Origin point x - * @param fromY {number} Origin point x - * @param cpX {number} Control point x - * @param cpY {number} Control point y - * @param toX {number} Destination point x - * @param toY {number} Destination point y - * @return {number[]} - */ -WebGLGraphics.quadraticBezierCurve = function (fromX, fromY, cpX, cpY, toX, toY) { - - var xa, - ya, - xb, - yb, - x, - y, - n = 20, - points = []; - - function getPt(n1 , n2, perc) { - var diff = n2 - n1; - - return n1 + ( diff * perc ); - } - - var j = 0; - for (var i = 0; i <= n; i++ ) { - j = i / n; - - // The Green Line - xa = getPt( fromX , cpX , j ); - ya = getPt( fromY , cpY , j ); - xb = getPt( cpX , toX , j ); - yb = getPt( cpY , toY , j ); - - // The Black Dot - x = getPt( xa , xb , j ); - y = getPt( ya , yb , j ); - - points.push(x, y); - } - return points; -}; - -/** - * Builds a circle to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object to draw - * @param webGLData {object} - */ -WebGLGraphics.buildCircle = function (graphicsData, webGLData) { - // need to convert points to a nice regular data - var circleData = graphicsData.shape; - var x = circleData.x; - var y = circleData.y; - var width; - var height; - - // TODO - bit hacky?? - if (graphicsData.type === CONST.SHAPES.CIRC) { - width = circleData.radius; - height = circleData.radius; - } - else { - width = circleData.width; - height = circleData.height; - } - - var totalSegs = 40; - var seg = (Math.PI * 2) / totalSegs ; - - var i = 0; - - if (graphicsData.fill) { - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var verts = webGLData.points; - var indices = webGLData.indices; - - var vecPos = verts.length/6; - - indices.push(vecPos); - - for (i = 0; i < totalSegs + 1 ; i++) { - verts.push(x,y, r, g, b, alpha); - - verts.push(x + Math.sin(seg * i) * width, - y + Math.cos(seg * i) * height, - r, g, b, alpha); - - indices.push(vecPos++, vecPos++); - } - - indices.push(vecPos-1); - } - - if (graphicsData.lineWidth) { - var tempPoints = graphicsData.points; - - graphicsData.points = []; - - for (i = 0; i < totalSegs + 1; i++) { - graphicsData.points.push(x + Math.sin(seg * i) * width, - y + Math.cos(seg * i) * height); - } - - WebGLGraphics.buildLine(graphicsData, webGLData); - - graphicsData.points = tempPoints; - } -}; - -/** - * Builds a line to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildLine = function (graphicsData, webGLData) { - // TODO OPTIMISE! - var i = 0; - var points = graphicsData.points; - - if (points.length === 0) { - return; - } - - // if the line width is an odd number add 0.5 to align to a whole pixel - if (graphicsData.lineWidth%2) { - for (i = 0; i < points.length; i++) { - points[i] += 0.5; - } - } - - // get first and last point.. figure out the middle! - var firstPoint = new math.Point(points[0], points[1]); - var lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]); - - // if the first point is the last point - gonna have issues :) - if (firstPoint.x === lastPoint.x && firstPoint.y === lastPoint.y) { - // need to clone as we are going to slightly modify the shape.. - points = points.slice(); - - points.pop(); - points.pop(); - - lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]); - - var midPointX = lastPoint.x + (firstPoint.x - lastPoint.x) *0.5; - var midPointY = lastPoint.y + (firstPoint.y - lastPoint.y) *0.5; - - points.unshift(midPointX, midPointY); - points.push(midPointX, midPointY); - } - - var verts = webGLData.points; - var indices = webGLData.indices; - var length = points.length / 2; - var indexCount = points.length; - var indexStart = verts.length/6; - - // DRAW the Line - var width = graphicsData.lineWidth / 2; - - // sort color - var color = utils.hex2rgb(graphicsData.lineColor); - var alpha = graphicsData.lineAlpha; - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var px, py, p1x, p1y, p2x, p2y, p3x, p3y; - var perpx, perpy, perp2x, perp2y, perp3x, perp3y; - var a1, b1, c1, a2, b2, c2; - var denom, pdist, dist; - - p1x = points[0]; - p1y = points[1]; - - p2x = points[2]; - p2y = points[3]; - - perpx = -(p1y - p2y); - perpy = p1x - p2x; - - dist = Math.sqrt(perpx*perpx + perpy*perpy); - - perpx /= dist; - perpy /= dist; - perpx *= width; - perpy *= width; - - // start - verts.push(p1x - perpx , p1y - perpy, - r, g, b, alpha); - - verts.push(p1x + perpx , p1y + perpy, - r, g, b, alpha); - - for (i = 1; i < length-1; i++) { - p1x = points[(i-1)*2]; - p1y = points[(i-1)*2 + 1]; - - p2x = points[(i)*2]; - p2y = points[(i)*2 + 1]; - - p3x = points[(i+1)*2]; - p3y = points[(i+1)*2 + 1]; - - perpx = -(p1y - p2y); - perpy = p1x - p2x; - - dist = Math.sqrt(perpx*perpx + perpy*perpy); - perpx /= dist; - perpy /= dist; - perpx *= width; - perpy *= width; - - perp2x = -(p2y - p3y); - perp2y = p2x - p3x; - - dist = Math.sqrt(perp2x*perp2x + perp2y*perp2y); - perp2x /= dist; - perp2y /= dist; - perp2x *= width; - perp2y *= width; - - a1 = (-perpy + p1y) - (-perpy + p2y); - b1 = (-perpx + p2x) - (-perpx + p1x); - c1 = (-perpx + p1x) * (-perpy + p2y) - (-perpx + p2x) * (-perpy + p1y); - a2 = (-perp2y + p3y) - (-perp2y + p2y); - b2 = (-perp2x + p2x) - (-perp2x + p3x); - c2 = (-perp2x + p3x) * (-perp2y + p2y) - (-perp2x + p2x) * (-perp2y + p3y); - - denom = a1*b2 - a2*b1; - - if (Math.abs(denom) < 0.1 ) { - - denom+=10.1; - verts.push(p2x - perpx , p2y - perpy, - r, g, b, alpha); - - verts.push(p2x + perpx , p2y + perpy, - r, g, b, alpha); - - continue; - } - - px = (b1*c2 - b2*c1)/denom; - py = (a2*c1 - a1*c2)/denom; - - - pdist = (px -p2x) * (px -p2x) + (py -p2y) + (py -p2y); - - - if (pdist > 140 * 140) { - perp3x = perpx - perp2x; - perp3y = perpy - perp2y; - - dist = Math.sqrt(perp3x*perp3x + perp3y*perp3y); - perp3x /= dist; - perp3y /= dist; - perp3x *= width; - perp3y *= width; - - verts.push(p2x - perp3x, p2y -perp3y); - verts.push(r, g, b, alpha); - - verts.push(p2x + perp3x, p2y +perp3y); - verts.push(r, g, b, alpha); - - verts.push(p2x - perp3x, p2y -perp3y); - verts.push(r, g, b, alpha); - - indexCount++; - } - else { - - verts.push(px , py); - verts.push(r, g, b, alpha); - - verts.push(p2x - (px-p2x), p2y - (py - p2y)); - verts.push(r, g, b, alpha); - } - } - - p1x = points[(length-2)*2]; - p1y = points[(length-2)*2 + 1]; - - p2x = points[(length-1)*2]; - p2y = points[(length-1)*2 + 1]; - - perpx = -(p1y - p2y); - perpy = p1x - p2x; - - dist = Math.sqrt(perpx*perpx + perpy*perpy); - perpx /= dist; - perpy /= dist; - perpx *= width; - perpy *= width; - - verts.push(p2x - perpx , p2y - perpy); - verts.push(r, g, b, alpha); - - verts.push(p2x + perpx , p2y + perpy); - verts.push(r, g, b, alpha); - - indices.push(indexStart); - - for (i = 0; i < indexCount; i++) { - indices.push(indexStart++); - } - - indices.push(indexStart-1); -}; - -/** - * Builds a complex polygon to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildComplexPoly = function (graphicsData, webGLData) { - //TODO - no need to copy this as it gets turned into a FLoat32Array anyways.. - var points = graphicsData.points.slice(); - - if (points.length < 6) { - return; - } - - // get first and last point.. figure out the middle! - var indices = webGLData.indices; - webGLData.points = points; - webGLData.alpha = graphicsData.fillAlpha; - webGLData.color = utils.hex2rgb(graphicsData.fillColor); - - // calclate the bounds.. - var minX = Infinity; - var maxX = -Infinity; - - var minY = Infinity; - var maxY = -Infinity; - - var x,y; - - // get size.. - for (var i = 0; i < points.length; i+=2) { - x = points[i]; - y = points[i+1]; - - minX = x < minX ? x : minX; - maxX = x > maxX ? x : maxX; - - minY = y < minY ? y : minY; - maxY = y > maxY ? y : maxY; - } - - // add a quad to the end cos there is no point making another buffer! - points.push(minX, minY, - maxX, minY, - maxX, maxY, - minX, maxY); - - // push a quad onto the end.. - - //TODO - this aint needed! - var length = points.length / 2; - for (i = 0; i < length; i++) { - indices.push( i ); - } - -}; - -/** - * Builds a polygon to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildPoly = function (graphicsData, webGLData) { - var points = graphicsData.points; - - if (points.length < 6) { - return; - } - // get first and last point.. figure out the middle! - var verts = webGLData.points; - var indices = webGLData.indices; - - var length = points.length / 2; - - // sort color - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var triangles = utils.PolyK.Triangulate(points); - - if (!triangles) { - return false; - } - - var vertPos = verts.length / 6; - - var i = 0; - - for (i = 0; i < triangles.length; i+=3) { - indices.push(triangles[i] + vertPos); - indices.push(triangles[i] + vertPos); - indices.push(triangles[i+1] + vertPos); - indices.push(triangles[i+2] +vertPos); - indices.push(triangles[i+2] + vertPos); - } - - for (i = 0; i < length; i++) { - verts.push(points[i * 2], points[i * 2 + 1], - r, g, b, alpha); - } - - return true; -}; - -WebGLGraphics.graphicsDataPool = []; - -},{"../../../const":4,"../../../math":12,"../../../utils":50,"./WebGLGraphicsData":40}],40:[function(require,module,exports){ -/** - * @class - * @private - */ -function WebGLGraphicsData(gl) { - this.gl = gl; - - //TODO does this need to be split before uploding?? - this.color = [0, 0, 0]; // color split! - this.points = []; - this.indices = []; - this.buffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - this.mode = 1; - this.alpha = 1; - this.dirty = true; -} - -WebGLGraphicsData.prototype.constructor = WebGLGraphicsData; -module.exports = WebGLGraphicsData; - -/** - * - */ -WebGLGraphicsData.prototype.reset = function () { - this.points.length = 0; - this.indices.length = 0; -}; - -/** - * - */ -WebGLGraphicsData.prototype.upload = function () { - var gl = this.gl; - -// this.lastIndex = graphics.graphicsData.length; - this.glPoints = new Float32Array(this.points); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.buffer); - gl.bufferData(gl.ARRAY_BUFFER, this.glPoints, gl.STATIC_DRAW); - - this.glIndicies = new Uint16Array(this.indices); - - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.glIndicies, gl.STATIC_DRAW); - - this.dirty = false; -}; - -},{}],41:[function(require,module,exports){ -var TextureUvs = require('../../../textures/TextureUvs'), - Shader = require('../shaders/Shader'); - -/** - * @author Mat Groves - * - * Big thanks to the very clever Matt DesLauriers https://github.com/mattdesl/ - * for creating the original pixi version! - * Also a thanks to https://github.com/bchevalier for tweaking the tint and alpha so that they now share 4 bytes on the vertex buffer - * - * Heavily inspired by LibGDX's WebGLSpriteBatch: - * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java - */ - -/** - * - * @class - * @private - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this sprite batch works for. - */ -function WebGLSpriteBatch(renderer) { - /** - * - * - * @member {WebGLRenderer} - */ - this.renderer = renderer; - - /** - * - * - * @member {number} - */ - this.vertSize = 5; - - /** - * The number of images in the SpriteBatch before it flushes. - * - * @member {number} - */ - this.size = 2000;//Math.pow(2, 16) / this.vertSize; - - // the total number of bytes in our batch - var numVerts = this.size * 4 * 4 * this.vertSize; - // the total number of indices in our batch - var numIndices = this.size * 6; - - /** - * Holds the vertices - * - * @member {ArrayBuffer} - */ - this.vertices = new ArrayBuffer(numVerts); - - /** - * View on the vertices as a Float32Array - * - * @member {Float32Array} - */ - this.positions = new Float32Array(this.vertices); - - /** - * View on the vertices as a Uint32Array - * - * @member {Uint32Array} - */ - this.colors = new Uint32Array(this.vertices); - - /** - * Holds the indices - * - * @member {Uint16Array} - */ - this.indices = new Uint16Array(numIndices); - - /** - * - * - * @member {number} - */ - this.lastIndexCount = 0; - - for (var i=0, j=0; i < numIndices; i += 6, j += 4) { - this.indices[i + 0] = j + 0; - this.indices[i + 1] = j + 1; - this.indices[i + 2] = j + 2; - this.indices[i + 3] = j + 0; - this.indices[i + 4] = j + 2; - this.indices[i + 5] = j + 3; - } - - /** - * - * - * @member {boolean} - */ - this.drawing = false; - - /** - * - * - * @member {number} - */ - this.currentBatchSize = 0; - - /** - * - * - * @member {BaseTexture} - */ - this.currentBaseTexture = null; - - /** - * - * - * @member {boolean} - */ - this.dirty = true; - - /** - * - * - * @member {Array} - */ - this.textures = []; - - /** - * - * - * @member {Array} - */ - this.blendModes = []; - - /** - * - * - * @member {Array} - */ - this.shaders = []; - - /** - * - * - * @member {Array} - */ - this.sprites = []; - - /** - * The default shader that is used if a sprite doesn't have a more specific one. - * - * @member {Shader} - */ - this.shader = null; - - // listen for context and update necessary buffers - var self = this; - this.renderer.on('context', function () { - self.setupContext(); - }); -} - -WebGLSpriteBatch.prototype.constructor = WebGLSpriteBatch; -module.exports = WebGLSpriteBatch; - -/** - * @param gl {WebGLContext} the current WebGL drawing context - */ -WebGLSpriteBatch.prototype.setupContext = function () { - var gl = this.renderer.gl; - - // setup default shader - this.shader = new Shader(gl); - - // create a couple of buffers - this.vertexBuffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - - // 65535 is max index, so 65535 / 6 = 10922. - - //upload the index data - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW); - - this.currentBlendMode = 99999; -}; - -/** - * - */ -WebGLSpriteBatch.prototype.begin = function () { - // this.shader = this.renderer.shaderManager.defaultShader; - - this.start(); -}; - -/** - */ -WebGLSpriteBatch.prototype.end = function () { - this.flush(); -}; - -/** - * @param sprite {Sprite} the sprite to render when using this spritebatch - */ -WebGLSpriteBatch.prototype.render = function (sprite) { - var texture = sprite.texture; - - //TODO set blend modes.. - // check texture.. - if (this.currentBatchSize >= this.size) { - this.flush(); - this.currentBaseTexture = texture.baseTexture; - } - - // get the uvs for the texture - var uvs = texture._uvs; - - // if the uvs have not updated then no point rendering just yet! - if (!uvs) { - return; - } - - // TODO trim?? - var aX = sprite.anchor.x; - var aY = sprite.anchor.y; - - var w0, w1, h0, h1; - - if (texture.trim) { - // if the sprite is trimmed then we need to add the extra space before transforming the sprite coords.. - var trim = texture.trim; - - w1 = trim.x - aX * trim.width; - w0 = w1 + texture.crop.width; - - h1 = trim.y - aY * trim.height; - h0 = h1 + texture.crop.height; - - } - else { - w0 = (texture.frame.width ) * (1-aX); - w1 = (texture.frame.width ) * -aX; - - h0 = texture.frame.height * (1-aY); - h1 = texture.frame.height * -aY; - } - - var index = this.currentBatchSize * 4 * this.vertSize; - - var resolution = texture.baseTexture.resolution; - - var worldTransform = sprite.worldTransform; - - var a = worldTransform.a / resolution; - var b = worldTransform.b / resolution; - var c = worldTransform.c / resolution; - var d = worldTransform.d / resolution; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var colors = this.colors; - var positions = this.positions; - - if (this.renderer.roundPixels) { - // xy - positions[index] = a * w1 + c * h1 + tx | 0; - positions[index+1] = d * h1 + b * w1 + ty | 0; - - // xy - positions[index+5] = a * w0 + c * h1 + tx | 0; - positions[index+6] = d * h1 + b * w0 + ty | 0; - - // xy - positions[index+10] = a * w0 + c * h0 + tx | 0; - positions[index+11] = d * h0 + b * w0 + ty | 0; - - // xy - positions[index+15] = a * w1 + c * h0 + tx | 0; - positions[index+16] = d * h0 + b * w1 + ty | 0; - } - else { - // xy - positions[index] = a * w1 + c * h1 + tx; - positions[index+1] = d * h1 + b * w1 + ty; - - // xy - positions[index+5] = a * w0 + c * h1 + tx; - positions[index+6] = d * h1 + b * w0 + ty; - - // xy - positions[index+10] = a * w0 + c * h0 + tx; - positions[index+11] = d * h0 + b * w0 + ty; - - // xy - positions[index+15] = a * w1 + c * h0 + tx; - positions[index+16] = d * h0 + b * w1 + ty; - } - - // uv - positions[index+2] = uvs.x0; - positions[index+3] = uvs.y0; - - // uv - positions[index+7] = uvs.x1; - positions[index+8] = uvs.y1; - - // uv - positions[index+12] = uvs.x2; - positions[index+13] = uvs.y2; - - // uv - positions[index+17] = uvs.x3; - positions[index+18] = uvs.y3; - - // color and alpha - var tint = sprite.tint; - colors[index+4] = colors[index+9] = colors[index+14] = colors[index+19] = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (sprite.worldAlpha * 255 << 24); - - // increment the batchsize - this.sprites[this.currentBatchSize++] = sprite; - - -}; - -/** - * Renders a TilingSprite using the spriteBatch. - * - * @param sprite {TilingSprite} the tilingSprite to render - */ -WebGLSpriteBatch.prototype.renderTilingSprite = function (tilingSprite) { - var texture = tilingSprite.tilingTexture; - - // check texture.. - if (this.currentBatchSize >= this.size) { - //return; - this.flush(); - this.currentBaseTexture = texture.baseTexture; - } - - // set the textures uvs temporarily - // TODO create a separate texture so that we can tile part of a texture - - if (!tilingSprite._uvs) { - tilingSprite._uvs = new TextureUvs(); - } - - var uvs = tilingSprite._uvs; - - tilingSprite.tilePosition.x %= texture.baseTexture.width * tilingSprite.tileScaleOffset.x; - tilingSprite.tilePosition.y %= texture.baseTexture.height * tilingSprite.tileScaleOffset.y; - - var offsetX = tilingSprite.tilePosition.x/(texture.baseTexture.width*tilingSprite.tileScaleOffset.x); - var offsetY = tilingSprite.tilePosition.y/(texture.baseTexture.height*tilingSprite.tileScaleOffset.y); - - var scaleX = (tilingSprite.width / texture.baseTexture.width) / (tilingSprite.tileScale.x * tilingSprite.tileScaleOffset.x); - var scaleY = (tilingSprite.height / texture.baseTexture.height) / (tilingSprite.tileScale.y * tilingSprite.tileScaleOffset.y); - - uvs.x0 = 0 - offsetX; - uvs.y0 = 0 - offsetY; - - uvs.x1 = (1 * scaleX) - offsetX; - uvs.y1 = 0 - offsetY; - - uvs.x2 = (1 * scaleX) - offsetX; - uvs.y2 = (1 * scaleY) - offsetY; - - uvs.x3 = 0 - offsetX; - uvs.y3 = (1 * scaleY) - offsetY; - - // get the tilingSprites current alpha and tint and combining them into a single color - var tint = tilingSprite.tint; - var color = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (tilingSprite.alpha * 255 << 24); - - var positions = this.positions; - var colors = this.colors; - - var width = tilingSprite.width; - var height = tilingSprite.height; - - // TODO trim?? - var aX = tilingSprite.anchor.x; - var aY = tilingSprite.anchor.y; - var w0 = width * (1-aX); - var w1 = width * -aX; - - var h0 = height * (1-aY); - var h1 = height * -aY; - - var index = this.currentBatchSize * 4 * this.vertSize; - - var resolution = texture.baseTexture.resolution; - - var worldTransform = tilingSprite.worldTransform; - - var a = worldTransform.a / resolution;//[0]; - var b = worldTransform.b / resolution;//[3]; - var c = worldTransform.c / resolution;//[1]; - var d = worldTransform.d / resolution;//[4]; - var tx = worldTransform.tx;//[2]; - var ty = worldTransform.ty;//[5]; - - // xy - positions[index++] = a * w1 + c * h1 + tx; - positions[index++] = d * h1 + b * w1 + ty; - // uv - positions[index++] = uvs.x0; - positions[index++] = uvs.y0; - // color - colors[index++] = color; - - // xy - positions[index++] = (a * w0 + c * h1 + tx); - positions[index++] = d * h1 + b * w0 + ty; - // uv - positions[index++] = uvs.x1; - positions[index++] = uvs.y1; - // color - colors[index++] = color; - - // xy - positions[index++] = a * w0 + c * h0 + tx; - positions[index++] = d * h0 + b * w0 + ty; - // uv - positions[index++] = uvs.x2; - positions[index++] = uvs.y2; - // color - colors[index++] = color; - - // xy - positions[index++] = a * w1 + c * h0 + tx; - positions[index++] = d * h0 + b * w1 + ty; - // uv - positions[index++] = uvs.x3; - positions[index++] = uvs.y3; - // color - colors[index++] = color; - - // increment the batchsize - this.sprites[this.currentBatchSize++] = tilingSprite; -}; - -/** - * Renders the content and empties the current batch. - * - */ -WebGLSpriteBatch.prototype.flush = function () { - // If the batch is length 0 then return as there is nothing to draw - if (this.currentBatchSize === 0) { - return; - } - - var gl = this.renderer.gl; - var shader; - - if (this.dirty) { - this.dirty = false; - // bind the main texture - gl.activeTexture(gl.TEXTURE0); - - // bind the buffers - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - - // this is the same for each shader? - var stride = this.vertSize * 4; - gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0); - gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 2 * 4); - - // color attributes will be interpreted as unsigned bytes and normalized - gl.vertexAttribPointer(this.shader.attributes.aColor, 4, gl.UNSIGNED_BYTE, true, stride, 4 * 4); - } - - // upload the verts to the buffer - if (this.currentBatchSize > ( this.size * 0.5 ) ) { - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices); - } - else { - var view = this.positions.subarray(0, this.currentBatchSize * 4 * this.vertSize); - gl.bufferSubData(gl.ARRAY_BUFFER, 0, view); - } - - var nextTexture, nextBlendMode, nextShader; - var batchSize = 0; - var start = 0; - - var currentBaseTexture = null; - var currentBlendMode = this.renderer.blendModeManager.currentBlendMode; - var currentShader = null; - - var blendSwap = false; - var shaderSwap = false; - var sprite; - - for (var i = 0, j = this.currentBatchSize; i < j; i++) { - - sprite = this.sprites[i]; - - nextTexture = sprite.texture.baseTexture; - nextBlendMode = sprite.blendMode; - nextShader = sprite.shader || this.shader; - - blendSwap = currentBlendMode !== nextBlendMode; - shaderSwap = currentShader !== nextShader; // should I use uuidS??? - - if (currentBaseTexture !== nextTexture || blendSwap || shaderSwap) { - this.renderBatch(currentBaseTexture, batchSize, start); - - start = i; - batchSize = 0; - currentBaseTexture = nextTexture; - - if (blendSwap) { - currentBlendMode = nextBlendMode; - this.renderer.blendModeManager.setBlendMode( currentBlendMode ); - } - - if (shaderSwap) { - currentShader = nextShader; - - shader = currentShader.shaders ? currentShader.shaders[gl.id] : currentShader; - - if (!shader) { - shader = new Shader(gl, null, currentShader.fragmentSrc, currentShader.uniforms); - currentShader.shaders[gl.id] = shader; - } - - // set shader function??? - this.renderer.shaderManager.setShader(shader); - - if (shader.dirty) { - shader.syncUniforms(); - } - - // both thease only need to be set if they are changing.. - // set the projection - var projection = this.renderer.projection; - gl.uniform2f(shader.uniforms.projectionVector._location, projection.x, projection.y); - - // TODO - this is temprorary! - var offsetVector = this.renderer.offset; - gl.uniform2f(shader.uniforms.offsetVector._location, offsetVector.x, offsetVector.y); - - // set the pointers - } - } - - batchSize++; - } - - this.renderBatch(currentBaseTexture, batchSize, start); - - // then reset the batch! - this.currentBatchSize = 0; -}; - -/** - * @param texture {Texture} - * @param size {number} - * @param startIndex {number} - */ -WebGLSpriteBatch.prototype.renderBatch = function (texture, size, startIndex) { - if (size === 0) { - return; - } - - var gl = this.renderer.gl; - - if (!texture._glTextures[gl.id]) { - this.renderer.updateTexture(texture); - } - else { - // bind the current texture - gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]); - } - - // now draw those suckas! - gl.drawElements(gl.TRIANGLES, size * 6, gl.UNSIGNED_SHORT, startIndex * 6 * 2); - - // increment the draw count - this.renderer.drawCount++; -}; - -/** - * - */ -WebGLSpriteBatch.prototype.stop = function () { - this.flush(); - this.dirty = true; -}; - -/** - * - */ -WebGLSpriteBatch.prototype.start = function () { - this.dirty = true; -}; - -/** - * Destroys the SpriteBatch. - * - */ -WebGLSpriteBatch.prototype.destroy = function () { - this.renderer.gl.deleteBuffer(this.vertexBuffer); - this.renderer.gl.deleteBuffer(this.indexBuffer); - - this.vertices = null; - this.indices = null; - - this.vertexBuffer = null; - this.indexBuffer = null; - - this.currentBaseTexture = null; - - this.renderer = null; -}; - -},{"../../../textures/TextureUvs":45,"../shaders/Shader":35}],42:[function(require,module,exports){ -var utils = require('../utils'), - CONST = require('../const'); - -/** - * A texture stores the information that represents an image. All textures have a base texture. - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param source {Image|Canvas} the source object of the texture. - * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values - */ -function BaseTexture(source, scaleMode) { - this.uuid = utils.uuid(); - - /** - * The Resolution of the texture. - * - * @member {number} - */ - this.resolution = 1; - - /** - * The width of the base texture set when the image has loaded - * - * @member {number} - * @readOnly - */ - this.width = 100; - - /** - * The height of the base texture set when the image has loaded - * - * @member {number} - * @readOnly - */ - this.height = 100; - - /** - * The scale mode to apply when scaling this texture - * - * @member {{number}} - * @default scaleModes.LINEAR - */ - this.scaleMode = scaleMode || CONST.scaleModes.DEFAULT; - - /** - * Set to true once the base texture has successfully loaded. - * - * This is never true if the underlying source fails to load or has no texture data. - * - * @member {boolean} - * @readOnly - */ - this.hasLoaded = false; - - /** - * Set to true if the source is currently loading. - * - * If an Image source is loading the 'loaded' or 'error' event will be - * dispatched when the operation ends. An underyling source that is - * immediately-available bypasses loading entirely. - * - * @member {boolean} - * @readonly - */ - this.isLoading = false; - - /** - * The image source that is used to create the texture. - * - * TODO: Make this a setter that calls loadSource(); - * - * @member {Image|Canvas} - * @readonly - */ - this.source = null; // set in loadSource, if at all - - /** - * Controls if RGB channels should be pre-multiplied by Alpha (WebGL only) - * - * @member {boolean} - * @default true - */ - this.premultipliedAlpha = true; - - /** - * @member {string} - */ - this.imageUrl = null; - - /** - * @member {boolean} - * @private - */ - this._powerOf2 = false; - - // used for webGL - - /** - * - * Set this to true if a mipmap of this texture needs to be generated. This value needs to be set before the texture is used - * Also the texture must be a power of two size to work - * - * @member {boolean} - */ - this.mipmap = false; - - /** - * A map of renderer IDs to webgl textures - * - * @member {object} - * @private - */ - this._glTextures = {}; - - /** - * Does the texture on the GPU need to be updated? - * - * @member {boolean} - * @private - */ - this._needsUpdate = false; - - // if no source passed don't try to load - if (source) { - this.loadSource(source); - } - - /** - * Fired when a not-immediately-available source finishes loading. - * - * @event loaded - * @protected - */ - - /** - * Fired when a not-immediately-available source fails to load. - * - * @event error - * @protected - */ -} - -BaseTexture.prototype.constructor = BaseTexture; -module.exports = BaseTexture; - -utils.eventTarget.mixin(BaseTexture.prototype); - -Object.defineProperties(BaseTexture.prototype, { - needsUpdate: { - get: function () { - return this._needsUpdate; - }, - set: function (val) { - this._needsUpdate = val; - - if (val) { - this.emit('update', this); - } - } - } -}); - -/** - * Load a source. - * - * If the source is not-immediately-available, such as an image that needs to be - * downloaded, then the 'loaded' or 'error' event will be dispatched in the future - * and `hasLoaded` will remain false after this call. - * - * The logic state after calling `loadSource` directly or indirectly (eg. `fromImage`, `new BaseTexture`) is: - * - * if (texture.hasLoaded) { - * // texture ready for use - * } else if (texture.isLoading) { - * // listen to 'loaded' and/or 'error' events on texture - * } else { - * // not loading, not going to load UNLESS the source is reloaded - * // (it may still make sense to listen to the events) - * } - * - * @protected - * @param source {Image|Canvas} the source object of the texture. - */ -BaseTexture.prototype.loadSource = function (source) { - var wasLoading = this.isLoading; - this.hasLoaded = false; - this.isLoading = false; - - if (wasLoading && this.source) { - this.source.onload = null; - this.source.onerror = null; - } - - this.source = source; - - // Apply source if loaded. Otherwise setup appropriate loading monitors. - if ((this.source.complete || this.source.getContext) && this.source.width && this.source.height) { - this._sourceLoaded(); - } - else if(!source.getContext) { - // Image fail / not ready - this.isLoading = true; - - var scope = this; - - source.onload = function () { - source.onload = null; - source.onerror = null; - - if(!scope.isLoading) { - return; - } - - scope.isLoading = false; - scope._sourceLoaded(); - - scope.emit('loaded', scope); - }; - - source.onerror = function () { - source.onload = null; - source.onerror = null; - - if(!scope.isLoading) { - return; - } - - scope.isLoading = false; - scope.emit('error', scope); - }; - - // Per http://www.w3.org/TR/html5/embedded-content-0.html#the-img-element - // "The value of `complete` can thus change while a script is executing." - // So complete needs to be re-checked after the callbacks have been added.. - if (source.complete) { - this.isLoading = false; - - // ..and if we're complete now, no need for callbacks - source.onload = null; - source.onerror = null; - - if (source.width && source.height) { - this._sourceLoaded(); - - // If any previous subscribers possible - if (wasLoading) { - this.emit('loaded', this); - } - } - else { - // If any previous subscribers possible - if (wasLoading) { - this.emit('error', this); - } - } - } - } -}; - -/** - * Used internally to update the width, height, and some other tracking vars once - * a source has successfully loaded. - * - * @private - */ -BaseTexture.prototype._sourceLoaded = function () { - this.hasLoaded = true; - - this.width = this.source.naturalWidth || this.source.width; - this.height = this.source.naturalHeight || this.source.height; - - this.needsUpdate = true; -}; - -/** - * Destroys this base texture - * - */ -BaseTexture.prototype.destroy = function () { - if (this.imageUrl) { - delete utils.BaseTextureCache[this.imageUrl]; - delete utils.TextureCache[this.imageUrl]; - this.imageUrl = null; - if (!navigator.isCocoonJS) { - this.source.src = ''; - } - } - else if (this.source && this.source._pixiId) { - delete utils.BaseTextureCache[this.source._pixiId]; - } - this.source = null; - - this.dispose(); -}; - -/** - * Frees the texture from WebGL memory without destroying this texture object. - * This means you can still use the texture later which will upload it to GPU - * memory again. - * - */ -BaseTexture.prototype.dispose = function () { - this.emit('dispose', this); -}; - -/** - * Changes the source image of the texture. - * The original source must be an Image element. - * - * @param newSrc {string} the path of the image - */ -BaseTexture.prototype.updateSourceImage = function (newSrc) { - this.source.src = newSrc; - this.loadSource(this.source); -}; - -/** - * Helper function that creates a base texture from the given image url. - * If the image is not in the base texture cache it will be created and loaded. - * - * @static - * @param imageUrl {string} The image url of the texture - * @param [crossorigin=(auto)] {boolean} Should use anonymouse CORS? Defaults to true if the URL is not a data-URI. - * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values - * @return BaseTexture - */ -BaseTexture.fromImage = function (imageUrl, crossorigin, scaleMode) { - var baseTexture = utils.BaseTextureCache[imageUrl]; - - if (crossorigin === undefined && imageUrl.indexOf('data:') !== 0) { - crossorigin = true; - } - - if (!baseTexture) { - // new Image() breaks tex loading in some versions of Chrome. - // See https://code.google.com/p/chromium/issues/detail?id=238071 - var image = new Image();//document.createElement('img'); - if (crossorigin) { - image.crossOrigin = ''; - } - - image.src = imageUrl; - baseTexture = new BaseTexture(image, scaleMode); - baseTexture.imageUrl = imageUrl; - utils.BaseTextureCache[imageUrl] = baseTexture; - - // if there is an @2x at the end of the url we are going to assume its a highres image - if ( imageUrl.indexOf(CONST.RETINA_PREFIX + '.') !== -1) { - baseTexture.resolution = 2; - } - } - - return baseTexture; -}; - -/** - * Helper function that creates a base texture from the given canvas element. - * - * @static - * @param canvas {Canvas} The canvas element source of the texture - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return BaseTexture - */ -BaseTexture.fromCanvas = function (canvas, scaleMode) { - if (!canvas._pixiId) { - canvas._pixiId = 'canvas_' + utils.uuid(); - } - - var baseTexture = utils.BaseTextureCache[canvas._pixiId]; - - if (!baseTexture) { - baseTexture = new BaseTexture(canvas, scaleMode); - utils.BaseTextureCache[canvas._pixiId] = baseTexture; - } - - return baseTexture; -}; - -},{"../const":4,"../utils":50}],43:[function(require,module,exports){ -var BaseTexture = require('./BaseTexture'), - Texture = require('./Texture'), - FilterTexture = require('../renderers/webgl/utils/FilterTexture'), - CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'), - math = require('../math'), - CONST = require('../const'); - -/** - * A RenderTexture is a special texture that allows any Pixi display object to be rendered to it. - * - * __Hint__: All DisplayObjects (i.e. Sprites) that render to a RenderTexture should be preloaded - * otherwise black rectangles will be drawn instead. - * - * A RenderTexture takes a snapshot of any Display Object given to its render method. The position - * and rotation of the given Display Objects is ignored. For example: - * - * ```js - * var renderTexture = new PIXI.RenderTexture(800, 600); - * var sprite = PIXI.Sprite.fromImage("spinObj_01.png"); - * - * sprite.position.x = 800/2; - * sprite.position.y = 600/2; - * sprite.anchor.x = 0.5; - * sprite.anchor.y = 0.5; - * - * renderTexture.render(sprite); - * ``` - * - * The Sprite in this case will be rendered to a position of 0,0. To render this sprite at its actual - * position a DisplayObjectContainer should be used: - * - * ```js - * var doc = new DisplayObjectContainer(); - * - * doc.addChild(sprite); - * - * renderTexture.render(doc); // Renders to center of renderTexture - * ``` - * - * @class - * @extends Texture - * @namespace PIXI - * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used for this RenderTexture - * @param [width=100] {number} The width of the render texture - * @param [height=100] {number} The height of the render texture - * @param [scaleMode] {number} See {@link scaleModes} for possible values - * @param [resolution=1] {number} The resolution of the texture being generated - */ -function RenderTexture(renderer, width, height, scaleMode, resolution) { - if (!renderer) { - throw new Error('Unable to create RenderTexture, you must pass a renderer into the constructor.'); - } - - /** - * The with of the render texture - * - * @member {number} - */ - this.width = width || 100; - - /** - * The height of the render texture - * - * @member {number} - */ - this.height = height || 100; - - /** - * The Resolution of the texture. - * - * @member {number} - */ - this.resolution = resolution || 1; - - /** - * The framing rectangle of the render texture - * - * @member {Rectangle} - */ - this.frame = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution); - - /** - * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering, - * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases) - * - * @member {Rectangle} - */ - this.crop = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution); - - /** - * Draw/render the given DisplayObject onto the texture. - * - * The displayObject and descendents are transformed during this operation. - * If `restoreWorldTransform` is true then the transformations will be restored before the - * method returns. Otherwise it is up to the calling code to correctly use or reset - * the transformed display objects. - * - * The display object is always rendered with a worldAlpha value of 1. - * - * @method - * @param displayObject {DisplayObject} The display object to render this texture on - * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering. - * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn - * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children - * transformations will be restored. Not restoring this information will be a little faster. - */ - this.render = null; - - /** - * The base texture object that this texture uses - * - * @member {BaseTexture} - */ - this.baseTexture = new BaseTexture(); - this.baseTexture.width = this.width * this.resolution; - this.baseTexture.height = this.height * this.resolution; - this.baseTexture._glTextures = []; - this.baseTexture.resolution = this.resolution; - - this.baseTexture.scaleMode = scaleMode || CONST.scaleModes.DEFAULT; - - this.baseTexture.hasLoaded = true; - - Texture.call(this, - this.baseTexture, - new math.Rectangle(0, 0, this.width, this.height) - ); - - /** - * The renderer this RenderTexture uses. A RenderTexture can only belong to one renderer at the moment if its webGL. - * - * @member {CanvasRenderer|WebGLRenderer} - */ - this.renderer = renderer; - - if (this.renderer.type === CONST.WEBGL_RENDERER) { - var gl = this.renderer.gl; - - this.textureBuffer = new FilterTexture(gl, this.width * this.resolution, this.height * this.resolution, this.baseTexture.scaleMode); - this.baseTexture._glTextures[gl.id] = this.textureBuffer.texture; - - this.render = this.renderWebGL; - this.projection = new math.Point(this.width*0.5, -this.height*0.5); - } - else { - this.render = this.renderCanvas; - this.textureBuffer = new CanvasBuffer(this.width* this.resolution, this.height* this.resolution); - this.baseTexture.source = this.textureBuffer.canvas; - } - - /** - * @member {boolean} - */ - this.valid = true; - - this._updateUvs(); -} - -RenderTexture.prototype = Object.create(Texture.prototype); -RenderTexture.prototype.constructor = RenderTexture; -module.exports = RenderTexture; - -/** - * Resizes the RenderTexture. - * - * @param width {number} The width to resize to. - * @param height {number} The height to resize to. - * @param updateBase {boolean} Should the baseTexture.width and height values be resized as well? - */ -RenderTexture.prototype.resize = function (width, height, updateBase) { - if (width === this.width && height === this.height) { - return; - } - - this.valid = (width > 0 && height > 0); - - this.width = this.frame.width = this.crop.width = width; - this.height = this.frame.height = this.crop.height = height; - - if (updateBase) { - this.baseTexture.width = this.width; - this.baseTexture.height = this.height; - } - - if (this.renderer.type === CONST.WEBGL_RENDERER) { - this.projection.x = this.width / 2; - this.projection.y = -this.height / 2; - } - - if (!this.valid) { - return; - } - - this.textureBuffer.resize(this.width * this.resolution, this.height * this.resolution); -}; - -/** - * Clears the RenderTexture. - * - */ -RenderTexture.prototype.clear = function () { - if (!this.valid) { - return; - } - - if (this.renderer.type === CONST.WEBGL_RENDERER) { - this.renderer.gl.bindFramebuffer(this.renderer.gl.FRAMEBUFFER, this.textureBuffer.frameBuffer); - } - - this.textureBuffer.clear(); -}; - -/** - * Internal method assigned to the `render` property if using a CanvasRenderer. - * - * @private - * @param displayObject {DisplayObject} The display object to render this texture on - * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering. - * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn - * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children - * transformations will be restored. Not restoring this information will be a little faster. - */ -RenderTexture.prototype.renderWebGL = function (displayObject, matrix, clear, restoreWorldTransform) { - if (!this.valid) { - return; - } - - if (typeof restoreWorldTransform === 'undefined') { - restoreWorldTransform = true; - } - - var tempAlpha, - tempTransform; - - if (restoreWorldTransform) { - tempAlpha = displayObject.worldAlpha; - tempTransform = displayObject.worldTransform.toArray(); - } - - //TOOD replace position with matrix.. - - //Lets create a nice matrix to apply to our display object. Frame buffers come in upside down so we need to flip the matrix - var wt = displayObject.worldTransform; - - wt.identity(); - wt.translate(0, this.projection.y * 2); - - if (matrix) { - wt.append(matrix); - } - - wt.scale(1,-1); - - // setWorld Alpha to ensure that the object is renderer at full opacity - displayObject.worldAlpha = 1; - - // Time to update all the children of the displayObject with the new matrix.. - var children = displayObject.children; - var i, j; - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - - // time for the webGL fun stuff! - var gl = this.renderer.gl; - - gl.viewport(0, 0, this.width * this.resolution, this.height * this.resolution); - - gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer ); - - if (clear) { - this.textureBuffer.clear(); - } - - this.renderer.spriteBatch.dirty = true; - - this.renderer.renderDisplayObject(displayObject, this.projection, this.textureBuffer.frameBuffer); - - this.renderer.spriteBatch.dirty = true; - - if (restoreWorldTransform) { - displayObject.worldAlpha = tempAlpha; - displayObject.worldTransform.fromArray(tempTransform); - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - } -}; - - -/** - * Internal method assigned to the `render` property if using a CanvasRenderer. - * - * @private - * @param displayObject {DisplayObject} The display object to render this texture on - * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering. - * @param [clear] {boolean} If true the texture will be cleared before the displayObject is drawn - * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children - * transformations will be restored. Not restoring this information will be a little faster. - */ -RenderTexture.prototype.renderCanvas = function (displayObject, matrix, clear, restoreWorldTransform) { - if (!this.valid) { - return; - } - - if (typeof restoreWorldTransform === 'undefined') { - restoreWorldTransform = true; - } - - var tempAlpha, - tempTransform; - - if (restoreWorldTransform) { - tempAlpha = displayObject.worldAlpha; - tempTransform = displayObject.worldTransform.toArray(); - } - - var wt = displayObject.worldTransform; - wt.identity(); - if (matrix) { - wt.append(matrix); - } - - // setWorld Alpha to ensure that the object is renderer at full opacity - displayObject.worldAlpha = 1; - - // Time to update all the children of the displayObject with the new matrix.. - var children = displayObject.children; - var i, j; - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - - if (clear) { - this.textureBuffer.clear(); - } - - var context = this.textureBuffer.context; - - var realResolution = this.renderer.resolution; - - this.renderer.resolution = this.resolution; - - this.renderer.renderDisplayObject(displayObject, context); - - this.renderer.resolution = realResolution; - - if (restoreWorldTransform) { - displayObject.worldAlpha = tempAlpha; - displayObject.worldTransform.fromArray(tempTransform); - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - } -}; - -/** - * Will return a HTML Image of the texture - * - * @return {Image} - */ -RenderTexture.prototype.getImage = function () { - var image = new Image(); - image.src = this.getBase64(); - return image; -}; - -/** - * Will return a a base64 encoded string of this texture. It works by calling RenderTexture.getCanvas and then running toDataURL on that. - * - * @return {string} A base64 encoded string of the texture. - */ -RenderTexture.prototype.getBase64 = function () { - return this.getCanvas().toDataURL(); -}; - -/** - * Creates a Canvas element, renders this RenderTexture to it and then returns it. - * - * @return {HTMLCanvasElement} A Canvas element with the texture rendered on. - */ -RenderTexture.prototype.getCanvas = function () { - if (this.renderer.type === CONST.WEBGL_RENDERER) { - var gl = this.renderer.gl; - var width = this.textureBuffer.width; - var height = this.textureBuffer.height; - - var webGLPixels = new Uint8Array(4 * width * height); - - gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer); - gl.readPixels(0, 0, width, height, gl.RGBA, gl.UNSIGNED_BYTE, webGLPixels); - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - - var tempCanvas = new CanvasBuffer(width, height); - var canvasData = tempCanvas.context.getImageData(0, 0, width, height); - canvasData.data.set(webGLPixels); - - tempCanvas.context.putImageData(canvasData, 0, 0); - - return tempCanvas.canvas; - } - else { - return this.textureBuffer.canvas; - } -}; - -},{"../const":4,"../math":12,"../renderers/canvas/utils/CanvasBuffer":21,"../renderers/webgl/utils/FilterTexture":37,"./BaseTexture":42,"./Texture":44}],44:[function(require,module,exports){ -var BaseTexture = require('./BaseTexture'), - VideoBaseTexture = require('./VideoBaseTexture'), - TextureUvs = require('./TextureUvs'), - eventTarget = require('../utils/eventTarget'), - math = require('../math'), - utils = require('../utils'); - -/** - * A texture stores the information that represents an image or part of an image. It cannot be added - * to the display list directly. Instead use it as the texture for a Sprite. If no frame is provided then the whole image is used. - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param baseTexture {BaseTexture} The base texture source to create the texture from - * @param [frame] {Rectangle} The rectangle frame of the texture to show - * @param [crop] {Rectangle} The area of original texture - * @param [trim] {Rectangle} Trimmed texture rectangle - */ -function Texture(baseTexture, frame, crop, trim) { - /** - * Does this Texture have any frame data assigned to it? - * - * @member {boolean} - */ - this.noFrame = false; - - if (!frame) { - this.noFrame = true; - frame = new math.Rectangle(0, 0, 1, 1); - } - - if (baseTexture instanceof Texture) { - baseTexture = baseTexture.baseTexture; - } - - /** - * The base texture that this texture uses. - * - * @member {BaseTexture} - */ - this.baseTexture = baseTexture; - - /** - * The frame specifies the region of the base texture that this texture uses - * - * @member {Rectangle} - * @private - */ - this._frame = frame; - - /** - * The texture trim data. - * - * @member {Rectangle} - */ - this.trim = trim; - - /** - * This will let the renderer know if the texture is valid. If it's not then it cannot be rendered. - * - * @member {boolean} - */ - this.valid = false; - - /** - * This will let a renderer know that a texture has been updated (used mainly for webGL uv updates) - * - * @member {boolean} - */ - this.requiresUpdate = false; - - /** - * The WebGL UV data cache. - * - * @member {object} - * @private - */ - this._uvs = null; - - /** - * The width of the Texture in pixels. - * - * @member {number} - */ - this.width = 0; - - /** - * The height of the Texture in pixels. - * - * @member {number} - */ - this.height = 0; - - /** - * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering, - * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases) - * - * @member {Rectangle} - */ - this.crop = crop || new math.Rectangle(0, 0, 1, 1); - - if (baseTexture.hasLoaded) { - if (this.noFrame) { - frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height); - } - this.frame = frame; - } - else { - baseTexture.addEventListener('loaded', this.onBaseTextureLoaded.bind(this)); - } -} - -Texture.prototype.constructor = Texture; -module.exports = Texture; - -eventTarget.mixin(Texture.prototype); - -Object.defineProperties(Texture.prototype, { - needsUpdate: { - get: function () { - return this.baseTexture.needsUpdate; - }, - set: function (val) { - this.baseTexture.needsUpdate = val; - } - }, - - frame: { - get: function () { - return this._frame; - }, - set: function (frame) { - this._frame = frame; - - this.noFrame = false; - - this.width = frame.width; - this.height = frame.height; - - this.crop.x = frame.x; - this.crop.y = frame.y; - this.crop.width = frame.width; - this.crop.height = frame.height; - - if (!this.trim && (frame.x + frame.width > this.baseTexture.width || frame.y + frame.height > this.baseTexture.height)) { - throw new Error('Texture Error: frame does not fit inside the base Texture dimensions ' + this); - } - - this.valid = frame && frame.width && frame.height && this.baseTexture.source && this.baseTexture.hasLoaded; - - if (this.trim) { - this.width = this.trim.width; - this.height = this.trim.height; - this._frame.width = this.trim.width; - this._frame.height = this.trim.height; - } - - if (this.valid) { - this._updateUvs(); - } - } - } -}); - -/** - * Called when the base texture is loaded - * - * @private - */ -Texture.prototype.onBaseTextureLoaded = function () { - var baseTexture = this.baseTexture; - baseTexture.removeEventListener('loaded', this.onLoaded); - - if (this.noFrame) { - this.frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height); - } - - this.dispatchEvent( { type: 'update', content: this } ); -}; - -/** - * Destroys this texture - * - * @param destroyBase {boolean} Whether to destroy the base texture as well - */ -Texture.prototype.destroy = function (destroyBase) { - if (destroyBase) { - this.baseTexture.destroy(); - } - - this.valid = false; -}; - -/** - * Updates the internal WebGL UV cache. - * - * @private - */ -Texture.prototype._updateUvs = function () { - if (!this._uvs) { - this._uvs = new TextureUvs(); - } - - var frame = this.crop; - var tw = this.baseTexture.width; - var th = this.baseTexture.height; - - this._uvs.x0 = frame.x / tw; - this._uvs.y0 = frame.y / th; - - this._uvs.x1 = (frame.x + frame.width) / tw; - this._uvs.y1 = frame.y / th; - - this._uvs.x2 = (frame.x + frame.width) / tw; - this._uvs.y2 = (frame.y + frame.height) / th; - - this._uvs.x3 = frame.x / tw; - this._uvs.y3 = (frame.y + frame.height) / th; -}; - -/** - * Helper function that creates a Texture object from the given image url. - * If the image is not in the texture cache it will be created and loaded. - * - * @static - * @param imageUrl {string} The image url of the texture - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return Texture - */ -Texture.fromImage = function (imageUrl, crossorigin, scaleMode) { - var texture = utils.TextureCache[imageUrl]; - - if (!texture) { - texture = new Texture(BaseTexture.fromImage(imageUrl, crossorigin, scaleMode)); - utils.TextureCache[imageUrl] = texture; - } - - return texture; -}; - -/** - * Helper function that returns a Texture objected based on the given frame id. - * If the frame id is not in the texture cache an error will be thrown. - * - * @static - * @param frameId {string} The frame id of the texture - * @return Texture - */ -Texture.fromFrame = function (frameId) { - var texture = utils.TextureCache[frameId]; - if (!texture) { - throw new Error('The frameId "' + frameId + '" does not exist in the texture cache '); - } - return texture; -}; - -/** - * Helper function that creates a new Texture based on the given canvas element. - * - * @static - * @param canvas {Canvas} The canvas element source of the texture - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return {Texture} - */ -Texture.fromCanvas = function (canvas, scaleMode) { - return new Texture(BaseTexture.fromCanvas(canvas, scaleMode)); -}; - -/** - * Helper function that creates a new Texture based on the given video element. - * - * @static - * @param video {HTMLVideoElement} - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return {Texture} A Texture - */ -Texture.fromVideo = function (video, scaleMode) { - return new Texture(VideoBaseTexture.baseTextureFromVideo(video, scaleMode)); -}; - -/** - * Adds a texture to the global utils.TextureCache. This cache is shared across the whole PIXI object. - * - * @static - * @param texture {Texture} The Texture to add to the cache. - * @param id {string} The id that the texture will be stored against. - */ -Texture.addTextureToCache = function (texture, id) { - utils.TextureCache[id] = texture; -}; - -/** - * Remove a texture from the global utils.TextureCache. - * - * @static - * @param id {string} The id of the texture to be removed - * @return {Texture} The texture that was removed - */ -Texture.removeTextureFromCache = function (id) { - var texture = utils.TextureCache[id]; - - delete utils.TextureCache[id]; - delete utils.BaseTextureCache[id]; - - return texture; -}; - -Texture.emptyTexture = new Texture(new BaseTexture()); - -},{"../math":12,"../utils":50,"../utils/eventTarget":49,"./BaseTexture":42,"./TextureUvs":45,"./VideoBaseTexture":46}],45:[function(require,module,exports){ -function TextureUvs() { - this.x0 = 0; - this.y0 = 0; - - this.x1 = 0; - this.y1 = 0; - - this.x2 = 0; - this.y2 = 0; - - this.x3 = 0; - this.y3 = 0; -} - -module.exports = TextureUvs; - -},{}],46:[function(require,module,exports){ -var BaseTexture = require('./BaseTexture'), - utils = require('../utils'); - -/** - * A texture of a [playing] Video. - * - * See the ["deus" demo](http://www.goodboydigital.com/pixijs/examples/deus/). - * - * @class - * @extends BaseTexture - * @namespace PIXI - * @param source {HTMLVideoElement} - * @param [scaleMode] {number} See {@link scaleModes} for possible values - */ -function VideoBaseTexture(source, scaleMode) { - if (!source){ - throw new Error('No video source element specified.'); - } - - // hook in here to check if video is already available. - // BaseTexture looks for a source.complete boolean, plus width & height. - - if ((source.readyState === source.HAVE_ENOUGH_DATA || source.readyState === source.HAVE_FUTURE_DATA) && source.width && source.height) { - source.complete = true; - } - - BaseTexture.call(this, source, scaleMode); - - this.autoUpdate = false; - - this._boundOnUpdate = this._onUpdate.bind(this); - this._boundOnCanPlay = this._onCanPlay.bind(this); - - if (!source.complete) { - source.addEventListener('canplay', this._boundOnCanPlay); - source.addEventListener('canplaythrough', this._boundOnCanPlay); - - // started playing.. - source.addEventListener('play', this._onPlayStart.bind(this)); - source.addEventListener('pause', this._onPlayStop.bind(this)); - } - - this.__loaded = false; -} - -VideoBaseTexture.prototype = Object.create(BaseTexture.prototype); -VideoBaseTexture.prototype.constructor = VideoBaseTexture; -module.exports = VideoBaseTexture; - -VideoBaseTexture.prototype._onUpdate = function () { - if (this.autoUpdate) { - window.requestAnimationFrame(this._boundOnUpdate); - this.needsUpdate = true; - } -}; - -VideoBaseTexture.prototype._onPlayStart = function () { - if (!this.autoUpdate) { - window.requestAnimationFrame(this._boundOnUpdate); - this.autoUpdate = true; - } -}; - -VideoBaseTexture.prototype._onPlayStop = function () { - this.autoUpdate = false; -}; - -VideoBaseTexture.prototype._onCanPlay = function () { - this.hasLoaded = true; - - if (this.source) { - this.source.removeEventListener('canplay', this._boundOnCanPlay); - this.source.removeEventListener('canplaythrough', this._boundOnCanPlay); - - this.width = this.source.videoWidth; - this.height = this.source.videoHeight; - - this.source.play(); - - // prevent multiple loaded dispatches.. - if(!this.__loaded){ - this.__loaded = true; - this.emit('loaded', this); - } - } -}; - -VideoBaseTexture.prototype.destroy = function () { - if (this.source && this.source._pixiId) { - utils.BaseTextureCache[ this.source._pixiId ] = null; - delete utils.BaseTextureCache[ this.source._pixiId ]; - - this.source._pixiId = null; - delete this.source._pixiId; - } - - BaseTexture.prototype.destroy.call(this); -}; - -/** - * Mimic Pixi BaseTexture.from.... method. - * - * @static - * @param video {HTMLVideoElement} - * @param scaleMode {number} See {@link scaleModes} for possible values - * @return {VideoBaseTexture} - */ -VideoBaseTexture.fromVideo = function (video, scaleMode) { - if (!video._pixiId) { - video._pixiId = 'video_' + utils.uuid(); - } - - var baseTexture = utils.BaseTextureCache[video._pixiId]; - - if (!baseTexture) { - baseTexture = new VideoBaseTexture(video, scaleMode); - utils.BaseTextureCache[ video._pixiId ] = baseTexture; - } - - return baseTexture; -}; - -/** - * Mimic Pixi BaseTexture.from.... method. - * - * This can be used in a couple ways, such as: - * - * ```js - * var texture = PIXI.VideoBaseTexture.fromUrl('http://mydomain.com/video.mp4'); - * - * var texture = PIXI.VideoBaseTexture.fromUrl({ src: 'http://mydomain.com/video.mp4', mime: 'video/mp4' }); - * - * var texture = PIXI.VideoBaseTexture.fromUrls(['/video.webm', '/video.mp4']); - * - * var texture = PIXI.VideoBaseTexture.fromUrls([ - * { src: '/video.webm', mime: 'video/webm' }, - * { src: '/video.mp4', mime: 'video/mp4' } - * ]); - * ``` - * - * @alias fromUrls - * @static - * @param videoSrc {string|object|string[]|object[]} The URL(s) for the video. - * @param [videoSrc.src] {string} One of the source urls for the video - * @param [videoSrc.mime] {string} The mimetype of the video (e.g. 'video/mp4'). If not specified - * the url's extension will be used as the second part of the mime type. - * @param scaleMode {number} See {@link scaleModes} for possible values - * @return {VideoBaseTexture} - */ -VideoBaseTexture.fromUrl = function (videoSrc, scaleMode) { - var video = document.createElement('video'); - - // array of objects or strings - if (Array.isArray(videoSrc)) { - for (var i = 0; i < videoSrc.length; ++i) { - video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime)); - } - } - // single object or string - else { - video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime)); - } - - video.load(); - video.play(); - - return VideoBaseTexture.textureFromVideo(video, scaleMode); -}; - -VideoBaseTexture.fromUrls = VideoBaseTexture.fromUrl; - -function createSource(path, type) { - if (!type) { - type = 'video/' + path.substr(path.lastIndexOf('.') + 1); - } - - var source = document.createElement('source'); - - source.src = path; - source.type = type; - - return source; -} - -},{"../utils":50,"./BaseTexture":42}],47:[function(require,module,exports){ -/** - * Creates an homogenous object for tracking events so users can know what to expect. - * - * @class - * @namespace PIXI - * @param target {object} The target object that the event is called on - * @param name {string} The string name of the event that was triggered - * @param data {object} Arbitrary event data to pass along - */ -function EventData(target, name, data) { - // for duck typing in the ".on()" function - this.__isEventObject = true; - - /** - * Tracks the state of bubbling propagation. Do not - * set this directly, instead use `event.stopPropagation()` - * - * @member {boolean} - * @private - * @readonly - */ - this.stopped = false; - - /** - * Tracks the state of sibling listener propagation. Do not - * set this directly, instead use `event.stopImmediatePropagation()` - * - * @member {boolean} - * @private - * @readonly - */ - this.stoppedImmediate = false; - - /** - * The original target the event triggered on. - * - * @member {object} - * @readonly - */ - this.target = target; - - /** - * The string name of the event that this represents. - * - * @member {string} - * @readonly - */ - this.type = name; - - /** - * The data that was passed in with this event. - * - * @member {object} - * @readonly - */ - this.data = data; - - /** - * The timestamp when the event occurred. - * - * @member {number} - * @readonly - */ - this.timeStamp = Date.now(); -} - -EventData.prototype.constructor = EventData; -module.exports = EventData; - -/** - * Stops the propagation of events up the scene graph (prevents bubbling). - * - */ -EventData.prototype.stopPropagation = function stopPropagation() { - this.stopped = true; -}; - -/** - * Stops the propagation of events to sibling listeners (no longer calls any listeners). - * - */ -EventData.prototype.stopImmediatePropagation = function stopImmediatePropagation() { - this.stoppedImmediate = true; -}; - -},{}],48:[function(require,module,exports){ -//TODO: Have Graphics use https://github.com/mattdesl/shape2d -// and https://github.com/mattdesl/shape2d-triangulate instead of custom code. - -/* - PolyK library - url: http://polyk.ivank.net - Released under MIT licence. - - Copyright (c) 2012 Ivan Kuckir - - Permission is hereby granted, free of charge, to any person - obtaining a copy of this software and associated documentation - files (the "Software"), to deal in the Software without - restriction, including without limitation the rights to use, - copy, modify, merge, publish, distribute, sublicense, and/or sell - copies of the Software, and to permit persons to whom the - Software is furnished to do so, subject to the following - conditions: - - The above copyright notice and this permission notice shall be - included in all copies or substantial portions of the Software. - - THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES - OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT - HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, - WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING - FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR - OTHER DEALINGS IN THE SOFTWARE. - - This is an amazing lib! - - Slightly modified by Mat Groves (matgroves.com); -*/ - -/** - * Based on the Polyk library http://polyk.ivank.net released under MIT licence. - * This is an amazing lib! - * Slightly modified by Mat Groves (matgroves.com); - * - * @namespace PIXI - */ -var PolyK = module.exports = {}; - -/** - * Triangulates shapes for webGL graphic fills. - * - */ -PolyK.Triangulate = function (p) { - var sign = true; - - var n = p.length >> 1; - if (n < 3) return []; - - var tgs = []; - var avl = []; - for (var i = 0; i < n; i++) avl.push(i); - - i = 0; - var al = n; - while(al > 3) { - var i0 = avl[(i+0)%al]; - var i1 = avl[(i+1)%al]; - var i2 = avl[(i+2)%al]; - - var ax = p[2*i0], ay = p[2*i0+1]; - var bx = p[2*i1], by = p[2*i1+1]; - var cx = p[2*i2], cy = p[2*i2+1]; - - var earFound = false; - if (PolyK._convex(ax, ay, bx, by, cx, cy, sign)) { - earFound = true; - for (var j = 0; j < al; j++) { - var vi = avl[j]; - if (vi === i0 || vi === i1 || vi === i2) continue; - - if (PolyK._PointInTriangle(p[2*vi], p[2*vi+1], ax, ay, bx, by, cx, cy)) { - earFound = false; - break; - } - } - } - - if (earFound) { - tgs.push(i0, i1, i2); - avl.splice((i+1)%al, 1); - al--; - i = 0; - } - else if (i++ > 3*al) { - // need to flip flip reverse it! - // reset! - if (sign) { - tgs = []; - avl = []; - for (i = 0; i < n; i++) avl.push(i); - - i = 0; - al = n; - - sign = false; - } - else { - // window.console.log("PIXI Warning: shape too complex to fill"); - return null; - } - } - } - - tgs.push(avl[0], avl[1], avl[2]); - return tgs; -}; - -/** - * Checks whether a point is within a triangle - * - * @param px {number} x coordinate of the point to test - * @param py {number} y coordinate of the point to test - * @param ax {number} x coordinate of the a point of the triangle - * @param ay {number} y coordinate of the a point of the triangle - * @param bx {number} x coordinate of the b point of the triangle - * @param by {number} y coordinate of the b point of the triangle - * @param cx {number} x coordinate of the c point of the triangle - * @param cy {number} y coordinate of the c point of the triangle - * @private - * @return {boolean} - */ -PolyK._PointInTriangle = function (px, py, ax, ay, bx, by, cx, cy) { - var v0x = cx-ax; - var v0y = cy-ay; - var v1x = bx-ax; - var v1y = by-ay; - var v2x = px-ax; - var v2y = py-ay; - - var dot00 = v0x*v0x+v0y*v0y; - var dot01 = v0x*v1x+v0y*v1y; - var dot02 = v0x*v2x+v0y*v2y; - var dot11 = v1x*v1x+v1y*v1y; - var dot12 = v1x*v2x+v1y*v2y; - - var invDenom = 1 / (dot00 * dot11 - dot01 * dot01); - var u = (dot11 * dot02 - dot01 * dot12) * invDenom; - var v = (dot00 * dot12 - dot01 * dot02) * invDenom; - - // Check if point is in triangle - return (u >= 0) && (v >= 0) && (u + v < 1); -}; - -/** - * Checks whether a shape is convex - * - * @private - * @return {boolean} - */ -PolyK._convex = function (ax, ay, bx, by, cx, cy, sign) { - return ((ay-by)*(cx-bx) + (bx-ax)*(cy-by) >= 0) === sign; -}; - -},{}],49:[function(require,module,exports){ -var EventData = require('./EventData'); - -/** - * Mixins event emitter functionality to an object. - * - * @mixin - * @namespace PIXI - * @example - * function MyEmitter() {} - * - * eventTarget.mixin(MyEmitter.prototype); - * - * var em = new MyEmitter(); - * em.emit('eventName', 'some data', 'some more data', {}, null, ...); - */ -function eventTarget(obj) { - /** - * Return a list of assigned event listeners. - * - * @param eventName {string} The events that should be listed. - * @return {Array} An array of listener functions - */ - obj.listeners = function listeners(eventName) { - this._listeners = this._listeners || {}; - - return this._listeners[eventName] ? this._listeners[eventName].slice() : []; - }; - - /** - * Emit an event to all registered event listeners. - * - * @alias dispatchEvent - * @param eventName {string} The name of the event. - * @return {boolean} Indication if we've emitted an event. - */ - obj.emit = obj.dispatchEvent = function emit(eventName, data) { - this._listeners = this._listeners || {}; - - // fast return when there are no listeners - if (!this._listeners[eventName]) { - return; - } - - //backwards compat with old method ".emit({ type: 'something' })" - if (typeof eventName === 'object') { - data = eventName; - eventName = eventName.type; - } - - //ensure we are using a real pixi event - if (!data || data.__isEventObject !== true) { - data = new EventData(this, eventName, data); - } - - //iterate the listeners - var listeners = this._listeners[eventName].slice(0), - length = listeners.length, - fn = listeners[0], - i; - - for (i = 0; i < length; fn = listeners[++i]) { - //call the event listener - fn.call(this, data); - - //if "stopImmediatePropagation" is called, stop calling sibling events - if (data.stoppedImmediate) { - return this; - } - } - - //if "stopPropagation" is called then don't bubble the event - if (data.stopped) { - return this; - } - - //bubble this event up the scene graph - if (this.parent && this.parent.emit) { - this.parent.emit.call(this.parent, eventName, data); - } - - return this; - }; - - /** - * Register a new EventListener for the given event. - * - * @alias addEventListener - * @param eventName {string} Name of the event. - * @param callback {Functon} fn Callback function. - */ - obj.on = obj.addEventListener = function on(eventName, fn) { - this._listeners = this._listeners || {}; - - (this._listeners[eventName] = this._listeners[eventName] || []) - .push(fn); - - return this; - }; - - /** - * Add an EventListener that's only called once. - * - * @param eventName {string} Name of the event. - * @param callback {Function} Callback function. - */ - obj.once = function once(eventName, fn) { - this._listeners = this._listeners || {}; - - var self = this; - function onceHandlerWrapper() { - fn.apply(self.off(eventName, onceHandlerWrapper), arguments); - } - onceHandlerWrapper._originalHandler = fn; - - return this.on(eventName, onceHandlerWrapper); - }; - - /** - * Remove event listeners. - * - * @alias removeEventListener - * @param eventName {string} The event we want to remove. - * @param callback {Function} The listener that we need to find. - */ - obj.off = obj.removeEventListener = function off(eventName, fn) { - this._listeners = this._listeners || {}; - - if (!this._listeners[eventName]) { - return this; - } - - var list = this._listeners[eventName], - i = fn ? list.length : 0; - - while(i-- > 0) { - if (list[i] === fn || list[i]._originalHandler === fn) { - list.splice(i, 1); - } - } - - if (list.length === 0) { - delete this._listeners[eventName]; - } - - return this; - }; - - /** - * Remove all listeners or only the listeners for the specified event. - * - * @param eventName {string} The event you want to remove all listeners for. - */ - obj.removeAllListeners = function removeAllListeners(eventName) { - this._listeners = this._listeners || {}; - - if (!this._listeners[eventName]) { - return this; - } - - delete this._listeners[eventName]; - - return this; - }; -} - -module.exports = { - /** - * Mixes in the properties of the eventTarget prototype onto another object - * - * @param object {object} The obj to mix into - */ - mixin: function mixin(obj) { - eventTarget(obj); - } -}; - -},{"./EventData":47}],50:[function(require,module,exports){ -var CONST = require('../const'); - -/** - * @namespace PIXI - */ -var utils = module.exports = { - _uid: 0, - _saidHello: false, - - PolyK: require('./PolyK'), - EventData: require('./EventData'), - eventTarget: require('./eventTarget'), - - /** - * Gets the next uuid - * - * @return {number} The next uuid to use. - */ - uuid: function () { - return ++utils._uid; - }, - - /** - * Converts a hex color number to an [R, G, B] array - * - * @param hex {number} - * @return {number[]} An array representing the [R, G, B] of the color. - */ - hex2rgb: function (hex, out) { - out = out || []; - - out[0] = (hex >> 16 & 0xFF) / 255; - out[1] = (hex >> 8 & 0xFF) / 255; - out[2] = (hex & 0xFF) / 255; - - return out; - }, - - /** - * Converts a hex color number to a string. - * - * @param hex {number} - * @return {string} The string color. - */ - hex2string: function (hex) { - hex = hex.toString(16); - hex = '000000'.substr(0, 6 - hex.length) + hex; - - return '#' + hex; - }, - - /** - * Converts a color as an [R, G, B] array to a hex number - * - * @param rgb {number[]} - * @return {number} The color number - */ - rgb2hex: function (rgb) { - return ((rgb[0]*255 << 16) + (rgb[1]*255 << 8) + rgb[2]*255); - }, - - /** - * Checks whether the Canvas BlendModes are supported by the current browser - * - * @return {boolean} whether they are supported - */ - canUseNewCanvasBlendModes: function () { - if (typeof document === 'undefined') { - return false; - } - - var canvas = document.createElement('canvas'), - context = canvas.getContext('2d'); - - canvas.width = 1; - canvas.height = 1; - - context.fillStyle = '#000'; - context.fillRect(0, 0, 1, 1); - - context.globalCompositeOperation = 'multiply'; - - context.fillStyle = '#fff'; - context.fillRect(0, 0, 1, 1); - - return context.getImageData(0,0,1,1).data[0] === 0; - }, - - /** - * Given a number, this function returns the closest number that is a power of two - * this function is taken from Starling Framework as its pretty neat ;) - * - * @param number {number} - * @return {number} the closest number that is a power of two - */ - getNextPowerOfTwo: function (number) { - // see: http://en.wikipedia.org/wiki/Power_of_two#Fast_algorithm_to_check_if_a_positive_number_is_a_power_of_two - if (number > 0 && (number & (number - 1)) === 0) { - return number; - } - else { - var result = 1; - - while (result < number) { - result <<= 1; - } - - return result; - } - }, - - /** - * checks if the given width and height make a power of two rectangle - * - * @param width {number} - * @param height {number} - * @return {boolean} - */ - isPowerOfTwo: function (width, height) { - return (width > 0 && (width & (width - 1)) === 0 && height > 0 && (height & (height - 1)) === 0); - }, - - /** - * Logs out the version and renderer information for this running instance of PIXI. - * If you don't want to see this message you can set `PIXI.utils._saidHello = true;` - * so the library thinks it already said it. Keep in mind that doing that will forever - * makes you a jerk face. - * - * @param {string} type - The string renderer type to log. - * @constant - * @static - */ - sayHello: function (type) { - if (utils._saidHello) { - return; - } - - if (navigator.userAgent.toLowerCase().indexOf('chrome') > -1) { - var args = [ - '%c %c %c Pixi.js ' + CONST.VERSION + ' - ' + type + ' %c ' + ' %c ' + ' http://www.pixijs.com/ %c %c ♥%c♥%c♥ ', - 'background: #ff66a5', - 'background: #ff66a5', - 'color: #ff66a5; background: #030307;', - 'background: #ff66a5', - 'background: #ffc3dc', - 'background: #ff66a5', - 'color: #ff2424; background: #fff', - 'color: #ff2424; background: #fff', - 'color: #ff2424; background: #fff' - ]; - - console.log.apply(console, args); //jshint ignore:line - } - else if (window.console) { - console.log('Pixi.js ' + CONST.VERSION + ' - ' + type + ' - http://www.pixijs.com/'); //jshint ignore:line - } - - utils._saidHello = true; - }, - - /** - * A wrapper for ajax requests to be handled cross browser - * - * TODO: Replace this wil superagent - * - * @class - * @namespace PIXI - */ - AjaxRequest: function () { - var activexmodes = ['Msxml2.XMLHTTP.6.0', 'Msxml2.XMLHTTP.3.0', 'Microsoft.XMLHTTP']; //activeX versions to check for in IE - - if (window.ActiveXObject) { //Test for support for ActiveXObject in IE first (as XMLHttpRequest in IE7 is broken) - for (var i=0; i= this.textures.length) { - this.gotoAndStop(this.textures.length - 1); - - if (this.onComplete) { - this.onComplete(); - } - } -}; - -/** - * A short hand way of creating a movieclip from an array of frame ids - * - * @static - * @param frames {string[]} the array of frames ids the movieclip will use as its texture frames - */ -MovieClip.fromFrames = function (frames) { - var textures = []; - - for (var i = 0; i < frames.length; ++i) { - textures.push(new core.Texture.fromFrame(frames[i])); - } - - return new MovieClip(textures); -}; - -/** - * A short hand way of creating a movieclip from an array of image ids - * - * @static - * @param images {string[]} the array of image urls the movieclip will use as its texture frames - */ -MovieClip.fromImages = function (images) { - var textures = []; - - for (var i = 0; i < images.length; ++i) { - textures.push(new core.Texture.fromImage(images[i])); - } - - return new MovieClip(textures); -}; - -},{"../core":9}],52:[function(require,module,exports){ -var Strip = require('./Strip'); - -/** - * - * @class - * @namespace PIXI - * @extends Strip - * @param {Texture} texture - The texture to use on the rope. - * @param {Array} points - An array of {Point}. - * - */ -function Rope(texture, points) { - Strip.call(this, texture); - this.points = points; - - this.vertices = new Float32Array(points.length * 4); - this.uvs = new Float32Array(points.length * 4); - this.colors = new Float32Array(points.length * 2); - this.indices = new Uint16Array(points.length * 2); - - this.refresh(); -} - - -// constructor -Rope.prototype = Object.create(Strip.prototype); -Rope.prototype.constructor = Rope; -module.exports = Rope; - -/** - * Refreshes - * - */ -Rope.prototype.refresh = function () { - var points = this.points; - - if (points.length < 1) { - return; - } - - var uvs = this.uvs; - - var indices = this.indices; - var colors = this.colors; - - // this.count -= 0.2; - - uvs[0] = 0; - uvs[1] = 0; - uvs[2] = 0; - uvs[3] = 1; - - colors[0] = 1; - colors[1] = 1; - - indices[0] = 0; - indices[1] = 1; - - var total = points.length, - point, index, amount; - - for (var i = 1; i < total; i++) { - point = points[i]; - index = i * 4; - // time to do some smart drawing! - amount = i / (total-1); - - if (i%2) { - uvs[index] = amount; - uvs[index+1] = 0; - - uvs[index+2] = amount; - uvs[index+3] = 1; - } - else { - uvs[index] = amount; - uvs[index+1] = 0; - - uvs[index+2] = amount; - uvs[index+3] = 1; - } - - index = i * 2; - colors[index] = 1; - colors[index+1] = 1; - - index = i * 2; - indices[index] = index; - indices[index + 1] = index + 1; - } -}; - -/* - * Updates the object transform for rendering - * - * @private - */ -Rope.prototype.updateTransform = function () { - var points = this.points; - - if (points.length < 1) { - return; - } - - var lastPoint = points[0]; - var nextPoint; - var perpX = 0; - var perpY = 0; - - // this.count -= 0.2; - - var vertices = this.vertices; - var total = points.length, - point, index, ratio, perpLength, num; - - for (var i = 0; i < total; i++) { - point = points[i]; - index = i * 4; - - if (i < points.length-1) { - nextPoint = points[i+1]; - } - else { - nextPoint = point; - } - - perpY = -(nextPoint.x - lastPoint.x); - perpX = nextPoint.y - lastPoint.y; - - ratio = (1 - (i / (total-1))) * 10; - - if (ratio > 1) { - ratio = 1; - } - - perpLength = Math.sqrt(perpX * perpX + perpY * perpY); - num = this.texture.height / 2; //(20 + Math.abs(Math.sin((i + this.count) * 0.3) * 50) )* ratio; - perpX /= perpLength; - perpY /= perpLength; - - perpX *= num; - perpY *= num; - - vertices[index] = point.x + perpX; - vertices[index+1] = point.y + perpY; - vertices[index+2] = point.x - perpX; - vertices[index+3] = point.y - perpY; - - lastPoint = point; - } - - this.displayObjectContainerUpdateTransform(); -}; - -/** - * Sets the texture that the Rope will use - * - * @param texture {Texture} the texture that will be used - */ -Rope.prototype.setTexture = function (texture) { - // stop current texture - this.texture = texture; - //this.updateFrame = true; -}; - -},{"./Strip":53}],53:[function(require,module,exports){ -var core = require('../core'); - -/** - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - * @param texture {Texture} The texture to use - * @param width {number} the width - * @param height {number} the height - * - */ -function Strip(texture) { - core.DisplayObjectContainer.call(this); - - /** - * The texture of the strip - * - * @member {Texture} - */ - this.texture = texture; - - // set up the main bits.. - this.uvs = new Float32Array([0, 1, - 1, 1, - 1, 0, - 0, 1]); - - this.vertices = new Float32Array([0, 0, - 100, 0, - 100, 100, - 0, 100]); - - this.colors = new Float32Array([1, 1, 1, 1]); - - this.indices = new Uint16Array([0, 1, 2, 3]); - - /** - * Whether the strip is dirty or not - * - * @member {boolean} - */ - this.dirty = true; - - /** - * The blend mode to be applied to the sprite. Set to blendModes.NORMAL to remove any blend mode. - * - * @member {number} - * @default CONST.blendModes.NORMAL; - */ - this.blendMode = core.CONST.blendModes.NORMAL; - - /** - * Triangles in canvas mode are automatically antialiased, use this value to force triangles to overlap a bit with each other. - * - * @member {number} - */ - this.canvasPadding = 0; - - this.drawMode = Strip.DrawModes.TRIANGLE_STRIP; -} - -// constructor -Strip.prototype = Object.create(core.DisplayObjectContainer.prototype); -Strip.prototype.constructor = Strip; -module.exports = Strip; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} - */ -Strip.prototype.renderWebGL = function (renderer) { - // if the sprite is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0) { - return; - } - - // render triangle strip.. - - renderer.spriteBatch.stop(); - - // init! init! - if (!this._vertexBuffer) { - this._initWebGL(renderer); - } - - renderer.shaderManager.setShader(renderer.shaderManager.stripShader); - - this._renderStrip(renderer); - - ///renderer.shaderManager.activateDefaultShader(); - - renderer.spriteBatch.start(); - - //TODO check culling -}; - -Strip.prototype._initWebGL = function (renderer) { - // build the strip! - var gl = renderer.gl; - - this._vertexBuffer = gl.createBuffer(); - this._indexBuffer = gl.createBuffer(); - this._uvBuffer = gl.createBuffer(); - this._colorBuffer = gl.createBuffer(); - - gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.uvs, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this._colorBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.colors, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); -}; - -Strip.prototype._renderStrip = function (renderer) { - var gl = renderer.gl; - var projection = renderer.projection, - offset = renderer.offset, - shader = renderer.shaderManager.stripShader; - - var drawMode = this.drawMode === Strip.DrawModes.TRIANGLE_STRIP ? gl.TRIANGLE_STRIP : gl.TRIANGLES; - - // gl.uniformMatrix4fv(shaderProgram.mvMatrixUniform, false, mat4Real); - - renderer.blendModeManager.setBlendMode(this.blendMode); - - - // set uniforms - gl.uniformMatrix3fv(shader.translationMatrix, false, this.worldTransform.toArray(true)); - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - gl.uniform1f(shader.alpha, this.worldAlpha); - - if (!this.dirty) { - - gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer); - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices); - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0); - - // update the uvs - gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer); - gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0); - - gl.activeTexture(gl.TEXTURE0); - - // check if a texture is dirty.. - if (this.texture.baseTexture._dirty[gl.id]) { - renderer.updateTexture(this.texture.baseTexture); - } - else { - // bind the current texture - gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]); - } - - // dont need to upload! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer); - - - } - else { - - this.dirty = false; - gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.STATIC_DRAW); - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0); - - // update the uvs - gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.uvs, gl.STATIC_DRAW); - gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0); - - gl.activeTexture(gl.TEXTURE0); - - // check if a texture is dirty.. - if (this.texture.baseTexture._dirty[gl.id]) { - renderer.updateTexture(this.texture.baseTexture); - } - else { - gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]); - } - - // dont need to upload! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); - - } - //console.log(gl.TRIANGLE_STRIP) - // - // - gl.drawElements(drawMode, this.indices.length, gl.UNSIGNED_SHORT, 0); - - -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} - */ -Strip.prototype.renderCanvas = function (renderer) { - var context = renderer.context; - - var transform = this.worldTransform; - - if (renderer.roundPixels) { - context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx | 0, transform.ty | 0); - } - else { - context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx, transform.ty); - } - - if (this.drawMode === Strip.DrawModes.TRIANGLE_STRIP) { - this._renderCanvasTriangleStrip(context); - } - else { - this._renderCanvasTriangles(context); - } -}; - -Strip.prototype._renderCanvasTriangleStrip = function (context) { - // draw triangles!! - var vertices = this.vertices; - var uvs = this.uvs; - - var length = vertices.length / 2; - // this.count++; - - for (var i = 0; i < length - 2; i++) { - // draw some triangles! - var index = i * 2; - this._renderCanvasDrawTriangle(context, vertices, uvs, index, (index + 2), (index + 4)); - } -}; - -Strip.prototype._renderCanvasTriangles = function (context) { - // draw triangles!! - var vertices = this.vertices; - var uvs = this.uvs; - var indices = this.indices; - - var length = indices.length; - // this.count++; - - for (var i = 0; i < length; i += 3) { - // draw some triangles! - var index0 = indices[i] * 2, index1 = indices[i + 1] * 2, index2 = indices[i + 2] * 2; - this._renderCanvasDrawTriangle(context, vertices, uvs, index0, index1, index2); - } -}; - -Strip.prototype._renderCanvasDrawTriangle = function (context, vertices, uvs, index0, index1, index2) { - var textureSource = this.texture.baseTexture.source; - var textureWidth = this.texture.width; - var textureHeight = this.texture.height; - - var x0 = vertices[index0], x1 = vertices[index1], x2 = vertices[index2]; - var y0 = vertices[index0 + 1], y1 = vertices[index1 + 1], y2 = vertices[index2 + 1]; - - var u0 = uvs[index0] * textureWidth, u1 = uvs[index1] * textureWidth, u2 = uvs[index2] * textureWidth; - var v0 = uvs[index0 + 1] * textureHeight, v1 = uvs[index1 + 1] * textureHeight, v2 = uvs[index2 + 1] * textureHeight; - - if (this.canvasPadding > 0) { - var paddingX = this.canvasPadding / this.worldTransform.a; - var paddingY = this.canvasPadding / this.worldTransform.d; - var centerX = (x0 + x1 + x2) / 3; - var centerY = (y0 + y1 + y2) / 3; - - var normX = x0 - centerX; - var normY = y0 - centerY; - - var dist = Math.sqrt(normX * normX + normY * normY); - x0 = centerX + (normX / dist) * (dist + paddingX); - y0 = centerY + (normY / dist) * (dist + paddingY); - - // - - normX = x1 - centerX; - normY = y1 - centerY; - - dist = Math.sqrt(normX * normX + normY * normY); - x1 = centerX + (normX / dist) * (dist + paddingX); - y1 = centerY + (normY / dist) * (dist + paddingY); - - normX = x2 - centerX; - normY = y2 - centerY; - - dist = Math.sqrt(normX * normX + normY * normY); - x2 = centerX + (normX / dist) * (dist + paddingX); - y2 = centerY + (normY / dist) * (dist + paddingY); - } - - context.save(); - context.beginPath(); - - - context.moveTo(x0, y0); - context.lineTo(x1, y1); - context.lineTo(x2, y2); - - context.closePath(); - - context.clip(); - - // Compute matrix transform - var delta = (u0 * v1) + (v0 * u2) + (u1 * v2) - (v1 * u2) - (v0 * u1) - (u0 * v2); - var deltaA = (x0 * v1) + (v0 * x2) + (x1 * v2) - (v1 * x2) - (v0 * x1) - (x0 * v2); - var deltaB = (u0 * x1) + (x0 * u2) + (u1 * x2) - (x1 * u2) - (x0 * u1) - (u0 * x2); - var deltaC = (u0 * v1 * x2) + (v0 * x1 * u2) + (x0 * u1 * v2) - (x0 * v1 * u2) - (v0 * u1 * x2) - (u0 * x1 * v2); - var deltaD = (y0 * v1) + (v0 * y2) + (y1 * v2) - (v1 * y2) - (v0 * y1) - (y0 * v2); - var deltaE = (u0 * y1) + (y0 * u2) + (u1 * y2) - (y1 * u2) - (y0 * u1) - (u0 * y2); - var deltaF = (u0 * v1 * y2) + (v0 * y1 * u2) + (y0 * u1 * v2) - (y0 * v1 * u2) - (v0 * u1 * y2) - (u0 * y1 * v2); - - context.transform(deltaA / delta, deltaD / delta, - deltaB / delta, deltaE / delta, - deltaC / delta, deltaF / delta); - - context.drawImage(textureSource, 0, 0); - context.restore(); -}; - - - -/** - * Renders a flat strip - * - * @param strip {Strip} The Strip to render - * @private - */ -Strip.prototype.renderStripFlat = function (strip) { - var context = this.context; - var vertices = strip.vertices; - - var length = vertices.length/2; - // this.count++; - - context.beginPath(); - for (var i=1; i < length-2; i++) { - // draw some triangles! - var index = i*2; - - var x0 = vertices[index], x1 = vertices[index+2], x2 = vertices[index+4]; - var y0 = vertices[index+1], y1 = vertices[index+3], y2 = vertices[index+5]; - - context.moveTo(x0, y0); - context.lineTo(x1, y1); - context.lineTo(x2, y2); - } - - context.fillStyle = '#FF0000'; - context.fill(); - context.closePath(); -}; - -/* -Strip.prototype.setTexture = function (texture) { - //TODO SET THE TEXTURES - //TODO VISIBILITY - - // stop current texture - this.texture = texture; - this.width = texture.frame.width; - this.height = texture.frame.height; - this.updateFrame = true; -}; - */ - -/** - * When the texture is updated, this event will fire to update the scale and frame - * - * @param event - * @private - */ - -Strip.prototype.onTextureUpdate = function () { - this.updateFrame = true; -}; - -/** - * Returns the bounds of the mesh as a rectangle. The bounds calculation takes the worldTransform into account. - * - * @param matrix {Matrix} the transformation matrix of the sprite - * @return {Rectangle} the framing rectangle - */ -Strip.prototype.getBounds = function (matrix) { - var worldTransform = matrix || this.worldTransform; - - var a = worldTransform.a; - var b = worldTransform.b; - var c = worldTransform.c; - var d = worldTransform.d; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var maxX = -Infinity; - var maxY = -Infinity; - - var minX = Infinity; - var minY = Infinity; - - var vertices = this.vertices; - for (var i = 0, n = vertices.length; i < n; i += 2) { - var rawX = vertices[i], rawY = vertices[i + 1]; - var x = (a * rawX) + (c * rawY) + tx; - var y = (d * rawY) + (b * rawX) + ty; - - minX = x < minX ? x : minX; - minY = y < minY ? y : minY; - - maxX = x > maxX ? x : maxX; - maxY = y > maxY ? y : maxY; - } - - if (minX === -Infinity || maxY === Infinity) { - return core.math.Rectangle.EMPTY; - } - - var bounds = this._bounds; - - bounds.x = minX; - bounds.width = maxX - minX; - - bounds.y = minY; - bounds.height = maxY - minY; - - // store a reference so that if this function gets called again in the render cycle we do not have to recalculate - this._currentBounds = bounds; - - return bounds; -}; - -/** - * Different drawing buffer modes supported - * - * @static - * @constant - * @property {object} DrawModes - * @property {number} DrawModes.TRIANGLE_STRIP - * @property {number} DrawModes.TRIANGLES - */ -Strip.DrawModes = { - TRIANGLE_STRIP: 0, - TRIANGLES: 1 -}; - -},{"../core":9}],54:[function(require,module,exports){ -var core = require('../core'); - -/** - * A tiling sprite is a fast way of rendering a tiling image - * - * @class - * @extends Sprite - * @namespace PIXI - * @param texture {Texture} the texture of the tiling sprite - * @param width {number} the width of the tiling sprite - * @param height {number} the height of the tiling sprite - */ -function TilingSprite(texture, width, height) { - core.Sprite.call( this, texture); - - /** - * The with of the tiling sprite - * - * @member {number} - */ - this._width = width || 100; - - /** - * The height of the tiling sprite - * - * @member {number} - */ - this._height = height || 100; - - /** - * The scaling of the image that is being tiled - * - * @member {Point} - */ - this.tileScale = new core.math.Point(1,1); - - /** - * A point that represents the scale of the texture object - * - * @member {Point} - */ - this.tileScaleOffset = new core.math.Point(1,1); - - /** - * The offset position of the image that is being tiled - * - * @member {Point} - */ - this.tilePosition = new core.math.Point(0,0); - - /** - * Whether this sprite is renderable or not - * - * @member {boolean} - * @default true - */ - this.renderable = true; - - /** - * The tint applied to the sprite. This is a hex value - * - * @member {number} - * @default 0xFFFFFF - */ - this.tint = 0xFFFFFF; - - /** - * The blend mode to be applied to the sprite - * - * @member {number} - * @default blendModes.NORMAL; - */ - this.blendMode = core.CONST.blendModes.NORMAL; -} - -TilingSprite.prototype = Object.create(core.Sprite.prototype); -TilingSprite.prototype.constructor = TilingSprite; -module.exports = TilingSprite; - - -Object.defineProperties(TilingSprite.prototype, { - /** - * The width of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof TilingSprite# - */ - width: { - get: function () { - return this._width; - }, - set: function (value) { - this._width = value; - } - }, - - /** - * The height of the TilingSprite, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof TilingSprite# - */ - height: { - get: function () { - return this._height; - }, - set: function (value) { - this._height = value; - } - }, - - texture: { - get: function () { - return this._texture; - }, - set: function (value) { - if (this._texture === value) { - return; - } - - this._texture = value; - this.refreshTexture = true; - this.cachedTint = 0xFFFFFF; - } - } -}); - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} - */ -TilingSprite.prototype.renderWebGL = function (renderer) { - if (!this.visible || this.alpha <= 0) { - return; - } - - var i, j; - - if (this._mask) { - renderer.spriteBatch.stop(); - renderer.maskManager.pushMask(this.mask, renderer); - renderer.spriteBatch.start(); - } - - if (this._filters) { - renderer.spriteBatch.flush(); - renderer.filterManager.pushFilter(this._filterBlock); - } - - - - if (!this.tilingTexture || this.refreshTexture) { - this.generateTilingTexture(true); - - if (this.tilingTexture && this.tilingTexture.needsUpdate) { - //TODO - tweaking - renderer.updateTexture(this.tilingTexture.baseTexture); - this.tilingTexture.needsUpdate = false; - // this.tilingTexture._uvs = null; - } - } - else { - renderer.spriteBatch.renderTilingSprite(this); - } - // simple render children! - for (i=0,j=this.children.length; i maxX ? x2 : maxX; - maxX = x3 > maxX ? x3 : maxX; - maxX = x4 > maxX ? x4 : maxX; - - maxY = y1; - maxY = y2 > maxY ? y2 : maxY; - maxY = y3 > maxY ? y3 : maxY; - maxY = y4 > maxY ? y4 : maxY; - - var bounds = this._bounds; - - bounds.x = minX; - bounds.width = maxX - minX; - - bounds.y = minY; - bounds.height = maxY - minY; - - // store a reference so that if this function gets called again in the render cycle we do not have to recalculate - this._currentBounds = bounds; - - return bounds; -}; - -/** - * When the texture is updated, this event will fire to update the scale and frame - * - * @param event - * @private - */ -TilingSprite.prototype.onTextureUpdate = function () { - // overriding the sprite version of this! -}; - -/** - * - * @param forcePowerOfTwo {boolean} Whether we want to force the texture to be a power of two - */ -TilingSprite.prototype.generateTilingTexture = function (forcePowerOfTwo) { - if (!this.texture.baseTexture.hasLoaded) { - return; - } - - var texture = this.originalTexture || this.texture; - var frame = texture.frame; - var targetWidth, targetHeight; - - // Check that the frame is the same size as the base texture. - var isFrame = frame.width !== texture.baseTexture.width || frame.height !== texture.baseTexture.height; - - var newTextureRequired = false; - - if (!forcePowerOfTwo) { - if (isFrame) { - targetWidth = frame.width; - targetHeight = frame.height; - - newTextureRequired = true; - } - } - else { - targetWidth = core.utils.getNextPowerOfTwo(frame.width); - targetHeight = core.utils.getNextPowerOfTwo(frame.height); - - // If the BaseTexture dimensions don't match the texture frame then we need a new texture anyway because it's part of a texture atlas - if (frame.width !== targetWidth || frame.height !== targetHeight || texture.baseTexture.width !== targetWidth || texture.baseTexture.height || targetHeight) { - newTextureRequired = true; - } - } - - if (newTextureRequired) { - var canvasBuffer; - - if (this.tilingTexture && this.tilingTexture.isTiling) { - canvasBuffer = this.tilingTexture.canvasBuffer; - canvasBuffer.resize(targetWidth, targetHeight); - this.tilingTexture.baseTexture.width = targetWidth; - this.tilingTexture.baseTexture.height = targetHeight; - this.tilingTexture.needsUpdate = true; - } - else { - canvasBuffer = new core.CanvasBuffer(targetWidth, targetHeight); - - this.tilingTexture = core.Texture.fromCanvas(canvasBuffer.canvas); - this.tilingTexture.canvasBuffer = canvasBuffer; - this.tilingTexture.isTiling = true; - } - - canvasBuffer.context.drawImage(texture.baseTexture.source, - texture.crop.x, - texture.crop.y, - texture.crop.width, - texture.crop.height, - 0, - 0, - targetWidth, - targetHeight); - - this.tileScaleOffset.x = frame.width / targetWidth; - this.tileScaleOffset.y = frame.height / targetHeight; - } - else { - // TODO - switching? - if (this.tilingTexture && this.tilingTexture.isTiling) { - // destroy the tiling texture! - // TODO could store this somewhere? - this.tilingTexture.destroy(true); - } - - this.tileScaleOffset.x = 1; - this.tileScaleOffset.y = 1; - this.tilingTexture = texture; - } - - this.refreshTexture = false; - - this.originalTexture = this.texture; - this.texture = this.tilingTexture; - - this.tilingTexture.baseTexture._powerOf2 = true; -}; - -},{"../core":9}],55:[function(require,module,exports){ -/** - * @file Main export of the PIXI extras library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - MovieClip: require('./MovieClip'), - Rope: require('./Rope'), - Strip: require('./Strip'), - TilingSprite: require('./TilingSprite') -}; - -},{"./MovieClip":51,"./Rope":52,"./Strip":53,"./TilingSprite":54}],56:[function(require,module,exports){ -/** - * This is the base class for creating a PIXI filter. Currently only WebGL supports filters. - * If you want to make a custom filter this should be your base class. - * - * @class - * @namespace PIXI - * @param fragmentSrc {string|string[]} The fragment source in an array of strings. - * @param uniforms {object} An object containing the uniforms for this filter. - */ -function AbstractFilter(fragmentSrc, uniforms) { - /** - * An array of passes - some filters contain a few steps this array simply stores the steps in a liniear fashion. - * For example the blur filter has two passes blurX and blurY. - * - * @member {AbstractFilter[]} - * @private - */ - this.passes = [this]; - - /** - * @member {Shader[]} - * @private - */ - this.shaders = []; - - /** - * @member {number} - */ - this.padding = 0; - - /** - * @member {object} - * @private - */ - this.uniforms = uniforms || {}; - - /** - * @member {string[]} - * @private - */ - this.fragmentSrc = typeof fragmentSrc === 'string' ? fragmentSrc.split('') : (fragmentSrc || []); -} - -AbstractFilter.prototype.constructor = AbstractFilter; -module.exports = AbstractFilter; - -/** - * Syncs the uniforms between the class object and the shaders. - * - */ -AbstractFilter.prototype.syncUniforms = function () { - for (var i = 0, j = this.shaders.length; i < j; ++i) { - this.shaders[i].dirty = true; - } -}; - -/* -AbstractFilter.prototype.apply = function (frameBuffer) { - // TODO :) -}; -*/ - -},{}],57:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The AlphaMaskFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object. - * You can use this filter to apply all manor of crazy warping effects - * Currently the r property of the texture is used to offset the x and the g property of the texture is used to offset the y. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment - */ -function AlphaMaskFilter(texture) { - AbstractFilter.call(this); - - texture.baseTexture._powerOf2 = true; - - // set the uniforms - this.uniforms = { - mask: { type: 'sampler2D', value: texture }, - mapDimensions: { type: '2f', value: { x: 1, y: 5112 } }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] }, - offset: { type: '2f', value: { x: 0, y: 0 } } - }; - - if (texture.baseTexture.hasLoaded) { - this.uniforms.mask.value.x = texture.width; - this.uniforms.mask.value.y = texture.height; - } - else { - this.boundLoadedFunction = this.onTextureLoaded.bind(this); - - texture.baseTexture.on('loaded', this.boundLoadedFunction); - } - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D uSampler;', - 'uniform sampler2D mask;', - 'uniform vec2 mapDimensions;', - 'uniform vec4 dimensions;', - 'uniform vec2 offset;', - - 'void main() {', - ' vec2 mapCords = vTextureCoord.xy;', - ' mapCords += (dimensions.zw + offset)/ dimensions.xy ;', - ' mapCords.y *= -1.0;', - ' mapCords.y += 1.0;', - ' mapCords *= dimensions.xy / mapDimensions;', - - ' vec4 original = texture2D(uSampler, vTextureCoord);', - ' float maskAlpha = texture2D(mask, mapCords).r;', - ' original *= maskAlpha;', - //' original.rgb *= maskAlpha;', - ' gl_FragColor = original;', - //' gl_FragColor = gl_FragColor;', - '}' - ]; -} - -AlphaMaskFilter.prototype = Object.create(AbstractFilter.prototype); -AlphaMaskFilter.prototype.constructor = AlphaMaskFilter; -module.exports = AlphaMaskFilter; - -/** - * Sets the map dimensions uniforms when the texture becomes available. - * - */ -AlphaMaskFilter.prototype.onTextureLoaded = function () { - this.uniforms.mapDimensions.value.x = this.uniforms.mask.value.width; - this.uniforms.mapDimensions.value.y = this.uniforms.mask.value.height; - - this.uniforms.mask.value.baseTexture.off('loaded', this.boundLoadedFunction); -}; - -Object.defineProperties(AlphaMaskFilter.prototype, { - /** - * The texture used for the displacement map. Must be power of 2 sized texture. - * - * @member {Texture} - * @memberof AlphaMaskFilter# - */ - map: { - get: function () { - return this.uniforms.mask.value; - }, - set: function (value) { - this.uniforms.mask.value = value; - } - }, - - /** - * The offset used to move the displacement map. - * - * @member {Point} - * @memberof AlphaMaskFilter# - */ - offset: { - get: function() { - return this.uniforms.offset.value; - }, - set: function(value) { - this.uniforms.offset.value = value; - } - } -}); - -},{"./AbstractFilter":56}],58:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original shader : https://www.shadertoy.com/view/lssGDj by @movAX13h - */ - -/** - * An ASCII filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function AsciiFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - dimensions: { type: '4fv', value: new Float32Array([10000, 100, 10, 10]) }, - pixelSize: { type: '1f', value: 8} - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'uniform vec4 dimensions;', - 'uniform float pixelSize;', - 'uniform sampler2D uSampler;', - - 'float character(float n, vec2 p)', - '{', - ' p = floor(p*vec2(4.0, -4.0) + 2.5);', - ' if (clamp(p.x, 0.0, 4.0) == p.x && clamp(p.y, 0.0, 4.0) == p.y)', - ' {', - ' if (int(mod(n/exp2(p.x + 5.0*p.y), 2.0)) == 1) return 1.0;', - ' }', - ' return 0.0;', - '}', - - 'void main()', - '{', - ' vec2 uv = gl_FragCoord.xy;', - ' vec3 col = texture2D(uSampler, floor( uv / pixelSize ) * pixelSize / dimensions.xy).rgb;', - - ' #ifdef HAS_GREENSCREEN', - ' float gray = (col.r + col.b)/2.0;', - ' #else', - ' float gray = (col.r + col.g + col.b)/3.0;', - ' #endif', - - ' float n = 65536.0; // .', - ' if (gray > 0.2) n = 65600.0; // :', - ' if (gray > 0.3) n = 332772.0; // *', - ' if (gray > 0.4) n = 15255086.0; // o', - ' if (gray > 0.5) n = 23385164.0; // &', - ' if (gray > 0.6) n = 15252014.0; // 8', - ' if (gray > 0.7) n = 13199452.0; // @', - ' if (gray > 0.8) n = 11512810.0; // #', - - ' vec2 p = mod( uv / ( pixelSize * 0.5 ), 2.0) - vec2(1.0);', - ' col = col * character(n, p);', - - ' gl_FragColor = vec4(col, 1.0);', - '}' - ]; -} - -AsciiFilter.prototype = Object.create(AbstractFilter.prototype); -AsciiFilter.prototype.constructor = AsciiFilter; -module.exports = AsciiFilter; - -Object.defineProperties(AsciiFilter.prototype, { - /** - * The pixel size used by the filter. - * - * @member {number} - * @memberof AsciiFilter# - */ - size: { - get: function () { - return this.uniforms.pixelSize.value; - }, - set: function (value) { - this.uniforms.pixelSize.value = value; - } - } -}); - -},{"./AbstractFilter":56}],59:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - BlurXFilter = require('./BlurXFilter'), - BlurYFilter = require('./BlurYFilter'); - -/** - * The BlurFilter applies a Gaussian blur to an object. - * The strength of the blur can be set for x- and y-axis separately. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function BlurFilter() { - AbstractFilter.call(this); - - this.blurXFilter = new BlurXFilter(); - this.blurYFilter = new BlurYFilter(); - - this.passes = [this.blurXFilter, this.blurYFilter]; -} - -BlurFilter.prototype = Object.create(AbstractFilter.prototype); -BlurFilter.prototype.constructor = BlurFilter; -module.exports = BlurFilter; - -Object.defineProperties(BlurFilter.prototype, { - /** - * Sets the strength of both the blurX and blurY properties simultaneously - * - * @member {number} - * @memberOf BlurFilter# - * @default 2 - */ - blur: { - get: function () { - return this.blurXFilter.blur; - }, - set: function (value) { - this.blurXFilter.blur = this.blurYFilter.blur = value; - } - }, - - /** - * Sets the strength of the blurX property - * - * @member {number} - * @memberOf BlurFilter# - * @default 2 - */ - blurX: { - get: function () { - return this.blurXFilter.blur; - }, - set: function (value) { - this.blurXFilter.blur = value; - } - }, - - /** - * Sets the strength of the blurY property - * - * @member {number} - * @memberOf BlurFilter# - * @default 2 - */ - blurY: { - get: function () { - return this.blurYFilter.blur; - }, - set: function (value) { - this.blurYFilter.blur = value; - } - } -}); - -},{"./AbstractFilter":56,"./BlurXFilter":60,"./BlurYFilter":61}],60:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - blurFactor = 1 / 7000; - -/** - * The BlurXFilter applies a horizontal Gaussian blur to an object. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function BlurXFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 1 / 512 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float blur;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec4 sum = vec4(0.0);', - - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - 4.0*blur, vTextureCoord.y)) * 0.05;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - 3.0*blur, vTextureCoord.y)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - 2.0*blur, vTextureCoord.y)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - blur, vTextureCoord.y)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + blur, vTextureCoord.y)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + 2.0*blur, vTextureCoord.y)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + 3.0*blur, vTextureCoord.y)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + 4.0*blur, vTextureCoord.y)) * 0.05;', - - ' gl_FragColor = sum;', - '}' - ]; -} - -BlurXFilter.prototype = Object.create(AbstractFilter.prototype); -BlurXFilter.prototype.constructor = BlurXFilter; -module.exports = BlurXFilter; - -Object.defineProperties(BlurXFilter.prototype, { - /** - * Sets the strength of both the blur. - * - * @member {number} - * @memberof BlurXFilter# - * @default 2 - */ - blur: { - get: function () { - return this.uniforms.blur.value / blurFactor; - }, - set: function (value) { - this.uniforms.blur.value = blurFactor * value; - } - } -}); - -},{"./AbstractFilter":56}],61:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - blurFactor = 1 / 7000; - -/** - * The BlurYFilter applies a vertical Gaussian blur to an object. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function BlurYFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 1 / 512 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float blur;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec4 sum = vec4(0.0);', - - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 4.0*blur)) * 0.05;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 3.0*blur)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 2.0*blur)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - blur)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + blur)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 2.0*blur)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 3.0*blur)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 4.0*blur)) * 0.05;', - - ' gl_FragColor = sum;', - '}' - ]; -} - -BlurYFilter.prototype = Object.create(AbstractFilter.prototype); -BlurYFilter.prototype.constructor = BlurYFilter; -module.exports = BlurYFilter; - -Object.defineProperties(BlurYFilter.prototype, { - /** - * Sets the strength of both the blur. - * - * @member {number} - * @memberof BlurYFilter - * @default 2 - */ - blur: { - get: function () { - return this.uniforms.blur.value / blurFactor; - }, - set: function (value) { - //this.padding = value; - this.uniforms.blur.value = blurFactor * value; - } - } -}); - -},{"./AbstractFilter":56}],62:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The ColorMatrixFilter class lets you apply a 4x4 matrix transformation on the RGBA - * color and alpha values of every pixel on your displayObject to produce a result - * with a new set of RGBA color and alpha values. It's pretty powerful! - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function ColorMatrixFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - matrix: { type: 'mat4', value: [1, 0, 0, 0, - 0, 1, 0, 0, - 0, 0, 1, 0, - 0, 0, 0, 1] } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float invert;', - 'uniform mat4 matrix;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord) * matrix;', - '}' - ]; -} - -ColorMatrixFilter.prototype = Object.create(AbstractFilter.prototype); -ColorMatrixFilter.prototype.constructor = ColorMatrixFilter; -module.exports = ColorMatrixFilter; - -Object.defineProperties(ColorMatrixFilter.prototype, { - /** - * Sets the matrix of the color matrix filter - * - * @member {number[]} - * @memberof ColorMatrixFilter# - * @default [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1] - */ - matrix: { - get: function () { - return this.uniforms.matrix.value; - }, - set: function (value) { - this.uniforms.matrix.value = value; - } - } -}); - -},{"./AbstractFilter":56}],63:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This lowers the color depth of your image by the given amount, producing an image with a smaller palette. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function ColorStepFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - step: { type: '1f', value: 5 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - 'uniform sampler2D uSampler;', - 'uniform float step;', - - 'void main(void) {', - ' vec4 color = texture2D(uSampler, vTextureCoord);', - ' color = floor(color * step) / step;', - ' gl_FragColor = color;', - '}' - ]; -} - -ColorStepFilter.prototype = Object.create(AbstractFilter.prototype); -ColorStepFilter.prototype.constructor = ColorStepFilter; -module.exports = ColorStepFilter; - -Object.defineProperties(ColorStepFilter.prototype, { - /** - * The number of steps to reduce the palette by. - * - * @member {number} - * @memberof ColorStepFilter# - */ - step: { - get: function () { - return this.uniforms.step.value; - }, - set: function (value) { - this.uniforms.step.value = value; - } - } -}); - -},{"./AbstractFilter":56}],64:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The ConvolutionFilter class applies a matrix convolution filter effect. - * A convolution combines pixels in the input image with neighboring pixels to produce a new image. - * A wide variety of image effects can be achieved through convolutions, including blurring, edge - * detection, sharpening, embossing, and beveling. The matrix should be specified as a 9 point Array. - * See http://docs.gimp.org/en/plug-in-convmatrix.html for more info. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param matrix {number[]} An array of values used for matrix transformation. Specified as a 9 point Array. - * @param width {number} Width of the object you are transforming - * @param height {number} Height of the object you are transforming - */ -function ConvolutionFilter(matrix, width, height) { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - matrix: { type: '1fv', value: new Float32Array(matrix) }, - texelSizeX: { type: '1f', value: 1 / width }, - texelSizeY: { type: '1f', value: 1 / height } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying mediump vec2 vTextureCoord;', - - 'uniform sampler2D texture;', - 'uniform float texelSizeX;', - 'uniform float texelSizeY;', - 'uniform float matrix[9];', - - 'vec2 px = vec2(texelSizeX, texelSizeY);', - - 'void main(void) {', - ' vec4 c11 = texture2D(texture, vTextureCoord - px);', // top left - ' vec4 c12 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y - px.y));', // top center - ' vec4 c13 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y - px.y));', // top right - - ' vec4 c21 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y) );', // mid left - ' vec4 c22 = texture2D(texture, vTextureCoord);', // mid center - ' vec4 c23 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y) );', // mid right - - ' vec4 c31 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y + px.y) );', // bottom left - ' vec4 c32 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y + px.y) );', // bottom center - ' vec4 c33 = texture2D(texture, vTextureCoord + px );', // bottom right - - ' gl_FragColor = ', - ' c11 * matrix[0] + c12 * matrix[1] + c13 * matrix[2] +', - ' c21 * matrix[3] + c22 * matrix[4] + c23 * matrix[5] +', - ' c31 * matrix[6] + c32 * matrix[7] + c33 * matrix[8];', - ' gl_FragColor.a = c22.a;', - '}' - ]; -} - -ConvolutionFilter.prototype = Object.create(AbstractFilter.prototype); -ConvolutionFilter.prototype.constructor = ConvolutionFilter; -module.exports = ConvolutionFilter; - -Object.defineProperties(ConvolutionFilter.prototype, { - /** - * An array of values used for matrix transformation. Specified as a 9 point Array. - * - * @member {number[]} - * @memberof ConvolutionFilter# - */ - matrix: { - get: function () { - return this.uniforms.matrix.value; - }, - set: function (value) { - this.uniforms.matrix.value = new Float32Array(value); - } - }, - - /** - * Width of the object you are transforming - * - * @member {number} - * @memberof ConvolutionFilter# - */ - width: { - get: function () { - return this.uniforms.texelSizeX.value; - }, - set: function (value) { - this.uniforms.texelSizeX.value = 1/value; - } - }, - - /** - * Height of the object you are transforming - * - * @member {number} - * @memberof ConvolutionFilter# - */ - height: { - get: function () { - return this.uniforms.texelSizeY.value; - }, - set: function (value) { - this.uniforms.texelSizeY.value = 1/value; - } - } -}); - -},{"./AbstractFilter":56}],65:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * A Cross Hatch effect filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function CrossHatchFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 1 / 512 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float blur;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' float lum = length(texture2D(uSampler, vTextureCoord.xy).rgb);', - - ' gl_FragColor = vec4(1.0, 1.0, 1.0, 1.0);', - - ' if (lum < 1.00) {', - ' if (mod(gl_FragCoord.x + gl_FragCoord.y, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - - ' if (lum < 0.75) {', - ' if (mod(gl_FragCoord.x - gl_FragCoord.y, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - - ' if (lum < 0.50) {', - ' if (mod(gl_FragCoord.x + gl_FragCoord.y - 5.0, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - - ' if (lum < 0.3) {', - ' if (mod(gl_FragCoord.x - gl_FragCoord.y - 5.0, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - '}' - ]; -} - -CrossHatchFilter.prototype = Object.create(AbstractFilter.prototype); -CrossHatchFilter.prototype.constructor = CrossHatchFilter; -module.exports = CrossHatchFilter; - -Object.defineProperties(CrossHatchFilter.prototype, { - /** - * Sets the strength of both the blur. - * - * @member {number} - * @memberof CrossHatchFilter# - * @default 2 - */ - blur: { - get: function () { - return this.uniforms.blur.value / (1/7000); - }, - set: function (value) { - //this.padding = value; - this.uniforms.blur.value = (1/7000) * value; - } - } -}); - -},{"./AbstractFilter":56}],66:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The DisplacementFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object. - * You can use this filter to apply all manor of crazy warping effects - * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment - */ -function DisplacementFilter(texture) { - AbstractFilter.call(this); - - texture.baseTexture._powerOf2 = true; - - // set the uniforms - this.uniforms = { - displacementMap: { type: 'sampler2D', value: texture }, - scale: { type: '2f', value: { x: 30, y: 30 } }, - offset: { type: '2f', value: { x: 0, y: 0 } }, - mapDimensions: { type: '2f', value: { x: 1, y: 5112 } }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] } - }; - - if (texture.baseTexture.hasLoaded) { - this.onTextureLoaded(); - } - else { - this.boundLoadedFunction = this.onTextureLoaded.bind(this); - - texture.baseTexture.on('loaded', this.boundLoadedFunction); - } - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D displacementMap;', - 'uniform sampler2D uSampler;', - 'uniform vec2 scale;', - 'uniform vec2 offset;', - 'uniform vec4 dimensions;', - 'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);', - // 'const vec2 textureDimensions = vec2(750.0, 750.0);', - - 'void main(void) {', - ' vec2 mapCords = vTextureCoord.xy;', - ' mapCords += (dimensions.zw + offset)/ dimensions.xy ;', - ' mapCords.y *= -1.0;', - ' mapCords.y += 1.0;', - - ' vec2 matSample = texture2D(displacementMap, mapCords).xy;', - ' matSample -= 0.5;', - ' matSample *= scale;', - ' matSample /= mapDimensions;', - - ' gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x + matSample.x, vTextureCoord.y + matSample.y));', - - //TODO: Is this needed? - ' gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb, 1.0);', - '}' - ]; -} - -DisplacementFilter.prototype = Object.create(AbstractFilter.prototype); -DisplacementFilter.prototype.constructor = DisplacementFilter; -module.exports = DisplacementFilter; - -/** - * Sets the map dimensions uniforms when the texture becomes available. - * - * @private - */ -DisplacementFilter.prototype.onTextureLoaded = function () { - this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width; - this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height; - - this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction); -}; - -Object.defineProperties(DisplacementFilter.prototype, { - /** - * The texture used for the displacement map. Must be power of 2 texture. - * - * @member {Texture} - * @memberof DisplacementFilter# - */ - map: { - get: function () { - return this.uniforms.displacementMap.value; - }, - set: function (value) { - this.uniforms.displacementMap.value = value; - } - }, - - /** - * The multiplier used to scale the displacement result from the map calculation. - * - * @member {Point} - * @memberof DisplacementFilter# - */ - scale: { - get: function () { - return this.uniforms.scale.value; - }, - set: function (value) { - this.uniforms.scale.value = value; - } - }, - - /** - * The offset used to move the displacement map. - * - * @member {Point} - * @memberof DisplacementFilter# - */ - offset: { - get: function () { - return this.uniforms.offset.value; - }, - set: function (value) { - this.uniforms.offset.value = value; - } - } -}); - -},{"./AbstractFilter":56}],67:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Mat Groves http://matgroves.com/ @Doormat23 - * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/fun/dotscreen.js - */ - -/** - * This filter applies a dotscreen effect making display objects appear to be made out of - * black and white halftone dots like an old printer. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function DotScreenFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - scale: { type: '1f', value: 1 }, - angle: { type: '1f', value: 5 }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform vec4 dimensions;', - 'uniform sampler2D uSampler;', - - 'uniform float angle;', - 'uniform float scale;', - - 'float pattern() {', - ' float s = sin(angle), c = cos(angle);', - ' vec2 tex = vTextureCoord * dimensions.xy;', - ' vec2 point = vec2(', - ' c * tex.x - s * tex.y,', - ' s * tex.x + c * tex.y', - ' ) * scale;', - ' return (sin(point.x) * sin(point.y)) * 4.0;', - '}', - - 'void main() {', - ' vec4 color = texture2D(uSampler, vTextureCoord);', - ' float average = (color.r + color.g + color.b) / 3.0;', - ' gl_FragColor = vec4(vec3(average * 10.0 - 5.0 + pattern()), color.a);', - '}' - ]; -} - -DotScreenFilter.prototype = Object.create(AbstractFilter.prototype); -DotScreenFilter.prototype.constructor = DotScreenFilter; -module.exports = DotScreenFilter; - -Object.defineProperties(DotScreenFilter.prototype, { - /** - * The scale of the effect. - * @member {number} - * @memberof DotScreenFilter# - */ - scale: { - get: function () { - return this.uniforms.scale.value; - }, - set: function (value) { - this.uniforms.scale.value = value; - } - }, - - /** - * The radius of the effect. - * @member {number} - * @memberof DotScreenFilter# - */ - angle: { - get: function () { - return this.uniforms.angle.value; - }, - set: function (value) { - this.uniforms.angle.value = value; - } - } -}); - -},{"./AbstractFilter":56}],68:[function(require,module,exports){ -/** - * A target and pass info object for filters. - * - * @class - * @namespace PIXI - */ -function FilterBlock() { - /** - * The visible state of this FilterBlock. - * - * @member {boolean} - */ - this.visible = true; - - /** - * The renderable state of this FilterBlock. - * - * @member {boolean} - */ - this.renderable = true; -} - -FilterBlock.prototype.constructor = FilterBlock; -module.exports = FilterBlock; - -},{}],69:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This greyscales the palette of your Display Objects. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function GrayFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - gray: { type: '1f', value: 1 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D uSampler;', - 'uniform float gray;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord);', - ' gl_FragColor.rgb = mix(gl_FragColor.rgb, vec3(0.2126*gl_FragColor.r + 0.7152*gl_FragColor.g + 0.0722*gl_FragColor.b), gray);', - // ' gl_FragColor = gl_FragColor;', - '}' - ]; -} - -GrayFilter.prototype = Object.create(AbstractFilter.prototype); -GrayFilter.prototype.constructor = GrayFilter; -module.exports = GrayFilter; - -Object.defineProperties(GrayFilter.prototype, { - /** - * The strength of the gray. 1 will make the object black and white, 0 will make the object its normal color. - * - * @member {number} - * @memberof GrayFilter# - */ - gray: { - get: function () { - return this.uniforms.gray.value; - }, - set: function (value) { - this.uniforms.gray.value = value; - } - } -}); - -},{"./AbstractFilter":56}],70:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This inverts your Display Objects colors. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function InvertFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - invert: { type: '1f', value: 1 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float invert;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord);', - ' gl_FragColor.rgb = mix( (vec3(1)-gl_FragColor.rgb) * gl_FragColor.a, gl_FragColor.rgb, 1.0 - invert);', - //' gl_FragColor.rgb = gl_FragColor.rgb * gl_FragColor.a;', - // ' gl_FragColor = gl_FragColor * vColor;', - '}' - ]; -} - -InvertFilter.prototype = Object.create(AbstractFilter.prototype); -InvertFilter.prototype.constructor = InvertFilter; -module.exports = InvertFilter; - -Object.defineProperties(InvertFilter.prototype, { - /** - * The strength of the invert. 1 will fully invert the colors, 0 will make the object its normal color - * - * @member {number} - * @memberof InvertFilter# - */ - invert: { - get: function () { - return this.uniforms.invert.value; - }, - set: function (value) { - this.uniforms.invert.value = value; - } - } -}); - -},{"./AbstractFilter":56}],71:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/adjust/noise.js - */ - -/** - * A Noise effect filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function NoiseFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - noise: { type: '1f', value: 0.5 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float noise;', - 'uniform sampler2D uSampler;', - - 'float rand(vec2 co) {', - ' return fract(sin(dot(co.xy ,vec2(12.9898,78.233))) * 43758.5453);', - '}', - - 'void main() {', - ' vec4 color = texture2D(uSampler, vTextureCoord);', - - ' float diff = (rand(vTextureCoord) - 0.5) * noise;', - ' color.r += diff;', - ' color.g += diff;', - ' color.b += diff;', - - ' gl_FragColor = color;', - '}' - ]; -} - -NoiseFilter.prototype = Object.create(AbstractFilter.prototype); -NoiseFilter.prototype.constructor = NoiseFilter; -module.exports = NoiseFilter; - -Object.defineProperties(NoiseFilter.prototype, { - /** - * The amount of noise to apply. - * - * @member {number} - * @memberof NoiseFilter# - * @default 0.5 - */ - noise: { - get: function () { - return this.uniforms.noise.value; - }, - set: function (value) { - this.dirty = true; - this.uniforms.noise.value = value; - } - } -}); - -},{"./AbstractFilter":56}],72:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The NormalMapFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object. - * You can use this filter to apply all manor of crazy warping effects - * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param texture {Texture} The texture used for the normal map, must be power of 2 texture at the moment - */ -function NormalMapFilter(texture) { - AbstractFilter.call(this); - - texture.baseTexture._powerOf2 = true; - - // set the uniforms - this.uniforms = { - displacementMap: { type: 'sampler2D', value: texture }, - scale: { type: '2f', value: { x: 15, y: 15 } }, - offset: { type: '2f', value: { x: 0, y: 0 } }, - mapDimensions: { type: '2f', value: { x: 1, y: 1 } }, - dimensions: { type: '4f', value: [0, 0, 0, 0] }, - // LightDir: { type: 'f3', value: [0, 1, 0] }, - LightPos: { type: '3f', value: [0, 1, 0] } - }; - - if (texture.baseTexture.hasLoaded) { - this.onTextureLoaded(); - } - else { - this.boundLoadedFunction = this.onTextureLoaded.bind(this); - - texture.baseTexture.on('loaded', this.boundLoadedFunction); - } - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying float vColor;', - - 'uniform sampler2D displacementMap;', - 'uniform sampler2D uSampler;', - - 'uniform vec4 dimensions;', - - 'const vec2 Resolution = vec2(1.0,1.0);', //resolution of screen - 'uniform vec3 LightPos;', //light position, normalized - 'const vec4 LightColor = vec4(1.0, 1.0, 1.0, 1.0);', //light RGBA -- alpha is intensity - 'const vec4 AmbientColor = vec4(1.0, 1.0, 1.0, 0.5);', //ambient RGBA -- alpha is intensity - 'const vec3 Falloff = vec3(0.0, 1.0, 0.2);', //attenuation coefficients - - 'uniform vec3 LightDir;',//' = vec3(1.0, 0.0, 1.0);', - - - 'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);', - - - 'void main(void) {', - ' vec2 mapCords = vTextureCoord.xy;', - - ' vec4 color = texture2D(uSampler, vTextureCoord.st);', - ' vec3 nColor = texture2D(displacementMap, vTextureCoord.st).rgb;', - - - ' mapCords *= vec2(dimensions.x/512.0, dimensions.y/512.0);', - - ' mapCords.y *= -1.0;', - ' mapCords.y += 1.0;', - - //RGBA of our diffuse color - ' vec4 DiffuseColor = texture2D(uSampler, vTextureCoord);', - - //RGB of our normal map - ' vec3 NormalMap = texture2D(displacementMap, mapCords).rgb;', - - //The delta position of light - //'vec3 LightDir = vec3(LightPos.xy - (gl_FragCoord.xy / Resolution.xy), LightPos.z);', - ' vec3 LightDir = vec3(LightPos.xy - (mapCords.xy), LightPos.z);', - //Correct for aspect ratio - // ' LightDir.x *= Resolution.x / Resolution.y;', - - //Determine distance (used for attenuation) BEFORE we normalize our LightDir - ' float D = length(LightDir);', - - //normalize our vectors - ' vec3 N = normalize(NormalMap * 2.0 - 1.0);', - ' vec3 L = normalize(LightDir);', - - //Pre-multiply light color with intensity - //Then perform 'N dot L' to determine our diffuse term - ' vec3 Diffuse = (LightColor.rgb * LightColor.a) * max(dot(N, L), 0.0);', - - //pre-multiply ambient color with intensity - ' vec3 Ambient = AmbientColor.rgb * AmbientColor.a;', - - //calculate attenuation - ' float Attenuation = 1.0 / ( Falloff.x + (Falloff.y*D) + (Falloff.z*D*D) );', - - //the calculation which brings it all together - ' vec3 Intensity = Ambient + Diffuse * Attenuation;', - ' vec3 FinalColor = DiffuseColor.rgb * Intensity;', - ' gl_FragColor = vColor * vec4(FinalColor, DiffuseColor.a);', - // ' gl_FragColor = vec4(1.0, 0.0, 0.0, Attenuation);',//vColor * vec4(FinalColor, DiffuseColor.a);', - - /*// normalise color - ' vec3 normal = normalize(nColor * 2.0 - 1.0);', - - ' vec3 deltaPos = vec3( (light.xy - gl_FragCoord.xy) / resolution.xy, light.z );', - - ' float lambert = clamp(dot(normal, lightDir), 0.0, 1.0);', - - ' float d = sqrt(dot(deltaPos, deltaPos));', - ' float att = 1.0 / ( attenuation.x + (attenuation.y*d) + (attenuation.z*d*d) );', - - ' vec3 result = (ambientColor * ambientIntensity) + (lightColor.rgb * lambert) * att;', - ' result *= color.rgb;', - - ' gl_FragColor = vec4(result, 1.0);',*/ - '}' - ]; -} - -NormalMapFilter.prototype = Object.create(AbstractFilter.prototype); -NormalMapFilter.prototype.constructor = NormalMapFilter; -module.exports = NormalMapFilter; - -/** - * Sets the map dimensions uniforms when the texture becomes available. - * - */ -NormalMapFilter.prototype.onTextureLoaded = function () { - this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width; - this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height; - - this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction); -}; - -Object.defineProperties(NormalMapFilter.prototype, { - /** - * The texture used for the displacement map. Must be power of 2 texture. - * - * @member {Texture} - * @memberof NormalMapFilter# - */ - map: { - get: function () { - return this.uniforms.displacementMap.value; - }, - set: function (value) { - this.uniforms.displacementMap.value = value; - } - }, - - /** - * The multiplier used to scale the displacement result from the map calculation. - * - * @member {Point} - * @memberof NormalMapFilter# - */ - scale: { - get: function () { - return this.uniforms.scale.value; - }, - set: function (value) { - this.uniforms.scale.value = value; - } - }, - - /** - * The offset used to move the displacement map. - * - * @member {Point} - * @memberof NormalMapFilter# - */ - offset: { - get: function () { - return this.uniforms.offset.value; - }, - set: function (value) { - this.uniforms.offset.value = value; - } - } -}); - -},{"./AbstractFilter":56}],73:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This filter applies a pixelate effect making display objects appear 'blocky'. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function PixelateFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - invert: { type: '1f', value: 0 }, - dimensions: { type: '4fv', value: new Float32Array([10000, 100, 10, 10]) }, - pixelSize: { type: '2f', value: { x: 10, y: 10 } } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform vec2 testDim;', - 'uniform vec4 dimensions;', - 'uniform vec2 pixelSize;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec2 coord = vTextureCoord;', - - ' vec2 size = dimensions.xy/pixelSize;', - - ' vec2 color = floor( ( vTextureCoord * size ) ) / size + pixelSize/dimensions.xy * 0.5;', - ' gl_FragColor = texture2D(uSampler, color);', - '}' - ]; -} - -PixelateFilter.prototype = Object.create(AbstractFilter.prototype); -PixelateFilter.prototype.constructor = PixelateFilter; -module.exports = PixelateFilter; - -Object.defineProperties(PixelateFilter.prototype, { - /** - * This a point that describes the size of the blocks. x is the width of the block and y is the height. - * - * @member {Point} - * @memberof PixelateFilter# - */ - size: { - get: function () { - return this.uniforms.pixelSize.value; - }, - set: function (value) { - this.uniforms.pixelSize.value = value; - } - } -}); - -},{"./AbstractFilter":56}],74:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * An RGB Split Filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function RGBSplitFilter() { - AbstractFilter.call(this); - - this.passes = [this]; - - // set the uniforms - this.uniforms = { - red: { type: '2f', value: { x: 20, y: 20 } }, - green: { type: '2f', value: { x: -20, y: 20 } }, - blue: { type: '2f', value: { x: 20, y: -20 } }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform vec2 red;', - 'uniform vec2 green;', - 'uniform vec2 blue;', - 'uniform vec4 dimensions;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor.r = texture2D(uSampler, vTextureCoord + red/dimensions.xy).r;', - ' gl_FragColor.g = texture2D(uSampler, vTextureCoord + green/dimensions.xy).g;', - ' gl_FragColor.b = texture2D(uSampler, vTextureCoord + blue/dimensions.xy).b;', - ' gl_FragColor.a = texture2D(uSampler, vTextureCoord).a;', - '}' - ]; -} - -RGBSplitFilter.prototype = Object.create(AbstractFilter.prototype); -RGBSplitFilter.prototype.constructor = RGBSplitFilter; -module.exports = RGBSplitFilter; - -Object.defineProperties(RGBSplitFilter.prototype, { - /** - * Red channel offset. - * - * @member {Point} - * @memberof RGBSplitFilter# - */ - red: { - get: function () { - return this.uniforms.red.value; - }, - set: function (value) { - this.uniforms.red.value = value; - } - }, - - /** - * Green channel offset. - * - * @member {Point} - * @memberof RGBSplitFilter# - */ - green: { - get: function () { - return this.uniforms.green.value; - }, - set: function (value) { - this.uniforms.green.value = value; - } - }, - - /** - * Blue offset. - * - * @member {Point} - * @memberof RGBSplitFilter# - */ - blue: { - get: function () { - return this.uniforms.blue.value; - }, - set: function (value) { - this.uniforms.blue.value = value; - } - } -}); - -},{"./AbstractFilter":56}],75:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This applies a sepia effect to your Display Objects. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function SepiaFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - sepia: { type: '1f', value: 1 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float sepia;', - 'uniform sampler2D uSampler;', - - 'const mat3 sepiaMatrix = mat3(0.3588, 0.7044, 0.1368, 0.2990, 0.5870, 0.1140, 0.2392, 0.4696, 0.0912);', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord);', - ' gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb * sepiaMatrix, sepia);', - '}' - ]; -} - -SepiaFilter.prototype = Object.create(AbstractFilter.prototype); -SepiaFilter.prototype.constructor = SepiaFilter; -module.exports = SepiaFilter; - -Object.defineProperties(SepiaFilter.prototype, { - /** - * The strength of the sepia. 1 will apply the full sepia effect, 0 will make the object its normal color. - * - * @member {number} - * @memberof SepiaFilter# - */ - sepia: { - get: function () { - return this.uniforms.sepia.value; - }, - set: function (value) { - this.uniforms.sepia.value = value; - } - } -}); - -},{"./AbstractFilter":56}],76:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * A Smart Blur Filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function SmartBlurFilter() { - AbstractFilter.call(this); - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - - 'uniform sampler2D uSampler;', - 'const vec2 delta = vec2(1.0/10.0, 0.0);', - - 'float random(vec3 scale, float seed) {', - ' return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);', - '}', - - - 'void main(void) {', - ' vec4 color = vec4(0.0);', - ' float total = 0.0;', - - ' float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);', - - ' for (float t = -30.0; t <= 30.0; t++) {', - ' float percent = (t + offset - 0.5) / 30.0;', - ' float weight = 1.0 - abs(percent);', - ' vec4 sample = texture2D(uSampler, vTextureCoord + delta * percent);', - ' sample.rgb *= sample.a;', - ' color += sample * weight;', - ' total += weight;', - ' }', - - ' gl_FragColor = color / total;', - ' gl_FragColor.rgb /= gl_FragColor.a + 0.00001;', - '}' - ]; -} - -SmartBlurFilter.prototype = Object.create(AbstractFilter.prototype); -SmartBlurFilter.prototype.constructor = SmartBlurFilter; -module.exports = SmartBlurFilter; - -},{"./AbstractFilter":56}],77:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - TiltShiftXFilter = require('./TiltShiftXFilter'), - TiltShiftYFilter = require('./TiltShiftYFilter'); - -/** - * @author Vico @vicocotea - * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/ - */ - -/** - * A TiltShift Filter. Manages the pass of both a TiltShiftXFilter and TiltShiftYFilter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TiltShiftFilter() { - AbstractFilter.call(this); - - this.tiltShiftXFilter = new TiltShiftXFilter(); - this.tiltShiftYFilter = new TiltShiftYFilter(); - - this.tiltShiftXFilter.updateDelta(); - this.tiltShiftXFilter.updateDelta(); - - this.passes = [this.tiltShiftXFilter, this.tiltShiftYFilter]; -} - -TiltShiftFilter.prototype = Object.create(AbstractFilter.prototype); -TiltShiftFilter.prototype.constructor = TiltShiftFilter; -module.exports = TiltShiftFilter; - -Object.defineProperties(TiltShiftFilter.prototype, { - /** - * The strength of the blur. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - blur: { - get: function () { - return this.tiltShiftXFilter.blur; - }, - set: function (value) { - this.tiltShiftXFilter.blur = this.tiltShiftYFilter.blur = value; - } - }, - - /** - * The strength of the gradient blur. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - gradientBlur: { - get: function () { - return this.tiltShiftXFilter.gradientBlur; - }, - set: function (value) { - this.tiltShiftXFilter.gradientBlur = this.tiltShiftYFilter.gradientBlur = value; - } - }, - - /** - * The Y value to start the effect at. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - start: { - get: function () { - return this.tiltShiftXFilter.start; - }, - set: function (value) { - this.tiltShiftXFilter.start = this.tiltShiftYFilter.start = value; - } - }, - - /** - * The Y value to end the effect at. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - end: { - get: function () { - return this.tiltShiftXFilter.end; - }, - set: function (value) { - this.tiltShiftXFilter.end = this.tiltShiftYFilter.end = value; - } - }, -}); - -},{"./AbstractFilter":56,"./TiltShiftXFilter":78,"./TiltShiftYFilter":79}],78:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/ - */ - -/** - * A TiltShiftXFilter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TiltShiftXFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 100 }, - gradientBlur: { type: '1f', value: 600 }, - start: { type: '2f', value: { x: 0, y: window.screenHeight / 2 } }, - end: { type: '2f', value: { x: 600, y: window.screenHeight / 2 } }, - delta: { type: '2f', value: { x: 30, y: 30 } }, - texSize: { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } } - }; - - this.updateDelta(); - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - - 'uniform sampler2D uSampler;', - 'uniform float blur;', - 'uniform float gradientBlur;', - 'uniform vec2 start;', - 'uniform vec2 end;', - 'uniform vec2 delta;', - 'uniform vec2 texSize;', - - 'float random(vec3 scale, float seed) {', - ' return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);', - '}', - - 'void main(void) {', - ' vec4 color = vec4(0.0);', - ' float total = 0.0;', - - ' float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);', - ' vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));', - ' float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;', - - ' for (float t = -30.0; t <= 30.0; t++) {', - ' float percent = (t + offset - 0.5) / 30.0;', - ' float weight = 1.0 - abs(percent);', - ' vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);', - ' sample.rgb *= sample.a;', - ' color += sample * weight;', - ' total += weight;', - ' }', - - ' gl_FragColor = color / total;', - ' gl_FragColor.rgb /= gl_FragColor.a + 0.00001;', - '}' - ]; -} - -TiltShiftXFilter.prototype = Object.create(AbstractFilter.prototype); -TiltShiftXFilter.prototype.constructor = TiltShiftXFilter; -module.exports = TiltShiftXFilter; - -/** - * Updates the filter delta values. - * - */ -TiltShiftXFilter.prototype.updateDelta = function () { - var dx = this.uniforms.end.value.x - this.uniforms.start.value.x; - var dy = this.uniforms.end.value.y - this.uniforms.start.value.y; - var d = Math.sqrt(dx * dx + dy * dy); - - this.uniforms.delta.value.x = dx / d; - this.uniforms.delta.value.y = dy / d; -}; - - -Object.defineProperties(TiltShiftXFilter.prototype, { - /** - * The strength of the blur. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - blur: { - get: function () { - return this.uniforms.blur.value; - }, - set: function (value) { - this.uniforms.blur.value = value; - } - }, - - /** - * The strength of the gradient blur. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - gradientBlur: { - get: function () { - return this.uniforms.gradientBlur.value; - }, - set: function (value) { - this.uniforms.gradientBlur.value = value; - } - }, - - /** - * The X value to start the effect at. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - start: { - get: function () { - return this.uniforms.start.value; - }, - set: function (value) { - this.uniforms.start.value = value; - this.updateDelta(); - } - }, - - /** - * The X value to end the effect at. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - end: { - get: function () { - return this.uniforms.end.value; - }, - set: function (value) { - this.uniforms.end.value = value; - this.updateDelta(); - } - } -}); - -},{"./AbstractFilter":56}],79:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/ - */ - -/** - * A TiltShiftYFilter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TiltShiftYFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 100 }, - gradientBlur: { type: '1f', value: 600 }, - start: { type: '2f', value: { x: 0, y: window.screenHeight / 2 } }, - end: { type: '2f', value: { x: 600, y: window.screenHeight / 2 } }, - delta: { type: '2f', value: { x: 30, y: 30 } }, - texSize: { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } } - }; - - this.updateDelta(); - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - - 'uniform sampler2D uSampler;', - 'uniform float blur;', - 'uniform float gradientBlur;', - 'uniform vec2 start;', - 'uniform vec2 end;', - 'uniform vec2 delta;', - 'uniform vec2 texSize;', - - 'float random(vec3 scale, float seed) {', - ' return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);', - '}', - - 'void main(void) {', - ' vec4 color = vec4(0.0);', - ' float total = 0.0;', - - ' float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);', - ' vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));', - ' float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;', - - ' for (float t = -30.0; t <= 30.0; t++) {', - ' float percent = (t + offset - 0.5) / 30.0;', - ' float weight = 1.0 - abs(percent);', - ' vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);', - ' sample.rgb *= sample.a;', - ' color += sample * weight;', - ' total += weight;', - ' }', - - ' gl_FragColor = color / total;', - ' gl_FragColor.rgb /= gl_FragColor.a + 0.00001;', - '}' - ]; -} - -TiltShiftYFilter.prototype = Object.create(AbstractFilter.prototype); -TiltShiftYFilter.prototype.constructor = TiltShiftYFilter; -module.exports = TiltShiftYFilter; - -/** - * Updates the filter delta values. - * - */ -TiltShiftYFilter.prototype.updateDelta = function (){ - var dx = this.uniforms.end.value.x - this.uniforms.start.value.x; - var dy = this.uniforms.end.value.y - this.uniforms.start.value.y; - var d = Math.sqrt(dx * dx + dy * dy); - this.uniforms.delta.value.x = -dy / d; - this.uniforms.delta.value.y = dx / d; -}; - -Object.defineProperties(TiltShiftYFilter.prototype, { - /** - * The strength of the blur. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - blur: { - get: function () { - return this.uniforms.blur.value; - }, - set: function (value) { - this.uniforms.blur.value = value; - } - }, - - /** - * The strength of the gradient blur. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - gradientBlur: { - get: function () { - return this.uniforms.gradientBlur.value; - }, - set: function (value) { - this.uniforms.gradientBlur.value = value; - } - }, - - /** - * The Y value to start the effect at. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - start: { - get: function () { - return this.uniforms.start.value; - }, - set: function (value) { - this.uniforms.start.value = value; - this.updateDelta(); - } - }, - - /** - * The Y value to end the effect at. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - end: { - get: function () { - return this.uniforms.end.value; - }, - set: function (value) { - this.uniforms.end.value = value; - this.updateDelta(); - } - } -}); - -},{"./AbstractFilter":56}],80:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This filter applies a twist effect making display objects appear twisted in the given direction. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TwistFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - radius: { type: '1f', value: 0.5}, - angle: { type: '1f', value: 5}, - offset: { type: '2f', value: { x: 0.5, y: 0.5 } } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float radius;', - 'uniform float angle;', - 'uniform vec2 offset;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec2 coord = vTextureCoord - offset;', - ' float distance = length(coord);', - - ' if (distance < radius) {', - ' float ratio = (radius - distance) / radius;', - ' float angleMod = ratio * ratio * angle;', - ' float s = sin(angleMod);', - ' float c = cos(angleMod);', - ' coord = vec2(coord.x * c - coord.y * s, coord.x * s + coord.y * c);', - ' }', - - ' gl_FragColor = texture2D(uSampler, coord+offset);', - '}' - ]; -} - -TwistFilter.prototype = Object.create(AbstractFilter.prototype); -TwistFilter.prototype.constructor = TwistFilter; -module.exports = TwistFilter; - -Object.defineProperties(TwistFilter.prototype, { - /** - * This point describes the the offset of the twist. - * - * @member {Point} - * @memberof TwistFilter# - */ - offset: { - get: function () { - return this.uniforms.offset.value; - }, - set: function (value) { - this.uniforms.offset.value = value; - } - }, - - /** - * This radius of the twist. - * - * @member {number} - * @memberof TwistFilter# - */ - radius: { - get: function () { - return this.uniforms.radius.value; - }, - set: function (value) { - this.uniforms.radius.value = value; - } - }, - - /** - * This angle of the twist. - * - * @member {number} - * @memberof TwistFilter# - */ - angle: { - get: function () { - return this.uniforms.angle.value; - }, - set: function (value) { - this.uniforms.angle.value = value; - } - } -}); - -},{"./AbstractFilter":56}],81:[function(require,module,exports){ -/** - * @file Main export of the PIXI filters library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - AbstractFilter: require('./AbstractFilter'), - AlphaMaskFilter: require('./AlphaMaskFilter'), - AsciiFilter: require('./AsciiFilter'), - BlurFilter: require('./BlurFilter'), - BlurXFilter: require('./BlurXFilter'), - BlurYFilter: require('./BlurYFilter'), - ColorMatrixFilter: require('./ColorMatrixFilter'), - ColorStepFilter: require('./ColorStepFilter'), - ConvolutionFilter: require('./ConvolutionFilter'), - CrossHatchFilter: require('./CrossHatchFilter'), - DisplacementFilter: require('./DisplacementFilter'), - DotScreenFilter: require('./DotScreenFilter'), - FilterBlock: require('./FilterBlock'), - GrayFilter: require('./GrayFilter'), - InvertFilter: require('./InvertFilter'), - NoiseFilter: require('./NoiseFilter'), - NormalMapFilter: require('./NormalMapFilter'), - PixelateFilter: require('./PixelateFilter'), - RGBSplitFilter: require('./RGBSplitFilter'), - SepiaFilter: require('./SepiaFilter'), - SmartBlurFilter: require('./SmartBlurFilter'), - TiltShiftFilter: require('./TiltShiftFilter'), - TiltShiftXFilter: require('./TiltShiftXFilter'), - TiltShiftYFilter: require('./TiltShiftYFilter'), - TwistFilter: require('./TwistFilter') -}; - -},{"./AbstractFilter":56,"./AlphaMaskFilter":57,"./AsciiFilter":58,"./BlurFilter":59,"./BlurXFilter":60,"./BlurYFilter":61,"./ColorMatrixFilter":62,"./ColorStepFilter":63,"./ConvolutionFilter":64,"./CrossHatchFilter":65,"./DisplacementFilter":66,"./DotScreenFilter":67,"./FilterBlock":68,"./GrayFilter":69,"./InvertFilter":70,"./NoiseFilter":71,"./NormalMapFilter":72,"./PixelateFilter":73,"./RGBSplitFilter":74,"./SepiaFilter":75,"./SmartBlurFilter":76,"./TiltShiftFilter":77,"./TiltShiftXFilter":78,"./TiltShiftYFilter":79,"./TwistFilter":80}],82:[function(require,module,exports){ -var core = require('../core'); - -/** - * Holds all information related to an Interaction event - * - * @class - * @namespace PIXI - */ -function InteractionData() { - /** - * This point stores the global coords of where the touch/mouse event happened - * - * @member {Point} - */ - this.global = new core.math.Point(); - - /** - * The target Sprite that was interacted with - * - * @member {Sprite} - */ - this.target = null; - - /** - * When passed to an event handler, this will be the original DOM Event that was captured - * - * @member {Event} - */ - this.originalEvent = null; -} - -InteractionData.prototype.constructor = InteractionData; -module.exports = InteractionData; - -/** - * This will return the local coordinates of the specified displayObject for this InteractionData - * - * @param displayObject {DisplayObject} The DisplayObject that you would like the local coords off - * @param [point] {Point} A Point object in which to store the value, optional (otherwise will create a new point) - * @return {Point} A point containing the coordinates of the InteractionData position relative to the DisplayObject - */ -InteractionData.prototype.getLocalPosition = function (displayObject, point) { - var worldTransform = displayObject.worldTransform; - var global = this.global; - - // do a cheeky transform to get the mouse coords; - var a00 = worldTransform.a, a01 = worldTransform.c, a02 = worldTransform.tx, - a10 = worldTransform.b, a11 = worldTransform.d, a12 = worldTransform.ty, - id = 1 / (a00 * a11 + a01 * -a10); - - point = point || new core.math.Point(); - - point.x = a11 * id * global.x + -a01 * id * global.y + (a12 * a01 - a02 * a11) * id; - point.y = a00 * id * global.y + -a10 * id * global.x + (-a12 * a00 + a02 * a10) * id; - - // set the mouse coords... - return point; -}; - -},{"../core":9}],83:[function(require,module,exports){ -var core = require('../core'), - InteractionData = require('./InteractionData'); - -// TODO: Obviously rewrite this... -var INTERACTION_FREQUENCY = 30; -var AUTO_PREVENT_DEFAULT = true; - -/** - * The interaction manager deals with mouse and touch events. Any DisplayObject can be interactive - * if its interactive parameter is set to true - * This manager also supports multitouch. - * - * @class - * @namespace PIXI - * @param stage {Stage} The stage to handle interactions - */ -function InteractionManager(stage) { - /** - * A reference to the stage - * - * @member {Stage} - */ - this.stage = stage; - - /** - * The mouse data - * - * @member {InteractionData} - */ - this.mouse = new InteractionData(); - - /** - * An object that stores current touches (InteractionData) by id reference - * - * @member {object} - */ - this.touches = {}; - - /** - * @member {Point} - * @private - */ - this.tempPoint = new core.math.Point(); - - /** - * @member {boolean} - * @default - */ - this.mouseoverEnabled = true; - - /** - * Tiny little interactiveData pool ! - * - * @member {Array} - */ - this.pool = []; - - /** - * An array containing all the iterative items from the our interactive tree - * - * @member {Array} - * @private - */ - this.interactiveItems = []; - - /** - * The DOM element to bind to. - * - * @member {HTMLElement} - * @private - */ - this.interactionDOMElement = null; - - /** - * Have events been attached to the dom element? - * - * @member {boolean} - * @private - */ - this.eventsAdded = false; - - //this will make it so that you don't have to call bind all the time - - /** - * @member {Function} - */ - this.onMouseMove = this.onMouseMove.bind( this ); - - /** - * @member {Function} - */ - this.onMouseDown = this.onMouseDown.bind(this); - - /** - * @member {Function} - */ - this.onMouseOut = this.onMouseOut.bind(this); - - /** - * @member {Function} - */ - this.onMouseUp = this.onMouseUp.bind(this); - - /** - * @member {Function} - */ - this.onTouchStart = this.onTouchStart.bind(this); - - /** - * @member {Function} - */ - this.onTouchEnd = this.onTouchEnd.bind(this); - - /** - * @member {Function} - */ - this.onTouchMove = this.onTouchMove.bind(this); - - /** - * @member {number} - */ - this.last = 0; - - /** - * The css style of the cursor that is being used - * @member {string} - */ - this.currentCursorStyle = 'inherit'; - - /** - * Is set to true when the mouse is moved out of the canvas - * @member {boolean} - */ - this.mouseOut = false; - - /** - * @member {number} - */ - this.resolution = 1; - - // used for hit testing - this._tempPoint = new core.math.Point(); -} - -InteractionManager.prototype.constructor = InteractionManager; -module.exports = InteractionManager; - -/** - * Collects an interactive sprite recursively to have their interactions managed - * - * @param displayObject {DisplayObject} the displayObject to collect - * @param iParent {DisplayObject} the display object's parent - * @private - */ -InteractionManager.prototype.collectInteractiveSprite = function (displayObject, iParent) { - var children = displayObject.children; - var length = children.length; - - // make an interaction tree... {item.__interactiveParent} - for (var i = length - 1; i >= 0; i--) { - var child = children[i]; - - // push all interactive bits - if (child._interactive) { - iParent.interactiveChildren = true; - //child.__iParent = iParent; - this.interactiveItems.push(child); - - if (child.children.length > 0) { - this.collectInteractiveSprite(child, child); - } - } - else { - child.__iParent = null; - if (child.children.length > 0) { - this.collectInteractiveSprite(child, iParent); - } - } - - } -}; - -/** - * Sets the DOM element which will receive mouse/touch events. This is useful for when you have - * other DOM elements on top of the renderers Canvas element. With this you'll be bale to deletegate - * another DOM element to receive those events. - * - * @param element {HTMLElement} the DOM element which will receive mouse and touch events. - * @param [resolution=1] {number} THe resolution of the new element (relative to the canvas). - * @private - */ -InteractionManager.prototype.setTargetElement = function (element, resolution) { - this.removeEvents(); - - this.interactionDOMElement = element; - - this.resolution = resolution || 1; - - this.addEvents(); -}; - -/** - * - * @private - */ -InteractionManager.prototype.addEvents = function () { - if (!this.interactionDOMElement) { - return; - } - - if (window.navigator.msPointerEnabled) { - this.interactionDOMElement.style['-ms-content-zooming'] = 'none'; - this.interactionDOMElement.style['-ms-touch-action'] = 'none'; - } - - this.interactionDOMElement.addEventListener('mousemove', this.onMouseMove, true); - this.interactionDOMElement.addEventListener('mousedown', this.onMouseDown, true); - this.interactionDOMElement.addEventListener('mouseout', this.onMouseOut, true); - - this.interactionDOMElement.addEventListener('touchstart', this.onTouchStart, true); - this.interactionDOMElement.addEventListener('touchend', this.onTouchEnd, true); - this.interactionDOMElement.addEventListener('touchmove', this.onTouchMove, true); - - window.addEventListener('mouseup', this.onMouseUp, true); - - this.eventsAdded = true; -}; - -/** - * - * @private - */ -InteractionManager.prototype.removeEvents = function () { - if (!this.interactionDOMElement) { - return; - } - - if (window.navigator.msPointerEnabled) { - this.interactionDOMElement.style['-ms-content-zooming'] = ''; - this.interactionDOMElement.style['-ms-touch-action'] = ''; - } - - this.interactionDOMElement.removeEventListener('mousemove', this.onMouseMove, true); - this.interactionDOMElement.removeEventListener('mousedown', this.onMouseDown, true); - this.interactionDOMElement.removeEventListener('mouseout', this.onMouseOut, true); - - this.interactionDOMElement.removeEventListener('touchstart', this.onTouchStart, true); - this.interactionDOMElement.removeEventListener('touchend', this.onTouchEnd, true); - this.interactionDOMElement.removeEventListener('touchmove', this.onTouchMove, true); - - this.interactionDOMElement = null; - - window.removeEventListener('mouseup', this.onMouseUp, true); - - this.eventsAdded = false; -}; - -/** - * updates the state of interactive objects - * - * @private - */ -InteractionManager.prototype.update = function () { - if (!this.interactionDOMElement) { - return; - } - - // frequency of 30fps?? - var now = Date.now(); - var diff = now - this.last; - diff = (diff * INTERACTION_FREQUENCY ) / 1000; - if (diff < 1) { - return; - } - - this.last = now; - - var i = 0; - - // ok.. so mouse events?? - // yes for now :) - // OPTIMISE - how often to check?? - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - // loop through interactive objects! - var length = this.interactiveItems.length; - var cursor = 'inherit'; - var over = false; - - for (i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - // OPTIMISATION - only calculate every time if the mousemove function exists.. - // OK so.. does the object have any other interactive functions? - // hit-test the clip! - // if (item.mouseover || item.mouseout || item.buttonMode) - // { - // ok so there are some functions so lets hit test it.. - item.__hit = this.hitTest(item, this.mouse); - this.mouse.target = item; - // ok so deal with interactions.. - // looks like there was a hit! - if (item.__hit && !over) { - if (item.buttonMode) { - cursor = item.defaultCursor; - } - - if (!item.interactiveChildren) { - over = true; - } - - if (!item.__isOver) { - if (item.mouseover) { - item.mouseover (this.mouse); - } - item.__isOver = true; - } - } - else { - if (item.__isOver) { - // roll out! - if (item.mouseout) { - item.mouseout (this.mouse); - } - item.__isOver = false; - } - } - } - - if (this.currentCursorStyle !== cursor) { - this.currentCursorStyle = cursor; - this.interactionDOMElement.style.cursor = cursor; - } -}; - -/** - * @private - */ -InteractionManager.prototype.rebuildInteractiveGraph = function () { - this.dirty = false; - - var len = this.interactiveItems.length; - - for (var i = 0; i < len; i++) { - this.interactiveItems[i].interactiveChildren = false; - } - - this.interactiveItems.length = 0; - - if (this.stage.interactive) { - this.interactiveItems.push(this.stage); - } - - // Go through and collect all the objects that are interactive.. - this.collectInteractiveSprite(this.stage, this.stage); -}; - -/** - * Is called when the mouse moves across the renderer element - * - * @param event {Event} The DOM event of the mouse moving - * @private - */ -InteractionManager.prototype.onMouseMove = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - // TODO optimize by not check EVERY TIME! maybe half as often? // - var rect = this.interactionDOMElement.getBoundingClientRect(); - - this.mouse.global.x = (event.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) / this.resolution; - this.mouse.global.y = (event.clientY - rect.top) * ( this.interactionDOMElement.height / rect.height) / this.resolution; - - var length = this.interactiveItems.length; - - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - // Call the function! - if (item.mousemove) { - item.mousemove(this.mouse); - } - } -}; - -/** - * Is called when the mouse button is pressed down on the renderer element - * - * @param event {Event} The DOM event of a mouse button being pressed down - * @private - */ -InteractionManager.prototype.onMouseDown = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - if (AUTO_PREVENT_DEFAULT) { - this.mouse.originalEvent.preventDefault(); - } - - // loop through interaction tree... - // hit test each item! -> - // get interactive items under point?? - //stage.__i - var length = this.interactiveItems.length; - - var e = this.mouse.originalEvent; - var isRightButton = e.button === 2 || e.which === 3; - var downFunction = isRightButton ? 'rightdown' : 'mousedown'; - var clickFunction = isRightButton ? 'rightclick' : 'click'; - var buttonIsDown = isRightButton ? '__rightIsDown' : '__mouseIsDown'; - var isDown = isRightButton ? '__isRightDown' : '__isDown'; - - // while - // hit test - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - if (item[downFunction] || item[clickFunction]) { - item[buttonIsDown] = true; - item.__hit = this.hitTest(item, this.mouse); - - if (item.__hit) { - //call the function! - if (item[downFunction]) { - item[downFunction](this.mouse); - } - item[isDown] = true; - - // just the one! - if (!item.interactiveChildren) { - break; - } - } - } - } -}; - -/** - * Is called when the mouse is moved out of the renderer element - * - * @param event {Event} The DOM event of a mouse being moved out - * @private - */ -InteractionManager.prototype.onMouseOut = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - var length = this.interactiveItems.length; - - this.interactionDOMElement.style.cursor = 'inherit'; - - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - if (item.__isOver) { - this.mouse.target = item; - if (item.mouseout) { - item.mouseout(this.mouse); - } - item.__isOver = false; - } - } - - this.mouseOut = true; - - // move the mouse to an impossible position - this.mouse.global.x = -10000; - this.mouse.global.y = -10000; -}; - -/** - * Is called when the mouse button is released on the renderer element - * - * @param event {Event} The DOM event of a mouse button being released - * @private - */ -InteractionManager.prototype.onMouseUp = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - var length = this.interactiveItems.length; - var up = false; - - var e = this.mouse.originalEvent; - var isRightButton = e.button === 2 || e.which === 3; - - var upFunction = isRightButton ? 'rightup' : 'mouseup'; - var clickFunction = isRightButton ? 'rightclick' : 'click'; - var upOutsideFunction = isRightButton ? 'rightupoutside' : 'mouseupoutside'; - var isDown = isRightButton ? '__isRightDown' : '__isDown'; - - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - if (item[clickFunction] || item[upFunction] || item[upOutsideFunction]) { - item.__hit = this.hitTest(item, this.mouse); - - if (item.__hit && !up) { - //call the function! - if (item[upFunction]) { - item[upFunction](this.mouse); - } - if (item[isDown]) { - if (item[clickFunction]) { - item[clickFunction](this.mouse); - } - } - - if (!item.interactiveChildren) { - up = true; - } - } - else { - if (item[isDown]) { - if (item[upOutsideFunction]) { - item[upOutsideFunction](this.mouse); - } - } - } - - item[isDown] = false; - } - } -}; - -/** - * Tests if the current mouse coordinates hit a sprite - * - * @param item {DisplayObject} The displayObject to test for a hit - * @param interactionData {InteractionData} The interactionData object to update in the case there is a hit - * @private - */ -InteractionManager.prototype.hitTest = function (item, interactionData) { - var global = interactionData.global; - - if (!item.worldVisible) { - return false; - } - - // map the global point to local space. - item.worldTransform.applyInverse(global, this._tempPoint); - - var x = this._tempPoint.x, - y = this._tempPoint.y, - i; - - interactionData.target = item; - - //a sprite or display object with a hit area defined - if (item.hitArea && item.hitArea.contains) { - return item.hitArea.contains(x, y); - } - // a sprite with no hitarea defined - else if (item instanceof core.Sprite) { - var width = item.texture.frame.width; - var height = item.texture.frame.height; - var x1 = -width * item.anchor.x; - var y1; - - if (x > x1 && x < x1 + width) { - y1 = -height * item.anchor.y; - - if (y > y1 && y < y1 + height) { - // set the target property if a hit is true! - return true; - } - } - } - else if (item instanceof core.Graphics) { - var graphicsData = item.graphicsData; - for (i = 0; i < graphicsData.length; i++) { - var data = graphicsData[i]; - - if (!data.fill) { - continue; - } - - // only deal with fills.. - if (data.shape) { - if (data.shape.contains(x, y)) { - //interactionData.target = item; - return true; - } - } - } - } - - var length = item.children.length; - - for (i = 0; i < length; i++) { - var tempItem = item.children[i]; - var hit = this.hitTest(tempItem, interactionData); - if (hit) { - // hmm.. TODO SET CORRECT TARGET? - interactionData.target = item; - return true; - } - } - return false; -}; - -/** - * Is called when a touch is moved across the renderer element - * - * @param event {Event} The DOM event of a touch moving across the renderer view - * @private - */ -InteractionManager.prototype.onTouchMove = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - var rect = this.interactionDOMElement.getBoundingClientRect(); - var changedTouches = event.changedTouches; - var touchData; - var i = 0; - - for (i = 0; i < changedTouches.length; i++) { - var touchEvent = changedTouches[i]; - touchData = this.touches[touchEvent.identifier]; - touchData.originalEvent = event; - - // update the touch position - touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution; - touchData.global.y = ( (touchEvent.clientY - rect.top) * (this.interactionDOMElement.height / rect.height) ) / this.resolution; - if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) { - //Support for CocoonJS fullscreen scale modes - touchData.global.x = touchEvent.clientX; - touchData.global.y = touchEvent.clientY; - } - - for (var j = 0; j < this.interactiveItems.length; j++) { - var item = this.interactiveItems[j]; - if (item.touchmove && item.__touchData && item.__touchData[touchEvent.identifier]) { - item.touchmove(touchData); - } - } - } -}; - -/** - * Is called when a touch is started on the renderer element - * - * @param event {Event} The DOM event of a touch starting on the renderer view - * @private - */ -InteractionManager.prototype.onTouchStart = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - var rect = this.interactionDOMElement.getBoundingClientRect(); - - if (AUTO_PREVENT_DEFAULT) { - event.preventDefault(); - } - - var changedTouches = event.changedTouches; - for (var i=0; i < changedTouches.length; i++) { - var touchEvent = changedTouches[i]; - - var touchData = this.pool.pop(); - if (!touchData) { - touchData = new InteractionData(); - } - - touchData.originalEvent = event; - - this.touches[touchEvent.identifier] = touchData; - touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution; - touchData.global.y = ( (touchEvent.clientY - rect.top) * (this.interactionDOMElement.height / rect.height) ) / this.resolution; - if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) { - //Support for CocoonJS fullscreen scale modes - touchData.global.x = touchEvent.clientX; - touchData.global.y = touchEvent.clientY; - } - - var length = this.interactiveItems.length; - - for (var j = 0; j < length; j++) { - var item = this.interactiveItems[j]; - - if (item.touchstart || item.tap) { - item.__hit = this.hitTest(item, touchData); - - if (item.__hit) { - //call the function! - if (item.touchstart) { - item.touchstart(touchData); - } - - item.__isDown = true; - item.__touchData = item.__touchData || {}; - item.__touchData[touchEvent.identifier] = touchData; - - if (!item.interactiveChildren) { - break; - } - } - } - } - } -}; - -/** - * Is called when a touch is ended on the renderer element - * - * @param event {Event} The DOM event of a touch ending on the renderer view - * @private - */ -InteractionManager.prototype.onTouchEnd = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - var rect = this.interactionDOMElement.getBoundingClientRect(); - var changedTouches = event.changedTouches; - - for (var i=0; i < changedTouches.length; i++) { - var touchEvent = changedTouches[i]; - var touchData = this.touches[touchEvent.identifier]; - var up = false; - touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution; - touchData.global.y = ( (touchEvent.clientY - rect.top) * (this.interactionDOMElement.height / rect.height) ) / this.resolution; - if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) { - //Support for CocoonJS fullscreen scale modes - touchData.global.x = touchEvent.clientX; - touchData.global.y = touchEvent.clientY; - } - - var length = this.interactiveItems.length; - for (var j = 0; j < length; j++) { - var item = this.interactiveItems[j]; - - if (item.__touchData && item.__touchData[touchEvent.identifier]) { - - item.__hit = this.hitTest(item, item.__touchData[touchEvent.identifier]); - - // so this one WAS down... - touchData.originalEvent = event; - // hitTest?? - - if (item.touchend || item.tap) { - if (item.__hit && !up) { - if (item.touchend) { - item.touchend(touchData); - } - if (item.__isDown && item.tap) { - item.tap(touchData); - } - if (!item.interactiveChildren) { - up = true; - } - } - else { - if (item.__isDown && item.touchendoutside) { - item.touchendoutside(touchData); - } - } - - item.__isDown = false; - } - - item.__touchData[touchEvent.identifier] = null; - } - } - // remove the touch.. - this.pool.push(touchData); - this.touches[touchEvent.identifier] = null; - } -}; - -},{"../core":9,"./InteractionData":82}],84:[function(require,module,exports){ -/** - * @file Main export of the PIXI interactions library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - InteractionData: require('./InteractionData'), - InteractionManager: require('./InteractionManager') -}; - -},{"./InteractionData":82,"./InteractionManager":83}],85:[function(require,module,exports){ -var core = require('../core'), - ImageLoader = require('./ImageLoader'); - -/** - * The atlas file loader is used to load in Texture Atlas data and parse it. When loaded this class will dispatch a 'loaded' event. If loading fails this class will dispatch an 'error' event. - * - * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format. - * - * It is highly recommended to use texture atlases (also know as 'sprite sheets') as it allowed sprites to be batched and drawn together for highly increased rendering speed. - * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId() - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function AtlasLoader(url, crossorigin) { - this.url = url; - this.baseUrl = url.replace(/[^\/]*$/, ''); - this.crossorigin = crossorigin; - this.loaded = false; -} - -AtlasLoader.prototype.constructor = AtlasLoader; -module.exports = AtlasLoader; - -core.utils.eventTarget.mixin(AtlasLoader.prototype); - - /** - * Starts loading the JSON file - * - */ -AtlasLoader.prototype.load = function () { - this.ajaxRequest = new core.utils.AjaxRequest(); - this.ajaxRequest.onreadystatechange = this.onAtlasLoaded.bind(this); - - this.ajaxRequest.open('GET', this.url, true); - - if (this.ajaxRequest.overrideMimeType) { - this.ajaxRequest.overrideMimeType('application/json'); - } - - this.ajaxRequest.send(null); -}; - -/** - * Invoked when the Atlas has fully loaded. Parses the JSON and builds the texture frames. - * - * @private - */ -AtlasLoader.prototype.onAtlasLoaded = function () { - if (this.ajaxRequest.readyState === 4) { - if (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1) { - this.atlas = { - meta : { - image : [] - }, - frames : [] - }; - var result = this.ajaxRequest.responseText.split(/\r?\n/); - var lineCount = -3; - - var currentImageId = 0; - var currentFrame = null; - var nameInNextLine = false; - - var i = 0, - j = 0, - selfOnLoaded = this.onLoaded.bind(this); - - // parser without rotation support yet! - for (i = 0; i < result.length; i++) { - result[i] = result[i].replace(/^\s+|\s+$/g, ''); - - if (result[i] === '') { - nameInNextLine = i+1; - } - - if (result[i].length > 0) { - if (nameInNextLine === i) { - this.atlas.meta.image.push(result[i]); - currentImageId = this.atlas.meta.image.length - 1; - this.atlas.frames.push({}); - lineCount = -3; - } else if (lineCount > 0) { - if (lineCount % 7 === 1) { // frame name - if (currentFrame != null) { //jshint ignore:line - this.atlas.frames[currentImageId][currentFrame.name] = currentFrame; - } - currentFrame = { name: result[i], frame : {} }; - } else { - var text = result[i].split(' '); - if (lineCount % 7 === 3) { // position - currentFrame.frame.x = Number(text[1].replace(',', '')); - currentFrame.frame.y = Number(text[2]); - } else if (lineCount % 7 === 4) { // size - currentFrame.frame.w = Number(text[1].replace(',', '')); - currentFrame.frame.h = Number(text[2]); - } else if (lineCount % 7 === 5) { // real size - var realSize = { - x : 0, - y : 0, - w : Number(text[1].replace(',', '')), - h : Number(text[2]) - }; - - if (realSize.w > currentFrame.frame.w || realSize.h > currentFrame.frame.h) { - currentFrame.trimmed = true; - currentFrame.realSize = realSize; - } else { - currentFrame.trimmed = false; - } - } - } - } - lineCount++; - } - } - - if (currentFrame != null) { //jshint ignore:line - this.atlas.frames[currentImageId][currentFrame.name] = currentFrame; - } - - if (this.atlas.meta.image.length > 0) { - this.images = []; - for (j = 0; j < this.atlas.meta.image.length; j++) { - // sprite sheet - var textureUrl = this.baseUrl + this.atlas.meta.image[j]; - var frameData = this.atlas.frames[j]; - this.images.push(new ImageLoader(textureUrl, this.crossorigin)); - - for (i in frameData) { - var rect = frameData[i].frame; - if (rect) { - core.utils.TextureCache[i] = new core.Texture(this.images[j].texture.baseTexture, { - x: rect.x, - y: rect.y, - width: rect.w, - height: rect.h - }); - if (frameData[i].trimmed) { - core.utils.TextureCache[i].realSize = frameData[i].realSize; - // trim in pixi not supported yet, todo update trim properties if it is done ... - core.utils.TextureCache[i].trim.x = 0; - core.utils.TextureCache[i].trim.y = 0; - } - } - } - } - - this.currentImageId = 0; - for (j = 0; j < this.images.length; j++) { - this.images[j].on('loaded', selfOnLoaded); - } - this.images[this.currentImageId].load(); - - } else { - this.onLoaded(); - } - - } else { - this.onError(); - } - } -}; - -/** - * Invoked when json file has loaded. - * - * @private - */ -AtlasLoader.prototype.onLoaded = function () { - if (this.images.length - 1 > this.currentImageId) { - this.currentImageId++; - this.images[this.currentImageId].load(); - } else { - this.loaded = true; - this.emit('loaded', { content: this }); - } -}; - -/** - * Invoked when an error occurs. - * - * @private - */ -AtlasLoader.prototype.onError = function () { - this.emit('error', { content: this }); -}; - -},{"../core":9,"./ImageLoader":87}],86:[function(require,module,exports){ -var core = require('../core'), - ImageLoader = require('./ImageLoader'); - -/** - * The xml loader is used to load in XML bitmap font data ('xml' or 'fnt') - * To generate the data you can use http://www.angelcode.com/products/bmfont/ - * This loader will also load the image file as the data. - * When loaded this class will dispatch a 'loaded' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the sprite sheet JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function BitmapFontLoader(url, crossorigin) { - /** - * The url of the bitmap font data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * The base url of the bitmap font data - * - * @member {String} - * @readOnly - */ - this.baseUrl = url.replace(/[^\/]*$/, ''); - - /** - * The texture of the bitmap font - * - * @member {Texture} - */ - this.texture = null; -} - -// constructor -BitmapFontLoader.prototype.constructor = BitmapFontLoader; -module.exports = BitmapFontLoader; - -core.utils.eventTarget.mixin(BitmapFontLoader.prototype); - -/** - * Loads the XML font data - * - */ -BitmapFontLoader.prototype.load = function () { - this.ajaxRequest = new core.utils.AjaxRequest(); - this.ajaxRequest.onreadystatechange = this.onXMLLoaded.bind(this); - - this.ajaxRequest.open('GET', this.url, true); - - if (this.ajaxRequest.overrideMimeType) { - this.ajaxRequest.overrideMimeType('application/xml'); - } - - this.ajaxRequest.send(null); -}; - -/** - * Invoked when the XML file is loaded, parses the data. - * - * @private - */ -BitmapFontLoader.prototype.onXMLLoaded = function () { - if (this.ajaxRequest.readyState === 4) { - if (this.ajaxRequest.status === 200 || window.location.protocol.indexOf('http') === -1) { - var responseXML = this.ajaxRequest.responseXML; - if (!responseXML || /MSIE 9/i.test(navigator.userAgent) || navigator.isCocoonJS) { - if (typeof(window.DOMParser) === 'function') { - var domparser = new DOMParser(); - responseXML = domparser.parseFromString(this.ajaxRequest.responseText, 'text/xml'); - } else { - var div = document.createElement('div'); - div.innerHTML = this.ajaxRequest.responseText; - responseXML = div; - } - } - - var textureUrl = this.baseUrl + responseXML.getElementsByTagName('page')[0].getAttribute('file'); - var image = new ImageLoader(textureUrl, this.crossorigin); - this.texture = image.texture.baseTexture; - - var data = {}; - var info = responseXML.getElementsByTagName('info')[0]; - var common = responseXML.getElementsByTagName('common')[0]; - data.font = info.getAttribute('face'); - data.size = parseInt(info.getAttribute('size'), 10); - data.lineHeight = parseInt(common.getAttribute('lineHeight'), 10); - data.chars = {}; - - //parse letters - var letters = responseXML.getElementsByTagName('char'); - - for (var i = 0; i < letters.length; i++) { - var charCode = parseInt(letters[i].getAttribute('id'), 10); - - var textureRect = new core.math.Rectangle( - parseInt(letters[i].getAttribute('x'), 10), - parseInt(letters[i].getAttribute('y'), 10), - parseInt(letters[i].getAttribute('width'), 10), - parseInt(letters[i].getAttribute('height'), 10) - ); - - data.chars[charCode] = { - xOffset: parseInt(letters[i].getAttribute('xoffset'), 10), - yOffset: parseInt(letters[i].getAttribute('yoffset'), 10), - xAdvance: parseInt(letters[i].getAttribute('xadvance'), 10), - kerning: {}, - texture: core.utils.TextureCache[charCode] = new core.Texture(this.texture, textureRect) - - }; - } - - //parse kernings - var kernings = responseXML.getElementsByTagName('kerning'); - for (i = 0; i < kernings.length; i++) { - var first = parseInt(kernings[i].getAttribute('first'), 10); - var second = parseInt(kernings[i].getAttribute('second'), 10); - var amount = parseInt(kernings[i].getAttribute('amount'), 10); - - data.chars[second].kerning[first] = amount; - - } - - core.BitmapText.fonts[data.font] = data; - - image.addEventListener('loaded', this.onLoaded.bind(this)); - image.load(); - } - } -}; - -/** - * Invoked when all files are loaded (xml/fnt and texture) - * - * @private - */ -BitmapFontLoader.prototype.onLoaded = function () { - this.emit('loaded', { content: this }); -}; - -},{"../core":9,"./ImageLoader":87}],87:[function(require,module,exports){ -var core = require('../core'); - -/** - * The image loader class is responsible for loading images file formats ('jpeg', 'jpg', 'png' and 'gif') - * Once the image has been loaded it is stored in the PIXI texture cache and can be accessed though Texture.fromFrame() and Sprite.fromFrame() - * When loaded this class will dispatch a 'loaded' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the image - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function ImageLoader(url, crossorigin) { - /** - * The texture being loaded - * - * @member {Texture} - */ - this.texture = core.Texture.fromImage(url, crossorigin); - - /** - * if the image is loaded with loadFramedSpriteSheet - * frames will contain the sprite sheet frames - * - * @member {Array} - * @readOnly - */ - this.frames = []; -} - -// constructor -ImageLoader.prototype.constructor = ImageLoader; -module.exports = ImageLoader; - -core.utils.eventTarget.mixin(ImageLoader.prototype); - -/** - * Loads image or takes it from cache - * - */ -ImageLoader.prototype.load = function () { - if (!this.texture.baseTexture.hasLoaded) { - this.texture.baseTexture.on('loaded', this.onLoaded.bind(this)); - this.texture.baseTexture.on('error', this.onError.bind(this)); - } - else { - this.onLoaded(); - } -}; - -/** - * Invoked when image file is loaded or it is already cached and ready to use - * - * @private - */ -ImageLoader.prototype.onLoaded = function () { - this.emit('loaded', { content: this }); -}; - -/** - * Invoked when image file failed loading - * - * @method onError - * @private - */ -ImageLoader.prototype.onError = function () { - this.emit('error', { content: this }); -}; - -/** - * Loads image and split it to uniform sized frames - * - * @param frameWidth {number} width of each frame - * @param frameHeight {number} height of each frame - * @param textureName {String} if given, the frames will be cached in - format - */ -ImageLoader.prototype.loadFramedSpriteSheet = function (frameWidth, frameHeight, textureName) { - this.frames = []; - - var cols = Math.floor(this.texture.width / frameWidth); - var rows = Math.floor(this.texture.height / frameHeight); - - var i=0; - for (var y = 0; y < rows; ++y) { - for (var x = 0; x < cols; ++x, ++i) { - var texture = new core.Texture( - this.texture.baseTexture, - new core.math.Rectangle( - x * frameWidth, - y * frameHeight, - frameWidth, - frameHeight - ) - ); - - this.frames.push(texture); - - if (textureName) { - core.utils.TextureCache[textureName + '-' + i] = texture; - } - } - } - - this.load(); -}; - -},{"../core":9}],88:[function(require,module,exports){ -var core = require('../core'), - spine = require('../spine/SpineRuntime'), - ImageLoader = require('./ImageLoader'), - SpineTextureLoader = require('./SpineTextureLoader'); - -/** - * The json file loader is used to load in JSON data and parse it - * When loaded this class will dispatch a 'loaded' event - * If loading fails this class will dispatch an 'error' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function JsonLoader(url, crossorigin) { - /** - * The url of the bitmap font data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * The base url of the bitmap font data - * - * @member {String} - * @readOnly - */ - this.baseUrl = url.replace(/[^\/]*$/, ''); - - /** - * Whether the data has loaded yet - * - * @member {boolean} - * @readOnly - */ - this.loaded = false; -} - -// constructor -JsonLoader.prototype.constructor = JsonLoader; -module.exports = JsonLoader; - -core.utils.eventTarget.mixin(JsonLoader.prototype); - -/** - * Loads the JSON data - * - */ -JsonLoader.prototype.load = function () { - if (window.XDomainRequest && this.crossorigin) { - this.ajaxRequest = new window.XDomainRequest(); - - // XDomainRequest has a few quirks. Occasionally it will abort requests - // A way to avoid this is to make sure ALL callbacks are set even if not used - // More info here: http://stackoverflow.com/questions/15786966/xdomainrequest-aborts-post-on-ie-9 - this.ajaxRequest.timeout = 3000; - - this.ajaxRequest.onerror = this.onError.bind(this); - this.ajaxRequest.ontimeout = this.onError.bind(this); - - this.ajaxRequest.onprogress = function () {}; - - this.ajaxRequest.onload = this.onJSONLoaded.bind(this); - } - else { - if (window.XMLHttpRequest) { - this.ajaxRequest = new window.XMLHttpRequest(); - } - else { - this.ajaxRequest = new window.ActiveXObject('Microsoft.XMLHTTP'); - } - - this.ajaxRequest.onreadystatechange = this.onReadyStateChanged.bind(this); - } - - this.ajaxRequest.open('GET',this.url,true); - - this.ajaxRequest.send(); -}; - -/** - * Bridge function to be able to use the more reliable onreadystatechange in XMLHttpRequest. - * - * @private - */ -JsonLoader.prototype.onReadyStateChanged = function () { - if (this.ajaxRequest.readyState === 4 && (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1)) { - this.onJSONLoaded(); - } -}; - -/** - * Invoke when JSON file is loaded - * - * @private - */ -JsonLoader.prototype.onJSONLoaded = function () { - if (!this.ajaxRequest.responseText) { - this.onError(); - return; - } - - this.json = JSON.parse(this.ajaxRequest.responseText); - - if (this.json.frames) { - // sprite sheet - var textureUrl = this.baseUrl + this.json.meta.image; - var image = new ImageLoader(textureUrl, this.crossorigin); - var frameData = this.json.frames; - - this.texture = image.texture.baseTexture; - image.addEventListener('loaded', this.onLoaded.bind(this)); - image.addEventListener('error', this.onError.bind(this)); - - for (var i in frameData) { - var rect = frameData[i].frame; - - if (rect) { - var textureSize = new core.math.Rectangle(rect.x, rect.y, rect.w, rect.h); - var crop = textureSize.clone(); - var trim = null; - - // Check to see if the sprite is trimmed - if (frameData[i].trimmed) { - var actualSize = frameData[i].sourceSize; - var realSize = frameData[i].spriteSourceSize; - trim = new core.math.Rectangle(realSize.x, realSize.y, actualSize.w, actualSize.h); - } - core.utils.TextureCache[i] = new core.Texture(this.texture, textureSize, crop, trim); - } - } - - image.load(); - - } - else if (this.json.bones) { - // check if the json was loaded before - if (core.utils.AnimCache[this.url]) { - this.onLoaded(); - } - else { - /** - * use a bit of hackery to load the atlas file, here we assume that the .json, .atlas and .png files - * that correspond to the spine file are in the same base URL and that the .json and .atlas files - * have the same name - */ - var atlasPath = this.url.substr(0, this.url.lastIndexOf('.')) + '.atlas'; - var atlasLoader = new JsonLoader(atlasPath, this.crossorigin); - // save a copy of the current object for future reference // - var originalLoader = this; - // before loading the file, replace the "onJSONLoaded" function for our own // - atlasLoader.onJSONLoaded = function () { - // at this point "this" points at the atlasLoader (JsonLoader) instance // - if (!this.ajaxRequest.responseText) { - this.onError(); // FIXME: hmm, this is funny because we are not responding to errors yet - return; - } - // create a new instance of a spine texture loader for this spine object // - var textureLoader = new SpineTextureLoader(this.url.substring(0, this.url.lastIndexOf('/'))); - // create a spine atlas using the loaded text and a spine texture loader instance // - var spineAtlas = new spine.Atlas(this.ajaxRequest.responseText, textureLoader); - // now we use an atlas attachment loader // - var attachmentLoader = new spine.AtlasAttachmentLoader(spineAtlas); - // spine animation - var spineJsonParser = new spine.SkeletonJson(attachmentLoader); - var skeletonData = spineJsonParser.readSkeletonData(originalLoader.json); - core.utils.AnimCache[originalLoader.url] = skeletonData; - originalLoader.spine = skeletonData; - originalLoader.spineAtlas = spineAtlas; - originalLoader.spineAtlasLoader = atlasLoader; - // wait for textures to finish loading if needed - if (textureLoader.loadingCount > 0) { - textureLoader.addEventListener('loadedBaseTexture', function (evt){ - if (evt.content.content.loadingCount <= 0) { - originalLoader.onLoaded(); - } - }); - } - else { - originalLoader.onLoaded(); - } - }; - // start the loading // - atlasLoader.load(); - } - } - else { - this.onLoaded(); - } -}; - -/** - * Invoke when json file loaded - * - * @private - */ -JsonLoader.prototype.onLoaded = function () { - this.loaded = true; - this.dispatchEvent({ - type: 'loaded', - content: this - }); -}; - -/** - * Invoke when error occured - * - * @private - */ -JsonLoader.prototype.onError = function () { - - this.dispatchEvent({ - type: 'error', - content: this - }); -}; - -},{"../core":9,"../spine/SpineRuntime":94,"./ImageLoader":87,"./SpineTextureLoader":90}],89:[function(require,module,exports){ -var core = require('../core'), - JsonLoader = require('./JsonLoader'); - -/** - * @author Mat Groves http://matgroves.com/ @Doormat23 - * based on pixi impact spine implementation made by Eemeli Kelokorpi (@ekelokorpi) https://github.com/ekelokorpi - * - * Awesome JS run time provided by EsotericSoftware - * https://github.com/EsotericSoftware/spine-runtimes - * - */ - -/** - * The Spine loader is used to load in JSON spine data - * To generate the data you need to use http://esotericsoftware.com/ and export in the "JSON" format - * Due to a clash of names You will need to change the extension of the spine file from *.json to *.anim for it to load - * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source - * You will need to generate a sprite sheet to accompany the spine data - * When loaded this class will dispatch a "loaded" event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function SpineLoader(url, crossorigin) { - /** - * The url of the bitmap font data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * Whether the data has loaded yet - * - * @member {boolean} - * @readOnly - */ - this.loaded = false; -} - -SpineLoader.prototype.constructor = SpineLoader; -module.exports = SpineLoader; - -core.utils.eventTarget.mixin(SpineLoader.prototype); - -/** - * Loads the JSON data - * - */ -SpineLoader.prototype.load = function () { - var scope = this; - var jsonLoader = new JsonLoader(this.url, this.crossorigin); - - jsonLoader.on('loaded', function (event) { - scope.json = event.data.content.json; - scope.onLoaded(); - }); - - jsonLoader.load(); -}; - -/** - * Invoked when JSON file is loaded. - * - * @private - */ -SpineLoader.prototype.onLoaded = function () { - this.loaded = true; - this.emit('loaded', { content: this }); -}; - -},{"../core":9,"./JsonLoader":88}],90:[function(require,module,exports){ -var core = require('../core'); - -/** - * Supporting class to load images from spine atlases as per spine spec. - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param basePath {string} Tha base path where to look for the images to be loaded - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function SpineTextureLoader(basePath, crossorigin) { - this.basePath = basePath; - this.crossorigin = crossorigin; - this.loadingCount = 0; -} - -SpineTextureLoader.prototype.constructor = SpineTextureLoader; -module.exports = SpineTextureLoader; - -core.utils.eventTarget.mixin(SpineTextureLoader.prototype); - -/** - * Starts loading a base texture as per spine specification - * - * @param page {spine.AtlasPage} Atlas page to which texture belongs - * @param file {string} The file to load, this is just the file path relative to the base path configured in the constructor - */ -SpineTextureLoader.prototype.load = function (page, file) { - page.rendererObject = core.BaseTexture.fromImage(this.basePath + '/' + file, this.crossorigin); - if (!page.rendererObject.hasLoaded) { - var scope = this; - ++scope.loadingCount; - page.rendererObject.addEventListener('loaded', function (){ - --scope.loadingCount; - scope.dispatchEvent({ - type: 'loadedBaseTexture', - content: scope - }); - }); - } -}; - -/** - * Unloads a previously loaded texture as per spine specification - * - * @param texture {BaseTexture} Texture object to destroy - */ -SpineTextureLoader.prototype.unload = function (texture) { - texture.destroy(true); -}; - -},{"../core":9}],91:[function(require,module,exports){ -var core = require('../core'), - JsonLoader = require('./JsonLoader'); - -/** - * The sprite sheet loader is used to load in JSON sprite sheet data - * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format - * There is a free version so thats nice, although the paid version is great value for money. - * It is highly recommended to use Sprite sheets (also know as a 'texture atlas') as it means sprites can be batched and drawn together for highly increased rendering speed. - * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId() - * This loader will load the image file that the Spritesheet points to as well as the data. - * When loaded this class will dispatch a 'loaded' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the sprite sheet JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function SpriteSheetLoader(url, crossorigin) { - - /** - * The url of the atlas data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * The base url of the bitmap font data - * - * @member {String} - * @readOnly - */ - this.baseUrl = url.replace(/[^\/]*$/, ''); - - /** - * The texture being loaded - * - * @member {Texture} - */ - this.texture = null; - - /** - * The frames of the sprite sheet - * - * @member {object} - */ - this.frames = {}; -} - -// constructor -SpriteSheetLoader.prototype.constructor = SpriteSheetLoader; -module.exports = SpriteSheetLoader; - -core.utils.eventTarget.mixin(SpriteSheetLoader.prototype); - -/** - * This will begin loading the JSON file - * - */ -SpriteSheetLoader.prototype.load = function () { - var scope = this; - var jsonLoader = new JsonLoader(this.url, this.crossorigin); - - jsonLoader.on('loaded', function (event) { - scope.json = event.data.content.json; - scope.onLoaded(); - }); - - jsonLoader.load(); -}; - -/** - * Invoke when all files are loaded (json and texture) - * - * @private - */ -SpriteSheetLoader.prototype.onLoaded = function () { - this.emit('loaded', { - content: this - }); -}; - -},{"../core":9,"./JsonLoader":88}],92:[function(require,module,exports){ -/** - * @file Main export of the PIXI loaders library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - AtlasLoader: require('./AtlasLoader'), - BitmapFontLoader: require('./BitmapFontLoader'), - ImageLoader: require('./ImageLoader'), - JsonLoader: require('./JsonLoader'), - SpineLoader: require('./SpineLoader'), - SpriteSheetLoader: require('./SpriteSheetLoader') -}; - -},{"./AtlasLoader":85,"./BitmapFontLoader":86,"./ImageLoader":87,"./JsonLoader":88,"./SpineLoader":89,"./SpriteSheetLoader":91}],93:[function(require,module,exports){ -var core = require('../core'), - spine = require('./SpineRuntime'); - -/* Esoteric Software SPINE wrapper for pixi.js */ - -spine.Bone.yDown = true; - -/** - * A class that enables the you to import and run your spine animations in pixi. - * Spine animation data needs to be loaded using the AssetLoader or SpineLoader before it can be used by this class - * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - * @param url {string} The url of the spine anim file to be used - */ -function Spine(url) { - core.DisplayObjectContainer.call(this); - - this.spineData = core.utils.AnimCache[url]; - - if (!this.spineData) { - throw new Error('Spine data must be preloaded using SpineLoader or AssetLoader: ' + url); - } - - this.skeleton = new spine.Skeleton(this.spineData); - this.skeleton.updateWorldTransform(); - - this.stateData = new spine.AnimationStateData(this.spineData); - this.state = new spine.AnimationState(this.stateData); - - this.slotContainers = []; - - for (var i = 0, n = this.skeleton.drawOrder.length; i < n; i++) { - var slot = this.skeleton.drawOrder[i]; - var attachment = slot.attachment; - var slotContainer = new core.DisplayObjectContainer(); - this.slotContainers.push(slotContainer); - this.addChild(slotContainer); - - if (attachment instanceof spine.RegionAttachment) { - var spriteName = attachment.rendererObject.name; - var sprite = this.createSprite(slot, attachment); - slot.currentSprite = sprite; - slot.currentSpriteName = spriteName; - slotContainer.addChild(sprite); - } - else if (attachment instanceof spine.MeshAttachment) { - var mesh = this.createMesh(slot, attachment); - slot.currentMesh = mesh; - slot.currentMeshName = attachment.name; - slotContainer.addChild(mesh); - } - else { - continue; - } - - } - - this.autoUpdate = true; -} - -Spine.prototype = Object.create(core.DisplayObjectContainer.prototype); -Spine.prototype.constructor = Spine; -module.exports = Spine; - -Object.defineProperties(Spine.prototype, { - /** - * If this flag is set to true, the spine animation will be autoupdated every time - * the object id drawn. The down side of this approach is that the delta time is - * automatically calculated and you could miss out on cool effects like slow motion, - * pause, skip ahead and the sorts. Most of these effects can be achieved even with - * autoupdate enabled but are harder to achieve. - * - * @member {boolean} - * @memberof Spine# - * @default true - */ - autoUpdate: { - get: function () { - return (this.updateTransform === Spine.prototype.autoUpdateTransform); - }, - - set: function (value) { - this.updateTransform = value ? Spine.prototype.autoUpdateTransform : core.DisplayObjectContainer.prototype.updateTransform; - } - } -}); - -/** - * Update the spine skeleton and its animations by delta time (dt) - * - * @param dt {number} Delta time. Time by which the animation should be updated - */ -Spine.prototype.update = function (dt) { - this.state.update(dt); - this.state.apply(this.skeleton); - this.skeleton.updateWorldTransform(); - - var drawOrder = this.skeleton.drawOrder; - for (var i = 0, n = drawOrder.length; i < n; i++) { - var slot = drawOrder[i]; - var attachment = slot.attachment; - var slotContainer = this.slotContainers[i]; - - if (!attachment) { - slotContainer.visible = false; - continue; - } - - var type = attachment.type; - if (type === spine.AttachmentType.region) { - if (attachment.rendererObject) { - if (!slot.currentSpriteName || slot.currentSpriteName !== attachment.rendererObject.name) { - var spriteName = attachment.rendererObject.name; - if (slot.currentSprite !== undefined) { - slot.currentSprite.visible = false; - } - slot.sprites = slot.sprites || {}; - if (slot.sprites[spriteName] !== undefined) { - slot.sprites[spriteName].visible = true; - } - else { - var sprite = this.createSprite(slot, attachment); - slotContainer.addChild(sprite); - } - slot.currentSprite = slot.sprites[spriteName]; - slot.currentSpriteName = spriteName; - } - } - - var bone = slot.bone; - - slotContainer.position.x = bone.worldX + attachment.x * bone.m00 + attachment.y * bone.m01; - slotContainer.position.y = bone.worldY + attachment.x * bone.m10 + attachment.y * bone.m11; - slotContainer.scale.x = bone.worldScaleX; - slotContainer.scale.y = bone.worldScaleY; - - slotContainer.rotation = -(slot.bone.worldRotation * spine.degRad); - - slot.currentSprite.tint = core.utils.rgb2hex([slot.r,slot.g,slot.b]); - } - else if (type === spine.AttachmentType.skinnedmesh) { - if (!slot.currentMeshName || slot.currentMeshName !== attachment.name) { - var meshName = attachment.name; - if (slot.currentMesh !== undefined) { - slot.currentMesh.visible = false; - } - - slot.meshes = slot.meshes || {}; - - if (slot.meshes[meshName] !== undefined) { - slot.meshes[meshName].visible = true; - } - else { - var mesh = this.createMesh(slot, attachment); - slotContainer.addChild(mesh); - } - - slot.currentMesh = slot.meshes[meshName]; - slot.currentMeshName = meshName; - } - - attachment.computeWorldVertices(slot.bone.skeleton.x, slot.bone.skeleton.y, slot, slot.currentMesh.vertices); - - } - else { - slotContainer.visible = false; - continue; - } - slotContainer.visible = true; - - slotContainer.alpha = slot.a; - } -}; - -/** - * When autoupdate is set to yes this function is used as pixi's updateTransform function - * - * @private - */ -Spine.prototype.autoUpdateTransform = function () { - this.lastTime = this.lastTime || Date.now(); - var timeDelta = (Date.now() - this.lastTime) * 0.001; - this.lastTime = Date.now(); - - this.update(timeDelta); - - core.DisplayObjectContainer.prototype.updateTransform.call(this); -}; - -/** - * Create a new sprite to be used with spine.RegionAttachment - * - * @param slot {spine.Slot} The slot to which the attachment is parented - * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent - * @private - */ -Spine.prototype.createSprite = function (slot, attachment) { - var descriptor = attachment.rendererObject; - var baseTexture = descriptor.page.rendererObject; - var spriteRect = new core.math.Rectangle(descriptor.x, - descriptor.y, - descriptor.rotate ? descriptor.height : descriptor.width, - descriptor.rotate ? descriptor.width : descriptor.height); - var spriteTexture = new core.Texture(baseTexture, spriteRect); - var sprite = new core.Sprite(spriteTexture); - - var baseRotation = descriptor.rotate ? Math.PI * 0.5 : 0.0; - sprite.scale.set(descriptor.width / descriptor.originalWidth, descriptor.height / descriptor.originalHeight); - sprite.rotation = baseRotation - (attachment.rotation * spine.degRad); - sprite.anchor.x = sprite.anchor.y = 0.5; - - slot.sprites = slot.sprites || {}; - slot.sprites[descriptor.name] = sprite; - return sprite; -}; - -/** - * - * @param slot {spine.Slot} The slot to which the attachment is parented - * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent - * @private - */ -Spine.prototype.createMesh = function (slot, attachment) { - var descriptor = attachment.rendererObject; - var baseTexture = descriptor.page.rendererObject; - var texture = new core.Texture(baseTexture); - - var strip = new core.Strip(texture); - strip.drawMode = core.Strip.DrawModes.TRIANGLES; - strip.canvasPadding = 1.5; - - strip.vertices = new Float32Array(attachment.uvs.length); - strip.uvs = attachment.uvs; - strip.indices = attachment.triangles; - - slot.meshes = slot.meshes || {}; - slot.meshes[attachment.name] = strip; - - return strip; -}; - -},{"../core":9,"./SpineRuntime":94}],94:[function(require,module,exports){ -/****************************************************************************** - * Spine Runtimes Software License - * Version 2.1 - * - * Copyright (c) 2013, Esoteric Software - * All rights reserved. - * - * You are granted a perpetual, non-exclusive, non-sublicensable and - * non-transferable license to install, execute and perform the Spine Runtimes - * Software (the "Software") solely for internal use. Without the written - * permission of Esoteric Software (typically granted by licensing Spine), you - * may not (a) modify, translate, adapt or otherwise create derivative works, - * improvements of the Software or develop new applications using the Software - * or (b) remove, delete, alter or obscure any trademarks or any copyright, - * trademark, patent or other intellectual property or proprietary rights - * notices on or in the Software, including any copy thereof. Redistributions - * in binary or source form must include this license and terms. - * - * THIS SOFTWARE IS PROVIDED BY ESOTERIC SOFTWARE "AS IS" AND ANY EXPRESS OR - * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF - * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO - * EVENT SHALL ESOTERIC SOFTARE BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, - * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, - * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; - * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, - * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR - * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF - * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - *****************************************************************************/ - -var spine = module.exports = { - radDeg: 180 / Math.PI, - degRad: Math.PI / 180, - temp: [], - Float32Array: (typeof(Float32Array) === 'undefined') ? Array : Float32Array, - Uint16Array: (typeof(Uint16Array) === 'undefined') ? Array : Uint16Array -}; - -spine.BoneData = function (name, parent) { - this.name = name; - this.parent = parent; -}; -spine.BoneData.prototype = { - length: 0, - x: 0, y: 0, - rotation: 0, - scaleX: 1, scaleY: 1, - inheritScale: true, - inheritRotation: true, - flipX: false, flipY: false -}; - -spine.SlotData = function (name, boneData) { - this.name = name; - this.boneData = boneData; -}; -spine.SlotData.prototype = { - r: 1, g: 1, b: 1, a: 1, - attachmentName: null, - additiveBlending: false -}; - -spine.IkConstraintData = function (name) { - this.name = name; - this.bones = []; -}; -spine.IkConstraintData.prototype = { - target: null, - bendDirection: 1, - mix: 1 -}; - -spine.Bone = function (boneData, skeleton, parent) { - this.data = boneData; - this.skeleton = skeleton; - this.parent = parent; - this.setToSetupPose(); -}; -spine.Bone.yDown = false; -spine.Bone.prototype = { - x: 0, y: 0, - rotation: 0, rotationIK: 0, - scaleX: 1, scaleY: 1, - flipX: false, flipY: false, - m00: 0, m01: 0, worldX: 0, // a b x - m10: 0, m11: 0, worldY: 0, // c d y - worldRotation: 0, - worldScaleX: 1, worldScaleY: 1, - worldFlipX: false, worldFlipY: false, - updateWorldTransform: function () { - var parent = this.parent; - if (parent) { - this.worldX = this.x * parent.m00 + this.y * parent.m01 + parent.worldX; - this.worldY = this.x * parent.m10 + this.y * parent.m11 + parent.worldY; - if (this.data.inheritScale) { - this.worldScaleX = parent.worldScaleX * this.scaleX; - this.worldScaleY = parent.worldScaleY * this.scaleY; - } else { - this.worldScaleX = this.scaleX; - this.worldScaleY = this.scaleY; - } - this.worldRotation = this.data.inheritRotation ? (parent.worldRotation + this.rotationIK) : this.rotationIK; - this.worldFlipX = parent.worldFlipX != this.flipX; - this.worldFlipY = parent.worldFlipY != this.flipY; - } else { - var skeletonFlipX = this.skeleton.flipX, skeletonFlipY = this.skeleton.flipY; - this.worldX = skeletonFlipX ? -this.x : this.x; - this.worldY = (skeletonFlipY != spine.Bone.yDown) ? -this.y : this.y; - this.worldScaleX = this.scaleX; - this.worldScaleY = this.scaleY; - this.worldRotation = this.rotationIK; - this.worldFlipX = skeletonFlipX != this.flipX; - this.worldFlipY = skeletonFlipY != this.flipY; - } - var radians = this.worldRotation * spine.degRad; - var cos = Math.cos(radians); - var sin = Math.sin(radians); - if (this.worldFlipX) { - this.m00 = -cos * this.worldScaleX; - this.m01 = sin * this.worldScaleY; - } else { - this.m00 = cos * this.worldScaleX; - this.m01 = -sin * this.worldScaleY; - } - if (this.worldFlipY != spine.Bone.yDown) { - this.m10 = -sin * this.worldScaleX; - this.m11 = -cos * this.worldScaleY; - } else { - this.m10 = sin * this.worldScaleX; - this.m11 = cos * this.worldScaleY; - } - }, - setToSetupPose: function () { - var data = this.data; - this.x = data.x; - this.y = data.y; - this.rotation = data.rotation; - this.rotationIK = this.rotation; - this.scaleX = data.scaleX; - this.scaleY = data.scaleY; - this.flipX = data.flipX; - this.flipY = data.flipY; - }, - worldToLocal: function (world) { - var dx = world[0] - this.worldX, dy = world[1] - this.worldY; - var m00 = this.m00, m10 = this.m10, m01 = this.m01, m11 = this.m11; - if (this.worldFlipX != (this.worldFlipY != spine.Bone.yDown)) { - m00 = -m00; - m11 = -m11; - } - var invDet = 1 / (m00 * m11 - m01 * m10); - world[0] = dx * m00 * invDet - dy * m01 * invDet; - world[1] = dy * m11 * invDet - dx * m10 * invDet; - }, - localToWorld: function (local) { - var localX = local[0], localY = local[1]; - local[0] = localX * this.m00 + localY * this.m01 + this.worldX; - local[1] = localX * this.m10 + localY * this.m11 + this.worldY; - } -}; - -spine.Slot = function (slotData, bone) { - this.data = slotData; - this.bone = bone; - this.setToSetupPose(); -}; -spine.Slot.prototype = { - r: 1, g: 1, b: 1, a: 1, - _attachmentTime: 0, - attachment: null, - attachmentVertices: [], - setAttachment: function (attachment) { - this.attachment = attachment; - this._attachmentTime = this.bone.skeleton.time; - this.attachmentVertices.length = 0; - }, - setAttachmentTime: function (time) { - this._attachmentTime = this.bone.skeleton.time - time; - }, - getAttachmentTime: function () { - return this.bone.skeleton.time - this._attachmentTime; - }, - setToSetupPose: function () { - var data = this.data; - this.r = data.r; - this.g = data.g; - this.b = data.b; - this.a = data.a; - - var slotDatas = this.bone.skeleton.data.slots; - for (var i = 0, n = slotDatas.length; i < n; i++) { - if (slotDatas[i] == data) { - this.setAttachment(!data.attachmentName ? null : this.bone.skeleton.getAttachmentBySlotIndex(i, data.attachmentName)); - break; - } - } - } -}; - -spine.IkConstraint = function (data, skeleton) { - this.data = data; - this.mix = data.mix; - this.bendDirection = data.bendDirection; - - this.bones = []; - for (var i = 0, n = data.bones.length; i < n; i++) - this.bones.push(skeleton.findBone(data.bones[i].name)); - this.target = skeleton.findBone(data.target.name); -}; -spine.IkConstraint.prototype = { - apply: function () { - var target = this.target; - var bones = this.bones; - switch (bones.length) { - case 1: - spine.IkConstraint.apply1(bones[0], target.worldX, target.worldY, this.mix); - break; - case 2: - spine.IkConstraint.apply2(bones[0], bones[1], target.worldX, target.worldY, this.bendDirection, this.mix); - break; - } - } -}; -/** Adjusts the bone rotation so the tip is as close to the target position as possible. The target is specified in the world - * coordinate system. */ -spine.IkConstraint.apply1 = function (bone, targetX, targetY, alpha) { - var parentRotation = (!bone.data.inheritRotation || !bone.parent) ? 0 : bone.parent.worldRotation; - var rotation = bone.rotation; - var rotationIK = Math.atan2(targetY - bone.worldY, targetX - bone.worldX) * spine.radDeg - parentRotation; - bone.rotationIK = rotation + (rotationIK - rotation) * alpha; -}; -/** Adjusts the parent and child bone rotations so the tip of the child is as close to the target position as possible. The - * target is specified in the world coordinate system. - * @param child Any descendant bone of the parent. */ -spine.IkConstraint.apply2 = function (parent, child, targetX, targetY, bendDirection, alpha) { - var childRotation = child.rotation, parentRotation = parent.rotation; - if (!alpha) { - child.rotationIK = childRotation; - parent.rotationIK = parentRotation; - return; - } - var positionX, positionY, tempPosition = spine.temp; - var parentParent = parent.parent; - if (parentParent) { - tempPosition[0] = targetX; - tempPosition[1] = targetY; - parentParent.worldToLocal(tempPosition); - targetX = (tempPosition[0] - parent.x) * parentParent.worldScaleX; - targetY = (tempPosition[1] - parent.y) * parentParent.worldScaleY; - } else { - targetX -= parent.x; - targetY -= parent.y; - } - if (child.parent == parent) { - positionX = child.x; - positionY = child.y; - } else { - tempPosition[0] = child.x; - tempPosition[1] = child.y; - child.parent.localToWorld(tempPosition); - parent.worldToLocal(tempPosition); - positionX = tempPosition[0]; - positionY = tempPosition[1]; - } - var childX = positionX * parent.worldScaleX, childY = positionY * parent.worldScaleY; - var offset = Math.atan2(childY, childX); - var len1 = Math.sqrt(childX * childX + childY * childY), len2 = child.data.length * child.worldScaleX; - // Based on code by Ryan Juckett with permission: Copyright (c) 2008-2009 Ryan Juckett, http://www.ryanjuckett.com/ - var cosDenom = 2 * len1 * len2; - if (cosDenom < 0.0001) { - child.rotationIK = childRotation + (Math.atan2(targetY, targetX) * spine.radDeg - parentRotation - childRotation) * alpha; - return; - } - var cos = (targetX * targetX + targetY * targetY - len1 * len1 - len2 * len2) / cosDenom; - if (cos < -1) - cos = -1; - else if (cos > 1) - cos = 1; - var childAngle = Math.acos(cos) * bendDirection; - var adjacent = len1 + len2 * cos, opposite = len2 * Math.sin(childAngle); - var parentAngle = Math.atan2(targetY * adjacent - targetX * opposite, targetX * adjacent + targetY * opposite); - var rotation = (parentAngle - offset) * spine.radDeg - parentRotation; - if (rotation > 180) - rotation -= 360; - else if (rotation < -180) // - rotation += 360; - parent.rotationIK = parentRotation + rotation * alpha; - rotation = (childAngle + offset) * spine.radDeg - childRotation; - if (rotation > 180) - rotation -= 360; - else if (rotation < -180) // - rotation += 360; - child.rotationIK = childRotation + (rotation + parent.worldRotation - child.parent.worldRotation) * alpha; -}; - -spine.Skin = function (name) { - this.name = name; - this.attachments = {}; -}; -spine.Skin.prototype = { - addAttachment: function (slotIndex, name, attachment) { - this.attachments[slotIndex + ":" + name] = attachment; - }, - getAttachment: function (slotIndex, name) { - return this.attachments[slotIndex + ":" + name]; - }, - _attachAll: function (skeleton, oldSkin) { - for (var key in oldSkin.attachments) { - var colon = key.indexOf(":"); - var slotIndex = parseInt(key.substring(0, colon)); - var name = key.substring(colon + 1); - var slot = skeleton.slots[slotIndex]; - if (slot.attachment && slot.attachment.name == name) { - var attachment = this.getAttachment(slotIndex, name); - if (attachment) slot.setAttachment(attachment); - } - } - } -}; - -spine.Animation = function (name, timelines, duration) { - this.name = name; - this.timelines = timelines; - this.duration = duration; -}; -spine.Animation.prototype = { - apply: function (skeleton, lastTime, time, loop, events) { - if (loop && this.duration != 0) { - time %= this.duration; - lastTime %= this.duration; - } - var timelines = this.timelines; - for (var i = 0, n = timelines.length; i < n; i++) - timelines[i].apply(skeleton, lastTime, time, events, 1); - }, - mix: function (skeleton, lastTime, time, loop, events, alpha) { - if (loop && this.duration != 0) { - time %= this.duration; - lastTime %= this.duration; - } - var timelines = this.timelines; - for (var i = 0, n = timelines.length; i < n; i++) - timelines[i].apply(skeleton, lastTime, time, events, alpha); - } -}; -spine.Animation.binarySearch = function (values, target, step) { - var low = 0; - var high = Math.floor(values.length / step) - 2; - if (!high) return step; - var current = high >>> 1; - while (true) { - if (values[(current + 1) * step] <= target) - low = current + 1; - else - high = current; - if (low == high) return (low + 1) * step; - current = (low + high) >>> 1; - } -}; -spine.Animation.binarySearch1 = function (values, target) { - var low = 0; - var high = values.length - 2; - if (!high) return 1; - var current = high >>> 1; - while (true) { - if (values[current + 1] <= target) - low = current + 1; - else - high = current; - if (low == high) return low + 1; - current = (low + high) >>> 1; - } -}; -spine.Animation.linearSearch = function (values, target, step) { - for (var i = 0, last = values.length - step; i <= last; i += step) - if (values[i] > target) return i; - return -1; -}; - -spine.Curves = function (frameCount) { - this.curves = []; // type, x, y, ... - //this.curves.length = (frameCount - 1) * 19/*BEZIER_SIZE*/; -}; -spine.Curves.prototype = { - setLinear: function (frameIndex) { - this.curves[frameIndex * 19/*BEZIER_SIZE*/] = 0/*LINEAR*/; - }, - setStepped: function (frameIndex) { - this.curves[frameIndex * 19/*BEZIER_SIZE*/] = 1/*STEPPED*/; - }, - /** Sets the control handle positions for an interpolation bezier curve used to transition from this keyframe to the next. - * cx1 and cx2 are from 0 to 1, representing the percent of time between the two keyframes. cy1 and cy2 are the percent of - * the difference between the keyframe's values. */ - setCurve: function (frameIndex, cx1, cy1, cx2, cy2) { - var subdiv1 = 1 / 10/*BEZIER_SEGMENTS*/, subdiv2 = subdiv1 * subdiv1, subdiv3 = subdiv2 * subdiv1; - var pre1 = 3 * subdiv1, pre2 = 3 * subdiv2, pre4 = 6 * subdiv2, pre5 = 6 * subdiv3; - var tmp1x = -cx1 * 2 + cx2, tmp1y = -cy1 * 2 + cy2, tmp2x = (cx1 - cx2) * 3 + 1, tmp2y = (cy1 - cy2) * 3 + 1; - var dfx = cx1 * pre1 + tmp1x * pre2 + tmp2x * subdiv3, dfy = cy1 * pre1 + tmp1y * pre2 + tmp2y * subdiv3; - var ddfx = tmp1x * pre4 + tmp2x * pre5, ddfy = tmp1y * pre4 + tmp2y * pre5; - var dddfx = tmp2x * pre5, dddfy = tmp2y * pre5; - - var i = frameIndex * 19/*BEZIER_SIZE*/; - var curves = this.curves; - curves[i++] = 2/*BEZIER*/; - - var x = dfx, y = dfy; - for (var n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) { - curves[i] = x; - curves[i + 1] = y; - dfx += ddfx; - dfy += ddfy; - ddfx += dddfx; - ddfy += dddfy; - x += dfx; - y += dfy; - } - }, - getCurvePercent: function (frameIndex, percent) { - percent = percent < 0 ? 0 : (percent > 1 ? 1 : percent); - var curves = this.curves; - var i = frameIndex * 19/*BEZIER_SIZE*/; - var type = curves[i]; - if (type === 0/*LINEAR*/) return percent; - if (type == 1/*STEPPED*/) return 0; - i++; - var x = 0; - for (var start = i, n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) { - x = curves[i]; - if (x >= percent) { - var prevX, prevY; - if (i == start) { - prevX = 0; - prevY = 0; - } else { - prevX = curves[i - 2]; - prevY = curves[i - 1]; - } - return prevY + (curves[i + 1] - prevY) * (percent - prevX) / (x - prevX); - } - } - var y = curves[i - 1]; - return y + (1 - y) * (percent - x) / (1 - x); // Last point is 1,1. - } -}; - -spine.RotateTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, angle, ... - this.frames.length = frameCount * 2; -}; -spine.RotateTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 2; - }, - setFrame: function (frameIndex, time, angle) { - frameIndex *= 2; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = angle; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var bone = skeleton.bones[this.boneIndex]; - - if (time >= frames[frames.length - 2]) { // Time is after last frame. - var amount = bone.data.rotation + frames[frames.length - 1] - bone.rotation; - while (amount > 180) - amount -= 360; - while (amount < -180) - amount += 360; - bone.rotation += amount * alpha; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 2); - var prevFrameValue = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex - 2/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 2 - 1, percent); - - var amount = frames[frameIndex + 1/*FRAME_VALUE*/] - prevFrameValue; - while (amount > 180) - amount -= 360; - while (amount < -180) - amount += 360; - amount = bone.data.rotation + (prevFrameValue + amount * percent) - bone.rotation; - while (amount > 180) - amount -= 360; - while (amount < -180) - amount += 360; - bone.rotation += amount * alpha; - } -}; - -spine.TranslateTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, x, y, ... - this.frames.length = frameCount * 3; -}; -spine.TranslateTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 3; - }, - setFrame: function (frameIndex, time, x, y) { - frameIndex *= 3; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = x; - this.frames[frameIndex + 2] = y; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var bone = skeleton.bones[this.boneIndex]; - - if (time >= frames[frames.length - 3]) { // Time is after last frame. - bone.x += (bone.data.x + frames[frames.length - 2] - bone.x) * alpha; - bone.y += (bone.data.y + frames[frames.length - 1] - bone.y) * alpha; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 3); - var prevFrameX = frames[frameIndex - 2]; - var prevFrameY = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent); - - bone.x += (bone.data.x + prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent - bone.x) * alpha; - bone.y += (bone.data.y + prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent - bone.y) * alpha; - } -}; - -spine.ScaleTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, x, y, ... - this.frames.length = frameCount * 3; -}; -spine.ScaleTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 3; - }, - setFrame: function (frameIndex, time, x, y) { - frameIndex *= 3; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = x; - this.frames[frameIndex + 2] = y; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var bone = skeleton.bones[this.boneIndex]; - - if (time >= frames[frames.length - 3]) { // Time is after last frame. - bone.scaleX += (bone.data.scaleX * frames[frames.length - 2] - bone.scaleX) * alpha; - bone.scaleY += (bone.data.scaleY * frames[frames.length - 1] - bone.scaleY) * alpha; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 3); - var prevFrameX = frames[frameIndex - 2]; - var prevFrameY = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent); - - bone.scaleX += (bone.data.scaleX * (prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent) - bone.scaleX) * alpha; - bone.scaleY += (bone.data.scaleY * (prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent) - bone.scaleY) * alpha; - } -}; - -spine.ColorTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, r, g, b, a, ... - this.frames.length = frameCount * 5; -}; -spine.ColorTimeline.prototype = { - slotIndex: 0, - getFrameCount: function () { - return this.frames.length / 5; - }, - setFrame: function (frameIndex, time, r, g, b, a) { - frameIndex *= 5; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = r; - this.frames[frameIndex + 2] = g; - this.frames[frameIndex + 3] = b; - this.frames[frameIndex + 4] = a; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var r, g, b, a; - if (time >= frames[frames.length - 5]) { - // Time is after last frame. - var i = frames.length - 1; - r = frames[i - 3]; - g = frames[i - 2]; - b = frames[i - 1]; - a = frames[i]; - } else { - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 5); - var prevFrameR = frames[frameIndex - 4]; - var prevFrameG = frames[frameIndex - 3]; - var prevFrameB = frames[frameIndex - 2]; - var prevFrameA = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex - 5/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 5 - 1, percent); - - r = prevFrameR + (frames[frameIndex + 1/*FRAME_R*/] - prevFrameR) * percent; - g = prevFrameG + (frames[frameIndex + 2/*FRAME_G*/] - prevFrameG) * percent; - b = prevFrameB + (frames[frameIndex + 3/*FRAME_B*/] - prevFrameB) * percent; - a = prevFrameA + (frames[frameIndex + 4/*FRAME_A*/] - prevFrameA) * percent; - } - var slot = skeleton.slots[this.slotIndex]; - if (alpha < 1) { - slot.r += (r - slot.r) * alpha; - slot.g += (g - slot.g) * alpha; - slot.b += (b - slot.b) * alpha; - slot.a += (a - slot.a) * alpha; - } else { - slot.r = r; - slot.g = g; - slot.b = b; - slot.a = a; - } - } -}; - -spine.AttachmentTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, ... - this.frames.length = frameCount; - this.attachmentNames = []; - this.attachmentNames.length = frameCount; -}; -spine.AttachmentTimeline.prototype = { - slotIndex: 0, - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, attachmentName) { - this.frames[frameIndex] = time; - this.attachmentNames[frameIndex] = attachmentName; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) { - if (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0); - return; - } else if (lastTime > time) // - lastTime = -1; - - var frameIndex = time >= frames[frames.length - 1] ? frames.length - 1 : spine.Animation.binarySearch1(frames, time) - 1; - if (frames[frameIndex] < lastTime) return; - - var attachmentName = this.attachmentNames[frameIndex]; - skeleton.slots[this.slotIndex].setAttachment( - !attachmentName ? null : skeleton.getAttachmentBySlotIndex(this.slotIndex, attachmentName)); - } -}; - -spine.EventTimeline = function (frameCount) { - this.frames = []; // time, ... - this.frames.length = frameCount; - this.events = []; - this.events.length = frameCount; -}; -spine.EventTimeline.prototype = { - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, event) { - this.frames[frameIndex] = time; - this.events[frameIndex] = event; - }, - /** Fires events for frames > lastTime and <= time. */ - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - if (!firedEvents) return; - - var frames = this.frames; - var frameCount = frames.length; - - if (lastTime > time) { // Fire events after last time for looped animations. - this.apply(skeleton, lastTime, Number.MAX_VALUE, firedEvents, alpha); - lastTime = -1; - } else if (lastTime >= frames[frameCount - 1]) // Last time is after last frame. - return; - if (time < frames[0]) return; // Time is before first frame. - - var frameIndex; - if (lastTime < frames[0]) - frameIndex = 0; - else { - frameIndex = spine.Animation.binarySearch1(frames, lastTime); - var frame = frames[frameIndex]; - while (frameIndex > 0) { // Fire multiple events with the same frame. - if (frames[frameIndex - 1] != frame) break; - frameIndex--; - } - } - var events = this.events; - for (; frameIndex < frameCount && time >= frames[frameIndex]; frameIndex++) - firedEvents.push(events[frameIndex]); - } -}; - -spine.DrawOrderTimeline = function (frameCount) { - this.frames = []; // time, ... - this.frames.length = frameCount; - this.drawOrders = []; - this.drawOrders.length = frameCount; -}; -spine.DrawOrderTimeline.prototype = { - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, drawOrder) { - this.frames[frameIndex] = time; - this.drawOrders[frameIndex] = drawOrder; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var frameIndex; - if (time >= frames[frames.length - 1]) // Time is after last frame. - frameIndex = frames.length - 1; - else - frameIndex = spine.Animation.binarySearch1(frames, time) - 1; - - var drawOrder = skeleton.drawOrder; - var slots = skeleton.slots; - var drawOrderToSetupIndex = this.drawOrders[frameIndex]; - if (!drawOrderToSetupIndex) { - for (var i = 0, n = slots.length; i < n; i++) - drawOrder[i] = slots[i]; - } else { - for (var i = 0, n = drawOrderToSetupIndex.length; i < n; i++) - drawOrder[i] = skeleton.slots[drawOrderToSetupIndex[i]]; - } - - } -}; - -spine.FfdTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; - this.frames.length = frameCount; - this.frameVertices = []; - this.frameVertices.length = frameCount; -}; -spine.FfdTimeline.prototype = { - slotIndex: 0, - attachment: 0, - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, vertices) { - this.frames[frameIndex] = time; - this.frameVertices[frameIndex] = vertices; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var slot = skeleton.slots[this.slotIndex]; - if (slot.attachment != this.attachment) return; - - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var frameVertices = this.frameVertices; - var vertexCount = frameVertices[0].length; - - var vertices = slot.attachmentVertices; - if (vertices.length != vertexCount) alpha = 1; - vertices.length = vertexCount; - - if (time >= frames[frames.length - 1]) { // Time is after last frame. - var lastVertices = frameVertices[frames.length - 1]; - if (alpha < 1) { - for (var i = 0; i < vertexCount; i++) - vertices[i] += (lastVertices[i] - vertices[i]) * alpha; - } else { - for (var i = 0; i < vertexCount; i++) - vertices[i] = lastVertices[i]; - } - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch1(frames, time); - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex - 1] - frameTime); - percent = this.curves.getCurvePercent(frameIndex - 1, percent < 0 ? 0 : (percent > 1 ? 1 : percent)); - - var prevVertices = frameVertices[frameIndex - 1]; - var nextVertices = frameVertices[frameIndex]; - - if (alpha < 1) { - for (var i = 0; i < vertexCount; i++) { - var prev = prevVertices[i]; - vertices[i] += (prev + (nextVertices[i] - prev) * percent - vertices[i]) * alpha; - } - } else { - for (var i = 0; i < vertexCount; i++) { - var prev = prevVertices[i]; - vertices[i] = prev + (nextVertices[i] - prev) * percent; - } - } - } -}; - -spine.IkConstraintTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, mix, bendDirection, ... - this.frames.length = frameCount * 3; -}; -spine.IkConstraintTimeline.prototype = { - ikConstraintIndex: 0, - getFrameCount: function () { - return this.frames.length / 3; - }, - setFrame: function (frameIndex, time, mix, bendDirection) { - frameIndex *= 3; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = mix; - this.frames[frameIndex + 2] = bendDirection; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var ikConstraint = skeleton.ikConstraints[this.ikConstraintIndex]; - - if (time >= frames[frames.length - 3]) { // Time is after last frame. - ikConstraint.mix += (frames[frames.length - 2] - ikConstraint.mix) * alpha; - ikConstraint.bendDirection = frames[frames.length - 1]; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 3); - var prevFrameMix = frames[frameIndex + -2/*PREV_FRAME_MIX*/]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent); - - var mix = prevFrameMix + (frames[frameIndex + 1/*FRAME_MIX*/] - prevFrameMix) * percent; - ikConstraint.mix += (mix - ikConstraint.mix) * alpha; - ikConstraint.bendDirection = frames[frameIndex + -1/*PREV_FRAME_BEND_DIRECTION*/]; - } -}; - -spine.FlipXTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, flip, ... - this.frames.length = frameCount * 2; -}; -spine.FlipXTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 2; - }, - setFrame: function (frameIndex, time, flip) { - frameIndex *= 2; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = flip ? 1 : 0; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) { - if (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0); - return; - } else if (lastTime > time) // - lastTime = -1; - var frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2; - if (frames[frameIndex] < lastTime) return; - skeleton.bones[boneIndex].flipX = frames[frameIndex + 1] != 0; - } -}; - -spine.FlipYTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, flip, ... - this.frames.length = frameCount * 2; -}; -spine.FlipYTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 2; - }, - setFrame: function (frameIndex, time, flip) { - frameIndex *= 2; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = flip ? 1 : 0; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) { - if (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0); - return; - } else if (lastTime > time) // - lastTime = -1; - var frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2; - if (frames[frameIndex] < lastTime) return; - skeleton.bones[boneIndex].flipY = frames[frameIndex + 1] != 0; - } -}; - -spine.SkeletonData = function () { - this.bones = []; - this.slots = []; - this.skins = []; - this.events = []; - this.animations = []; - this.ikConstraints = []; -}; -spine.SkeletonData.prototype = { - name: null, - defaultSkin: null, - width: 0, height: 0, - version: null, hash: null, - /** @return May be null. */ - findBone: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].name == boneName) return bones[i]; - return null; - }, - /** @return -1 if the bone was not found. */ - findBoneIndex: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].name == boneName) return i; - return -1; - }, - /** @return May be null. */ - findSlot: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) { - if (slots[i].name == slotName) return slot[i]; - } - return null; - }, - /** @return -1 if the bone was not found. */ - findSlotIndex: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) - if (slots[i].name == slotName) return i; - return -1; - }, - /** @return May be null. */ - findSkin: function (skinName) { - var skins = this.skins; - for (var i = 0, n = skins.length; i < n; i++) - if (skins[i].name == skinName) return skins[i]; - return null; - }, - /** @return May be null. */ - findEvent: function (eventName) { - var events = this.events; - for (var i = 0, n = events.length; i < n; i++) - if (events[i].name == eventName) return events[i]; - return null; - }, - /** @return May be null. */ - findAnimation: function (animationName) { - var animations = this.animations; - for (var i = 0, n = animations.length; i < n; i++) - if (animations[i].name == animationName) return animations[i]; - return null; - }, - /** @return May be null. */ - findIkConstraint: function (ikConstraintName) { - var ikConstraints = this.ikConstraints; - for (var i = 0, n = ikConstraints.length; i < n; i++) - if (ikConstraints[i].name == ikConstraintName) return ikConstraints[i]; - return null; - } -}; - -spine.Skeleton = function (skeletonData) { - this.data = skeletonData; - - this.bones = []; - for (var i = 0, n = skeletonData.bones.length; i < n; i++) { - var boneData = skeletonData.bones[i]; - var parent = !boneData.parent ? null : this.bones[skeletonData.bones.indexOf(boneData.parent)]; - this.bones.push(new spine.Bone(boneData, this, parent)); - } - - this.slots = []; - this.drawOrder = []; - for (var i = 0, n = skeletonData.slots.length; i < n; i++) { - var slotData = skeletonData.slots[i]; - var bone = this.bones[skeletonData.bones.indexOf(slotData.boneData)]; - var slot = new spine.Slot(slotData, bone); - this.slots.push(slot); - this.drawOrder.push(slot); - } - - this.ikConstraints = []; - for (var i = 0, n = skeletonData.ikConstraints.length; i < n; i++) - this.ikConstraints.push(new spine.IkConstraint(skeletonData.ikConstraints[i], this)); - - this.boneCache = []; - this.updateCache(); -}; -spine.Skeleton.prototype = { - x: 0, y: 0, - skin: null, - r: 1, g: 1, b: 1, a: 1, - time: 0, - flipX: false, flipY: false, - /** Caches information about bones and IK constraints. Must be called if bones or IK constraints are added or removed. */ - updateCache: function () { - var ikConstraints = this.ikConstraints; - var ikConstraintsCount = ikConstraints.length; - - var arrayCount = ikConstraintsCount + 1; - var boneCache = this.boneCache; - if (boneCache.length > arrayCount) boneCache.length = arrayCount; - for (var i = 0, n = boneCache.length; i < n; i++) - boneCache[i].length = 0; - while (boneCache.length < arrayCount) - boneCache[boneCache.length] = []; - - var nonIkBones = boneCache[0]; - var bones = this.bones; - - outer: - for (var i = 0, n = bones.length; i < n; i++) { - var bone = bones[i]; - var current = bone; - do { - for (var ii = 0; ii < ikConstraintsCount; ii++) { - var ikConstraint = ikConstraints[ii]; - var parent = ikConstraint.bones[0]; - var child= ikConstraint.bones[ikConstraint.bones.length - 1]; - while (true) { - if (current == child) { - boneCache[ii].push(bone); - boneCache[ii + 1].push(bone); - continue outer; - } - if (child == parent) break; - child = child.parent; - } - } - current = current.parent; - } while (current); - nonIkBones[nonIkBones.length] = bone; - } - }, - /** Updates the world transform for each bone. */ - updateWorldTransform: function () { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) { - var bone = bones[i]; - bone.rotationIK = bone.rotation; - } - var i = 0, last = this.boneCache.length - 1; - while (true) { - var cacheBones = this.boneCache[i]; - for (var ii = 0, nn = cacheBones.length; ii < nn; ii++) - cacheBones[ii].updateWorldTransform(); - if (i == last) break; - this.ikConstraints[i].apply(); - i++; - } - }, - /** Sets the bones and slots to their setup pose values. */ - setToSetupPose: function () { - this.setBonesToSetupPose(); - this.setSlotsToSetupPose(); - }, - setBonesToSetupPose: function () { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - bones[i].setToSetupPose(); - - var ikConstraints = this.ikConstraints; - for (var i = 0, n = ikConstraints.length; i < n; i++) { - var ikConstraint = ikConstraints[i]; - ikConstraint.bendDirection = ikConstraint.data.bendDirection; - ikConstraint.mix = ikConstraint.data.mix; - } - }, - setSlotsToSetupPose: function () { - var slots = this.slots; - var drawOrder = this.drawOrder; - for (var i = 0, n = slots.length; i < n; i++) { - drawOrder[i] = slots[i]; - slots[i].setToSetupPose(i); - } - }, - /** @return May return null. */ - getRootBone: function () { - return this.bones.length ? this.bones[0] : null; - }, - /** @return May be null. */ - findBone: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].data.name == boneName) return bones[i]; - return null; - }, - /** @return -1 if the bone was not found. */ - findBoneIndex: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].data.name == boneName) return i; - return -1; - }, - /** @return May be null. */ - findSlot: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) - if (slots[i].data.name == slotName) return slots[i]; - return null; - }, - /** @return -1 if the bone was not found. */ - findSlotIndex: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) - if (slots[i].data.name == slotName) return i; - return -1; - }, - setSkinByName: function (skinName) { - var skin = this.data.findSkin(skinName); - if (!skin) throw "Skin not found: " + skinName; - this.setSkin(skin); - }, - /** Sets the skin used to look up attachments before looking in the {@link SkeletonData#getDefaultSkin() default skin}. - * Attachments from the new skin are attached if the corresponding attachment from the old skin was attached. If there was - * no old skin, each slot's setup mode attachment is attached from the new skin. - * @param newSkin May be null. */ - setSkin: function (newSkin) { - if (newSkin) { - if (this.skin) - newSkin._attachAll(this, this.skin); - else { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) { - var slot = slots[i]; - var name = slot.data.attachmentName; - if (name) { - var attachment = newSkin.getAttachment(i, name); - if (attachment) slot.setAttachment(attachment); - } - } - } - } - this.skin = newSkin; - }, - /** @return May be null. */ - getAttachmentBySlotName: function (slotName, attachmentName) { - return this.getAttachmentBySlotIndex(this.data.findSlotIndex(slotName), attachmentName); - }, - /** @return May be null. */ - getAttachmentBySlotIndex: function (slotIndex, attachmentName) { - if (this.skin) { - var attachment = this.skin.getAttachment(slotIndex, attachmentName); - if (attachment) return attachment; - } - if (this.data.defaultSkin) return this.data.defaultSkin.getAttachment(slotIndex, attachmentName); - return null; - }, - /** @param attachmentName May be null. */ - setAttachment: function (slotName, attachmentName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) { - var slot = slots[i]; - if (slot.data.name == slotName) { - var attachment = null; - if (attachmentName) { - attachment = this.getAttachmentBySlotIndex(i, attachmentName); - if (!attachment) throw "Attachment not found: " + attachmentName + ", for slot: " + slotName; - } - slot.setAttachment(attachment); - return; - } - } - throw "Slot not found: " + slotName; - }, - /** @return May be null. */ - findIkConstraint: function (ikConstraintName) { - var ikConstraints = this.ikConstraints; - for (var i = 0, n = ikConstraints.length; i < n; i++) - if (ikConstraints[i].data.name == ikConstraintName) return ikConstraints[i]; - return null; - }, - update: function (delta) { - this.time += delta; - } -}; - -spine.EventData = function (name) { - this.name = name; -}; -spine.EventData.prototype = { - intValue: 0, - floatValue: 0, - stringValue: null -}; - -spine.Event = function (data) { - this.data = data; -}; -spine.Event.prototype = { - intValue: 0, - floatValue: 0, - stringValue: null -}; - -spine.AttachmentType = { - region: 0, - boundingbox: 1, - mesh: 2, - skinnedmesh: 3 -}; - -spine.RegionAttachment = function (name) { - this.name = name; - this.offset = []; - this.offset.length = 8; - this.uvs = []; - this.uvs.length = 8; -}; -spine.RegionAttachment.prototype = { - type: spine.AttachmentType.region, - x: 0, y: 0, - rotation: 0, - scaleX: 1, scaleY: 1, - width: 0, height: 0, - r: 1, g: 1, b: 1, a: 1, - path: null, - rendererObject: null, - regionOffsetX: 0, regionOffsetY: 0, - regionWidth: 0, regionHeight: 0, - regionOriginalWidth: 0, regionOriginalHeight: 0, - setUVs: function (u, v, u2, v2, rotate) { - var uvs = this.uvs; - if (rotate) { - uvs[2/*X2*/] = u; - uvs[3/*Y2*/] = v2; - uvs[4/*X3*/] = u; - uvs[5/*Y3*/] = v; - uvs[6/*X4*/] = u2; - uvs[7/*Y4*/] = v; - uvs[0/*X1*/] = u2; - uvs[1/*Y1*/] = v2; - } else { - uvs[0/*X1*/] = u; - uvs[1/*Y1*/] = v2; - uvs[2/*X2*/] = u; - uvs[3/*Y2*/] = v; - uvs[4/*X3*/] = u2; - uvs[5/*Y3*/] = v; - uvs[6/*X4*/] = u2; - uvs[7/*Y4*/] = v2; - } - }, - updateOffset: function () { - var regionScaleX = this.width / this.regionOriginalWidth * this.scaleX; - var regionScaleY = this.height / this.regionOriginalHeight * this.scaleY; - var localX = -this.width / 2 * this.scaleX + this.regionOffsetX * regionScaleX; - var localY = -this.height / 2 * this.scaleY + this.regionOffsetY * regionScaleY; - var localX2 = localX + this.regionWidth * regionScaleX; - var localY2 = localY + this.regionHeight * regionScaleY; - var radians = this.rotation * spine.degRad; - var cos = Math.cos(radians); - var sin = Math.sin(radians); - var localXCos = localX * cos + this.x; - var localXSin = localX * sin; - var localYCos = localY * cos + this.y; - var localYSin = localY * sin; - var localX2Cos = localX2 * cos + this.x; - var localX2Sin = localX2 * sin; - var localY2Cos = localY2 * cos + this.y; - var localY2Sin = localY2 * sin; - var offset = this.offset; - offset[0/*X1*/] = localXCos - localYSin; - offset[1/*Y1*/] = localYCos + localXSin; - offset[2/*X2*/] = localXCos - localY2Sin; - offset[3/*Y2*/] = localY2Cos + localXSin; - offset[4/*X3*/] = localX2Cos - localY2Sin; - offset[5/*Y3*/] = localY2Cos + localX2Sin; - offset[6/*X4*/] = localX2Cos - localYSin; - offset[7/*Y4*/] = localYCos + localX2Sin; - }, - computeVertices: function (x, y, bone, vertices) { - x += bone.worldX; - y += bone.worldY; - var m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11; - var offset = this.offset; - vertices[0/*X1*/] = offset[0/*X1*/] * m00 + offset[1/*Y1*/] * m01 + x; - vertices[1/*Y1*/] = offset[0/*X1*/] * m10 + offset[1/*Y1*/] * m11 + y; - vertices[2/*X2*/] = offset[2/*X2*/] * m00 + offset[3/*Y2*/] * m01 + x; - vertices[3/*Y2*/] = offset[2/*X2*/] * m10 + offset[3/*Y2*/] * m11 + y; - vertices[4/*X3*/] = offset[4/*X3*/] * m00 + offset[5/*X3*/] * m01 + x; - vertices[5/*X3*/] = offset[4/*X3*/] * m10 + offset[5/*X3*/] * m11 + y; - vertices[6/*X4*/] = offset[6/*X4*/] * m00 + offset[7/*Y4*/] * m01 + x; - vertices[7/*Y4*/] = offset[6/*X4*/] * m10 + offset[7/*Y4*/] * m11 + y; - } -}; - -spine.MeshAttachment = function (name) { - this.name = name; -}; -spine.MeshAttachment.prototype = { - type: spine.AttachmentType.mesh, - vertices: null, - uvs: null, - regionUVs: null, - triangles: null, - hullLength: 0, - r: 1, g: 1, b: 1, a: 1, - path: null, - rendererObject: null, - regionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false, - regionOffsetX: 0, regionOffsetY: 0, - regionWidth: 0, regionHeight: 0, - regionOriginalWidth: 0, regionOriginalHeight: 0, - edges: null, - width: 0, height: 0, - updateUVs: function () { - var width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV; - var n = this.regionUVs.length; - if (!this.uvs || this.uvs.length != n) { - this.uvs = new spine.Float32Array(n); - } - if (this.regionRotate) { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width; - this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height; - } - } else { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i] * width; - this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height; - } - } - }, - computeWorldVertices: function (x, y, slot, worldVertices) { - var bone = slot.bone; - x += bone.worldX; - y += bone.worldY; - var m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11; - var vertices = this.vertices; - var verticesCount = vertices.length; - if (slot.attachmentVertices.length == verticesCount) vertices = slot.attachmentVertices; - for (var i = 0; i < verticesCount; i += 2) { - var vx = vertices[i]; - var vy = vertices[i + 1]; - worldVertices[i] = vx * m00 + vy * m01 + x; - worldVertices[i + 1] = vx * m10 + vy * m11 + y; - } - } -}; - -spine.SkinnedMeshAttachment = function (name) { - this.name = name; -}; -spine.SkinnedMeshAttachment.prototype = { - type: spine.AttachmentType.skinnedmesh, - bones: null, - weights: null, - uvs: null, - regionUVs: null, - triangles: null, - hullLength: 0, - r: 1, g: 1, b: 1, a: 1, - path: null, - rendererObject: null, - regionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false, - regionOffsetX: 0, regionOffsetY: 0, - regionWidth: 0, regionHeight: 0, - regionOriginalWidth: 0, regionOriginalHeight: 0, - edges: null, - width: 0, height: 0, - updateUVs: function (u, v, u2, v2, rotate) { - var width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV; - var n = this.regionUVs.length; - if (!this.uvs || this.uvs.length != n) { - this.uvs = new spine.Float32Array(n); - } - if (this.regionRotate) { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width; - this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height; - } - } else { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i] * width; - this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height; - } - } - }, - computeWorldVertices: function (x, y, slot, worldVertices) { - var skeletonBones = slot.bone.skeleton.bones; - var weights = this.weights; - var bones = this.bones; - - var w = 0, v = 0, b = 0, f = 0, n = bones.length, nn; - var wx, wy, bone, vx, vy, weight; - if (!slot.attachmentVertices.length) { - for (; v < n; w += 2) { - wx = 0; - wy = 0; - nn = bones[v++] + v; - for (; v < nn; v++, b += 3) { - bone = skeletonBones[bones[v]]; - vx = weights[b]; - vy = weights[b + 1]; - weight = weights[b + 2]; - wx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight; - wy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight; - } - worldVertices[w] = wx + x; - worldVertices[w + 1] = wy + y; - } - } else { - var ffd = slot.attachmentVertices; - for (; v < n; w += 2) { - wx = 0; - wy = 0; - nn = bones[v++] + v; - for (; v < nn; v++, b += 3, f += 2) { - bone = skeletonBones[bones[v]]; - vx = weights[b] + ffd[f]; - vy = weights[b + 1] + ffd[f + 1]; - weight = weights[b + 2]; - wx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight; - wy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight; - } - worldVertices[w] = wx + x; - worldVertices[w + 1] = wy + y; - } - } - } -}; - -spine.BoundingBoxAttachment = function (name) { - this.name = name; - this.vertices = []; -}; -spine.BoundingBoxAttachment.prototype = { - type: spine.AttachmentType.boundingbox, - computeWorldVertices: function (x, y, bone, worldVertices) { - x += bone.worldX; - y += bone.worldY; - var m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11; - var vertices = this.vertices; - for (var i = 0, n = vertices.length; i < n; i += 2) { - var px = vertices[i]; - var py = vertices[i + 1]; - worldVertices[i] = px * m00 + py * m01 + x; - worldVertices[i + 1] = px * m10 + py * m11 + y; - } - } -}; - -spine.AnimationStateData = function (skeletonData) { - this.skeletonData = skeletonData; - this.animationToMixTime = {}; -}; -spine.AnimationStateData.prototype = { - defaultMix: 0, - setMixByName: function (fromName, toName, duration) { - var from = this.skeletonData.findAnimation(fromName); - if (!from) throw "Animation not found: " + fromName; - var to = this.skeletonData.findAnimation(toName); - if (!to) throw "Animation not found: " + toName; - this.setMix(from, to, duration); - }, - setMix: function (from, to, duration) { - this.animationToMixTime[from.name + ":" + to.name] = duration; - }, - getMix: function (from, to) { - var key = from.name + ":" + to.name; - return this.animationToMixTime.hasOwnProperty(key) ? this.animationToMixTime[key] : this.defaultMix; - } -}; - -spine.TrackEntry = function () {}; -spine.TrackEntry.prototype = { - next: null, previous: null, - animation: null, - loop: false, - delay: 0, time: 0, lastTime: -1, endTime: 0, - timeScale: 1, - mixTime: 0, mixDuration: 0, mix: 1, - onStart: null, onEnd: null, onComplete: null, onEvent: null -}; - -spine.AnimationState = function (stateData) { - this.data = stateData; - this.tracks = []; - this.events = []; -}; -spine.AnimationState.prototype = { - onStart: null, - onEnd: null, - onComplete: null, - onEvent: null, - timeScale: 1, - update: function (delta) { - delta *= this.timeScale; - for (var i = 0; i < this.tracks.length; i++) { - var current = this.tracks[i]; - if (!current) continue; - - current.time += delta * current.timeScale; - if (current.previous) { - var previousDelta = delta * current.previous.timeScale; - current.previous.time += previousDelta; - current.mixTime += previousDelta; - } - - var next = current.next; - if (next) { - next.time = current.lastTime - next.delay; - if (next.time >= 0) this.setCurrent(i, next); - } else { - // End non-looping animation when it reaches its end time and there is no next entry. - if (!current.loop && current.lastTime >= current.endTime) this.clearTrack(i); - } - } - }, - apply: function (skeleton) { - for (var i = 0; i < this.tracks.length; i++) { - var current = this.tracks[i]; - if (!current) continue; - - this.events.length = 0; - - var time = current.time; - var lastTime = current.lastTime; - var endTime = current.endTime; - var loop = current.loop; - if (!loop && time > endTime) time = endTime; - - var previous = current.previous; - if (!previous) { - if (current.mix == 1) - current.animation.apply(skeleton, current.lastTime, time, loop, this.events); - else - current.animation.mix(skeleton, current.lastTime, time, loop, this.events, current.mix); - } else { - var previousTime = previous.time; - if (!previous.loop && previousTime > previous.endTime) previousTime = previous.endTime; - previous.animation.apply(skeleton, previousTime, previousTime, previous.loop, null); - - var alpha = current.mixTime / current.mixDuration * current.mix; - if (alpha >= 1) { - alpha = 1; - current.previous = null; - } - current.animation.mix(skeleton, current.lastTime, time, loop, this.events, alpha); - } - - for (var ii = 0, nn = this.events.length; ii < nn; ii++) { - var event = this.events[ii]; - if (current.onEvent) current.onEvent(i, event); - if (this.onEvent) this.onEvent(i, event); - } - - // Check if completed the animation or a loop iteration. - if (loop ? (lastTime % endTime > time % endTime) : (lastTime < endTime && time >= endTime)) { - var count = Math.floor(time / endTime); - if (current.onComplete) current.onComplete(i, count); - if (this.onComplete) this.onComplete(i, count); - } - - current.lastTime = current.time; - } - }, - clearTracks: function () { - for (var i = 0, n = this.tracks.length; i < n; i++) - this.clearTrack(i); - this.tracks.length = 0; - }, - clearTrack: function (trackIndex) { - if (trackIndex >= this.tracks.length) return; - var current = this.tracks[trackIndex]; - if (!current) return; - - if (current.onEnd) current.onEnd(trackIndex); - if (this.onEnd) this.onEnd(trackIndex); - - this.tracks[trackIndex] = null; - }, - _expandToIndex: function (index) { - if (index < this.tracks.length) return this.tracks[index]; - while (index >= this.tracks.length) - this.tracks.push(null); - return null; - }, - setCurrent: function (index, entry) { - var current = this._expandToIndex(index); - if (current) { - var previous = current.previous; - current.previous = null; - - if (current.onEnd) current.onEnd(index); - if (this.onEnd) this.onEnd(index); - - entry.mixDuration = this.data.getMix(current.animation, entry.animation); - if (entry.mixDuration > 0) { - entry.mixTime = 0; - // If a mix is in progress, mix from the closest animation. - if (previous && current.mixTime / current.mixDuration < 0.5) - entry.previous = previous; - else - entry.previous = current; - } - } - - this.tracks[index] = entry; - - if (entry.onStart) entry.onStart(index); - if (this.onStart) this.onStart(index); - }, - setAnimationByName: function (trackIndex, animationName, loop) { - var animation = this.data.skeletonData.findAnimation(animationName); - if (!animation) throw "Animation not found: " + animationName; - return this.setAnimation(trackIndex, animation, loop); - }, - /** Set the current animation. Any queued animations are cleared. */ - setAnimation: function (trackIndex, animation, loop) { - var entry = new spine.TrackEntry(); - entry.animation = animation; - entry.loop = loop; - entry.endTime = animation.duration; - this.setCurrent(trackIndex, entry); - return entry; - }, - addAnimationByName: function (trackIndex, animationName, loop, delay) { - var animation = this.data.skeletonData.findAnimation(animationName); - if (!animation) throw "Animation not found: " + animationName; - return this.addAnimation(trackIndex, animation, loop, delay); - }, - /** Adds an animation to be played delay seconds after the current or last queued animation. - * @param delay May be <= 0 to use duration of previous animation minus any mix duration plus the negative delay. */ - addAnimation: function (trackIndex, animation, loop, delay) { - var entry = new spine.TrackEntry(); - entry.animation = animation; - entry.loop = loop; - entry.endTime = animation.duration; - - var last = this._expandToIndex(trackIndex); - if (last) { - while (last.next) - last = last.next; - last.next = entry; - } else - this.tracks[trackIndex] = entry; - - if (delay <= 0) { - if (last) - delay += last.endTime - this.data.getMix(last.animation, animation); - else - delay = 0; - } - entry.delay = delay; - - return entry; - }, - /** May be null. */ - getCurrent: function (trackIndex) { - if (trackIndex >= this.tracks.length) return null; - return this.tracks[trackIndex]; - } -}; - -spine.SkeletonJson = function (attachmentLoader) { - this.attachmentLoader = attachmentLoader; -}; -spine.SkeletonJson.prototype = { - scale: 1, - readSkeletonData: function (root, name) { - var skeletonData = new spine.SkeletonData(); - skeletonData.name = name; - - // Skeleton. - var skeletonMap = root["skeleton"]; - if (skeletonMap) { - skeletonData.hash = skeletonMap["hash"]; - skeletonData.version = skeletonMap["spine"]; - skeletonData.width = skeletonMap["width"] || 0; - skeletonData.height = skeletonMap["height"] || 0; - } - - // Bones. - var bones = root["bones"]; - for (var i = 0, n = bones.length; i < n; i++) { - var boneMap = bones[i]; - var parent = null; - if (boneMap["parent"]) { - parent = skeletonData.findBone(boneMap["parent"]); - if (!parent) throw "Parent bone not found: " + boneMap["parent"]; - } - var boneData = new spine.BoneData(boneMap["name"], parent); - boneData.length = (boneMap["length"] || 0) * this.scale; - boneData.x = (boneMap["x"] || 0) * this.scale; - boneData.y = (boneMap["y"] || 0) * this.scale; - boneData.rotation = (boneMap["rotation"] || 0); - boneData.scaleX = boneMap.hasOwnProperty("scaleX") ? boneMap["scaleX"] : 1; - boneData.scaleY = boneMap.hasOwnProperty("scaleY") ? boneMap["scaleY"] : 1; - boneData.inheritScale = boneMap.hasOwnProperty("inheritScale") ? boneMap["inheritScale"] : true; - boneData.inheritRotation = boneMap.hasOwnProperty("inheritRotation") ? boneMap["inheritRotation"] : true; - skeletonData.bones.push(boneData); - } - - // IK constraints. - var ik = root["ik"]; - if (ik) { - for (var i = 0, n = ik.length; i < n; i++) { - var ikMap = ik[i]; - var ikConstraintData = new spine.IkConstraintData(ikMap["name"]); - - var bones = ikMap["bones"]; - for (var ii = 0, nn = bones.length; ii < nn; ii++) { - var bone = skeletonData.findBone(bones[ii]); - if (!bone) throw "IK bone not found: " + bones[ii]; - ikConstraintData.bones.push(bone); - } - - ikConstraintData.target = skeletonData.findBone(ikMap["target"]); - if (!ikConstraintData.target) throw "Target bone not found: " + ikMap["target"]; - - ikConstraintData.bendDirection = (!ikMap.hasOwnProperty("bendPositive") || ikMap["bendPositive"]) ? 1 : -1; - ikConstraintData.mix = ikMap.hasOwnProperty("mix") ? ikMap["mix"] : 1; - - skeletonData.ikConstraints.push(ikConstraintData); - } - } - - // Slots. - var slots = root["slots"]; - for (var i = 0, n = slots.length; i < n; i++) { - var slotMap = slots[i]; - var boneData = skeletonData.findBone(slotMap["bone"]); - if (!boneData) throw "Slot bone not found: " + slotMap["bone"]; - var slotData = new spine.SlotData(slotMap["name"], boneData); - - var color = slotMap["color"]; - if (color) { - slotData.r = this.toColor(color, 0); - slotData.g = this.toColor(color, 1); - slotData.b = this.toColor(color, 2); - slotData.a = this.toColor(color, 3); - } - - slotData.attachmentName = slotMap["attachment"]; - slotData.additiveBlending = slotMap["additive"] && slotMap["additive"] == "true"; - - skeletonData.slots.push(slotData); - } - - // Skins. - var skins = root["skins"]; - for (var skinName in skins) { - if (!skins.hasOwnProperty(skinName)) continue; - var skinMap = skins[skinName]; - var skin = new spine.Skin(skinName); - for (var slotName in skinMap) { - if (!skinMap.hasOwnProperty(slotName)) continue; - var slotIndex = skeletonData.findSlotIndex(slotName); - var slotEntry = skinMap[slotName]; - for (var attachmentName in slotEntry) { - if (!slotEntry.hasOwnProperty(attachmentName)) continue; - var attachment = this.readAttachment(skin, attachmentName, slotEntry[attachmentName]); - if (attachment) skin.addAttachment(slotIndex, attachmentName, attachment); - } - } - skeletonData.skins.push(skin); - if (skin.name == "default") skeletonData.defaultSkin = skin; - } - - // Events. - var events = root["events"]; - for (var eventName in events) { - if (!events.hasOwnProperty(eventName)) continue; - var eventMap = events[eventName]; - var eventData = new spine.EventData(eventName); - eventData.intValue = eventMap["int"] || 0; - eventData.floatValue = eventMap["float"] || 0; - eventData.stringValue = eventMap["string"] || null; - skeletonData.events.push(eventData); - } - - // Animations. - var animations = root["animations"]; - for (var animationName in animations) { - if (!animations.hasOwnProperty(animationName)) continue; - this.readAnimation(animationName, animations[animationName], skeletonData); - } - - return skeletonData; - }, - readAttachment: function (skin, name, map) { - name = map["name"] || name; - - var type = spine.AttachmentType[map["type"] || "region"]; - var path = map["path"] || name; - - var scale = this.scale; - if (type == spine.AttachmentType.region) { - var region = this.attachmentLoader.newRegionAttachment(skin, name, path); - if (!region) return null; - region.path = path; - region.x = (map["x"] || 0) * scale; - region.y = (map["y"] || 0) * scale; - region.scaleX = map.hasOwnProperty("scaleX") ? map["scaleX"] : 1; - region.scaleY = map.hasOwnProperty("scaleY") ? map["scaleY"] : 1; - region.rotation = map["rotation"] || 0; - region.width = (map["width"] || 0) * scale; - region.height = (map["height"] || 0) * scale; - - var color = map["color"]; - if (color) { - region.r = this.toColor(color, 0); - region.g = this.toColor(color, 1); - region.b = this.toColor(color, 2); - region.a = this.toColor(color, 3); - } - - region.updateOffset(); - return region; - } else if (type == spine.AttachmentType.mesh) { - var mesh = this.attachmentLoader.newMeshAttachment(skin, name, path); - if (!mesh) return null; - mesh.path = path; - mesh.vertices = this.getFloatArray(map, "vertices", scale); - mesh.triangles = this.getIntArray(map, "triangles"); - mesh.regionUVs = this.getFloatArray(map, "uvs", 1); - mesh.updateUVs(); - - color = map["color"]; - if (color) { - mesh.r = this.toColor(color, 0); - mesh.g = this.toColor(color, 1); - mesh.b = this.toColor(color, 2); - mesh.a = this.toColor(color, 3); - } - - mesh.hullLength = (map["hull"] || 0) * 2; - if (map["edges"]) mesh.edges = this.getIntArray(map, "edges"); - mesh.width = (map["width"] || 0) * scale; - mesh.height = (map["height"] || 0) * scale; - return mesh; - } else if (type == spine.AttachmentType.skinnedmesh) { - var mesh = this.attachmentLoader.newSkinnedMeshAttachment(skin, name, path); - if (!mesh) return null; - mesh.path = path; - - var uvs = this.getFloatArray(map, "uvs", 1); - var vertices = this.getFloatArray(map, "vertices", 1); - var weights = []; - var bones = []; - for (var i = 0, n = vertices.length; i < n; ) { - var boneCount = vertices[i++] | 0; - bones[bones.length] = boneCount; - for (var nn = i + boneCount * 4; i < nn; ) { - bones[bones.length] = vertices[i]; - weights[weights.length] = vertices[i + 1] * scale; - weights[weights.length] = vertices[i + 2] * scale; - weights[weights.length] = vertices[i + 3]; - i += 4; - } - } - mesh.bones = bones; - mesh.weights = weights; - mesh.triangles = this.getIntArray(map, "triangles"); - mesh.regionUVs = uvs; - mesh.updateUVs(); - - color = map["color"]; - if (color) { - mesh.r = this.toColor(color, 0); - mesh.g = this.toColor(color, 1); - mesh.b = this.toColor(color, 2); - mesh.a = this.toColor(color, 3); - } - - mesh.hullLength = (map["hull"] || 0) * 2; - if (map["edges"]) mesh.edges = this.getIntArray(map, "edges"); - mesh.width = (map["width"] || 0) * scale; - mesh.height = (map["height"] || 0) * scale; - return mesh; - } else if (type == spine.AttachmentType.boundingbox) { - var attachment = this.attachmentLoader.newBoundingBoxAttachment(skin, name); - var vertices = map["vertices"]; - for (var i = 0, n = vertices.length; i < n; i++) - attachment.vertices.push(vertices[i] * scale); - return attachment; - } - throw "Unknown attachment type: " + type; - }, - readAnimation: function (name, map, skeletonData) { - var timelines = []; - var duration = 0; - - var slots = map["slots"]; - for (var slotName in slots) { - if (!slots.hasOwnProperty(slotName)) continue; - var slotMap = slots[slotName]; - var slotIndex = skeletonData.findSlotIndex(slotName); - - for (var timelineName in slotMap) { - if (!slotMap.hasOwnProperty(timelineName)) continue; - var values = slotMap[timelineName]; - if (timelineName == "color") { - var timeline = new spine.ColorTimeline(values.length); - timeline.slotIndex = slotIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var color = valueMap["color"]; - var r = this.toColor(color, 0); - var g = this.toColor(color, 1); - var b = this.toColor(color, 2); - var a = this.toColor(color, 3); - timeline.setFrame(frameIndex, valueMap["time"], r, g, b, a); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 5 - 5]); - - } else if (timelineName == "attachment") { - var timeline = new spine.AttachmentTimeline(values.length); - timeline.slotIndex = slotIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - timeline.setFrame(frameIndex++, valueMap["time"], valueMap["name"]); - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]); - - } else - throw "Invalid timeline type for a slot: " + timelineName + " (" + slotName + ")"; - } - } - - var bones = map["bones"]; - for (var boneName in bones) { - if (!bones.hasOwnProperty(boneName)) continue; - var boneIndex = skeletonData.findBoneIndex(boneName); - if (boneIndex == -1) throw "Bone not found: " + boneName; - var boneMap = bones[boneName]; - - for (var timelineName in boneMap) { - if (!boneMap.hasOwnProperty(timelineName)) continue; - var values = boneMap[timelineName]; - if (timelineName == "rotate") { - var timeline = new spine.RotateTimeline(values.length); - timeline.boneIndex = boneIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - timeline.setFrame(frameIndex, valueMap["time"], valueMap["angle"]); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]); - - } else if (timelineName == "translate" || timelineName == "scale") { - var timeline; - var timelineScale = 1; - if (timelineName == "scale") - timeline = new spine.ScaleTimeline(values.length); - else { - timeline = new spine.TranslateTimeline(values.length); - timelineScale = this.scale; - } - timeline.boneIndex = boneIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var x = (valueMap["x"] || 0) * timelineScale; - var y = (valueMap["y"] || 0) * timelineScale; - timeline.setFrame(frameIndex, valueMap["time"], x, y); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 3 - 3]); - - } else if (timelineName == "flipX" || timelineName == "flipY") { - var x = timelineName == "flipX"; - var timeline = x ? new spine.FlipXTimeline(values.length) : new spine.FlipYTimeline(values.length); - timeline.boneIndex = boneIndex; - - var field = x ? "x" : "y"; - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - timeline.setFrame(frameIndex, valueMap["time"], valueMap[field] || false); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]); - } else - throw "Invalid timeline type for a bone: " + timelineName + " (" + boneName + ")"; - } - } - - var ikMap = map["ik"]; - for (var ikConstraintName in ikMap) { - if (!ikMap.hasOwnProperty(ikConstraintName)) continue; - var ikConstraint = skeletonData.findIkConstraint(ikConstraintName); - var values = ikMap[ikConstraintName]; - var timeline = new spine.IkConstraintTimeline(values.length); - timeline.ikConstraintIndex = skeletonData.ikConstraints.indexOf(ikConstraint); - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var mix = valueMap.hasOwnProperty("mix") ? valueMap["mix"] : 1; - var bendDirection = (!valueMap.hasOwnProperty("bendPositive") || valueMap["bendPositive"]) ? 1 : -1; - timeline.setFrame(frameIndex, valueMap["time"], mix, bendDirection); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.frameCount * 3 - 3]); - } - - var ffd = map["ffd"]; - for (var skinName in ffd) { - var skin = skeletonData.findSkin(skinName); - var slotMap = ffd[skinName]; - for (slotName in slotMap) { - var slotIndex = skeletonData.findSlotIndex(slotName); - var meshMap = slotMap[slotName]; - for (var meshName in meshMap) { - var values = meshMap[meshName]; - var timeline = new spine.FfdTimeline(values.length); - var attachment = skin.getAttachment(slotIndex, meshName); - if (!attachment) throw "FFD attachment not found: " + meshName; - timeline.slotIndex = slotIndex; - timeline.attachment = attachment; - - var isMesh = attachment.type == spine.AttachmentType.mesh; - var vertexCount; - if (isMesh) - vertexCount = attachment.vertices.length; - else - vertexCount = attachment.weights.length / 3 * 2; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var vertices; - if (!valueMap["vertices"]) { - if (isMesh) - vertices = attachment.vertices; - else { - vertices = []; - vertices.length = vertexCount; - } - } else { - var verticesValue = valueMap["vertices"]; - var vertices = []; - vertices.length = vertexCount; - var start = valueMap["offset"] || 0; - var nn = verticesValue.length; - if (this.scale == 1) { - for (var ii = 0; ii < nn; ii++) - vertices[ii + start] = verticesValue[ii]; - } else { - for (var ii = 0; ii < nn; ii++) - vertices[ii + start] = verticesValue[ii] * this.scale; - } - if (isMesh) { - var meshVertices = attachment.vertices; - for (var ii = 0, nn = vertices.length; ii < nn; ii++) - vertices[ii] += meshVertices[ii]; - } - } - - timeline.setFrame(frameIndex, valueMap["time"], vertices); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines[timelines.length] = timeline; - duration = Math.max(duration, timeline.frames[timeline.frameCount - 1]); - } - } - } - - var drawOrderValues = map["drawOrder"]; - if (!drawOrderValues) drawOrderValues = map["draworder"]; - if (drawOrderValues) { - var timeline = new spine.DrawOrderTimeline(drawOrderValues.length); - var slotCount = skeletonData.slots.length; - var frameIndex = 0; - for (var i = 0, n = drawOrderValues.length; i < n; i++) { - var drawOrderMap = drawOrderValues[i]; - var drawOrder = null; - if (drawOrderMap["offsets"]) { - drawOrder = []; - drawOrder.length = slotCount; - for (var ii = slotCount - 1; ii >= 0; ii--) - drawOrder[ii] = -1; - var offsets = drawOrderMap["offsets"]; - var unchanged = []; - unchanged.length = slotCount - offsets.length; - var originalIndex = 0, unchangedIndex = 0; - for (var ii = 0, nn = offsets.length; ii < nn; ii++) { - var offsetMap = offsets[ii]; - var slotIndex = skeletonData.findSlotIndex(offsetMap["slot"]); - if (slotIndex == -1) throw "Slot not found: " + offsetMap["slot"]; - // Collect unchanged items. - while (originalIndex != slotIndex) - unchanged[unchangedIndex++] = originalIndex++; - // Set changed items. - drawOrder[originalIndex + offsetMap["offset"]] = originalIndex++; - } - // Collect remaining unchanged items. - while (originalIndex < slotCount) - unchanged[unchangedIndex++] = originalIndex++; - // Fill in unchanged items. - for (var ii = slotCount - 1; ii >= 0; ii--) - if (drawOrder[ii] == -1) drawOrder[ii] = unchanged[--unchangedIndex]; - } - timeline.setFrame(frameIndex++, drawOrderMap["time"], drawOrder); - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]); - } - - var events = map["events"]; - if (events) { - var timeline = new spine.EventTimeline(events.length); - var frameIndex = 0; - for (var i = 0, n = events.length; i < n; i++) { - var eventMap = events[i]; - var eventData = skeletonData.findEvent(eventMap["name"]); - if (!eventData) throw "Event not found: " + eventMap["name"]; - var event = new spine.Event(eventData); - event.intValue = eventMap.hasOwnProperty("int") ? eventMap["int"] : eventData.intValue; - event.floatValue = eventMap.hasOwnProperty("float") ? eventMap["float"] : eventData.floatValue; - event.stringValue = eventMap.hasOwnProperty("string") ? eventMap["string"] : eventData.stringValue; - timeline.setFrame(frameIndex++, eventMap["time"], event); - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]); - } - - skeletonData.animations.push(new spine.Animation(name, timelines, duration)); - }, - readCurve: function (timeline, frameIndex, valueMap) { - var curve = valueMap["curve"]; - if (!curve) - timeline.curves.setLinear(frameIndex); - else if (curve == "stepped") - timeline.curves.setStepped(frameIndex); - else if (curve instanceof Array) - timeline.curves.setCurve(frameIndex, curve[0], curve[1], curve[2], curve[3]); - }, - toColor: function (hexString, colorIndex) { - if (hexString.length != 8) throw "Color hexidecimal length must be 8, recieved: " + hexString; - return parseInt(hexString.substring(colorIndex * 2, (colorIndex * 2) + 2), 16) / 255; - }, - getFloatArray: function (map, name, scale) { - var list = map[name]; - var values = new spine.Float32Array(list.length); - var i = 0, n = list.length; - if (scale == 1) { - for (; i < n; i++) - values[i] = list[i]; - } else { - for (; i < n; i++) - values[i] = list[i] * scale; - } - return values; - }, - getIntArray: function (map, name) { - var list = map[name]; - var values = new spine.Uint16Array(list.length); - for (var i = 0, n = list.length; i < n; i++) - values[i] = list[i] | 0; - return values; - } -}; - -spine.Atlas = function (atlasText, textureLoader) { - this.textureLoader = textureLoader; - this.pages = []; - this.regions = []; - - var reader = new spine.AtlasReader(atlasText); - var tuple = []; - tuple.length = 4; - var page = null; - while (true) { - var line = reader.readLine(); - if (line === null) break; - line = reader.trim(line); - if (!line.length) - page = null; - else if (!page) { - page = new spine.AtlasPage(); - page.name = line; - - if (reader.readTuple(tuple) == 2) { // size is only optional for an atlas packed with an old TexturePacker. - page.width = parseInt(tuple[0]); - page.height = parseInt(tuple[1]); - reader.readTuple(tuple); - } - page.format = spine.Atlas.Format[tuple[0]]; - - reader.readTuple(tuple); - page.minFilter = spine.Atlas.TextureFilter[tuple[0]]; - page.magFilter = spine.Atlas.TextureFilter[tuple[1]]; - - var direction = reader.readValue(); - page.uWrap = spine.Atlas.TextureWrap.clampToEdge; - page.vWrap = spine.Atlas.TextureWrap.clampToEdge; - if (direction == "x") - page.uWrap = spine.Atlas.TextureWrap.repeat; - else if (direction == "y") - page.vWrap = spine.Atlas.TextureWrap.repeat; - else if (direction == "xy") - page.uWrap = page.vWrap = spine.Atlas.TextureWrap.repeat; - - textureLoader.load(page, line, this); - - this.pages.push(page); - - } else { - var region = new spine.AtlasRegion(); - region.name = line; - region.page = page; - - region.rotate = reader.readValue() == "true"; - - reader.readTuple(tuple); - var x = parseInt(tuple[0]); - var y = parseInt(tuple[1]); - - reader.readTuple(tuple); - var width = parseInt(tuple[0]); - var height = parseInt(tuple[1]); - - region.u = x / page.width; - region.v = y / page.height; - if (region.rotate) { - region.u2 = (x + height) / page.width; - region.v2 = (y + width) / page.height; - } else { - region.u2 = (x + width) / page.width; - region.v2 = (y + height) / page.height; - } - region.x = x; - region.y = y; - region.width = Math.abs(width); - region.height = Math.abs(height); - - if (reader.readTuple(tuple) == 4) { // split is optional - region.splits = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])]; - - if (reader.readTuple(tuple) == 4) { // pad is optional, but only present with splits - region.pads = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])]; - - reader.readTuple(tuple); - } - } - - region.originalWidth = parseInt(tuple[0]); - region.originalHeight = parseInt(tuple[1]); - - reader.readTuple(tuple); - region.offsetX = parseInt(tuple[0]); - region.offsetY = parseInt(tuple[1]); - - region.index = parseInt(reader.readValue()); - - this.regions.push(region); - } - } -}; -spine.Atlas.prototype = { - findRegion: function (name) { - var regions = this.regions; - for (var i = 0, n = regions.length; i < n; i++) - if (regions[i].name == name) return regions[i]; - return null; - }, - dispose: function () { - var pages = this.pages; - for (var i = 0, n = pages.length; i < n; i++) - this.textureLoader.unload(pages[i].rendererObject); - }, - updateUVs: function (page) { - var regions = this.regions; - for (var i = 0, n = regions.length; i < n; i++) { - var region = regions[i]; - if (region.page != page) continue; - region.u = region.x / page.width; - region.v = region.y / page.height; - if (region.rotate) { - region.u2 = (region.x + region.height) / page.width; - region.v2 = (region.y + region.width) / page.height; - } else { - region.u2 = (region.x + region.width) / page.width; - region.v2 = (region.y + region.height) / page.height; - } - } - } -}; - -spine.Atlas.Format = { - alpha: 0, - intensity: 1, - luminanceAlpha: 2, - rgb565: 3, - rgba4444: 4, - rgb888: 5, - rgba8888: 6 -}; - -spine.Atlas.TextureFilter = { - nearest: 0, - linear: 1, - mipMap: 2, - mipMapNearestNearest: 3, - mipMapLinearNearest: 4, - mipMapNearestLinear: 5, - mipMapLinearLinear: 6 -}; - -spine.Atlas.TextureWrap = { - mirroredRepeat: 0, - clampToEdge: 1, - repeat: 2 -}; - -spine.AtlasPage = function () {}; -spine.AtlasPage.prototype = { - name: null, - format: null, - minFilter: null, - magFilter: null, - uWrap: null, - vWrap: null, - rendererObject: null, - width: 0, - height: 0 -}; - -spine.AtlasRegion = function () {}; -spine.AtlasRegion.prototype = { - page: null, - name: null, - x: 0, y: 0, - width: 0, height: 0, - u: 0, v: 0, u2: 0, v2: 0, - offsetX: 0, offsetY: 0, - originalWidth: 0, originalHeight: 0, - index: 0, - rotate: false, - splits: null, - pads: null -}; - -spine.AtlasReader = function (text) { - this.lines = text.split(/\r\n|\r|\n/); -}; -spine.AtlasReader.prototype = { - index: 0, - trim: function (value) { - return value.replace(/^\s+|\s+$/g, ""); - }, - readLine: function () { - if (this.index >= this.lines.length) return null; - return this.lines[this.index++]; - }, - readValue: function () { - var line = this.readLine(); - var colon = line.indexOf(":"); - if (colon == -1) throw "Invalid line: " + line; - return this.trim(line.substring(colon + 1)); - }, - /** Returns the number of tuple values read (1, 2 or 4). */ - readTuple: function (tuple) { - var line = this.readLine(); - var colon = line.indexOf(":"); - if (colon == -1) throw "Invalid line: " + line; - var i = 0, lastMatch = colon + 1; - for (; i < 3; i++) { - var comma = line.indexOf(",", lastMatch); - if (comma == -1) break; - tuple[i] = this.trim(line.substr(lastMatch, comma - lastMatch)); - lastMatch = comma + 1; - } - tuple[i] = this.trim(line.substring(lastMatch)); - return i + 1; - } -}; - -spine.AtlasAttachmentLoader = function (atlas) { - this.atlas = atlas; -}; -spine.AtlasAttachmentLoader.prototype = { - newRegionAttachment: function (skin, name, path) { - var region = this.atlas.findRegion(path); - if (!region) throw "Region not found in atlas: " + path + " (region attachment: " + name + ")"; - var attachment = new spine.RegionAttachment(name); - attachment.rendererObject = region; - attachment.setUVs(region.u, region.v, region.u2, region.v2, region.rotate); - attachment.regionOffsetX = region.offsetX; - attachment.regionOffsetY = region.offsetY; - attachment.regionWidth = region.width; - attachment.regionHeight = region.height; - attachment.regionOriginalWidth = region.originalWidth; - attachment.regionOriginalHeight = region.originalHeight; - return attachment; - }, - newMeshAttachment: function (skin, name, path) { - var region = this.atlas.findRegion(path); - if (!region) throw "Region not found in atlas: " + path + " (mesh attachment: " + name + ")"; - var attachment = new spine.MeshAttachment(name); - attachment.rendererObject = region; - attachment.regionU = region.u; - attachment.regionV = region.v; - attachment.regionU2 = region.u2; - attachment.regionV2 = region.v2; - attachment.regionRotate = region.rotate; - attachment.regionOffsetX = region.offsetX; - attachment.regionOffsetY = region.offsetY; - attachment.regionWidth = region.width; - attachment.regionHeight = region.height; - attachment.regionOriginalWidth = region.originalWidth; - attachment.regionOriginalHeight = region.originalHeight; - return attachment; - }, - newSkinnedMeshAttachment: function (skin, name, path) { - var region = this.atlas.findRegion(path); - if (!region) throw "Region not found in atlas: " + path + " (skinned mesh attachment: " + name + ")"; - var attachment = new spine.SkinnedMeshAttachment(name); - attachment.rendererObject = region; - attachment.regionU = region.u; - attachment.regionV = region.v; - attachment.regionU2 = region.u2; - attachment.regionV2 = region.v2; - attachment.regionRotate = region.rotate; - attachment.regionOffsetX = region.offsetX; - attachment.regionOffsetY = region.offsetY; - attachment.regionWidth = region.width; - attachment.regionHeight = region.height; - attachment.regionOriginalWidth = region.originalWidth; - attachment.regionOriginalHeight = region.originalHeight; - return attachment; - }, - newBoundingBoxAttachment: function (skin, name) { - return new spine.BoundingBoxAttachment(name); - } -}; - -spine.SkeletonBounds = function () { - this.polygonPool = []; - this.polygons = []; - this.boundingBoxes = []; -}; -spine.SkeletonBounds.prototype = { - minX: 0, minY: 0, maxX: 0, maxY: 0, - update: function (skeleton, updateAabb) { - var slots = skeleton.slots; - var slotCount = slots.length; - var x = skeleton.x, y = skeleton.y; - var boundingBoxes = this.boundingBoxes; - var polygonPool = this.polygonPool; - var polygons = this.polygons; - - boundingBoxes.length = 0; - for (var i = 0, n = polygons.length; i < n; i++) - polygonPool.push(polygons[i]); - polygons.length = 0; - - for (var i = 0; i < slotCount; i++) { - var slot = slots[i]; - var boundingBox = slot.attachment; - if (boundingBox.type != spine.AttachmentType.boundingbox) continue; - boundingBoxes.push(boundingBox); - - var poolCount = polygonPool.length, polygon; - if (poolCount > 0) { - polygon = polygonPool[poolCount - 1]; - polygonPool.splice(poolCount - 1, 1); - } else - polygon = []; - polygons.push(polygon); - - polygon.length = boundingBox.vertices.length; - boundingBox.computeWorldVertices(x, y, slot.bone, polygon); - } - - if (updateAabb) this.aabbCompute(); - }, - aabbCompute: function () { - var polygons = this.polygons; - var minX = Number.MAX_VALUE, minY = Number.MAX_VALUE, maxX = Number.MIN_VALUE, maxY = Number.MIN_VALUE; - for (var i = 0, n = polygons.length; i < n; i++) { - var vertices = polygons[i]; - for (var ii = 0, nn = vertices.length; ii < nn; ii += 2) { - var x = vertices[ii]; - var y = vertices[ii + 1]; - minX = Math.min(minX, x); - minY = Math.min(minY, y); - maxX = Math.max(maxX, x); - maxY = Math.max(maxY, y); - } - } - this.minX = minX; - this.minY = minY; - this.maxX = maxX; - this.maxY = maxY; - }, - /** Returns true if the axis aligned bounding box contains the point. */ - aabbContainsPoint: function (x, y) { - return x >= this.minX && x <= this.maxX && y >= this.minY && y <= this.maxY; - }, - /** Returns true if the axis aligned bounding box intersects the line segment. */ - aabbIntersectsSegment: function (x1, y1, x2, y2) { - var minX = this.minX, minY = this.minY, maxX = this.maxX, maxY = this.maxY; - if ((x1 <= minX && x2 <= minX) || (y1 <= minY && y2 <= minY) || (x1 >= maxX && x2 >= maxX) || (y1 >= maxY && y2 >= maxY)) - return false; - var m = (y2 - y1) / (x2 - x1); - var y = m * (minX - x1) + y1; - if (y > minY && y < maxY) return true; - y = m * (maxX - x1) + y1; - if (y > minY && y < maxY) return true; - var x = (minY - y1) / m + x1; - if (x > minX && x < maxX) return true; - x = (maxY - y1) / m + x1; - if (x > minX && x < maxX) return true; - return false; - }, - /** Returns true if the axis aligned bounding box intersects the axis aligned bounding box of the specified bounds. */ - aabbIntersectsSkeleton: function (bounds) { - return this.minX < bounds.maxX && this.maxX > bounds.minX && this.minY < bounds.maxY && this.maxY > bounds.minY; - }, - /** Returns the first bounding box attachment that contains the point, or null. When doing many checks, it is usually more - * efficient to only call this method if {@link #aabbContainsPoint(float, float)} returns true. */ - containsPoint: function (x, y) { - var polygons = this.polygons; - for (var i = 0, n = polygons.length; i < n; i++) - if (this.polygonContainsPoint(polygons[i], x, y)) return this.boundingBoxes[i]; - return null; - }, - /** Returns the first bounding box attachment that contains the line segment, or null. When doing many checks, it is usually - * more efficient to only call this method if {@link #aabbIntersectsSegment(float, float, float, float)} returns true. */ - intersectsSegment: function (x1, y1, x2, y2) { - var polygons = this.polygons; - for (var i = 0, n = polygons.length; i < n; i++) - if (polygons[i].intersectsSegment(x1, y1, x2, y2)) return this.boundingBoxes[i]; - return null; - }, - /** Returns true if the polygon contains the point. */ - polygonContainsPoint: function (polygon, x, y) { - var nn = polygon.length; - var prevIndex = nn - 2; - var inside = false; - for (var ii = 0; ii < nn; ii += 2) { - var vertexY = polygon[ii + 1]; - var prevY = polygon[prevIndex + 1]; - if ((vertexY < y && prevY >= y) || (prevY < y && vertexY >= y)) { - var vertexX = polygon[ii]; - if (vertexX + (y - vertexY) / (prevY - vertexY) * (polygon[prevIndex] - vertexX) < x) inside = !inside; - } - prevIndex = ii; - } - return inside; - }, - /** Returns true if the polygon contains the line segment. */ - polygonIntersectsSegment: function (polygon, x1, y1, x2, y2) { - var nn = polygon.length; - var width12 = x1 - x2, height12 = y1 - y2; - var det1 = x1 * y2 - y1 * x2; - var x3 = polygon[nn - 2], y3 = polygon[nn - 1]; - for (var ii = 0; ii < nn; ii += 2) { - var x4 = polygon[ii], y4 = polygon[ii + 1]; - var det2 = x3 * y4 - y3 * x4; - var width34 = x3 - x4, height34 = y3 - y4; - var det3 = width12 * height34 - height12 * width34; - var x = (det1 * width34 - width12 * det2) / det3; - if (((x >= x3 && x <= x4) || (x >= x4 && x <= x3)) && ((x >= x1 && x <= x2) || (x >= x2 && x <= x1))) { - var y = (det1 * height34 - height12 * det2) / det3; - if (((y >= y3 && y <= y4) || (y >= y4 && y <= y3)) && ((y >= y1 && y <= y2) || (y >= y2 && y <= y1))) return true; - } - x3 = x4; - y3 = y4; - } - return false; - }, - getPolygon: function (attachment) { - var index = this.boundingBoxes.indexOf(attachment); - return index == -1 ? null : this.polygons[index]; - }, - getWidth: function () { - return this.maxX - this.minX; - }, - getHeight: function () { - return this.maxY - this.minY; - } -}; - -},{}],95:[function(require,module,exports){ -/** - * @file Main export of the PIXI spine library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - Spine: require('./Spine') -}; - -},{"./Spine":93}],96:[function(require,module,exports){ -var core = require('../core'); - -/** - * A BitmapText object will create a line or multiple lines of text using bitmap font. To - * split a line you can use '\n', '\r' or '\r\n' in your string. You can generate the fnt files using: - * - * http://www.angelcode.com/products/bmfont/ for windows or - * http://www.bmglyph.com/ for mac. - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - * @param text {string} The copy that you would like the text to display - * @param style {object} The style parameters - * @param style.font {string} The size (optional) and bitmap font id (required) eq 'Arial' or '20px Arial' (must have loaded previously) - * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text - */ -function BitmapText(text, style) { - core.DisplayObjectContainer.call(this); - - /** - * The width of the overall text, different from fontSize, - * which is defined in the style object - * - * @member {number} - * @readOnly - */ - this.textWidth = 0; - - /** - * The height of the overall text, different from fontSize, - * which is defined in the style object - * - * @member {number} - * @readOnly - */ - this.textHeight = 0; - - /** - * Private tracker for the letter sprite pool. - * - * @member {Sprite[]} - * @private - */ - this._pool = []; - - /** - * Private tracker for the current style. - * - * @member {object} - * @private - */ - this._style = { - tint: style.tint, - align: style.align, - fontName: null, - fontSize: 0 - }; - this.font = style.font; // run font setter - - /** - * Private tracker for the current text. - * - * @member {string} - * @private - */ - this._text = text; - - /** - * The dirty state of this object. - * - * @member {boolean} - */ - this.dirty = false; - - this.updateText(); -} - -// constructor -BitmapText.prototype = Object.create(core.DisplayObjectContainer.prototype); -BitmapText.prototype.constructor = BitmapText; -module.exports = BitmapText; - -Object.defineProperties(BitmapText.prototype, { - /** - * The tint of the BitmapText object - * - * @member {number} - * @memberof BitmapText# - */ - tint: { - get: function () { - return this._style.tint; - }, - set: function (value) { - this._style.tint = (typeof value === 'number' && value >= 0) ? value : 0xFFFFFF; - - this.dirty = true; - } - }, - - /** - * The tint of the BitmapText object - * - * @member {string} - * @default 'left' - * @memberof BitmapText# - */ - align: { - get: function () { - return this._style.align; - }, - set: function (value) { - this._style.align = value; - - this.dirty = true; - } - }, - - /** - * The tint of the BitmapText object - * - * @member {Font} - * @memberof BitmapText# - */ - font: { - get: function () { - return this._style.font; - }, - set: function (value) { - value = value.split(' '); - - // TODO - This should be object-based not string based like it has been. - this._style.fontName = value[value.length - 1]; - this._style.fontSize = value.length >= 2 ? parseInt(value[value.length - 2], 10) : BitmapText.fonts[this.fontName].size; - - this.dirty = true; - } - }, - - /** - * The text of the BitmapText object - * - * @member {string} - * @memberof BitmapText# - */ - text: { - get: function () { - return this._text; - }, - set: function (value) { - this._text = value; - - this.dirty = true; - } - } -}); - -/** - * Renders text and updates it when needed - * - * @private - */ -BitmapText.prototype.updateText = function () { - var data = BitmapText.fonts[this.fontName]; - var pos = new core.math.Point(); - var prevCharCode = null; - var chars = []; - var maxLineWidth = 0; - var lineWidths = []; - var line = 0; - var scale = this.fontSize / data.size; - - for (var i = 0; i < this.text.length; i++) { - var charCode = this.text.charCodeAt(i); - - if (/(?:\r\n|\r|\n)/.test(this.text.charAt(i))) { - lineWidths.push(pos.x); - maxLineWidth = Math.max(maxLineWidth, pos.x); - line++; - - pos.x = 0; - pos.y += data.lineHeight; - prevCharCode = null; - continue; - } - - var charData = data.chars[charCode]; - - if (!charData) { - continue; - } - - if (prevCharCode && charData.kerning[prevCharCode]) { - pos.x += charData.kerning[prevCharCode]; - } - - chars.push({texture:charData.texture, line: line, charCode: charCode, position: new core.math.Point(pos.x + charData.xOffset, pos.y + charData.yOffset)}); - pos.x += charData.xAdvance; - - prevCharCode = charCode; - } - - lineWidths.push(pos.x); - maxLineWidth = Math.max(maxLineWidth, pos.x); - - var lineAlignOffsets = []; - - for (i = 0; i <= line; i++) { - var alignOffset = 0; - - if (this.style.align === 'right') { - alignOffset = maxLineWidth - lineWidths[i]; - } - else if (this.style.align === 'center') { - alignOffset = (maxLineWidth - lineWidths[i]) / 2; - } - - lineAlignOffsets.push(alignOffset); - } - - var lenChildren = this.children.length; - var lenChars = chars.length; - var tint = this.tint; - - for (i = 0; i < lenChars; i++) { - var c = i < lenChildren ? this.children[i] : this._pool.pop(); // get old child if have. if not - take from pool. - - if (c) { - c.setTexture(chars[i].texture); // check if got one before. - } - else { - c = new core.Sprite(chars[i].texture); // if no create new one. - } - - c.position.x = (chars[i].position.x + lineAlignOffsets[chars[i].line]) * scale; - c.position.y = chars[i].position.y * scale; - c.scale.x = c.scale.y = scale; - c.tint = tint; - - if (!c.parent) { - this.addChild(c); - } - } - - // remove unnecessary children. - // and put their into the pool. - while(this.children.length > lenChars) { - var child = this.getChildAt(this.children.length - 1); - this._pool.push(child); - this.removeChild(child); - } - - this.textWidth = maxLineWidth * scale; - this.textHeight = (pos.y + data.lineHeight) * scale; -}; - -/** - * Updates the transform of this object - * - * @private - */ -BitmapText.prototype.updateTransform = function () { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - this.displayObjectContainerUpdateTransform(); -}; - -BitmapText.fonts = {}; - -},{"../core":9}],97:[function(require,module,exports){ -var core = require('../core'); - -/** - * A Text Object will create a line or multiple lines of text. To split a line you can use '\n' in your text string, - * or add a wordWrap property set to true and and wordWrapWidth property with a value in the style object. - * - * @class - * @extends Sprite - * @namespace PIXI - * @param text {string} The copy that you would like the text to display - * @param [style] {object} The style parameters - * @param [style.font] {string} default 'bold 20px Arial' The style and size of the font - * @param [style.fill='black'] {String|Number} A canvas fillstyle that will be used on the text e.g 'red', '#00FF00' - * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text - * @param [style.stroke] {String|Number} A canvas fillstyle that will be used on the text stroke e.g 'blue', '#FCFF00' - * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke) - * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used - * @param [style.wordWrapWidth=100] {number} The width at which text will wrap, it needs wordWrap to be set to true - * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text - * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00' - * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow - * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow - */ -function Text(text, style) { - /** - * The canvas element that everything is drawn to - * - * @member {HTMLCanvasElement} - */ - this.canvas = document.createElement('canvas'); - - /** - * The canvas 2d context that everything is drawn with - * @member {HTMLCanvasElement} - */ - this.context = this.canvas.getContext('2d'); - - /** - * The resolution of the canvas. - * @member {number} - */ - this.resolution = 1; - - core.Sprite.call(this, core.Texture.fromCanvas(this.canvas)); - - this.setText(text); - this.setStyle(style); -} - -// constructor -Text.prototype = Object.create(core.Sprite.prototype); -Text.prototype.constructor = Text; -module.exports = Text; - -Object.defineProperties(Text.prototype, { - /** - * The width of the Text, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof Text# - */ - width: { - get: function () { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - return this.scale.x * this.texture.frame.width; - }, - set: function (value) { - this.scale.x = value / this.texture.frame.width; - this._width = value; - } - }, - - /** - * The height of the Text, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof Text# - */ - height: { - get: function () { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - return this.scale.y * this.texture.frame.height; - }, - set: function (value) { - this.scale.y = value / this.texture.frame.height; - this._height = value; - } - } -}); - -/** - * Set the style of the text - * - * @param [style] {object} The style parameters - * @param [style.font='bold 20pt Arial'] {string} The style and size of the font - * @param [style.fill='black'] {object} A canvas fillstyle that will be used on the text eg 'red', '#00FF00' - * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text - * @param [style.stroke='black'] {string} A canvas fillstyle that will be used on the text stroke eg 'blue', '#FCFF00' - * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke) - * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used - * @param [style.wordWrapWidth=100] {number} The width at which text will wrap - * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text - * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00' - * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow - * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow - */ -Text.prototype.setStyle = function (style) { - style = style || {}; - style.font = style.font || 'bold 20pt Arial'; - style.fill = style.fill || 'black'; - style.align = style.align || 'left'; - style.stroke = style.stroke || 'black'; //provide a default, see: https://github.com/GoodBoyDigital/pixi.js/issues/136 - style.strokeThickness = style.strokeThickness || 0; - style.wordWrap = style.wordWrap || false; - style.wordWrapWidth = style.wordWrapWidth || 100; - - style.dropShadow = style.dropShadow || false; - style.dropShadowAngle = style.dropShadowAngle || Math.PI / 6; - style.dropShadowDistance = style.dropShadowDistance || 4; - style.dropShadowColor = style.dropShadowColor || 'black'; - - this.style = style; - this.dirty = true; -}; - -/** - * Set the copy for the text object. To split a line you can use '\n'. - * - * @param text {string} The copy that you would like the text to display - */ -Text.prototype.setText = function (text) { - this.text = text.toString() || ' '; - this.dirty = true; -}; - -/** - * Renders text and updates it when needed - * - * @private - */ -Text.prototype.updateText = function () { - this.texture.baseTexture.resolution = this.resolution; - - this.context.font = this.style.font; - - var outputText = this.text; - - // word wrap - // preserve original text - if (this.style.wordWrap) { - outputText = this.wordWrap(this.text); - } - - //split text into lines - var lines = outputText.split(/(?:\r\n|\r|\n)/); - - //calculate text width - var lineWidths = []; - var maxLineWidth = 0; - var fontProperties = this.determineFontProperties(this.style.font); - for (var i = 0; i < lines.length; i++) { - var lineWidth = this.context.measureText(lines[i]).width; - lineWidths[i] = lineWidth; - maxLineWidth = Math.max(maxLineWidth, lineWidth); - } - - var width = maxLineWidth + this.style.strokeThickness; - if (this.style.dropShadow) { - width += this.style.dropShadowDistance; - } - - this.canvas.width = ( width + this.context.lineWidth ) * this.resolution; - - //calculate text height - var lineHeight = fontProperties.fontSize + this.style.strokeThickness; - - var height = lineHeight * lines.length; - if (this.style.dropShadow) { - height += this.style.dropShadowDistance; - } - - this.canvas.height = height * this.resolution; - - this.context.scale( this.resolution, this.resolution); - - if (navigator.isCocoonJS) { - this.context.clearRect(0, 0, this.canvas.width, this.canvas.height); - } - - // used for debugging.. - //this.context.fillStyle ="#FF0000" - //this.context.fillRect(0, 0, this.canvas.width,this.canvas.height); - - this.context.font = this.style.font; - this.context.strokeStyle = this.style.stroke; - this.context.lineWidth = this.style.strokeThickness; - this.context.textBaseline = 'alphabetic'; - //this.context.lineJoin = 'round'; - - var linePositionX; - var linePositionY; - - if (this.style.dropShadow) { - this.context.fillStyle = this.style.dropShadowColor; - - var xShadowOffset = Math.cos(this.style.dropShadowAngle) * this.style.dropShadowDistance; - var yShadowOffset = Math.sin(this.style.dropShadowAngle) * this.style.dropShadowDistance; - - for (i = 0; i < lines.length; i++) { - linePositionX = this.style.strokeThickness / 2; - linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent; - - if (this.style.align === 'right') { - linePositionX += maxLineWidth - lineWidths[i]; - } - else if (this.style.align === 'center') { - linePositionX += (maxLineWidth - lineWidths[i]) / 2; - } - - if (this.style.fill) { - this.context.fillText(lines[i], linePositionX + xShadowOffset, linePositionY + yShadowOffset); - } - - // if (dropShadow) - } - } - - //set canvas text styles - this.context.fillStyle = this.style.fill; - - //draw lines line by line - for (i = 0; i < lines.length; i++) { - linePositionX = this.style.strokeThickness / 2; - linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent; - - if (this.style.align === 'right') { - linePositionX += maxLineWidth - lineWidths[i]; - } - else if (this.style.align === 'center') { - linePositionX += (maxLineWidth - lineWidths[i]) / 2; - } - - if (this.style.stroke && this.style.strokeThickness) { - this.context.strokeText(lines[i], linePositionX, linePositionY); - } - - if (this.style.fill) { - this.context.fillText(lines[i], linePositionX, linePositionY); - } - - // if (dropShadow) - } - - this.updateTexture(); -}; - -/** - * Updates texture size based on canvas size - * - * @private - */ -Text.prototype.updateTexture = function () { - this.texture.baseTexture.width = this.canvas.width; - this.texture.baseTexture.height = this.canvas.height; - this.texture.crop.width = this.texture.frame.width = this.canvas.width; - this.texture.crop.height = this.texture.frame.height = this.canvas.height; - - this._width = this.canvas.width; - this._height = this.canvas.height; - - // update the dirty base textures - this.texture.baseTexture.dirty(); -}; - -/** - * Renders the object using the WebGL renderer -* - * @param renderer {WebGLRenderer} - */ -Text.prototype.renderWebGL = function (renderer) { - if (this.dirty) { - this.resolution = renderer.resolution; - - this.updateText(); - this.dirty = false; - } - - core.Sprite.prototype.renderWebGL.call(this, renderer); -}; - -/** - * Renders the object using the Canvas renderer -* - * @param renderer {CanvasRenderer} - */ -Text.prototype.renderCanvas = function (renderer) { - if (this.dirty) { - this.resolution = renderer.resolution; - - this.updateText(); - this.dirty = false; - } - - core.Sprite.prototype.renderCanvas.call(this, renderer); -}; - -/** - * Calculates the ascent, descent and fontSize of a given fontStyle -* - * @param fontStyle {object} - * @private - */ -Text.prototype.determineFontProperties = function (fontStyle) { - var properties = Text.fontPropertiesCache[fontStyle]; - - if (!properties) { - properties = {}; - - var canvas = Text.fontPropertiesCanvas; - var context = Text.fontPropertiesContext; - - context.font = fontStyle; - - var width = Math.ceil(context.measureText('|Mq').width); - var baseline = Math.ceil(context.measureText('M').width); - var height = 2 * baseline; - - baseline = baseline * 1.4 | 0; - - canvas.width = width; - canvas.height = height; - - context.fillStyle = '#f00'; - context.fillRect(0, 0, width, height); - - context.font = fontStyle; - - context.textBaseline = 'alphabetic'; - context.fillStyle = '#000'; - context.fillText('|MÉq', 0, baseline); - - var imagedata = context.getImageData(0, 0, width, height).data; - var pixels = imagedata.length; - var line = width * 4; - - var i, j; - - var idx = 0; - var stop = false; - - // ascent. scan from top to bottom until we find a non red pixel - for (i = 0; i < baseline; i++) { - for (j = 0; j < line; j += 4) { - if (imagedata[idx + j] !== 255) { - stop = true; - break; - } - } - if (!stop) { - idx += line; - } - else { - break; - } - } - - properties.ascent = baseline - i; - - idx = pixels - line; - stop = false; - - // descent. scan from bottom to top until we find a non red pixel - for (i = height; i > baseline; i--) { - for (j = 0; j < line; j += 4) { - if (imagedata[idx + j] !== 255) { - stop = true; - break; - } - } - if (!stop) { - idx -= line; - } - else { - break; - } - } - - properties.descent = i - baseline; - //TODO might need a tweak. kind of a temp fix! - properties.descent += 6; - properties.fontSize = properties.ascent + properties.descent; - - Text.fontPropertiesCache[fontStyle] = properties; - } - - return properties; -}; - -/** - * Applies newlines to a string to have it optimally fit into the horizontal - * bounds set by the Text object's wordWrapWidth property. - * - * @param text {string} - * @private - */ -Text.prototype.wordWrap = function (text) { - // Greedy wrapping algorithm that will wrap words as the line grows longer - // than its horizontal bounds. - var result = ''; - var lines = text.split('\n'); - for (var i = 0; i < lines.length; i++) { - var spaceLeft = this.style.wordWrapWidth; - var words = lines[i].split(' '); - for (var j = 0; j < words.length; j++) { - var wordWidth = this.context.measureText(words[j]).width; - var wordWidthWithSpace = wordWidth + this.context.measureText(' ').width; - if (j === 0 || wordWidthWithSpace > spaceLeft) { - // Skip printing the newline if it's the first word of the line that is - // greater than the word wrap width. - if (j > 0) { - result += '\n'; - } - result += words[j]; - spaceLeft = this.style.wordWrapWidth - wordWidth; - } - else { - spaceLeft -= wordWidthWithSpace; - result += ' ' + words[j]; - } - } - - if (i < lines.length-1) { - result += '\n'; - } - } - return result; -}; - -/** - * Returns the bounds of the Text as a rectangle. The bounds calculation takes the worldTransform into account. - * - * @param matrix {Matrix} the transformation matrix of the Text - * @return {Rectangle} the framing rectangle - */ -Text.prototype.getBounds = function (matrix) { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - return core.Sprite.prototype.getBounds.call(this, matrix); -}; - -/** - * Destroys this text object. - * - * @param destroyBaseTexture {boolean} whether to destroy the base texture as well - */ -Text.prototype.destroy = function (destroyBaseTexture) { - // make sure to reset the the context and canvas.. dont want this hanging around in memory! - this.context = null; - this.canvas = null; - - this.texture.destroy(destroyBaseTexture === undefined ? true : destroyBaseTexture); -}; - -Text.fontPropertiesCache = {}; -Text.fontPropertiesCanvas = document.createElement('canvas'); -Text.fontPropertiesContext = Text.fontPropertiesCanvas.getContext('2d'); - -},{"../core":9}],98:[function(require,module,exports){ -/** - * @file Main export of the PIXI text library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - Text: require('./Text'), - BitmapText: require('./BitmapText') -}; - -},{"./BitmapText":96,"./Text":97}]},{},[1])(1) -}); -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","src/index","node_modules/webgl-enabled/index.js","package.json","src/core/const.js","src/core/display/DisplayObject.js","src/core/display/DisplayObjectContainer.js","src/core/display/Sprite.js","src/core/display/SpriteBatch.js","src/core/index.js","src/core/math/Matrix.js","src/core/math/Point.js","src/core/math/index.js","src/core/math/shapes/Circle.js","src/core/math/shapes/Ellipse.js","src/core/math/shapes/Polygon.js","src/core/math/shapes/Rectangle.js","src/core/math/shapes/RoundedRectangle.js","src/core/primitives/Graphics.js","src/core/primitives/GraphicsData.js","src/core/renderers/canvas/CanvasRenderer.js","src/core/renderers/canvas/utils/CanvasBuffer.js","src/core/renderers/canvas/utils/CanvasGraphics.js","src/core/renderers/canvas/utils/CanvasMaskManager.js","src/core/renderers/canvas/utils/CanvasTinter.js","src/core/renderers/webgl/WebGLRenderer.js","src/core/renderers/webgl/managers/WebGLBlendModeManager.js","src/core/renderers/webgl/managers/WebGLFilterManager.js","src/core/renderers/webgl/managers/WebGLManager.js","src/core/renderers/webgl/managers/WebGLMaskManager.js","src/core/renderers/webgl/managers/WebGLShaderManager.js","src/core/renderers/webgl/managers/WebGLStencilManager.js","src/core/renderers/webgl/shaders/ComplexPrimitiveShader.js","src/core/renderers/webgl/shaders/FastShader.js","src/core/renderers/webgl/shaders/PrimitiveShader.js","src/core/renderers/webgl/shaders/Shader.js","src/core/renderers/webgl/shaders/StripShader.js","src/core/renderers/webgl/utils/FilterTexture.js","src/core/renderers/webgl/utils/WebGLFastSpriteBatch.js","src/core/renderers/webgl/utils/WebGLGraphics.js","src/core/renderers/webgl/utils/WebGLGraphicsData.js","src/core/renderers/webgl/utils/WebGLSpriteBatch.js","src/core/textures/BaseTexture.js","src/core/textures/RenderTexture.js","src/core/textures/Texture.js","src/core/textures/TextureUvs.js","src/core/textures/VideoBaseTexture.js","src/core/utils/EventData.js","src/core/utils/PolyK.js","src/core/utils/eventTarget.js","src/core/utils/index.js","src/extras/MovieClip.js","src/extras/Rope.js","src/extras/Strip.js","src/extras/TilingSprite.js","src/extras/index.js","src/filters/AbstractFilter.js","src/filters/AlphaMaskFilter.js","src/filters/AsciiFilter.js","src/filters/BlurFilter.js","src/filters/BlurXFilter.js","src/filters/BlurYFilter.js","src/filters/ColorMatrixFilter.js","src/filters/ColorStepFilter.js","src/filters/ConvolutionFilter.js","src/filters/CrossHatchFilter.js","src/filters/DisplacementFilter.js","src/filters/DotScreenFilter.js","src/filters/FilterBlock.js","src/filters/GrayFilter.js","src/filters/InvertFilter.js","src/filters/NoiseFilter.js","src/filters/NormalMapFilter.js","src/filters/PixelateFilter.js","src/filters/RGBSplitFilter.js","src/filters/SepiaFilter.js","src/filters/SmartBlurFilter.js","src/filters/TiltShiftFilter.js","src/filters/TiltShiftXFilter.js","src/filters/TiltShiftYFilter.js","src/filters/TwistFilter.js","src/filters/index.js","src/interaction/InteractionData.js","src/interaction/InteractionManager.js","src/interaction/index.js","src/loaders/AtlasLoader.js","src/loaders/BitmapFontLoader.js","src/loaders/ImageLoader.js","src/loaders/JsonLoader.js","src/loaders/SpineLoader.js","src/loaders/SpineTextureLoader.js","src/loaders/SpriteSheetLoader.js","src/loaders/index.js","src/spine/Spine.js","src/spine/SpineRuntime.js","src/spine/index.js","src/text/BitmapText.js","src/text/Text.js","src/text/index.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/YA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5kBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3iCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/iBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1xBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3mBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClkFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7dA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","module.exports = {\n    core:           require('./core'),\n    extras:         require('./extras'),\n    filters:        require('./filters'),\n    interaction:    require('./interaction'),\n    loaders:        require('./loaders'),\n    spine:          require('./spine'),\n    text:           require('./text')\n};\n","module.exports = function webglEnabled() {\n  try {\n    var canvas = document.createElement('canvas');\n    return !!(window.WebGLRenderingContext && (canvas.getContext('webgl') || canvas.getContext('experimental-webgl')));\n  } catch (e) {\n    return false;\n  }\n};\n","module.exports={\n  \"name\": \"pixi.js\",\n  \"version\": \"3.0.0\",\n  \"description\": \"Pixi.js is a fast lightweight 2D library that works across all devices.\",\n  \"author\": \"Mat Groves\",\n  \"contributors\": [\n    \"Chad Engler <chad@pantherdev.com>\",\n    \"Richard Davey <rdavey@gmail.com>\"\n  ],\n  \"main\": \"./src/index.js\",\n  \"homepage\": \"http://goodboydigital.com/\",\n  \"bugs\": \"https://github.com/GoodBoyDigital/pixi.js/issues\",\n  \"license\": \"MIT\",\n  \"repository\": {\n    \"type\": \"git\",\n    \"url\": \"https://github.com/GoodBoyDigital/pixi.js.git\"\n  },\n  \"scripts\": {\n    \"test\": \"gulp test\",\n    \"docs\": \"./node_modules/.bin/jsdoc -c ./gulp/util/jsdoc.conf.json\"\n  },\n  \"devDependencies\": {\n    \"browserify\": \"^8.0.2\",\n    \"chai\": \"^1.10.0\",\n    \"del\": \"^1.1.0\",\n    \"gulp\": \"^3.8.10\",\n    \"gulp-jshint\": \"^1.9.0\",\n    \"gulp-plumber\": \"^0.6.6\",\n    \"gulp-rename\": \"^1.2.0\",\n    \"gulp-uglify\": \"^1.0.2\",\n    \"gulp-util\": \"^3.0.1\",\n    \"ink-docstrap\": \"^0.4.12\",\n    \"jsdoc\": \"^3.3.0-alpha13\",\n    \"jshint-summary\": \"^0.4.0\",\n    \"karma\": \"^0.12.28\",\n    \"karma-firefox-launcher\": \"^0.1.0\",\n    \"karma-mocha\": \"^0.1.10\",\n    \"karma-spec-reporter\": \"^0.0.16\",\n    \"mocha\": \"^2.1.0\",\n    \"require-dir\": \"^0.1.0\",\n    \"run-sequence\": \"^1.0.2\",\n    \"vinyl-buffer\": \"^1.0.0\",\n    \"vinyl-source-stream\": \"^1.0.0\",\n    \"watchify\": \"^2.2.1\"\n  },\n  \"dependencies\": {\n    \"webgl-enabled\": \"^1.0.2\"\n  }\n}\n","/**\n * Constant values used in pixi\n *\n * @mixin const\n */\nmodule.exports = {\n    /**\n     * Constant to identify the WEBGL Renderer Type\n     *\n     * @static\n     * @constant\n     * @property {number} WEBGL_RENDERER\n     */\n    WEBGL_RENDERER: 1,\n\n    /**\n     * Constant to identify the CANVAS Renderer Type\n     *\n     * @static\n     * @constant\n     * @property {number} CANVAS_RENDERER\n     */\n    CANVAS_RENDERER: 2,\n\n    /**\n     * String of the current PIXI version\n     *\n     * @static\n     * @constant\n     * @property {string} VERSION\n     */\n    VERSION: require('../../package.json').version,\n\n    /**\n     * Various blend modes supported by PIXI. IMPORTANT - The WebGL renderer only supports\n     * the NORMAL, ADD, MULTIPLY and SCREEN blend modes. Anything else will silently act like\n     * NORMAL.\n     *\n     * @static\n     * @constant\n     * @property {object} blendModes\n     * @property {number} blendModes.NORMAL\n     * @property {number} blendModes.ADD\n     * @property {number} blendModes.MULTIPLY\n     * @property {number} blendModes.SCREEN\n     * @property {number} blendModes.OVERLAY\n     * @property {number} blendModes.DARKEN\n     * @property {number} blendModes.LIGHTEN\n     * @property {number} blendModes.COLOR_DODGE\n     * @property {number} blendModes.COLOR_BURN\n     * @property {number} blendModes.HARD_LIGHT\n     * @property {number} blendModes.SOFT_LIGHT\n     * @property {number} blendModes.DIFFERENCE\n     * @property {number} blendModes.EXCLUSION\n     * @property {number} blendModes.HUE\n     * @property {number} blendModes.SATURATION\n     * @property {number} blendModes.COLOR\n     * @property {number} blendModes.LUMINOSITY\n     */\n    blendModes: {\n        NORMAL:         0,\n        ADD:            1,\n        MULTIPLY:       2,\n        SCREEN:         3,\n        OVERLAY:        4,\n        DARKEN:         5,\n        LIGHTEN:        6,\n        COLOR_DODGE:    7,\n        COLOR_BURN:     8,\n        HARD_LIGHT:     9,\n        SOFT_LIGHT:     10,\n        DIFFERENCE:     11,\n        EXCLUSION:      12,\n        HUE:            13,\n        SATURATION:     14,\n        COLOR:          15,\n        LUMINOSITY:     16\n    },\n\n    /**\n     * The scale modes that are supported by pixi.\n     *\n     * The DEFAULT scale mode affects the default scaling mode of future operations.\n     * It can be re-assigned to either LINEAR or NEAREST, depending upon suitability.\n     *\n     * @static\n     * @constant\n     * @property {object} scaleModes\n     * @property {number} scaleModes.DEFAULT=LINEAR\n     * @property {number} scaleModes.LINEAR Smooth scaling\n     * @property {number} scaleModes.NEAREST Pixelating scaling\n     */\n    scaleModes: {\n        DEFAULT:    0,\n        LINEAR:     0,\n        NEAREST:    1\n    },\n\n    /**\n     * The prefix that denotes a URL is for a retina asset\n     *\n     * @static\n     * @constant\n     * @property {string} RETINA_PREFIX\n     */\n    RETINA_PREFIX: '@2x',\n\n    /**\n     * The default render options if none are supplied to {@link PIXI.WebGLRenderer}\n     * or {@link PIXI.CanvasRenderer}.\n     *\n     * @static\n     * @constant\n     * @property {object} defaultRenderOptions\n     * @property {HTMLCanvasElement} defaultRenderOptions.view=null\n     * @property {boolean} defaultRenderOptions.transparent=false\n     * @property {boolean} defaultRenderOptions.antialias=false\n     * @property {boolean} defaultRenderOptions.preserveDrawingBuffer=false\n     * @property {number} defaultRenderOptions.resolution=1\n     * @property {number} defaultRenderOptions.backgroundColor=0x000000\n     * @property {boolean} defaultRenderOptions.clearBeforeRender=true\n     * @property {boolean} defaultRenderOptions.autoResize=false\n     */\n    defaultRenderOptions: {\n        view: null,\n        resolution: 1,\n        antialias: false,\n        autoResize: false,\n        transparent: false,\n        backgroundColor: 0x000000,\n        clearBeforeRender: true,\n        preserveDrawingBuffer: false\n    },\n\n    /**\n     * Constants that identify shapes, mainly to prevent `instanceof` calls.\n     *\n     * @static\n     * @constant\n     * @property {object} SHAPES\n     * @property {object} SHAPES.POLY=0\n     * @property {object} SHAPES.RECT=1\n     * @property {object} SHAPES.CIRC=2\n     * @property {object} SHAPES.ELIP=3\n     * @property {object} SHAPES.RREC=4\n     */\n    SHAPES: {\n        POLY: 0,\n        RECT: 1,\n        CIRC: 2,\n        ELIP: 3,\n        RREC: 4\n    }\n};\n","var math = require('../math');\n\n/**\n * The base class for all objects that are rendered on the screen.\n * This is an abstract class and should not be used on its own rather it should be extended.\n *\n * @class\n * @namespace PIXI\n */\nfunction DisplayObject() {\n    /**\n     * The coordinate of the object relative to the local coordinates of the parent.\n     *\n     * @member {Point}\n     */\n    this.position = new math.Point();\n\n    /**\n     * The scale factor of the object.\n     *\n     * @member {Point}\n     */\n    this.scale = new math.Point(1, 1);\n\n    /**\n     * The pivot point of the displayObject that it rotates around\n     *\n     * @member {Point}\n     */\n    this.pivot = new math.Point(0, 0);\n\n    /**\n     * The rotation of the object in radians.\n     *\n     * @member {number}\n     */\n    this.rotation = 0;\n\n    /**\n     * The opacity of the object.\n     *\n     * @member {number}\n     */\n    this.alpha = 1;\n\n    /**\n     * The visibility of the object. If false the object will not be drawn, and\n     * the updateTransform function will not be called.\n     *\n     * @member {boolean}\n     */\n    this.visible = true;\n\n    /**\n     * Can this object be rendered, if false the object will not be drawn but the updateTransform\n     * methods will still be called.\n     *\n     * @member {boolean}\n     */\n    this.renderable = false;\n\n    /**\n     * The display object container that contains this display object.\n     *\n     * @member {DisplayObjectContainer}\n     * @readOnly\n     */\n    this.parent = null;\n\n    /**\n     * The multiplied alpha of the displayObject\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.worldAlpha = 1;\n\n    /**\n     * Current transform of the object based on world (parent) factors\n     *\n     * @member {Matrix}\n     * @readOnly\n     */\n    this.worldTransform = new math.Matrix();\n\n    /**\n     * The area the filter is applied to. This is used as more of an optimisation\n     * rather than figuring out the dimensions of the displayObject each frame you can set this rectangle\n     *\n     * @member {Rectangle}\n     */\n    this.filterArea = null;\n\n    /**\n     * cached sin rotation\n     *\n     * @member {number}\n     * @private\n     */\n    this._sr = 0;\n\n    /**\n     * cached cos rotation\n     *\n     * @member {number}\n     * @private\n     */\n    this._cr = 1;\n\n    /**\n     * The original, cached bounds of the object\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._bounds = new math.Rectangle(0, 0, 1, 1);\n\n    /**\n     * The most up-to-date bounds of the object\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._currentBounds = null;\n\n    /**\n     * The original, cached mask of the object\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._mask = null;\n\n    /**\n     * Cached internal flag.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this._cacheIsDirty = false;\n}\n\n// constructor\nDisplayObject.prototype.constructor = DisplayObject;\nmodule.exports = DisplayObject;\n\nObject.defineProperties(DisplayObject.prototype, {\n    /**\n     * The position of the displayObject on the x axis relative to the local coordinates of the parent.\n     *\n     * @member {number}\n     * @memberof DisplayObject#\n     */\n    x: {\n        get: function () {\n            return this.position.x;\n        },\n        set: function (value) {\n            this.position.x = value;\n        }\n    },\n\n    /**\n     * The position of the displayObject on the y axis relative to the local coordinates of the parent.\n     *\n     * @member {number}\n     * @memberof DisplayObject#\n     */\n    y: {\n        get: function () {\n            return this.position.y;\n        },\n        set: function (value) {\n            this.position.y = value;\n        }\n    },\n\n    /**\n     * Indicates if the sprite is globally visible.\n     *\n     * @member {boolean}\n     * @memberof DisplayObject#\n     * @readonly\n     */\n    worldVisible: {\n        get: function () {\n            var item = this;\n\n            do {\n                if (!item.visible) {\n                    return false;\n                }\n\n                item = item.parent;\n            } while(item);\n\n            return true;\n        }\n    },\n\n    /**\n     * Sets a mask for the displayObject. A mask is an object that limits the visibility of an object to the shape of the mask applied to it.\n     * In PIXI a regular mask must be a PIXI.Graphics object. This allows for much faster masking in canvas as it utilises shape clipping.\n     * To remove a mask, set this property to null.\n     *\n     * @member {Graphics}\n     * @memberof DisplayObject#\n     */\n    mask: {\n        get: function () {\n            return this._mask;\n        },\n        set: function (value) {\n            if (this._mask) {\n                this._mask.isMask = false;\n            }\n\n            this._mask = value;\n\n            if (this._mask) {\n                this._mask.isMask = true;\n            }\n        }\n    },\n\n    /**\n     * Sets the filters for the displayObject.\n     * * IMPORTANT: This is a webGL only feature and will be ignored by the canvas renderer.\n     * To remove filters simply set this property to 'null'\n     *\n     * @member {Filter[]}\n     * @memberof DisplayObject#\n     */\n    filters: {\n        get: function () {\n            return this._filters;\n        },\n        set: function (value) {\n            if (value) {\n                // now put all the passes in one place..\n                var passes = [];\n\n                for (var i = 0; i < value.length; i++) {\n                    var filterPasses = value[i].passes;\n\n                    for (var j = 0; j < filterPasses.length; j++) {\n                        passes.push(filterPasses[j]);\n                    }\n                }\n\n                // TODO change this as it is legacy\n                this._filterBlock = { target: this, filterPasses: passes };\n            }\n\n            this._filters = value;\n        }\n    }\n});\n\n/*\n * Updates the object transform for rendering\n *\n * TODO - Optimization pass!\n *\n * @private\n */\nDisplayObject.prototype.updateTransform = function () {\n    if (!this.parent) {\n        return;\n    }\n\n    // create some matrix refs for easy access\n    var pt = this.parent.worldTransform;\n    var wt = this.worldTransform;\n\n    // temporary matrix variables\n    var a, b, c, d, tx, ty;\n\n    // so if rotation is between 0 then we can simplify the multiplication process..\n    if (this.rotation % math.PI_2) {\n        // check to see if the rotation is the same as the previous render. This means we only need to use sin and cos when rotation actually changes\n        if (this.rotation !== this.rotationCache) {\n            this.rotationCache = this.rotation;\n            this._sr = Math.sin(this.rotation);\n            this._cr = Math.cos(this.rotation);\n        }\n\n        // get the matrix values of the displayobject based on its transform properties..\n        a  =  this._cr * this.scale.x;\n        b  =  this._sr * this.scale.x;\n        c  = -this._sr * this.scale.y;\n        d  =  this._cr * this.scale.y;\n        tx =  this.position.x;\n        ty =  this.position.y;\n\n        // check for pivot.. not often used so geared towards that fact!\n        if (this.pivot.x || this.pivot.y) {\n            tx -= this.pivot.x * a + this.pivot.y * c;\n            ty -= this.pivot.x * b + this.pivot.y * d;\n        }\n\n        // concat the parent matrix with the objects transform.\n        wt.a  = a  * pt.a + b  * pt.c;\n        wt.b  = a  * pt.b + b  * pt.d;\n        wt.c  = c  * pt.a + d  * pt.c;\n        wt.d  = c  * pt.b + d  * pt.d;\n        wt.tx = tx * pt.a + ty * pt.c + pt.tx;\n        wt.ty = tx * pt.b + ty * pt.d + pt.ty;\n    }\n    else {\n        // lets do the fast version as we know there is no rotation..\n        a  = this.scale.x;\n        d  = this.scale.y;\n\n        tx = this.position.x - this.pivot.x * a;\n        ty = this.position.y - this.pivot.y * d;\n\n        wt.a  = a  * pt.a;\n        wt.b  = a  * pt.b;\n        wt.c  = d  * pt.c;\n        wt.d  = d  * pt.d;\n        wt.tx = tx * pt.a + ty * pt.c + pt.tx;\n        wt.ty = tx * pt.b + ty * pt.d + pt.ty;\n    }\n\n    // multiply the alphas..\n    this.worldAlpha = this.alpha * this.parent.worldAlpha;\n};\n\n// performance increase to avoid using call.. (10x faster)\nDisplayObject.prototype.displayObjectUpdateTransform = DisplayObject.prototype.updateTransform;\n\n/**\n * Retrieves the bounds of the displayObject as a rectangle object\n *\n * @param matrix {Matrix}\n * @return {Rectangle} the rectangular bounding area\n */\nDisplayObject.prototype.getBounds = function (/* matrix */) {\n    return math.Rectangle.EMPTY;\n};\n\n/**\n * Retrieves the local bounds of the displayObject as a rectangle object\n *\n * @return {Rectangle} the rectangular bounding area\n */\nDisplayObject.prototype.getLocalBounds = function () {\n    return this.getBounds(math.Matrix.IDENTITY);\n};\n\n/**\n * Calculates the global position of the display object\n *\n * @param position {Point} The world origin to calculate from\n * @return {Point} A point object representing the position of this object\n */\nDisplayObject.prototype.toGlobal = function (position) {\n    // don't need to u[date the lot\n    this.displayObjectUpdateTransform();\n    return this.worldTransform.apply(position);\n};\n\n/**\n * Calculates the local position of the display object relative to another point\n *\n * @param position {Point} The world origin to calculate from\n * @param [from] {DisplayObject} The DisplayObject to calculate the global position from\n * @return {Point} A point object representing the position of this object\n */\nDisplayObject.prototype.toLocal = function (position, from) {\n    if (from) {\n        position = from.toGlobal(position);\n    }\n\n    // don't need to update the lot\n    this.displayObjectUpdateTransform();\n    return this.worldTransform.applyInverse(position);\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer} The renderer\n * @private\n */\nDisplayObject.prototype.renderWebGL = function (/* renderer */) {\n    // OVERWRITE;\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer} The renderer\n * @private\n */\nDisplayObject.prototype.renderCanvas = function (/* renderer */) {\n    // OVERWRITE;\n};\n","var math = require('../math'),\n    DisplayObject = require('./DisplayObject'),\n    RenderTexture = require('../textures/RenderTexture'),\n    // Sprite = require('./Sprite'),\n    _tempMatrix = new math.Matrix();\n\n/**\n * A DisplayObjectContainer represents a collection of display objects.\n * It is the base class of all display objects that act as a container for other objects.\n *\n * @class\n * @extends DisplayObject\n * @namespace PIXI\n */\nfunction DisplayObjectContainer() {\n    DisplayObject.call(this);\n\n    /**\n     * The array of children of this container.\n     *\n     * @member {DisplayObject[]}\n     * @readonly\n     */\n    this.children = [];\n\n    /**\n     * Cached internal flag.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this._cacheAsBitmap = false;\n\n    this._cachedSprite = null;\n}\n\n// constructor\nDisplayObjectContainer.prototype = Object.create(DisplayObject.prototype);\nDisplayObjectContainer.prototype.constructor = DisplayObjectContainer;\nmodule.exports = DisplayObjectContainer;\n\nObject.defineProperties(DisplayObjectContainer.prototype, {\n    /**\n     * The width of the displayObjectContainer, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof DisplayObjectContainer#\n     */\n    width: {\n        get: function () {\n            return this.scale.x * this.getLocalBounds().width;\n        },\n        set: function (value) {\n\n            var width = this.getLocalBounds().width;\n\n            if(width !== 0) {\n                this.scale.x = value / width;\n            }\n            else {\n                this.scale.x = 1;\n            }\n\n\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the displayObjectContainer, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof DisplayObjectContainer#\n     */\n    height: {\n        get: function () {\n            return  this.scale.y * this.getLocalBounds().height;\n        },\n        set: function (value) {\n\n            var height = this.getLocalBounds().height;\n\n            if (height !== 0) {\n                this.scale.y = value / height ;\n            }\n            else {\n                this.scale.y = 1;\n            }\n\n            this._height = value;\n        }\n    },\n\n    /**\n     * Set if this display object is cached as a bitmap.\n     * This basically takes a snap shot of the display object as it is at that moment. It can provide a performance benefit for complex static displayObjects.\n     * To remove simply set this property to 'null'\n     *\n     * @member {boolean}\n     * @memberof DisplayObject#\n     */\n    cacheAsBitmap: {\n        get: function () {\n            return this._cacheAsBitmap;\n        },\n        set: function (value) {\n            if (this._cacheAsBitmap === value) {\n                return;\n            }\n\n            if (value) {\n                this._generateCachedSprite();\n            }\n            else {\n                this._destroyCachedSprite();\n            }\n\n            this._cacheAsBitmap = value;\n        }\n    }\n});\n\n/**\n * Adds a child to the container.\n *\n * @param child {DisplayObject} The DisplayObject to add to the container\n * @return {DisplayObject} The child that was added.\n */\nDisplayObjectContainer.prototype.addChild = function (child) {\n    return this.addChildAt(child, this.children.length);\n};\n\n/**\n * Adds a child to the container at a specified index. If the index is out of bounds an error will be thrown\n *\n * @param child {DisplayObject} The child to add\n * @param index {Number} The index to place the child in\n * @return {DisplayObject} The child that was added.\n */\nDisplayObjectContainer.prototype.addChildAt = function (child, index) {\n    // prevent adding self as child\n    if (child === this) {\n        return;\n    }\n\n    if (index >= 0 && index <= this.children.length) {\n        if (child.parent) {\n            child.parent.removeChild(child);\n        }\n\n        child.parent = this;\n\n        this.children.splice(index, 0, child);\n\n        return child;\n    }\n    else {\n        throw new Error(child + 'addChildAt: The index '+ index +' supplied is out of bounds ' + this.children.length);\n    }\n};\n\n/**\n * Swaps the position of 2 Display Objects within this container.\n *\n * @param child {DisplayObject}\n * @param child2 {DisplayObject}\n */\nDisplayObjectContainer.prototype.swapChildren = function (child, child2) {\n    if (child === child2) {\n        return;\n    }\n\n    var index1 = this.getChildIndex(child);\n    var index2 = this.getChildIndex(child2);\n\n    if (index1 < 0 || index2 < 0) {\n        throw new Error('swapChildren: Both the supplied DisplayObjects must be a child of the caller.');\n    }\n\n    this.children[index1] = child2;\n    this.children[index2] = child;\n};\n\n/**\n * Returns the index position of a child DisplayObject instance\n *\n * @param child {DisplayObject} The DisplayObject instance to identify\n * @return {Number} The index position of the child display object to identify\n */\nDisplayObjectContainer.prototype.getChildIndex = function (child) {\n    var index = this.children.indexOf(child);\n\n    if (index === -1) {\n        throw new Error('The supplied DisplayObject must be a child of the caller');\n    }\n\n    return index;\n};\n\n/**\n * Changes the position of an existing child in the display object container\n *\n * @param child {DisplayObject} The child DisplayObject instance for which you want to change the index number\n * @param index {Number} The resulting index number for the child display object\n */\nDisplayObjectContainer.prototype.setChildIndex = function (child, index) {\n    if (index < 0 || index >= this.children.length) {\n        throw new Error('The supplied index is out of bounds');\n    }\n\n    var currentIndex = this.getChildIndex(child);\n\n    this.children.splice(currentIndex, 1); //remove from old position\n    this.children.splice(index, 0, child); //add at new position\n};\n\n/**\n * Returns the child at the specified index\n *\n * @param index {Number} The index to get the child from\n * @return {DisplayObject} The child at the given index, if any.\n */\nDisplayObjectContainer.prototype.getChildAt = function (index) {\n    if (index < 0 || index >= this.children.length) {\n        throw new Error('getChildAt: Supplied index ' + index + ' does not exist in the child list, or the supplied DisplayObject must be a child of the caller');\n    }\n\n    return this.children[index];\n};\n\n/**\n * Removes a child from the container.\n *\n * @param child {DisplayObject} The DisplayObject to remove\n * @return {DisplayObject} The child that was removed.\n */\nDisplayObjectContainer.prototype.removeChild = function (child) {\n    var index = this.children.indexOf(child);\n\n    if (index === -1) {\n        return;\n    }\n\n    return this.removeChildAt(index);\n};\n\n/**\n * Removes a child from the specified index position.\n *\n * @param index {Number} The index to get the child from\n * @return {DisplayObject} The child that was removed.\n */\nDisplayObjectContainer.prototype.removeChildAt = function (index) {\n    var child = this.getChildAt(index);\n\n    child.parent = null;\n    this.children.splice(index, 1);\n\n    return child;\n};\n\n/**\n * Removes all children from this container that are within the begin and end indexes.\n *\n * @param beginIndex {Number} The beginning position. Default value is 0.\n * @param endIndex {Number} The ending position. Default value is size of the container.\n */\nDisplayObjectContainer.prototype.removeChildren = function (beginIndex, endIndex) {\n    var begin = beginIndex || 0;\n    var end = typeof endIndex === 'number' ? endIndex : this.children.length;\n    var range = end - begin;\n\n    if (range > 0 && range <= end) {\n        var removed = this.children.splice(begin, range);\n\n        for (var i = 0; i < removed.length; ++i) {\n            removed[i].parent = null;\n        }\n\n        return removed;\n    }\n    else if (range === 0 && this.children.length === 0) {\n        return [];\n    }\n    else {\n        throw new RangeError('removeChildren: numeric values are outside the acceptable range.');\n    }\n};\n\n/**\n * Generates and updates the cached sprite for this object.\n *\n */\nDisplayObjectContainer.prototype.updateCachedSprite = function () {\n    this._generateCachedSprite();\n};\n\n/**\n * Useful function that returns a texture of the displayObject object that can then be used to create sprites\n * This can be quite useful if your displayObject is static / complicated and needs to be reused multiple times.\n *\n * @param resolution {Number} The resolution of the texture being generated\n * @param scaleMode {Number} See {{#crossLink \"PIXI/scaleModes:property\"}}PIXI.scaleModes{{/crossLink}} for possible values\n * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used to generate the texture.\n * @return {Texture} a texture of the graphics object\n */\nDisplayObjectContainer.prototype.generateTexture = function (resolution, scaleMode, renderer) {\n    var bounds = this.getLocalBounds();\n\n    var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0, renderer, scaleMode, resolution);\n\n    _tempMatrix.tx = -bounds.x;\n    _tempMatrix.ty = -bounds.y;\n\n    renderTexture.render(this, _tempMatrix);\n\n    return renderTexture;\n};\n\n/*\n * Updates the transform on all children of this container for rendering\n *\n * @private\n */\nDisplayObjectContainer.prototype.updateTransform = function () {\n    if (!this.visible) {\n        return;\n    }\n\n    this.displayObjectUpdateTransform();\n\n    if (this._cacheAsBitmap) {\n        return;\n    }\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].updateTransform();\n    }\n};\n\n// performance increase to avoid using call.. (10x faster)\nDisplayObjectContainer.prototype.displayObjectContainerUpdateTransform = DisplayObjectContainer.prototype.updateTransform;\n\n/**\n * Retrieves the bounds of the displayObjectContainer as a rectangle. The bounds calculation takes all visible children into consideration.\n *\n * @return {Rectangle} The rectangular bounding area\n */\nDisplayObjectContainer.prototype.getBounds = function () {\n    if (this.children.length === 0) {\n        return math.Rectangle.EMPTY;\n    }\n\n    // TODO the bounds have already been calculated this render session so return what we have\n\n    var minX = Infinity;\n    var minY = Infinity;\n\n    var maxX = -Infinity;\n    var maxY = -Infinity;\n\n    var childBounds;\n    var childMaxX;\n    var childMaxY;\n\n    var childVisible = false;\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        var child = this.children[i];\n\n        if (!child.visible) {\n            continue;\n        }\n\n        childVisible = true;\n\n        childBounds = this.children[i].getBounds();\n\n        minX = minX < childBounds.x ? minX : childBounds.x;\n        minY = minY < childBounds.y ? minY : childBounds.y;\n\n        childMaxX = childBounds.width + childBounds.x;\n        childMaxY = childBounds.height + childBounds.y;\n\n        maxX = maxX > childMaxX ? maxX : childMaxX;\n        maxY = maxY > childMaxY ? maxY : childMaxY;\n    }\n\n    if (!childVisible) {\n        return math.Rectangle.EMPTY;\n    }\n\n    this._bounds.x = minX;\n    this._bounds.y = minY;\n    this._bounds.width = maxX - minX;\n    this._bounds.height = maxY - minY;\n\n    // TODO: store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    //this._currentBounds = bounds;\n\n    return this._bounds;\n};\n\n/**\n * Retrieves the non-global local bounds of the displayObjectContainer as a rectangle.\n * The calculation takes all visible children into consideration.\n *\n * @return {Rectangle} The rectangular bounding area\n */\nDisplayObjectContainer.prototype.getLocalBounds = function () {\n    var matrixCache = this.worldTransform;\n\n    this.worldTransform = math.Matrix.IDENTITY;\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].updateTransform();\n    }\n\n    this.worldTransform = matrixCache;\n\n    return this.getBounds();\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * TODO - Optimization pass!\n *\n * @param renderer {WebGLRenderer} The renderer\n */\nDisplayObjectContainer.prototype.renderWebGL = function (renderer) {\n    // if the object is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        this._renderCachedSprite(renderer);\n        return;\n    }\n\n    var i, j;\n\n    // do a quick check to see if this element has a mask or a filter.\n    if (this._mask || this._filters) {\n        // push filter first as we need to ensure the stencil buffer is correct for any masking\n        if (this._filters) {\n            renderer.spriteBatch.flush();\n            renderer.filterManager.pushFilter(this._filterBlock);\n        }\n\n        if (this._mask) {\n            renderer.spriteBatch.stop();\n            renderer.maskManager.pushMask(this.mask, renderer);\n            renderer.spriteBatch.start();\n        }\n\n        // add this object to the batch, only rendered if it has a texture.\n        if (this.texture) {\n            renderer.spriteBatch.render(this);\n        }\n\n        // now loop through the children and make sure they get rendered\n        for (i = 0, j = this.children.length; i < j; i++) {\n            this.children[i].renderWebGL(renderer);\n        }\n\n        // time to stop the sprite batch as either a mask element or a filter draw will happen next\n        renderer.spriteBatch.stop();\n\n        if (this._mask) {\n            renderer.maskManager.popMask(this._mask, renderer);\n        }\n\n        if (this._filters) {\n            renderer.filterManager.popFilter();\n        }\n\n        renderer.spriteBatch.start();\n    }\n    else {\n        if (this.texture) {\n            renderer.spriteBatch.render(this);\n        }\n\n        // simple render children!\n        for (i = 0, j = this.children.length; i < j; ++i) {\n            this.children[i].renderWebGL(renderer);\n        }\n\n    }\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer} The renderer\n */\nDisplayObjectContainer.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        this._renderCachedSprite(renderer);\n        return;\n    }\n\n    if (this._mask) {\n        renderer.maskManager.pushMask(this._mask, renderer);\n    }\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].renderCanvas(renderer);\n    }\n\n    if (this._mask) {\n        renderer.maskManager.popMask(renderer);\n    }\n};\n\n/**\n * Internal method.\n *\n * @param renderer {WebGLRenderer|CanvasRenderer} The renderer\n * @private\n */\nDisplayObjectContainer.prototype._renderCachedSprite = function (renderer) {\n    this._cachedSprite.worldAlpha = this.worldAlpha;\n\n    if (renderer.gl) {\n        this._cachedSprite.renderWebGL(renderer);\n    }\n    else {\n        this._cachedSprite.renderCanvas(renderer);\n    }\n};\n\n/**\n * Internal method.\n *\n * @private\n */\nDisplayObjectContainer.prototype._generateCachedSprite = function () {\n    var bounds = this.getLocalBounds();\n\n    if (!this._cachedSprite) {\n        // TODO - RenderTexture now *requires* a renderer instance, so this is like broken\n        // because `renderer` isn't actually in scope here :P\n        var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0);\n\n        this._cachedSprite = new Sprite(renderTexture);\n        this._cachedSprite.worldTransform = this.worldTransform;\n    }\n    else {\n        this._cachedSprite.texture.resize(bounds.width | 0, bounds.height | 0);\n    }\n\n    var tempFilters = this._filters;\n    this._filters = null;\n\n    this._cachedSprite.filters = tempFilters;\n\n    _tempMatrix.tx = -bounds.x;\n    _tempMatrix.ty = -bounds.y;\n\n    this._cachedSprite.texture.render(this, _tempMatrix, true);\n\n    this._cachedSprite.anchor.x = -(bounds.x / bounds.width);\n    this._cachedSprite.anchor.y = -(bounds.y / bounds.height);\n\n    this._filters = tempFilters;\n};\n\n/**\n * Destroys the cached sprite.\n *\n * @private\n */\nDisplayObjectContainer.prototype._destroyCachedSprite = function () {\n    if (!this._cachedSprite) {\n        return;\n    }\n\n    // TODO: Pool this sprite\n    this._cachedSprite.destroy(true, true);\n    this._cachedSprite = null;\n};\n","var math = require('../math'),\n    Texture = require('../textures/Texture'),\n    DisplayObjectContainer = require('./DisplayObjectContainer'),\n    CanvasTinter = require('../renderers/canvas/utils/CanvasTinter'),\n    utils = require('../utils'),\n    CONST = require('../const');\n\n/**\n * The Sprite object is the base for all textured objects that are rendered to the screen\n *\n * A sprite can be created directly from an image like this:\n *\n * ```js\n * var sprite = new Sprite.fromImage('assets/image.png');\n * ```\n *\n * @class Sprite\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param texture {Texture} The texture for this sprite\n */\nfunction Sprite(texture) {\n    DisplayObjectContainer.call(this);\n\n    /**\n     * The anchor sets the origin point of the texture.\n     * The default is 0,0 this means the texture's origin is the top left\n     * Setting than anchor to 0.5,0.5 means the textures origin is centered\n     * Setting the anchor to 1,1 would mean the textures origin points will be the bottom right corner\n     *\n     * @member {Point}\n     */\n    this.anchor = new math.Point();\n\n    /**\n     * The texture that the sprite is using\n     *\n     * @member {Texture}\n     * @private\n     */\n    this._texture = null;\n\n    /**\n     * The width of the sprite (this is initially set by the texture)\n     *\n     * @member {number}\n     * @private\n     */\n    this._width = 0;\n\n    /**\n     * The height of the sprite (this is initially set by the texture)\n     *\n     * @member {number}\n     * @private\n     */\n    this._height = 0;\n\n    /**\n     * The tint applied to the sprite. This is a hex value. A value of 0xFFFFFF will remove any tint effect.\n     *\n     * @member {number}\n     * @default 0xFFFFFF\n     */\n    this.tint = 0xFFFFFF;\n\n    /**\n     * The blend mode to be applied to the sprite. Set to CONST.blendModes.NORMAL to remove any blend mode.\n     *\n     * @member {number}\n     * @default CONST.blendModes.NORMAL;\n     */\n    this.blendMode = CONST.blendModes.NORMAL;\n\n    /**\n     * The shader that will be used to render the sprite. Set to null to remove a current shader.\n     *\n     * @member {AbstractFilter}\n     */\n    this.shader = null;\n\n    this.renderable = true;\n\n    // call texture setter\n    this.texture = texture || Texture.EMPTY;\n}\n\nSprite.prototype.destroy = function (destroyTexture, destroyBaseTexture) {\n    DisplayObjectContainer.prototype.destroy.call(this);\n\n    this.anchor = null;\n\n    if (destroyTexture) {\n        this._texture.destroy(destroyBaseTexture);\n    }\n\n    this._texture = null;\n    this.shader = null;\n};\n\n// constructor\nSprite.prototype = Object.create(DisplayObjectContainer.prototype);\nSprite.prototype.constructor = Sprite;\nmodule.exports = Sprite;\n\nObject.defineProperties(Sprite.prototype, {\n    /**\n     * The width of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member\n     * @memberof Sprite#\n     */\n    width: {\n        get: function () {\n            return this.scale.x * this.texture.frame.width;\n        },\n        set: function (value) {\n            this.scale.x = value / this.texture.frame.width;\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member\n     * @memberof Sprite#\n     */\n    height: {\n        get: function () {\n            return  this.scale.y * this.texture.frame.height;\n        },\n        set: function (value) {\n            this.scale.y = value / this.texture.frame.height;\n            this._height = value;\n        }\n    },\n\n    /**\n     * The height of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member\n     * @memberof Sprite#\n     */\n    texture: {\n        get: function () {\n            return  this._texture;\n        },\n        set: function (value) {\n            if (this._texture === value) {\n                return;\n            }\n\n            this._texture = value;\n            this.cachedTint = 0xFFFFFF;\n\n            if (value) {\n                // wait for the texture to load\n                if (value.baseTexture.hasLoaded) {\n                    this._onTextureUpdate();\n                }\n                else {\n                    value.once('update', this._onTextureUpdate.bind(this));\n                }\n            }\n        }\n    },\n});\n\n/**\n * When the texture is updated, this event will fire to update the scale and frame\n *\n * @private\n */\nSprite.prototype._onTextureUpdate = function () {\n    // so if _width is 0 then width was not set..\n    if (this._width) {\n        this.scale.x = this._width / this.texture.frame.width;\n    }\n\n    if (this._height) {\n        this.scale.y = this._height / this.texture.frame.height;\n    }\n};\n\n/**\n * Returns the bounds of the Sprite as a rectangle. The bounds calculation takes the worldTransform into account.\n *\n * @param matrix {Matrix} the transformation matrix of the sprite\n * @return {Rectangle} the framing rectangle\n */\nSprite.prototype.getBounds = function (matrix) {\n    var width = this.texture.frame.width;\n    var height = this.texture.frame.height;\n\n    var w0 = width * (1-this.anchor.x);\n    var w1 = width * -this.anchor.x;\n\n    var h0 = height * (1-this.anchor.y);\n    var h1 = height * -this.anchor.y;\n\n    var worldTransform = matrix || this.worldTransform ;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var minX,\n        maxX,\n        minY,\n        maxY;\n\n    if(b === 0 && c === 0)\n    {\n        // scale may be negative!\n        if (a < 0) {\n            a *= -1;\n        }\n\n        if (d < 0) {\n            d *= -1;\n        }\n\n        // this means there is no rotation going on right? RIGHT?\n        // if thats the case then we can avoid checking the bound values! yay\n        minX = a * w1 + tx;\n        maxX = a * w0 + tx;\n        minY = d * h1 + ty;\n        maxY = d * h0 + ty;\n    }\n    else\n    {\n        var x1 = a * w1 + c * h1 + tx;\n        var y1 = d * h1 + b * w1 + ty;\n\n        var x2 = a * w0 + c * h1 + tx;\n        var y2 = d * h1 + b * w0 + ty;\n\n        var x3 = a * w0 + c * h0 + tx;\n        var y3 = d * h0 + b * w0 + ty;\n\n        var x4 =  a * w1 + c * h0 + tx;\n        var y4 =  d * h0 + b * w1 + ty;\n\n        minX = x1;\n        minX = x2 < minX ? x2 : minX;\n        minX = x3 < minX ? x3 : minX;\n        minX = x4 < minX ? x4 : minX;\n\n        minY = y1;\n        minY = y2 < minY ? y2 : minY;\n        minY = y3 < minY ? y3 : minY;\n        minY = y4 < minY ? y4 : minY;\n\n        maxX = x1;\n        maxX = x2 > maxX ? x2 : maxX;\n        maxX = x3 > maxX ? x3 : maxX;\n        maxX = x4 > maxX ? x4 : maxX;\n\n        maxY = y1;\n        maxY = y2 > maxY ? y2 : maxY;\n        maxY = y3 > maxY ? y3 : maxY;\n        maxY = y4 > maxY ? y4 : maxY;\n    }\n\n    var bounds = this._bounds;\n\n    bounds.x = minX;\n    bounds.width = maxX - minX;\n\n    bounds.y = minY;\n    bounds.height = maxY - minY;\n\n    // store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    this._currentBounds = bounds;\n\n    return bounds;\n};\n\n/**\n* Renders the object using the Canvas renderer\n*\n* @param renderer {CanvasRenderer} The renderer\n*/\nSprite.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0 || this.texture.crop.width <= 0 || this.texture.crop.height <= 0) {\n        return;\n    }\n\n    if (this.blendMode !== renderer.currentBlendMode) {\n        renderer.currentBlendMode = this.blendMode;\n        renderer.context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode];\n    }\n\n    if (this._mask) {\n        renderer.maskManager.pushMask(this._mask, renderer);\n    }\n\n    //  Ignore null sources\n    if (this.texture.valid) {\n        var resolution = this.texture.baseTexture.resolution / renderer.resolution;\n\n        renderer.context.globalAlpha = this.worldAlpha;\n\n        // If smoothingEnabled is supported and we need to change the smoothing property for this texture\n        if (renderer.smoothProperty && renderer.scaleMode !== this.texture.baseTexture.scaleMode) {\n            renderer.scaleMode = this.texture.baseTexture.scaleMode;\n            renderer.context[renderer.smoothProperty] = (renderer.scaleMode === CONST.scaleModes.LINEAR);\n        }\n\n        // If the texture is trimmed we offset by the trim x/y, otherwise we use the frame dimensions\n        var dx = (this.texture.trim ? this.texture.trim.x : 0) - (this.anchor.x * this.texture.trim.width);\n        var dy = (this.texture.trim ? this.texture.trim.y : 0) - (this.anchor.y * this.texture.trim.height);\n\n        // Allow for pixel rounding\n        if (renderer.roundPixels) {\n            renderer.context.setTransform(\n                this.worldTransform.a,\n                this.worldTransform.b,\n                this.worldTransform.c,\n                this.worldTransform.d,\n                (this.worldTransform.tx * renderer.resolution) | 0,\n                (this.worldTransform.ty * renderer.resolution) | 0\n            );\n\n            dx = dx | 0;\n            dy = dy | 0;\n        }\n        else {\n            renderer.context.setTransform(\n                this.worldTransform.a,\n                this.worldTransform.b,\n                this.worldTransform.c,\n                this.worldTransform.d,\n                this.worldTransform.tx * renderer.resolution,\n                this.worldTransform.ty * renderer.resolution\n            );\n        }\n\n        if (this.tint !== 0xFFFFFF) {\n            if (this.cachedTint !== this.tint) {\n                this.cachedTint = this.tint;\n\n                // TODO clean up caching - how to clean up the caches?\n                this.tintedTexture = CanvasTinter.getTintedTexture(this, this.tint);\n            }\n\n            renderer.context.drawImage(\n                this.tintedTexture,\n                0,\n                0,\n                this.texture.crop.width,\n                this.texture.crop.height,\n                dx / resolution,\n                dy / resolution,\n                this.texture.crop.width / resolution,\n                this.texture.crop.height / resolution\n            );\n        }\n        else {\n            renderer.context.drawImage(\n                this.texture.baseTexture.source,\n                this.texture.crop.x,\n                this.texture.crop.y,\n                this.texture.crop.width,\n                this.texture.crop.height,\n                dx / resolution,\n                dy / resolution,\n                this.texture.crop.width / resolution,\n                this.texture.crop.height / resolution\n            );\n        }\n    }\n\n    for (var i = 0, j = this.children.length; i < j; i++) {\n        this.children[i].renderCanvas(renderer);\n    }\n\n    if (this._mask) {\n        renderer.maskManager.popMask(renderer);\n    }\n};\n\n// some helper functions..\n\n/**\n * Helper function that creates a sprite that will contain a texture from the TextureCache based on the frameId\n * The frame ids are created when a Texture packer file has been loaded\n *\n * @static\n * @param frameId {String} The frame Id of the texture in the cache\n * @return {Sprite} A new Sprite using a texture from the texture cache matching the frameId\n */\nSprite.fromFrame = function (frameId) {\n    var texture = utils.TextureCache[frameId];\n\n    if (!texture) {\n        throw new Error('The frameId \"' + frameId + '\" does not exist in the texture cache' + this);\n    }\n\n    return new Sprite(texture);\n};\n\n/**\n * Helper function that creates a sprite that will contain a texture based on an image url\n * If the image is not in the texture cache it will be loaded\n *\n * @static\n * @param imageId {String} The image url of the texture\n * @return {Sprite} A new Sprite using a texture from the texture cache matching the image id\n */\nSprite.fromImage = function (imageId, crossorigin, scaleMode) {\n    return new Sprite(Texture.fromImage(imageId, crossorigin, scaleMode));\n};\n","var DisplayObjectContainer = require('./DisplayObjectContainer'),\n    WebGLFastSpriteBatch = require('../renderers/webgl/utils/WebGLFastSpriteBatch');\n\n/**\n * The SpriteBatch class is a really fast version of the DisplayObjectContainer built solely for speed,\n * so use when you need a lot of sprites or particles. The tradeoff of the SpriteBatch is that advanced\n * functionality will not work. SpriteBatch implements only the basic object transform (position, scale, rotation).\n * Any other functionality like tinting, masking, etc will not work on sprites in this batch.\n *\n * It's extremely easy to use :\n *\n * ```js\n * var container = new SpriteBatch();\n *\n * for(var i = 0; i < 100; ++i) {\n *     var sprite = new PIXI.Sprite.fromImage(\"myImage.png\");\n *     container.addChild(sprite);\n * }\n * ```\n *\n * And here you have a hundred sprites that will be renderer at the speed of light.\n *\n * @class\n * @namespace PIXI\n */\n\n//TODO RENAME to PARTICLE CONTAINER?\nfunction SpriteBatch() {\n    DisplayObjectContainer.call(this);\n}\n\nSpriteBatch.prototype = Object.create(DisplayObjectContainer.prototype);\nSpriteBatch.prototype.constructor = SpriteBatch;\nmodule.exports = SpriteBatch;\n\n/**\n * Updates the object transform for rendering\n *\n * @private\n */\nSpriteBatch.prototype.updateTransform = function () {\n    // TODO don't need to!\n    this.displayObjectUpdateTransform();\n    //  PIXI.DisplayObjectContainer.prototype.updateTransform.call( this );\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer} The webgl renderer\n * @private\n */\nSpriteBatch.prototype.renderWebGL = function (renderer) {\n    if (!this.visible || this.alpha <= 0 || !this.children.length) {\n        return;\n    }\n\n    renderer.spriteBatch.stop();\n\n    renderer.shaderManager.setShader(renderer.shaderManager.fastShader);\n\n    renderer.fastSpriteBatch.begin(this);\n    renderer.fastSpriteBatch.render(this);\n\n    renderer.spriteBatch.start();\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer} The canvas renderer\n * @private\n */\nSpriteBatch.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0 || !this.children.length) {\n        return;\n    }\n\n    var context = renderer.context;\n    var transform = this.worldTransform;\n    var isRotated = true;\n\n    context.globalAlpha = this.worldAlpha;\n\n    this.displayObjectUpdateTransform();\n\n    for (var i = 0; i < this.children.length; ++i) {\n        var child = this.children[i];\n\n        if (!child.visible) {\n            continue;\n        }\n\n        var frame = child.texture.frame;\n\n        context.globalAlpha = this.worldAlpha * child.alpha;\n\n        if (child.rotation % (Math.PI * 2) === 0) {\n            // this is the fastest  way to optimise! - if rotation is 0 then we can avoid any kind of setTransform call\n            if (isRotated) {\n                context.setTransform(\n                    transform.a,\n                    transform.b,\n                    transform.c,\n                    transform.d,\n                    transform.tx,\n                    transform.ty\n                );\n\n                isRotated = false;\n            }\n\n            context.drawImage(\n                child.texture.baseTexture.source,\n                frame.x,\n                frame.y,\n                frame.width,\n                frame.height,\n                ((child.anchor.x) * (-frame.width * child.scale.x) + child.position.x  + 0.5) | 0,\n                ((child.anchor.y) * (-frame.height * child.scale.y) + child.position.y  + 0.5) | 0,\n                frame.width * child.scale.x,\n                frame.height * child.scale.y\n            );\n        }\n        else {\n            if (!isRotated) {\n                isRotated = true;\n            }\n\n            child.displayObjectUpdateTransform();\n\n            var childTransform = child.worldTransform;\n\n            if (renderer.roundPixels) {\n                context.setTransform(\n                    childTransform.a,\n                    childTransform.b,\n                    childTransform.c,\n                    childTransform.d,\n                    childTransform.tx | 0,\n                    childTransform.ty | 0\n                );\n            }\n            else {\n                context.setTransform(\n                    childTransform.a,\n                    childTransform.b,\n                    childTransform.c,\n                    childTransform.d,\n                    childTransform.tx,\n                    childTransform.ty\n                );\n            }\n\n            context.drawImage(\n                child.texture.baseTexture.source,\n                frame.x,\n                frame.y,\n                frame.width,\n                frame.height,\n                ((child.anchor.x) * (-frame.width) + 0.5) | 0,\n                ((child.anchor.y) * (-frame.height) + 0.5) | 0,\n                frame.width,\n                frame.height\n            );\n        }\n    }\n};\n","/**\n * @file        Main export of the PIXI core library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nvar core = module.exports = {\n    CONST: require('./const'),\n\n    // utils\n    utils: require('./utils'),\n    math: require('./math'),\n\n    // display\n    DisplayObject:          require('./display/DisplayObject'),\n    DisplayObjectContainer: require('./display/DisplayObjectContainer'),\n    Sprite:                 require('./display/Sprite'),\n    SpriteBatch:            require('./display/SpriteBatch'),\n\n    // primitives\n    Graphics:               require('./primitives/Graphics'),\n    GraphicsData:           require('./primitives/GraphicsData'),\n\n    // textures\n    Texture:                require('./textures/Texture'),\n    BaseTexture:            require('./textures/BaseTexture'),\n    RenderTexture:          require('./textures/RenderTexture'),\n    VideoBaseTexture:       require('./textures/VideoBaseTexture'),\n\n    // renderers - canvas\n    CanvasRenderer:         require('./renderers/canvas/CanvasRenderer'),\n    CanvasGraphics:         require('./renderers/canvas/utils/CanvasGraphics'),\n    CanvasBuffer:           require('./renderers/canvas/utils/CanvasBuffer'),\n\n    // renderers - webgl\n    WebGLRenderer:         require('./renderers/webgl/WebGLRenderer'),\n    WebGLGraphics:         require('./renderers/webgl/utils/WebGLGraphics'),\n\n    /**\n     * This helper function will automatically detect which renderer you should be using.\n     * WebGL is the preferred renderer as it is a lot faster. If webGL is not supported by\n     * the browser then this function will return a canvas renderer\n     *\n     * @param width=800 {number} the width of the renderers view\n     * @param height=600 {number} the height of the renderers view\n     * @param [options] {object} The optional renderer parameters\n     * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n     * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n     * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment)\n     * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you\n     *      need to call toDataUrl on the webgl context\n     * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n     * @param [noWebGL=false] {Boolean} prevents selection of WebGL renderer, even if such is present\n     *\n     * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer\n     */\n    autoDetectRenderer: function (width, height, options, noWebGL) {\n        width = width || 800;\n        height = height || 600;\n\n        if (!noWebGL && require('webgl-enabled')()) {\n            return new core.WebGLRenderer(width, height, options);\n        }\n\n        return new core.CanvasRenderer(width, height, options);\n    },\n\n    /**\n     * This helper function will automatically detect which renderer you should be using. This function is very\n     * similar to the autoDetectRenderer function except that is will return a canvas renderer for android.\n     * Even thought both android chrome supports webGL the canvas implementation perform better at the time of writing.\n     * This function will likely change and update as webGL performance improves on these devices.\n     *\n     * @param width=800 {number} the width of the renderers view\n     * @param height=600 {number} the height of the renderers view\n     * @param [options] {object} The optional renderer parameters\n     * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n     * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n     * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment)\n     * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you\n     *      need to call toDataUrl on the webgl context\n     * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n     *\n     * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer\n     */\n    autoDetectRecommendedRenderer: function (width, height, options) {\n        var isAndroid = /Android/i.test(navigator.userAgent);\n\n        return core.autoDetectRenderer(width, height, options, isAndroid);\n    }\n};\n","var Point = require('./Point');\n\n/**\n * The Matrix class is now an object, which makes it a lot faster,\n * here is a representation of it :\n * | a | b | tx|\n * | c | d | ty|\n * | 0 | 0 | 1 |\n *\n * @class\n * @namespace PIXI\n */\nfunction Matrix() {\n    /**\n     * @member {number}\n     * @default 1\n     */\n    this.a = 1;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.b = 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.c = 0;\n\n    /**\n     * @member {number}\n     * @default 1\n     */\n    this.d = 1;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.tx = 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.ty = 0;\n}\n\nMatrix.prototype.constructor = Matrix;\nmodule.exports = Matrix;\n\n/**\n * Creates a Matrix object based on the given array. The Element to Matrix mapping order is as follows:\n *\n * a = array[0]\n * b = array[1]\n * c = array[3]\n * d = array[4]\n * tx = array[2]\n * ty = array[5]\n *\n * @param array {number[]} The array that the matrix will be populated from.\n */\nMatrix.prototype.fromArray = function (array) {\n    this.a = array[0];\n    this.b = array[1];\n    this.c = array[3];\n    this.d = array[4];\n    this.tx = array[2];\n    this.ty = array[5];\n};\n\n/**\n * Creates an array from the current Matrix object.\n *\n * @param transpose {boolean} Whether we need to transpose the matrix or not\n * @return {number[]} the newly created array which contains the matrix\n */\nMatrix.prototype.toArray = function (transpose) {\n    if (!this.array) {\n        this.array = new Float32Array(9);\n    }\n\n    var array = this.array;\n\n    if (transpose) {\n        array[0] = this.a;\n        array[1] = this.b;\n        array[2] = 0;\n        array[3] = this.c;\n        array[4] = this.d;\n        array[5] = 0;\n        array[6] = this.tx;\n        array[7] = this.ty;\n        array[8] = 1;\n    }\n    else {\n        array[0] = this.a;\n        array[1] = this.c;\n        array[2] = this.tx;\n        array[3] = this.b;\n        array[4] = this.d;\n        array[5] = this.ty;\n        array[6] = 0;\n        array[7] = 0;\n        array[8] = 1;\n    }\n\n    return array;\n};\n\n/**\n * Get a new position with the current transformation applied.\n * Can be used to go from a child's coordinate space to the world coordinate space. (e.g. rendering)\n *\n * @param pos {Point} The origin\n * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input)\n * @return {Point} The new point, transformed through this matrix\n */\nMatrix.prototype.apply = function (pos, newPos) {\n    newPos = newPos || new Point();\n\n    newPos.x = this.a * pos.x + this.c * pos.y + this.tx;\n    newPos.y = this.b * pos.x + this.d * pos.y + this.ty;\n\n    return newPos;\n};\n\n/**\n * Get a new position with the inverse of the current transformation applied.\n * Can be used to go from the world coordinate space to a child's coordinate space. (e.g. input)\n *\n * @param pos {Point} The origin\n * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input)\n * @return {Point} The new point, inverse-transformed through this matrix\n */\nMatrix.prototype.applyInverse = function (pos, newPos) {\n    newPos = newPos || new Point();\n\n    var id = 1 / (this.a * this.d + this.c * -this.b);\n\n    newPos.x = this.d * id * pos.x + -this.c * id * pos.y + (this.ty * this.c - this.tx * this.d) * id;\n    newPos.y = this.a * id * pos.y + -this.b * id * pos.x + (-this.ty * this.a + this.tx * this.b) * id;\n\n    return newPos;\n};\n\n/**\n * Translates the matrix on the x and y.\n *\n * @param {number} x\n * @param {number} y\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.translate = function (x, y) {\n    this.tx += x;\n    this.ty += y;\n\n    return this;\n};\n\n/**\n * Applies a scale transformation to the matrix.\n *\n * @param {number} x The amount to scale horizontally\n * @param {number} y The amount to scale vertically\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.scale = function (x, y) {\n    this.a *= x;\n    this.d *= y;\n    this.c *= x;\n    this.b *= y;\n    this.tx *= x;\n    this.ty *= y;\n\n    return this;\n};\n\n\n/**\n * Applies a rotation transformation to the matrix.\n *\n * @param {number} angle - The angle in radians.\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.rotate = function (angle) {\n    var cos = Math.cos( angle );\n    var sin = Math.sin( angle );\n\n    var a1 = this.a;\n    var c1 = this.c;\n    var tx1 = this.tx;\n\n    this.a = a1 * cos-this.b * sin;\n    this.b = a1 * sin+this.b * cos;\n    this.c = c1 * cos-this.d * sin;\n    this.d = c1 * sin+this.d * cos;\n    this.tx = tx1 * cos - this.ty * sin;\n    this.ty = tx1 * sin + this.ty * cos;\n\n    return this;\n};\n\n/**\n * Appends the given Matrix to this Matrix.\n *\n * @param {Matrix} matrix\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.append = function (matrix) {\n    var a1 = this.a;\n    var b1 = this.b;\n    var c1 = this.c;\n    var d1 = this.d;\n\n    this.a  = matrix.a * a1 + matrix.b * c1;\n    this.b  = matrix.a * b1 + matrix.b * d1;\n    this.c  = matrix.c * a1 + matrix.d * c1;\n    this.d  = matrix.c * b1 + matrix.d * d1;\n\n    this.tx = matrix.tx * a1 + matrix.ty * c1 + this.tx;\n    this.ty = matrix.tx * b1 + matrix.ty * d1 + this.ty;\n\n    return this;\n};\n\n/**\n * Resets this Matix to an identity (default) matrix.\n *\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.identity = function () {\n    this.a = 1;\n    this.b = 0;\n    this.c = 0;\n    this.d = 1;\n    this.tx = 0;\n    this.ty = 0;\n\n    return this;\n};\n\nMatrix.IDENTITY = new Matrix();\n","/**\n * The Point object represents a location in a two-dimensional coordinate system, where x represents\n * the horizontal axis and y represents the vertical axis.\n *\n * @class\n * @namespace PIXI\n * @param [x=0] {number} position of the point on the x axis\n * @param [y=0] {number} position of the point on the y axis\n */\nfunction Point(x, y) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n}\n\nPoint.prototype.constructor = Point;\nmodule.exports = Point;\n\n/**\n * Creates a clone of this point\n *\n * @return {Point} a copy of the point\n */\nPoint.prototype.clone = function () {\n    return new Point(this.x, this.y);\n};\n\n/**\n * Sets the point to a new x and y position.\n * If y is omitted, both x and y will be set to x.\n *\n * @param [x=0] {number} position of the point on the x axis\n * @param [y=0] {number} position of the point on the y axis\n */\nPoint.prototype.set = function (x, y) {\n    this.x = x || 0;\n    this.y = y || ( (y !== 0) ? this.x : 0 ) ;\n};\n","/**\n * @namespace PIXI.math\n */\nmodule.exports = {\n    /**\n     * @property {number} PI_2 - Math.PI x 2\n     * @constant\n     * @static\n     */\n    PI_2: Math.PI * 2,\n\n    /**\n     * @property {number} RAD_TO_DEG - Constant conversion factor for converting radians to degrees\n     * @constant\n     * @static\n     */\n    RAD_TO_DEG: 180 / Math.PI,\n\n    /**\n     * @property {Number} DEG_TO_RAD - Constant conversion factor for converting degrees to radians\n     * @constant\n     * @static\n     */\n    DEG_TO_RAD: Math.PI / 180,\n\n    Point:      require('./Point'),\n    Matrix:     require('./Matrix'),\n\n    Circle:     require('./shapes/Circle'),\n    Ellipse:    require('./shapes/Ellipse'),\n    Polygon:    require('./shapes/Polygon'),\n    Rectangle:  require('./shapes/Rectangle'),\n    RoundedRectangle: require('./shapes/RoundedRectangle')\n};\n","var Rectangle = require('./Rectangle'),\n    CONST = require('../../const');\n\n/**\n * The Circle object can be used to specify a hit area for displayObjects\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the center of this circle\n * @param y {number} The Y coordinate of the center of this circle\n * @param radius {number} The radius of the circle\n */\nfunction Circle(x, y, radius) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.radius = radius || 0;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.CIRC;\n}\n\nCircle.prototype.constructor = Circle;\nmodule.exports = Circle;\n\n/**\n * Creates a clone of this Circle instance\n *\n * @method clone\n * @return {Circle} a copy of the Circle\n */\nCircle.prototype.clone = function () {\n    return new Circle(this.x, this.y, this.radius);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this circle\n *\n * @method contains\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this Circle\n */\nCircle.prototype.contains = function (x, y) {\n    if (this.radius <= 0) {\n        return false;\n    }\n\n    var dx = (this.x - x),\n        dy = (this.y - y),\n        r2 = this.radius * this.radius;\n\n    dx *= dx;\n    dy *= dy;\n\n    return (dx + dy <= r2);\n};\n\n/**\n* Returns the framing rectangle of the circle as a Rectangle object\n*\n* @method getBounds\n* @return {Rectangle} the framing rectangle\n*/\nCircle.prototype.getBounds = function () {\n    return new Rectangle(this.x - this.radius, this.y - this.radius, this.radius * 2, this.radius * 2);\n};\n","var Rectangle = require('./Rectangle'),\n    CONST = require('../../const');\n\n/**\n * The Ellipse object can be used to specify a hit area for displayObjects\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the center of the ellipse\n * @param y {number} The Y coordinate of the center of the ellipse\n * @param width {number} The half width of this ellipse\n * @param height {number} The half height of this ellipse\n */\nfunction Ellipse(x, y, width, height) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.width = width || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.height = height || 0;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.ELIP;\n}\n\nEllipse.prototype.constructor = Ellipse;\nmodule.exports = Ellipse;\n\n/**\n * Creates a clone of this Ellipse instance\n *\n * @method clone\n * @return {Ellipse} a copy of the ellipse\n */\nEllipse.prototype.clone = function () {\n    return new Ellipse(this.x, this.y, this.width, this.height);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this ellipse\n *\n * @method contains\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coords are within this ellipse\n */\nEllipse.prototype.contains = function (x, y) {\n    if (this.width <= 0 || this.height <= 0) {\n        return false;\n    }\n\n    //normalize the coords to an ellipse with center 0,0\n    var normx = ((x - this.x) / this.width),\n        normy = ((y - this.y) / this.height);\n\n    normx *= normx;\n    normy *= normy;\n\n    return (normx + normy <= 1);\n};\n\n/**\n* Returns the framing rectangle of the ellipse as a Rectangle object\n*\n* @method getBounds\n* @return {Rectangle} the framing rectangle\n*/\nEllipse.prototype.getBounds = function () {\n    return new Rectangle(this.x - this.width, this.y - this.height, this.width, this.height);\n};\n","var Point = require('../Point'),\n    CONST = require('../../const');\n\n/**\n * @class\n * @namespace PIXI\n * @param points* {Point[]|number[]|Point...|number...} This can be an array of Points that form the polygon,\n *      a flat array of numbers that will be interpreted as [x,y, x,y, ...], or the arguments passed can be\n *      all the points of the polygon e.g. `new Polygon(new Point(), new Point(), ...)`, or the\n *      arguments passed can be flat x,y values e.g. `new Polygon(x,y, x,y, x,y, ...)` where `x` and `y` are\n *      Numbers.\n */\nfunction Polygon(points) {\n    //if points isn't an array, use arguments as the array\n    if (!(points instanceof Array)) {\n        points = Array.prototype.slice.call(arguments);\n    }\n\n    //if this is a flat array of numbers, convert it to points\n    if (points[0] instanceof Point) {\n        var p = [];\n        for (var i = 0, il = points.length; i < il; i++) {\n            p.push(points[i].x, points[i].y);\n        }\n\n        points = p;\n    }\n\n    this.closed = true;\n\n    /**\n     * An array of the points of this polygon\n     *\n     * @member {Point[]}\n     */\n    this.points = points;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.POLY;\n}\n\nPolygon.prototype.constructor = Polygon;\nmodule.exports = Polygon;\n\n/**\n * Creates a clone of this polygon\n *\n * @return {Polygon} a copy of the polygon\n */\nPolygon.prototype.clone = function () {\n    return new Polygon(this.points.slice());\n};\n\n/**\n * Checks whether the x and y coordinates passed to this function are contained within this polygon\n *\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this polygon\n */\nPolygon.prototype.contains = function (x, y) {\n    var inside = false;\n\n    // use some raycasting to test hits\n    // https://github.com/substack/point-in-polygon/blob/master/index.js\n    var length = this.points.length / 2;\n\n    for (var i = 0, j = length - 1; i < length; j = i++) {\n        var xi = this.points[i * 2], yi = this.points[i * 2 + 1],\n            xj = this.points[j * 2], yj = this.points[j * 2 + 1],\n            intersect = ((yi > y) !== (yj > y)) && (x < (xj - xi) * (y - yi) / (yj - yi) + xi);\n\n        if (intersect) {\n            inside = !inside;\n        }\n    }\n\n    return inside;\n};\n","var CONST = require('../../const');\n\n/**\n * the Rectangle object is an area defined by its position, as indicated by its top-left corner point (x, y) and by its width and its height.\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the upper-left corner of the rectangle\n * @param y {number} The Y coordinate of the upper-left corner of the rectangle\n * @param width {number} The overall width of this rectangle\n * @param height {number} The overall height of this rectangle\n */\nfunction Rectangle(x, y, width, height) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.width = width || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.height = height || 0;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.RECT;\n}\n\nRectangle.prototype.constructor = Rectangle;\nmodule.exports = Rectangle;\n\n/**\n * A constant empty rectangle.\n *\n * @static\n * @constant\n */\nRectangle.EMPTY = new Rectangle(0, 0, 0, 0);\n\n\n/**\n * Creates a clone of this Rectangle\n *\n * @return {Rectangle} a copy of the rectangle\n */\nRectangle.prototype.clone = function () {\n    return new Rectangle(this.x, this.y, this.width, this.height);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this Rectangle\n *\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this Rectangle\n */\nRectangle.prototype.contains = function (x, y) {\n    if (this.width <= 0 || this.height <= 0) {\n        return false;\n    }\n\n    if (x >= this.x && x <= this.x + this.width) {\n        if (y >= this.y && y <= this.y + this.height) {\n            return true;\n        }\n    }\n\n    return false;\n};\n","var CONST = require('../../const');\n\n/**\n * The Rounded Rectangle object is an area defined by its position and has nice rounded corners, as indicated by its top-left corner point (x, y) and by its width and its height.\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the upper-left corner of the rounded rectangle\n * @param y {number} The Y coordinate of the upper-left corner of the rounded rectangle\n * @param width {number} The overall width of this rounded rectangle\n * @param height {number} The overall height of this rounded rectangle\n * @param radius {number} Controls the radius of the rounded corners\n */\nfunction RoundedRectangle(x, y, width, height, radius) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.width = width || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.height = height || 0;\n\n    /**\n     * @member {number}\n     * @default 20\n     */\n    this.radius = radius || 20;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.RREC;\n}\n\nRoundedRectangle.prototype.constructor = RoundedRectangle;\nmodule.exports = RoundedRectangle;\n\n/**\n * Creates a clone of this Rounded Rectangle\n *\n * @return {RoundedRectangle} a copy of the rounded rectangle\n */\nRoundedRectangle.prototype.clone = function () {\n    return new RoundedRectangle(this.x, this.y, this.width, this.height, this.radius);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this Rounded Rectangle\n *\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this Rounded Rectangle\n */\nRoundedRectangle.prototype.contains = function (x, y) {\n    if (this.width <= 0 || this.height <= 0) {\n        return false;\n    }\n\n    if (x >= this.x && x <= this.x + this.width) {\n        if (y >= this.y && y <= this.y + this.height) {\n            return true;\n        }\n    }\n\n    return false;\n};\n","var DisplayObjectContainer = require('../display/DisplayObjectContainer'),\n    Sprite = require('../display/Sprite'),\n    Texture = require('../textures/Texture'),\n    CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'),\n    CanvasGraphics = require('../renderers/canvas/utils/CanvasGraphics'),\n    WebGLGraphics = require('../renderers/webgl/utils/WebGLGraphics'),\n    GraphicsData = require('./GraphicsData'),\n    math = require('../math'),\n    CONST = require('../const');\n\n/**\n * The Graphics class contains methods used to draw primitive shapes such as lines, circles and\n * rectangles to the display, and color and fill them.\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n */\nfunction Graphics() {\n    DisplayObjectContainer.call(this);\n\n    this.renderable = true;\n\n    /**\n     * The alpha value used when filling the Graphics object.\n     *\n     * @member {number}\n     * @default 1\n     */\n    this.fillAlpha = 1;\n\n    /**\n     * The width (thickness) of any lines drawn.\n     *\n     * @member {number}\n     * @default 0\n     */\n    this.lineWidth = 0;\n\n    /**\n     * The color of any lines drawn.\n     *\n     * @member {string}\n     * @default 0\n     */\n    this.lineColor = 0;\n\n    /**\n     * Graphics data\n     *\n     * @member {GraphicsData[]}\n     * @private\n     */\n    this.graphicsData = [];\n\n    /**\n     * The tint applied to the graphic shape. This is a hex value. Apply a value of 0xFFFFFF to reset the tint.\n     *\n     * @member {number}\n     * @default 0xFFFFFF\n     */\n    this.tint = 0xFFFFFF;\n\n    /**\n     * The blend mode to be applied to the graphic shape. Apply a value of blendModes.NORMAL to reset the blend mode.\n     *\n     * @member {number}\n     * @default CONST.blendModes.NORMAL;\n     */\n    this.blendMode = CONST.blendModes.NORMAL;\n\n    /**\n     * Current path\n     *\n     * @member {GraphicsData}\n     * @private\n     */\n    this.currentPath = null;\n\n    /**\n     * Array containing some WebGL-related properties used by the WebGL renderer.\n     *\n     * @member {object<number, object>}\n     * @private\n     */\n    // TODO - _webgl should use a prototype object, not a random undocumented object...\n    this._webGL = {};\n\n    /**\n     * Whether this shape is being used as a mask.\n     *\n     * @member {boolean}\n     */\n    this.isMask = false;\n\n    /**\n     * The bounds' padding used for bounds calculation.\n     *\n     * @member {number}\n     */\n    this.boundsPadding = 0;\n\n    /**\n     * A cache of the local bounds to prevent recalculation.\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._localBounds = new math.Rectangle(0,0,1,1);\n\n    /**\n     * Used to detect if the graphics object has changed. If this is set to true then the graphics\n     * object will be recalculated.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.dirty = true;\n\n    /**\n     * Used to detect if the WebGL graphics object has changed. If this is set to true then the\n     * graphics object will be recalculated.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.glDirty = false;\n\n    /**\n     * Used to detect if the cached sprite object needs to be updated.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.cachedSpriteDirty = false;\n}\n\n// constructor\nGraphics.prototype = Object.create(DisplayObjectContainer.prototype);\nGraphics.prototype.constructor = Graphics;\nmodule.exports = Graphics;\n\nObject.defineProperties(Graphics.prototype, {\n    /**\n     * When cacheAsBitmap is set to true the graphics object will be rendered as if it was a sprite.\n     * This is useful if your graphics element does not change often, as it will speed up the rendering\n     * of the object in exchange for taking up texture memory. It is also useful if you need the graphics\n     * object to be anti-aliased, because it will be rendered using canvas. This is not recommended if\n     * you are constantly redrawing the graphics element.\n     *\n     * @member {boolean}\n     * @memberof Graphics#\n     * @default false\n     * @private\n     */\n    cacheAsBitmap: {\n        get: function () {\n            return this._cacheAsBitmap;\n        },\n        set: function (value) {\n            this._cacheAsBitmap = value;\n\n            if (this._cacheAsBitmap) {\n                this._generateCachedSprite();\n            }\n            else {\n                this.destroyCachedSprite();\n                this.dirty = true;\n            }\n        }\n    }\n});\n\n/**\n * Creates a new Graphics object with the same values as this one.\n *\n * @return {Graphics}\n */\nGraphicsData.prototype.clone = function () {\n    var clone = new Graphics();\n\n    clone.renderable    = this.renderable;\n    clone.fillAlpha     = this.fillAlpha;\n    clone.lineWidth     = this.lineWidth;\n    clone.lineColor     = this.lineColor;\n    clone.tint          = this.tint;\n    clone.blendMode     = this.blendMode;\n    clone.isMask        = this.isMask;\n    clone.boundsPadding = this.boundsPadding;\n    clone.dirty         = this.dirty;\n    clone.glDirty       = this.glDirty;\n    clone.cachedSpriteDirty = this.cachedSpriteDirty;\n\n    // copy graphics data\n    for (var i = 0; i < this.graphicsData.length; ++i) {\n        clone.graphicsData.push(this.graphicsData.clone());\n    }\n\n    clone.currentPath = clone.graphicsData[clone.graphicsData.length - 1];\n\n    clone.updateLocalBounds();\n\n    return clone;\n};\n\n/**\n * Specifies the line style used for subsequent calls to Graphics methods such as the lineTo() method or the drawCircle() method.\n *\n * @param lineWidth {number} width of the line to draw, will update the objects stored style\n * @param color {number} color of the line to draw, will update the objects stored style\n * @param alpha {number} alpha of the line to draw, will update the objects stored style\n * @return {Graphics}\n */\nGraphics.prototype.lineStyle = function (lineWidth, color, alpha) {\n    this.lineWidth = lineWidth || 0;\n    this.lineColor = color || 0;\n    this.lineAlpha = (arguments.length < 3) ? 1 : alpha;\n\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length) {\n            // halfway through a line? start a new one!\n            this.drawShape( new math.Polygon( this.currentPath.shape.points.slice(-2) ));\n        }\n        else {\n            // otherwise its empty so lets just set the line properties\n            this.currentPath.lineWidth = this.lineWidth;\n            this.currentPath.lineColor = this.lineColor;\n            this.currentPath.lineAlpha = this.lineAlpha;\n        }\n    }\n\n    return this;\n};\n\n/**\n * Moves the current drawing position to x, y.\n *\n * @param x {number} the X coordinate to move to\n * @param y {number} the Y coordinate to move to\n * @return {Graphics}\n  */\nGraphics.prototype.moveTo = function (x, y) {\n    this.drawShape(new math.Polygon([x,y]));\n\n    return this;\n};\n\n/**\n * Draws a line using the current line style from the current drawing position to (x, y);\n * The current drawing position is then set to (x, y).\n *\n * @param x {number} the X coordinate to draw to\n * @param y {number} the Y coordinate to draw to\n * @return {Graphics}\n */\nGraphics.prototype.lineTo = function (x, y) {\n    this.currentPath.shape.points.push(x, y);\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * Calculate the points for a quadratic bezier curve and then draws it.\n * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c\n *\n * @param cpX {number} Control point x\n * @param cpY {number} Control point y\n * @param toX {number} Destination point x\n * @param toY {number} Destination point y\n * @return {Graphics}\n */\nGraphics.prototype.quadraticCurveTo = function (cpX, cpY, toX, toY) {\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length === 0) {\n            this.currentPath.shape.points = [0, 0];\n        }\n    }\n    else {\n        this.moveTo(0,0);\n    }\n\n    var xa,\n        ya,\n        n = 20,\n        points = this.currentPath.shape.points;\n\n    if (points.length === 0) {\n        this.moveTo(0, 0);\n    }\n\n    var fromX = points[points.length-2];\n    var fromY = points[points.length-1];\n\n    var j = 0;\n    for (var i = 1; i <= n; ++i) {\n        j = i / n;\n\n        xa = fromX + ( (cpX - fromX) * j );\n        ya = fromY + ( (cpY - fromY) * j );\n\n        points.push( xa + ( ((cpX + ( (toX - cpX) * j )) - xa) * j ),\n                     ya + ( ((cpY + ( (toY - cpY) * j )) - ya) * j ) );\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * Calculate the points for a bezier curve and then draws it.\n *\n * @param cpX {number} Control point x\n * @param cpY {number} Control point y\n * @param cpX2 {number} Second Control point x\n * @param cpY2 {number} Second Control point y\n * @param toX {number} Destination point x\n * @param toY {number} Destination point y\n * @return {Graphics}\n */\nGraphics.prototype.bezierCurveTo = function (cpX, cpY, cpX2, cpY2, toX, toY) {\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length === 0) {\n            this.currentPath.shape.points = [0, 0];\n        }\n    }\n    else {\n        this.moveTo(0,0);\n    }\n\n    var n = 20,\n        dt,\n        dt2,\n        dt3,\n        t2,\n        t3,\n        points = this.currentPath.shape.points;\n\n    var fromX = points[points.length-2];\n    var fromY = points[points.length-1];\n\n    var j = 0;\n\n    for (var i = 1; i <= n; ++i) {\n        j = i / n;\n\n        dt = (1 - j);\n        dt2 = dt * dt;\n        dt3 = dt2 * dt;\n\n        t2 = j * j;\n        t3 = t2 * j;\n\n        points.push( dt3 * fromX + 3 * dt2 * j * cpX + 3 * dt * t2 * cpX2 + t3 * toX,\n                     dt3 * fromY + 3 * dt2 * j * cpY + 3 * dt * t2 * cpY2 + t3 * toY);\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * The arcTo() method creates an arc/curve between two tangents on the canvas.\n *\n * \"borrowed\" from https://code.google.com/p/fxcanvas/ - thanks google!\n *\n * @param x1 {number} The x-coordinate of the beginning of the arc\n * @param y1 {number} The y-coordinate of the beginning of the arc\n * @param x2 {number} The x-coordinate of the end of the arc\n * @param y2 {number} The y-coordinate of the end of the arc\n * @param radius {number} The radius of the arc\n * @return {Graphics}\n */\nGraphics.prototype.arcTo = function (x1, y1, x2, y2, radius) {\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length === 0) {\n            this.currentPath.shape.points.push(x1, y1);\n        }\n    }\n    else {\n        this.moveTo(x1, y1);\n    }\n\n    var points = this.currentPath.shape.points,\n        fromX = points[points.length-2],\n        fromY = points[points.length-1],\n        a1 = fromY - y1,\n        b1 = fromX - x1,\n        a2 = y2   - y1,\n        b2 = x2   - x1,\n        mm = Math.abs(a1 * b2 - b1 * a2);\n\n    if (mm < 1.0e-8 || radius === 0) {\n        if (points[points.length-2] !== x1 || points[points.length-1] !== y1) {\n            points.push(x1, y1);\n        }\n    }\n    else {\n        var dd = a1 * a1 + b1 * b1,\n            cc = a2 * a2 + b2 * b2,\n            tt = a1 * a2 + b1 * b2,\n            k1 = radius * Math.sqrt(dd) / mm,\n            k2 = radius * Math.sqrt(cc) / mm,\n            j1 = k1 * tt / dd,\n            j2 = k2 * tt / cc,\n            cx = k1 * b2 + k2 * b1,\n            cy = k1 * a2 + k2 * a1,\n            px = b1 * (k2 + j1),\n            py = a1 * (k2 + j1),\n            qx = b2 * (k1 + j2),\n            qy = a2 * (k1 + j2),\n            startAngle = Math.atan2(py - cy, px - cx),\n            endAngle   = Math.atan2(qy - cy, qx - cx);\n\n        this.arc(cx + x1, cy + y1, radius, startAngle, endAngle, b1 * a2 > b2 * a1);\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * The arc method creates an arc/curve (used to create circles, or parts of circles).\n *\n * @param cx {number} The x-coordinate of the center of the circle\n * @param cy {number} The y-coordinate of the center of the circle\n * @param radius {number} The radius of the circle\n * @param startAngle {number} The starting angle, in radians (0 is at the 3 o'clock position of the arc's circle)\n * @param endAngle {number} The ending angle, in radians\n * @param anticlockwise {boolean} Optional. Specifies whether the drawing should be counterclockwise or clockwise. False is default, and indicates clockwise, while true indicates counter-clockwise.\n * @return {Graphics}\n */\nGraphics.prototype.arc = function (cx, cy, radius, startAngle, endAngle, anticlockwise) {\n    var startX = cx + Math.cos(startAngle) * radius;\n    var startY = cy + Math.sin(startAngle) * radius;\n    var points;\n\n    // TODO - This if-else makes no sense. It uses currentPath in the else where it doesn't exist...\n    if (this.currentPath) {\n        points = this.currentPath.shape.points;\n\n        if (points.length === 0) {\n            points.push(startX, startY);\n        }\n        else if (points[points.length-2] !== startX || points[points.length-1] !== startY) {\n            points.push(startX, startY);\n        }\n    }\n    else {\n        this.moveTo(startX, startY);\n        points = this.currentPath.shape.points;\n    }\n\n    if (startAngle === endAngle) {\n        return this;\n    }\n\n    if (!anticlockwise && endAngle <= startAngle) {\n        endAngle += Math.PI * 2;\n    }\n    else if (anticlockwise && startAngle <= endAngle) {\n        startAngle += Math.PI * 2;\n    }\n\n    var sweep = anticlockwise ? (startAngle - endAngle) *-1 : (endAngle - startAngle);\n    var segs = (Math.abs(sweep)/ (Math.PI * 2)) * 40;\n\n    if (sweep === 0) {\n        return this;\n    }\n\n    var theta = sweep/(segs*2);\n    var theta2 = theta*2;\n\n    var cTheta = Math.cos(theta);\n    var sTheta = Math.sin(theta);\n\n    var segMinus = segs - 1;\n\n    var remainder = ( segMinus % 1 ) / segMinus;\n\n    for (var i = 0; i <= segMinus; ++i) {\n        var real =  i + remainder * i;\n        var angle = ((theta) + startAngle + (theta2 * real));\n\n        var c = Math.cos(angle);\n        var s = -Math.sin(angle);\n\n        points.push(( (cTheta *  c) + (sTheta * s) ) * radius + cx,\n                    ( (cTheta * -s) + (sTheta * c) ) * radius + cy);\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * Specifies a simple one-color fill that subsequent calls to other Graphics methods\n * (such as lineTo() or drawCircle()) use when drawing.\n *\n * @param color {number} the color of the fill\n * @param alpha {number} the alpha of the fill\n * @return {Graphics}\n */\nGraphics.prototype.beginFill = function (color, alpha) {\n    this.filling = true;\n    this.fillColor = color || 0;\n    this.fillAlpha = (alpha === undefined) ? 1 : alpha;\n\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length <= 2) {\n            this.currentPath.fill = this.filling;\n            this.currentPath.fillColor = this.fillColor;\n            this.currentPath.fillAlpha = this.fillAlpha;\n        }\n    }\n    return this;\n};\n\n/**\n * Applies a fill to the lines and shapes that were added since the last call to the beginFill() method.\n *\n * @return {Graphics}\n */\nGraphics.prototype.endFill = function () {\n    this.filling = false;\n    this.fillColor = null;\n    this.fillAlpha = 1;\n\n    return this;\n};\n\n/**\n *\n * @param x {number} The X coord of the top-left of the rectangle\n * @param y {number} The Y coord of the top-left of the rectangle\n * @param width {number} The width of the rectangle\n * @param height {number} The height of the rectangle\n * @return {Graphics}\n */\nGraphics.prototype.drawRect = function ( x, y, width, height ) {\n    this.drawShape(new math.Rectangle(x,y, width, height));\n\n    return this;\n};\n\n/**\n *\n * @param x {number} The X coord of the top-left of the rectangle\n * @param y {number} The Y coord of the top-left of the rectangle\n * @param width {number} The width of the rectangle\n * @param height {number} The height of the rectangle\n * @param radius {number} Radius of the rectangle corners\n */\nGraphics.prototype.drawRoundedRect = function ( x, y, width, height, radius ) {\n    this.drawShape(new math.RoundedRectangle(x, y, width, height, radius));\n\n    return this;\n};\n\n/**\n * Draws a circle.\n *\n * @param x {number} The X coordinate of the center of the circle\n * @param y {number} The Y coordinate of the center of the circle\n * @param radius {number} The radius of the circle\n * @return {Graphics}\n */\nGraphics.prototype.drawCircle = function (x, y, radius) {\n    this.drawShape(new math.Circle(x,y, radius));\n\n    return this;\n};\n\n/**\n * Draws an ellipse.\n *\n * @param x {number} The X coordinate of the center of the ellipse\n * @param y {number} The Y coordinate of the center of the ellipse\n * @param width {number} The half width of the ellipse\n * @param height {number} The half height of the ellipse\n * @return {Graphics}\n */\nGraphics.prototype.drawEllipse = function (x, y, width, height) {\n    this.drawShape(new math.Ellipse(x, y, width, height));\n\n    return this;\n};\n\n/**\n * Draws a polygon using the given path.\n *\n * @param path {Array} The path data used to construct the polygon.\n * @return {Graphics}\n */\nGraphics.prototype.drawPolygon = function (path) {\n    if (!(path instanceof Array)) {\n        path = Array.prototype.slice.call(arguments);\n    }\n\n    this.drawShape(new math.Polygon(path));\n\n    return this;\n};\n\n/**\n * Clears the graphics that were drawn to this Graphics object, and resets fill and line style settings.\n *\n * @return {Graphics}\n */\nGraphics.prototype.clear = function () {\n    this.lineWidth = 0;\n    this.filling = false;\n\n    this.dirty = true;\n    this.clearDirty = true;\n    this.graphicsData = [];\n\n    return this;\n};\n\n/**\n * Useful function that returns a texture of the graphics object that can then be used to create sprites\n * This can be quite useful if your geometry is complicated and needs to be reused multiple times.\n *\n * @param resolution {number} The resolution of the texture being generated\n * @param scaleMode {number} Should be one of the scaleMode consts\n * @return {Texture} a texture of the graphics object\n */\nGraphics.prototype.generateTexture = function (resolution, scaleMode) {\n    resolution = resolution || 1;\n\n    var bounds = this.getBounds();\n\n    var canvasBuffer = new CanvasBuffer(bounds.width * resolution, bounds.height * resolution);\n\n    var texture = Texture.fromCanvas(canvasBuffer.canvas, scaleMode);\n    texture.baseTexture.resolution = resolution;\n\n    canvasBuffer.context.scale(resolution, resolution);\n\n    canvasBuffer.context.translate(-bounds.x,-bounds.y);\n\n    CanvasGraphics.renderGraphics(this, canvasBuffer.context);\n\n    return texture;\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer}\n */\nGraphics.prototype.renderWebGL = function (renderer) {\n    // if the sprite is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0 || this.isMask === true) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        if (this.dirty || this.cachedSpriteDirty) {\n            this._generateCachedSprite();\n\n            // we will also need to update the texture on the gpu too!\n            this.updateCachedSpriteTexture();\n\n            this.cachedSpriteDirty = false;\n            this.dirty = false;\n        }\n\n        this._cachedSprite.worldAlpha = this.worldAlpha;\n\n        Sprite.prototype.renderWebGL.call(this._cachedSprite, renderer);\n\n        return;\n    }\n    else {\n        renderer.spriteBatch.stop();\n        renderer.blendModeManager.setBlendMode(this.blendMode);\n\n        if (this._mask) {\n            renderer.maskManager.pushMask(this._mask, renderer);\n        }\n\n        if (this._filters) {\n            renderer.filterManager.pushFilter(this._filterBlock);\n        }\n\n        // check blend mode\n        if (this.blendMode !== renderer.spriteBatch.currentBlendMode) {\n            renderer.spriteBatch.currentBlendMode = this.blendMode;\n\n            var blendModeWebGL = renderer.blendModes[renderer.spriteBatch.currentBlendMode];\n\n            renderer.spriteBatch.gl.blendFunc(blendModeWebGL[0], blendModeWebGL[1]);\n        }\n\n        // check if the webgl graphic needs to be updated\n        if (this.glDirty) {\n            this.dirty = true;\n            this.glDirty = false;\n        }\n\n        WebGLGraphics.renderGraphics(this, renderer);\n\n        // only render if it has children!\n        if (this.children.length) {\n            renderer.spriteBatch.start();\n\n             // simple render children!\n            for (var i = 0, j = this.children.length; i < j; ++i) {\n                this.children[i].renderWebGL(renderer);\n            }\n\n            renderer.spriteBatch.stop();\n        }\n\n        if (this._filters) {\n            renderer.filterManager.popFilter();\n        }\n\n        if (this._mask) {\n            renderer.maskManager.popMask(this.mask, renderer);\n        }\n\n        renderer.drawCount++;\n\n        renderer.spriteBatch.start();\n    }\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer}\n * @private\n */\nGraphics.prototype.renderCanvas = function (renderer) {\n    // if the sprite is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0 || this.isMask === true) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        if (this.dirty || this.cachedSpriteDirty) {\n            this._generateCachedSprite();\n\n            // we will also need to update the texture\n            this.updateCachedSpriteTexture();\n\n            this.cachedSpriteDirty = false;\n            this.dirty = false;\n        }\n\n        this._cachedSprite.alpha = this.alpha;\n\n        Sprite.prototype.renderCanvas.call(this._cachedSprite, renderer);\n\n        return;\n    }\n    else {\n        var context = renderer.context;\n        var transform = this.worldTransform;\n\n        if (this.blendMode !== renderer.currentBlendMode) {\n            renderer.currentBlendMode = this.blendMode;\n            context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode];\n        }\n\n        if (this._mask) {\n            renderer.maskManager.pushMask(this._mask, renderer);\n        }\n\n        var resolution = renderer.resolution;\n        context.setTransform(\n            transform.a * resolution,\n            transform.b * resolution,\n            transform.c * resolution,\n            transform.d * resolution,\n            transform.tx * resolution,\n            transform.ty * resolution\n        );\n\n        CanvasGraphics.renderGraphics(this, context);\n\n        for (var i = 0, j = this.children.length; i < j; ++i) {\n            this.children[i].renderCanvas(renderer);\n        }\n\n        if (this._mask) {\n            renderer.maskManager.popMask(renderer);\n        }\n    }\n};\n\n/**\n * Retrieves the bounds of the graphic shape as a rectangle object\n *\n * @return {Rectangle} the rectangular bounding area\n */\nGraphics.prototype.getBounds = function (matrix) {\n    // return an empty object if the item is a mask!\n    if (this.isMask) {\n        return math.Rectangle.EMPTY;\n    }\n\n    if (this.dirty) {\n        this.updateLocalBounds();\n\n        this.glDirty = true;\n        this.cachedSpriteDirty = true;\n        this.dirty = false;\n    }\n\n    var bounds = this._localBounds;\n\n    var w0 = bounds.x;\n    var w1 = bounds.width + bounds.x;\n\n    var h0 = bounds.y;\n    var h1 = bounds.height + bounds.y;\n\n    var worldTransform = matrix || this.worldTransform;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var x1 = a * w1 + c * h1 + tx;\n    var y1 = d * h1 + b * w1 + ty;\n\n    var x2 = a * w0 + c * h1 + tx;\n    var y2 = d * h1 + b * w0 + ty;\n\n    var x3 = a * w0 + c * h0 + tx;\n    var y3 = d * h0 + b * w0 + ty;\n\n    var x4 =  a * w1 + c * h0 + tx;\n    var y4 =  d * h0 + b * w1 + ty;\n\n    var maxX = x1;\n    var maxY = y1;\n\n    var minX = x1;\n    var minY = y1;\n\n    minX = x2 < minX ? x2 : minX;\n    minX = x3 < minX ? x3 : minX;\n    minX = x4 < minX ? x4 : minX;\n\n    minY = y2 < minY ? y2 : minY;\n    minY = y3 < minY ? y3 : minY;\n    minY = y4 < minY ? y4 : minY;\n\n    maxX = x2 > maxX ? x2 : maxX;\n    maxX = x3 > maxX ? x3 : maxX;\n    maxX = x4 > maxX ? x4 : maxX;\n\n    maxY = y2 > maxY ? y2 : maxY;\n    maxY = y3 > maxY ? y3 : maxY;\n    maxY = y4 > maxY ? y4 : maxY;\n\n    this._bounds.x = minX;\n    this._bounds.width = maxX - minX;\n\n    this._bounds.y = minY;\n    this._bounds.height = maxY - minY;\n\n    return this._bounds;\n};\n\n/**\n * Update the bounds of the object\n *\n */\nGraphics.prototype.updateLocalBounds = function () {\n    var minX = Infinity;\n    var maxX = -Infinity;\n\n    var minY = Infinity;\n    var maxY = -Infinity;\n\n    if (this.graphicsData.length) {\n        var shape, points, x, y, w, h;\n\n        for (var i = 0; i < this.graphicsData.length; i++) {\n            var data = this.graphicsData[i];\n            var type = data.type;\n            var lineWidth = data.lineWidth;\n            shape = data.shape;\n\n            if (type === CONST.SHAPES.RECT || type === CONST.SHAPES.RREC) {\n                x = shape.x - lineWidth/2;\n                y = shape.y - lineWidth/2;\n                w = shape.width + lineWidth;\n                h = shape.height + lineWidth;\n\n                minX = x < minX ? x : minX;\n                maxX = x + w > maxX ? x + w : maxX;\n\n                minY = y < minY ? y : minY;\n                maxY = y + h > maxY ? y + h : maxY;\n            }\n            else if (type === CONST.SHAPES.CIRC) {\n                x = shape.x;\n                y = shape.y;\n                w = shape.radius + lineWidth/2;\n                h = shape.radius + lineWidth/2;\n\n                minX = x - w < minX ? x - w : minX;\n                maxX = x + w > maxX ? x + w : maxX;\n\n                minY = y - h < minY ? y - h : minY;\n                maxY = y + h > maxY ? y + h : maxY;\n            }\n            else if (type === CONST.SHAPES.ELIP) {\n                x = shape.x;\n                y = shape.y;\n                w = shape.width + lineWidth/2;\n                h = shape.height + lineWidth/2;\n\n                minX = x - w < minX ? x - w : minX;\n                maxX = x + w > maxX ? x + w : maxX;\n\n                minY = y - h < minY ? y - h : minY;\n                maxY = y + h > maxY ? y + h : maxY;\n            }\n            else {\n                // POLY\n                points = shape.points;\n\n                for (var j = 0; j < points.length; j += 2) {\n                    x = points[j];\n                    y = points[j+1];\n\n                    minX = x-lineWidth < minX ? x-lineWidth : minX;\n                    maxX = x+lineWidth > maxX ? x+lineWidth : maxX;\n\n                    minY = y-lineWidth < minY ? y-lineWidth : minY;\n                    maxY = y+lineWidth > maxY ? y+lineWidth : maxY;\n                }\n            }\n        }\n    }\n    else {\n        minX = 0;\n        maxX = 0;\n        minY = 0;\n        maxY = 0;\n    }\n\n    var padding = this.boundsPadding;\n\n    this._localBounds.x = minX - padding;\n    this._localBounds.width = (maxX - minX) + padding * 2;\n\n    this._localBounds.y = minY - padding;\n    this._localBounds.height = (maxY - minY) + padding * 2;\n};\n\n/**\n * Generates the cached sprite when the sprite has cacheAsBitmap = true\n *\n * @private\n */\nGraphics.prototype._generateCachedSprite = function () {\n    var bounds = this.getLocalBounds();\n\n    if (!this._cachedSprite) {\n        var canvasBuffer = new CanvasBuffer(bounds.width, bounds.height);\n        var texture = Texture.fromCanvas(canvasBuffer.canvas);\n\n        this._cachedSprite = new Sprite(texture);\n        this._cachedSprite.buffer = canvasBuffer;\n\n        this._cachedSprite.worldTransform = this.worldTransform;\n    }\n    else {\n        this._cachedSprite.buffer.resize(bounds.width, bounds.height);\n    }\n\n    // leverage the anchor to account for the offset of the element\n    this._cachedSprite.anchor.x = -( bounds.x / bounds.width );\n    this._cachedSprite.anchor.y = -( bounds.y / bounds.height );\n\n    // this._cachedSprite.buffer.context.save();\n    this._cachedSprite.buffer.context.translate(-bounds.x,-bounds.y);\n\n    // make sure we set the alpha of the graphics to 1 for the render..\n    this.worldAlpha = 1;\n\n    // now render the graphic..\n    CanvasGraphics.renderGraphics(this, this._cachedSprite.buffer.context);\n\n    this._cachedSprite.alpha = this.alpha;\n};\n\n/**\n * Updates texture size based on canvas size\n *\n * @private\n */\nGraphics.prototype.updateCachedSpriteTexture = function () {\n    var cachedSprite = this._cachedSprite;\n    var texture = cachedSprite.texture;\n    var canvas = cachedSprite.buffer.canvas;\n\n    texture.baseTexture.width = canvas.width;\n    texture.baseTexture.height = canvas.height;\n    texture.crop.width = texture.frame.width = canvas.width;\n    texture.crop.height = texture.frame.height = canvas.height;\n\n    cachedSprite._width = canvas.width;\n    cachedSprite._height = canvas.height;\n\n    // update the dirty base textures\n    texture.baseTexture.dirty();\n};\n\n/**\n * Destroys a previous cached sprite.\n *\n */\nGraphics.prototype.destroyCachedSprite = function () {\n    this._cachedSprite.texture.destroy(true);\n\n    // let the gc collect the unused sprite\n    // TODO could be object pooled!\n    this._cachedSprite = null;\n};\n\n/**\n * Draws the given shape to this Graphics object. Can be any of Circle, Rectangle, Ellipse, Line or Polygon.\n *\n * @param {Circle|Rectangle|Ellipse|Line|Polygon} shape The Shape object to draw.\n * @return {GraphicsData} The generated GraphicsData object.\n */\nGraphics.prototype.drawShape = function (shape) {\n    if (this.currentPath) {\n        // check current path!\n        if (this.currentPath.shape.points.length <= 2) {\n            this.graphicsData.pop();\n        }\n    }\n\n    this.currentPath = null;\n\n    var data = new GraphicsData(this.lineWidth, this.lineColor, this.lineAlpha, this.fillColor, this.fillAlpha, this.filling, shape);\n\n    this.graphicsData.push(data);\n\n    if (data.type === CONST.SHAPES.POLY) {\n        data.shape.closed = this.filling;\n        this.currentPath = data;\n    }\n\n    this.dirty = true;\n\n    return data;\n};\n","/**\n * A GraphicsData object.\n *\n * @class\n * @namespace PIXI\n */\nfunction GraphicsData(lineWidth, lineColor, lineAlpha, fillColor, fillAlpha, fill, shape) {\n    this.lineWidth = lineWidth;\n    this.lineColor = lineColor;\n    this.lineAlpha = lineAlpha;\n    this._lineTint = lineColor;\n\n    this.fillColor = fillColor;\n    this.fillAlpha = fillAlpha;\n    this._fillTint = fillColor;\n    this.fill = fill;\n\n    this.shape = shape;\n    this.type = shape.type;\n}\n\nGraphicsData.prototype.constructor = GraphicsData;\nmodule.exports = GraphicsData;\n\n/**\n * Creates a new GraphicsData object with the same values as this one.\n *\n * @return {GraphicsData}\n */\nGraphicsData.prototype.clone = function () {\n    return new GraphicsData(\n        this.lineWidth,\n        this.lineColor,\n        this.lineAlpha,\n        this.fillColor,\n        this.fillAlpha,\n        this.fill,\n        this.shape\n    );\n};\n","var CanvasMaskManager = require('./utils/CanvasMaskManager'),\n    utils = require('../../utils'),\n    CONST = require('../../const');\n\n/**\n * The CanvasRenderer draws the scene and all its content onto a 2d canvas. This renderer should be used for browsers that do not support webGL.\n * Don't forget to add the CanvasRenderer.view to your DOM or you will not see anything :)\n *\n * @class\n * @namespace PIXI\n * @param [width=800] {number} the width of the canvas view\n * @param [height=600] {number} the height of the canvas view\n * @param [options] {object} The optional renderer parameters\n * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false\n * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n * @param [options.clearBeforeRender=true] {boolean} This sets if the CanvasRenderer will clear the canvas or not before the new render pass.\n */\nfunction CanvasRenderer(width, height, options) {\n    utils.sayHello('Canvas');\n\n    if (options) {\n        for (var i in CONST.defaultRenderOptions) {\n            if (typeof options[i] === 'undefined') {\n                options[i] = CONST.defaultRenderOptions[i];\n            }\n        }\n    }\n    else {\n        options = CONST.defaultRenderOptions;\n    }\n\n    /**\n     * The renderer type.\n     *\n     * @member {number}\n     */\n    this.type = CONST.CANVAS_RENDERER;\n\n    /**\n     * The resolution of the canvas.\n     *\n     * @member {number}\n     */\n    this.resolution = options.resolution;\n\n    /**\n     * This sets if the CanvasRenderer will clear the canvas or not before the new render pass.\n     * If the scene is NOT transparent Pixi will use a canvas sized fillRect operation every frame to set the canvas background color.\n     * If the scene is transparent Pixi will use clearRect to clear the canvas every frame.\n     * Disable this by setting this to false. For example if your game has a canvas filling background image you often don't need this set.\n     *\n     * @member {boolean}\n     * @default\n     */\n    this.clearBeforeRender = options.clearBeforeRender;\n\n    /**\n     * The background color as a number.\n     *\n     * @member {number}\n     * @private\n     */\n    this._backgroundColor = 0x000000;\n\n    /**\n     * The background color as a string.\n     *\n     * @member {string}\n     * @private\n     */\n    this._backgroundColorString = '#000000';\n\n    this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter\n\n    /**\n     * Whether the render view is transparent\n     *\n     * @member {boolean}\n     */\n    this.transparent = options.transparent;\n\n    /**\n     * Whether the render view should be resized automatically\n     *\n     * @member {boolean}\n     */\n    this.autoResize = options.autoResize || false;\n\n\n    /**\n     * The width of the canvas view\n     *\n     * @member {number}\n     * @default 800\n     */\n    this.width = width || 800;\n\n    /**\n     * The height of the canvas view\n     *\n     * @member {number}\n     * @default 600\n     */\n    this.height = height || 600;\n\n    this.width *= this.resolution;\n    this.height *= this.resolution;\n\n    /**\n     * The canvas element that everything is drawn to.\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.view = options.view || document.createElement('canvas');\n\n    /**\n     * The canvas 2d context that everything is drawn with\n     * @member {CanvasRenderingContext2D}\n     */\n    this.context = this.view.getContext('2d', { alpha: this.transparent });\n\n    /**\n     * Boolean flag controlling canvas refresh.\n     *\n     * @member {boolean}\n     */\n    this.refresh = true;\n\n    this.view.width = this.width * this.resolution;\n    this.view.height = this.height * this.resolution;\n\n    /**\n     * Internal var.\n     *\n     * @member {number}\n     */\n    this.count = 0;\n\n    /**\n     * Instance of a CanvasMaskManager, handles masking when using the canvas renderer\n     * @member {CanvasMaskManager}\n     */\n    this.maskManager = new CanvasMaskManager();\n\n    /**\n     * If true Pixi will Math.floor() x/y values when rendering, stopping pixel interpolation.\n     * Handy for crisp pixel art and speed on legacy devices.\n     *\n     * @member {boolean}\n     */\n    this.roundPixels = false;\n\n    this.scaleMode = null;\n\n    this.smoothProperty = null;\n\n    if (this.context.imageSmoothingEnabled) {\n        this.smoothProperty = 'imageSmoothingEnabled';\n    }\n    else if (this.context.webkitImageSmoothingEnabled) {\n        this.smoothProperty = 'webkitImageSmoothingEnabled';\n    }\n    else if (this.context.mozImageSmoothingEnabled) {\n        this.smoothProperty = 'mozImageSmoothingEnabled';\n    }\n    else if (this.context.oImageSmoothingEnabled) {\n        this.smoothProperty = 'oImageSmoothingEnabled';\n    }\n    else if (this.context.msImageSmoothingEnabled) {\n        this.smoothProperty = 'msImageSmoothingEnabled';\n    }\n\n    this.currentBlendMode = CONST.blendModes.NORMAL;\n\n    this.blendModes = null;\n\n    this._mapBlendModes();\n\n    this.resize(width, height);\n}\n\n// constructor\nCanvasRenderer.prototype.constructor = CanvasRenderer;\nmodule.exports = CanvasRenderer;\n\nObject.defineProperties(CanvasRenderer.prototype, {\n    /**\n     * The background color to fill if not transparent\n     *\n     * @member {number}\n     * @memberof CanvasRenderer#\n     */\n    backgroundColor: {\n        get: function () {\n            return this._backgroundColor;\n        },\n        set: function (val) {\n            this._backgroundColor = val;\n            this._backgroundColorString = utils.hex2string(val);\n        }\n    }\n});\n\n/**\n * Renders the object to this canvas view\n *\n * @param object {DisplayObject} the object to be rendered\n */\nCanvasRenderer.prototype.render = function (object) {\n    object.updateTransform();\n\n    this.context.setTransform(1,0,0,1,0,0);\n\n    this.context.globalAlpha = 1;\n\n    this.currentBlendMode = CONST.blendModes.NORMAL;\n    this.context.globalCompositeOperation = this.blendModes[CONST.blendModes.NORMAL];\n\n    if (navigator.isCocoonJS && this.view.screencanvas) {\n        this.context.fillStyle = 'black';\n        this.context.clear();\n    }\n\n    if (this.clearBeforeRender) {\n        if (this.transparent) {\n            this.context.clearRect(0, 0, this.width, this.height);\n        }\n        else {\n            this.context.fillStyle = this._backgroundColorString;\n            this.context.fillRect(0, 0, this.width , this.height);\n        }\n    }\n\n    this.renderDisplayObject(object);\n};\n\n/**\n * Removes everything from the renderer and optionally removes the Canvas DOM element.\n *\n * @param [removeView=false] {boolean} Removes the Canvas element from the DOM.\n */\nCanvasRenderer.prototype.destroy = function (removeView) {\n    if (removeView && this.view.parent) {\n        this.view.parent.removeChild(this.view);\n    }\n\n    this.view = null;\n    this.context = null;\n    this.maskManager = null;\n};\n\n/**\n * Resizes the canvas view to the specified width and height\n *\n * @param width {number} the new width of the canvas view\n * @param height {number} the new height of the canvas view\n */\nCanvasRenderer.prototype.resize = function (width, height) {\n    this.width = width * this.resolution;\n    this.height = height * this.resolution;\n\n    this.view.width = this.width;\n    this.view.height = this.height;\n\n    if (this.autoResize) {\n        this.view.style.width = this.width / this.resolution + 'px';\n        this.view.style.height = this.height / this.resolution + 'px';\n    }\n};\n\n/**\n * Renders a display object\n *\n * @param displayObject {DisplayObject} The displayObject to render\n * @private\n */\nCanvasRenderer.prototype.renderDisplayObject = function (displayObject) {\n    displayObject.renderCanvas(this);\n};\n\n/**\n * Maps Pixi blend modes to canvas blend modes.\n *\n * @private\n */\nCanvasRenderer.prototype._mapBlendModes = function () {\n    if (!this.blendModes) {\n        this.blendModes = {};\n\n        if (utils.canUseNewCanvasBlendModes()) {\n            this.blendModes[CONST.blendModes.NORMAL]        = 'source-over';\n            this.blendModes[CONST.blendModes.ADD]           = 'lighter'; //IS THIS OK???\n            this.blendModes[CONST.blendModes.MULTIPLY]      = 'multiply';\n            this.blendModes[CONST.blendModes.SCREEN]        = 'screen';\n            this.blendModes[CONST.blendModes.OVERLAY]       = 'overlay';\n            this.blendModes[CONST.blendModes.DARKEN]        = 'darken';\n            this.blendModes[CONST.blendModes.LIGHTEN]       = 'lighten';\n            this.blendModes[CONST.blendModes.COLOR_DODGE]   = 'color-dodge';\n            this.blendModes[CONST.blendModes.COLOR_BURN]    = 'color-burn';\n            this.blendModes[CONST.blendModes.HARD_LIGHT]    = 'hard-light';\n            this.blendModes[CONST.blendModes.SOFT_LIGHT]    = 'soft-light';\n            this.blendModes[CONST.blendModes.DIFFERENCE]    = 'difference';\n            this.blendModes[CONST.blendModes.EXCLUSION]     = 'exclusion';\n            this.blendModes[CONST.blendModes.HUE]           = 'hue';\n            this.blendModes[CONST.blendModes.SATURATION]    = 'saturation';\n            this.blendModes[CONST.blendModes.COLOR]         = 'color';\n            this.blendModes[CONST.blendModes.LUMINOSITY]    = 'luminosity';\n        }\n        else {\n            // this means that the browser does not support the cool new blend modes in canvas 'cough' ie 'cough'\n            this.blendModes[CONST.blendModes.NORMAL]        = 'source-over';\n            this.blendModes[CONST.blendModes.ADD]           = 'lighter'; //IS THIS OK???\n            this.blendModes[CONST.blendModes.MULTIPLY]      = 'source-over';\n            this.blendModes[CONST.blendModes.SCREEN]        = 'source-over';\n            this.blendModes[CONST.blendModes.OVERLAY]       = 'source-over';\n            this.blendModes[CONST.blendModes.DARKEN]        = 'source-over';\n            this.blendModes[CONST.blendModes.LIGHTEN]       = 'source-over';\n            this.blendModes[CONST.blendModes.COLOR_DODGE]   = 'source-over';\n            this.blendModes[CONST.blendModes.COLOR_BURN]    = 'source-over';\n            this.blendModes[CONST.blendModes.HARD_LIGHT]    = 'source-over';\n            this.blendModes[CONST.blendModes.SOFT_LIGHT]    = 'source-over';\n            this.blendModes[CONST.blendModes.DIFFERENCE]    = 'source-over';\n            this.blendModes[CONST.blendModes.EXCLUSION]     = 'source-over';\n            this.blendModes[CONST.blendModes.HUE]           = 'source-over';\n            this.blendModes[CONST.blendModes.SATURATION]    = 'source-over';\n            this.blendModes[CONST.blendModes.COLOR]         = 'source-over';\n            this.blendModes[CONST.blendModes.LUMINOSITY]    = 'source-over';\n        }\n    }\n};\n","/**\n * Creates a Canvas element of the given size.\n *\n * @class\n * @namespace PIXI\n * @param width {number} the width for the newly created canvas\n * @param height {number} the height for the newly created canvas\n */\nfunction CanvasBuffer(width, height) {\n    /**\n     * The Canvas object that belongs to this CanvasBuffer.\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.canvas = document.createElement('canvas');\n\n    /**\n     * A CanvasRenderingContext2D object representing a two-dimensional rendering context.\n     *\n     * @member {CanvasRenderingContext2D}\n     */\n    this.context = this.canvas.getContext('2d');\n\n    this.canvas.width = width;\n    this.canvas.height = height;\n}\n\nCanvasBuffer.prototype.constructor = CanvasBuffer;\nmodule.exports = CanvasBuffer;\n\nObject.defineProperties(CanvasBuffer.prototype, {\n    /**\n     * The width of the canvas buffer in pixels.\n     *\n     * @member {number}\n     * @memberof CanvasBuffer#\n     */\n    width: {\n        get: function () {\n            return this.canvas.width;\n        },\n        set: function (val) {\n            this.canvas.width = val;\n        }\n    },\n    /**\n     * The height of the canvas buffer in pixels.\n     *\n     * @member {number}\n     * @memberof CanvasBuffer#\n     */\n    height: {\n        get: function () {\n            return this.canvas.height;\n        },\n        set: function (val) {\n            this.canvas.height = val;\n        }\n    }\n});\n\n/**\n * Clears the canvas that was created by the CanvasBuffer class.\n *\n * @private\n */\nCanvasBuffer.prototype.clear = function () {\n    this.context.setTransform(1, 0, 0, 1, 0, 0);\n    this.context.clearRect(0,0, this.canvas.width, this.canvas.height);\n};\n\n/**\n * Resizes the canvas to the specified width and height.\n *\n * @param width {number} the new width of the canvas\n * @param height {number} the new height of the canvas\n */\nCanvasBuffer.prototype.resize = function (width, height) {\n    this.canvas.width = width;\n    this.canvas.height = height;\n};\n","var CONST = require('../../../const');\n\n/**\n * A set of functions used by the canvas renderer to draw the primitive graphics data.\n *\n * @namespace PIXI\n */\nvar CanvasGraphics = module.exports = {};\n\n/*\n * Renders a Graphics object to a canvas.\n *\n * @param graphics {Graphics} the actual graphics object to render\n * @param context {CanvasRenderingContext2D} the 2d drawing method of the canvas\n */\nCanvasGraphics.renderGraphics = function (graphics, context) {\n    var worldAlpha = graphics.worldAlpha;\n\n    if (graphics.dirty) {\n        this.updateGraphicsTint(graphics);\n        graphics.dirty = false;\n    }\n\n    for (var i = 0; i < graphics.graphicsData.length; i++) {\n        var data = graphics.graphicsData[i];\n        var shape = data.shape;\n\n        var fillColor = data._fillTint;\n        var lineColor = data._lineTint;\n\n        context.lineWidth = data.lineWidth;\n\n        if (data.type === CONST.SHAPES.POLY) {\n            context.beginPath();\n\n            var points = shape.points;\n\n            context.moveTo(points[0], points[1]);\n\n            for (var j=1; j < points.length/2; j++) {\n                context.lineTo(points[j * 2], points[j * 2 + 1]);\n            }\n\n            if (shape.closed) {\n                context.lineTo(points[0], points[1]);\n            }\n\n            // if the first and last point are the same close the path - much neater :)\n            if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) {\n                context.closePath();\n            }\n\n            if (data.fill) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n        else if (data.type === CONST.SHAPES.RECT) {\n\n            if (data.fillColor || data.fillColor === 0) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fillRect(shape.x, shape.y, shape.width, shape.height);\n\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.strokeRect(shape.x, shape.y, shape.width, shape.height);\n            }\n        }\n        else if (data.type === CONST.SHAPES.CIRC) {\n            // TODO - need to be Undefined!\n            context.beginPath();\n            context.arc(shape.x, shape.y, shape.radius,0,2*Math.PI);\n            context.closePath();\n\n            if (data.fill) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n        else if (data.type === CONST.SHAPES.ELIP) {\n            // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas\n\n            var w = shape.width * 2;\n            var h = shape.height * 2;\n\n            var x = shape.x - w/2;\n            var y = shape.y - h/2;\n\n            context.beginPath();\n\n            var kappa = 0.5522848,\n                ox = (w / 2) * kappa, // control point offset horizontal\n                oy = (h / 2) * kappa, // control point offset vertical\n                xe = x + w,           // x-end\n                ye = y + h,           // y-end\n                xm = x + w / 2,       // x-middle\n                ym = y + h / 2;       // y-middle\n\n            context.moveTo(x, ym);\n            context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y);\n            context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym);\n            context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye);\n            context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym);\n\n            context.closePath();\n\n            if (data.fill) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n        else if (data.type === CONST.SHAPES.RREC) {\n            var rx = shape.x;\n            var ry = shape.y;\n            var width = shape.width;\n            var height = shape.height;\n            var radius = shape.radius;\n\n            var maxRadius = Math.min(width, height) / 2 | 0;\n            radius = radius > maxRadius ? maxRadius : radius;\n\n            context.beginPath();\n            context.moveTo(rx, ry + radius);\n            context.lineTo(rx, ry + height - radius);\n            context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height);\n            context.lineTo(rx + width - radius, ry + height);\n            context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius);\n            context.lineTo(rx + width, ry + radius);\n            context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry);\n            context.lineTo(rx + radius, ry);\n            context.quadraticCurveTo(rx, ry, rx, ry + radius);\n            context.closePath();\n\n            if (data.fillColor || data.fillColor === 0) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n    }\n};\n\n/*\n * Renders a graphics mask\n *\n * @private\n * @param graphics {Graphics} the graphics which will be used as a mask\n * @param context {CanvasRenderingContext2D} the context 2d method of the canvas\n */\nCanvasGraphics.renderGraphicsMask = function (graphics, context) {\n    var len = graphics.graphicsData.length;\n\n    if (len === 0) {\n        return;\n    }\n\n    context.beginPath();\n\n    for (var i = 0; i < len; i++) {\n        var data = graphics.graphicsData[i];\n        var shape = data.shape;\n\n        if (data.type === CONST.SHAPES.POLY) {\n\n            var points = shape.points;\n\n            context.moveTo(points[0], points[1]);\n\n            for (var j=1; j < points.length/2; j++) {\n                context.lineTo(points[j * 2], points[j * 2 + 1]);\n            }\n\n            // if the first and last point are the same close the path - much neater :)\n            if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) {\n                context.closePath();\n            }\n\n        }\n        else if (data.type === CONST.SHAPES.RECT) {\n            context.rect(shape.x, shape.y, shape.width, shape.height);\n            context.closePath();\n        }\n        else if (data.type === CONST.SHAPES.CIRC) {\n            // TODO - need to be Undefined!\n            context.arc(shape.x, shape.y, shape.radius, 0, 2 * Math.PI);\n            context.closePath();\n        }\n        else if (data.type === CONST.SHAPES.ELIP) {\n\n            // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas\n\n            var w = shape.width * 2;\n            var h = shape.height * 2;\n\n            var x = shape.x - w/2;\n            var y = shape.y - h/2;\n\n            var kappa = 0.5522848,\n                ox = (w / 2) * kappa, // control point offset horizontal\n                oy = (h / 2) * kappa, // control point offset vertical\n                xe = x + w,           // x-end\n                ye = y + h,           // y-end\n                xm = x + w / 2,       // x-middle\n                ym = y + h / 2;       // y-middle\n\n            context.moveTo(x, ym);\n            context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y);\n            context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym);\n            context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye);\n            context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym);\n            context.closePath();\n        }\n        else if (data.type === CONST.SHAPES.RREC) {\n\n            var rx = shape.x;\n            var ry = shape.y;\n            var width = shape.width;\n            var height = shape.height;\n            var radius = shape.radius;\n\n            var maxRadius = Math.min(width, height) / 2 | 0;\n            radius = radius > maxRadius ? maxRadius : radius;\n\n            context.moveTo(rx, ry + radius);\n            context.lineTo(rx, ry + height - radius);\n            context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height);\n            context.lineTo(rx + width - radius, ry + height);\n            context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius);\n            context.lineTo(rx + width, ry + radius);\n            context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry);\n            context.lineTo(rx + radius, ry);\n            context.quadraticCurveTo(rx, ry, rx, ry + radius);\n            context.closePath();\n        }\n    }\n};\n\nCanvasGraphics.updateGraphicsTint = function (graphics) {\n    if (graphics.tint === 0xFFFFFF) {\n        return;\n    }\n\n    var tintR = (graphics.tint >> 16 & 0xFF) / 255;\n    var tintG = (graphics.tint >> 8 & 0xFF) / 255;\n    var tintB = (graphics.tint & 0xFF)/ 255;\n\n    for (var i = 0; i < graphics.graphicsData.length; i++) {\n        var data = graphics.graphicsData[i];\n\n        var fillColor = data.fillColor | 0;\n        var lineColor = data.lineColor | 0;\n\n        /*\n        var colorR = (fillColor >> 16 & 0xFF) / 255;\n        var colorG = (fillColor >> 8 & 0xFF) / 255;\n        var colorB = (fillColor & 0xFF) / 255;\n\n        colorR *= tintR;\n        colorG *= tintG;\n        colorB *= tintB;\n\n        fillColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255);\n\n        colorR = (lineColor >> 16 & 0xFF) / 255;\n        colorG = (lineColor >> 8 & 0xFF) / 255;\n        colorB = (lineColor & 0xFF) / 255;\n\n        colorR *= tintR;\n        colorG *= tintG;\n        colorB *= tintB;\n\n        lineColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255);\n        */\n\n        // super inline cos im an optimization NAZI :)\n        data._fillTint = (((fillColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((fillColor >> 8 & 0xFF) / 255 * tintG*255 << 8) +  (fillColor & 0xFF) / 255 * tintB*255);\n        data._lineTint = (((lineColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((lineColor >> 8 & 0xFF) / 255 * tintG*255 << 8) +  (lineColor & 0xFF) / 255 * tintB*255);\n\n    }\n};\n\n","var CanvasGraphics = require('./CanvasGraphics');\n\n/**\n * A set of functions used to handle masking.\n *\n * @class\n * @namespace PIXI\n */\nfunction CanvasMaskManager() {}\n\nCanvasMaskManager.prototype.constructor = CanvasMaskManager;\nmodule.exports = CanvasMaskManager;\n\n/**\n * This method adds it to the current stack of masks.\n *\n * @param maskData {object} the maskData that will be pushed\n * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use.\n */\nCanvasMaskManager.prototype.pushMask = function (maskData, renderer) {\n    renderer.context.save();\n\n    var cacheAlpha = maskData.alpha;\n    var transform = maskData.worldTransform;\n    var resolution = renderer.resolution;\n\n    renderer.context.setTransform(\n        transform.a * resolution,\n        transform.b * resolution,\n        transform.c * resolution,\n        transform.d * resolution,\n        transform.tx * resolution,\n        transform.ty * resolution\n    );\n\n    CanvasGraphics.renderGraphicsMask(maskData, renderer.context);\n\n    renderer.context.clip();\n\n    maskData.worldAlpha = cacheAlpha;\n};\n\n/**\n * Restores the current drawing context to the state it was before the mask was applied.\n *\n * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use.\n */\nCanvasMaskManager.prototype.popMask = function (renderer) {\n    renderer.context.restore();\n};\n","var utils = require('../../../utils');\n\n/**\n * Utility methods for Sprite/Texture tinting.\n *\n * @namespace PIXI\n */\nvar CanvasTinter = module.exports = {};\n\n/**\n * Basically this method just needs a sprite and a color and tints the sprite with the given color.\n *\n * @param sprite {Sprite} the sprite to tint\n * @param color {number} the color to use to tint the sprite with\n * @return {HTMLCanvasElement} The tinted canvas\n */\nCanvasTinter.getTintedTexture = function (sprite, color) {\n    var texture = sprite.texture;\n\n    color = CanvasTinter.roundColor(color);\n\n    var stringColor = '#' + ('00000' + ( color | 0).toString(16)).substr(-6);\n\n    texture.tintCache = texture.tintCache || {};\n\n    if (texture.tintCache[stringColor]) {\n        return texture.tintCache[stringColor];\n    }\n\n     // clone texture..\n    var canvas = CanvasTinter.canvas || document.createElement('canvas');\n\n    //CanvasTinter.tintWithPerPixel(texture, stringColor, canvas);\n    CanvasTinter.tintMethod(texture, color, canvas);\n\n    if (CanvasTinter.convertTintToImage) {\n        // is this better?\n        var tintImage = new Image();\n        tintImage.src = canvas.toDataURL();\n\n        texture.tintCache[stringColor] = tintImage;\n    }\n    else {\n        texture.tintCache[stringColor] = canvas;\n        // if we are not converting the texture to an image then we need to lose the reference to the canvas\n        CanvasTinter.canvas = null;\n    }\n\n    return canvas;\n};\n\n/**\n * Tint a texture using the 'multiply' operation.\n *\n * @param texture {Texture} the texture to tint\n * @param color {number} the color to use to tint the sprite with\n * @param canvas {HTMLCanvasElement} the current canvas\n */\nCanvasTinter.tintWithMultiply = function (texture, color, canvas) {\n    var context = canvas.getContext( '2d' );\n\n    var crop = texture.crop;\n\n    canvas.width = crop.width;\n    canvas.height = crop.height;\n\n    context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6);\n\n    context.fillRect(0, 0, crop.width, crop.height);\n\n    context.globalCompositeOperation = 'multiply';\n\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n\n    context.globalCompositeOperation = 'destination-atop';\n\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n};\n\n/**\n * Tint a texture using the 'overlay' operation.\n *\n * @param texture {Texture} the texture to tint\n * @param color {number} the color to use to tint the sprite with\n * @param canvas {HTMLCanvasElement} the current canvas\n */\nCanvasTinter.tintWithOverlay = function (texture, color, canvas) {\n    var context = canvas.getContext( '2d' );\n\n    var crop = texture.crop;\n\n    canvas.width = crop.width;\n    canvas.height = crop.height;\n\n    context.globalCompositeOperation = 'copy';\n    context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6);\n    context.fillRect(0, 0, crop.width, crop.height);\n\n    context.globalCompositeOperation = 'destination-atop';\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n\n    // context.globalCompositeOperation = 'copy';\n};\n\n/**\n * Tint a texture pixel per pixel.\n *\n * @param texture {Texture} the texture to tint\n * @param color {number} the color to use to tint the sprite with\n * @param canvas {HTMLCanvasElement} the current canvas\n */\nCanvasTinter.tintWithPerPixel = function (texture, color, canvas) {\n    var context = canvas.getContext( '2d' );\n\n    var crop = texture.crop;\n\n    canvas.width = crop.width;\n    canvas.height = crop.height;\n\n    context.globalCompositeOperation = 'copy';\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n\n    var rgbValues = utils.hex2rgb(color);\n    var r = rgbValues[0], g = rgbValues[1], b = rgbValues[2];\n\n    var pixelData = context.getImageData(0, 0, crop.width, crop.height);\n\n    var pixels = pixelData.data;\n\n    for (var i = 0; i < pixels.length; i += 4) {\n        pixels[i+0] *= r;\n        pixels[i+1] *= g;\n        pixels[i+2] *= b;\n    }\n\n    context.putImageData(pixelData, 0, 0);\n};\n\n/**\n * Rounds the specified color according to the CanvasTinter.cacheStepsPerColorChannel.\n *\n * @param color {number} the color to round, should be a hex color\n */\nCanvasTinter.roundColor = function (color) {\n    var step = CanvasTinter.cacheStepsPerColorChannel;\n\n    var rgbValues = utils.hex2rgb(color);\n\n    rgbValues[0] = Math.min(255, (rgbValues[0] / step) * step);\n    rgbValues[1] = Math.min(255, (rgbValues[1] / step) * step);\n    rgbValues[2] = Math.min(255, (rgbValues[2] / step) * step);\n\n    return utils.rgb2hex(rgbValues);\n};\n\n/**\n * Number of steps which will be used as a cap when rounding colors.\n *\n * @member\n */\nCanvasTinter.cacheStepsPerColorChannel = 8;\n\n/**\n * Tint cache boolean flag.\n *\n * @member\n */\nCanvasTinter.convertTintToImage = false;\n\n/**\n * Whether or not the Canvas BlendModes are supported, consequently the ability to tint using the multiply method.\n *\n * @member\n */\nCanvasTinter.canUseMultiply = utils.canUseNewCanvasBlendModes();\n\n/**\n * The tinting method that will be used.\n *\n */\nCanvasTinter.tintMethod = CanvasTinter.canUseMultiply ? CanvasTinter.tintWithMultiply :  CanvasTinter.tintWithPerPixel;\n","var WebGLSpriteBatch = require('./utils/WebGLSpriteBatch'),\n    WebGLFastSpriteBatch = require('./utils/WebGLFastSpriteBatch'),\n    WebGLShaderManager = require('./managers/WebGLShaderManager'),\n    WebGLMaskManager = require('./managers/WebGLMaskManager'),\n    WebGLFilterManager = require('./managers/WebGLFilterManager'),\n    WebGLStencilManager = require('./managers/WebGLStencilManager'),\n    WebGLBlendModeManager = require('./managers/WebGLBlendModeManager'),\n    math = require('../../math'),\n    utils = require('../../utils'),\n    CONST = require('../../const');\n\n/**\n * The WebGLRenderer draws the scene and all its content onto a webGL enabled canvas. This renderer\n * should be used for browsers that support webGL. This Render works by automatically managing webGLBatchs.\n * So no need for Sprite Batches or Sprite Clouds.\n * Don't forget to add the view to your DOM or you will not see anything :)\n *\n * @class\n * @namespace PIXI\n * @param [width=0] {number} the width of the canvas view\n * @param [height=0] {number} the height of the canvas view\n * @param [options] {object} The optional renderer parameters\n * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false\n * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment)\n * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you need to call toDataUrl on the webgl context\n * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n */\nfunction WebGLRenderer(width, height, options) {\n    utils.sayHello('webGL');\n\n    if (options) {\n        for (var i in CONST.defaultRenderOptions) {\n            if (typeof options[i] === 'undefined') {\n                options[i] = CONST.defaultRenderOptions[i];\n            }\n        }\n    }\n    else {\n        options = CONST.defaultRenderOptions;\n    }\n\n    this.uuid = utils.uuid();\n\n    /**\n     * @member {number}\n     */\n    this.type = CONST.WEBGL_RENDERER;\n\n    /**\n     * The resolution of the renderer\n     *\n     * @member {number}\n     * @default 1\n     */\n    this.resolution = options.resolution;\n\n    // do a catch.. only 1 webGL renderer..\n\n    /**\n     * Whether the render view is transparent\n     *\n     * @member {boolean}\n     */\n    this.transparent = options.transparent;\n\n    /**\n     * The background color as a number.\n     *\n     * @member {number}\n     * @private\n     */\n    this._backgroundColor = 0x000000;\n\n    /**\n     * The background color as an [R, G, B] array.\n     *\n     * @member {number[]}\n     * @private\n     */\n    this._backgroundColorRgb = [0, 0, 0];\n\n    this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter\n\n    /**\n     * Whether the render view should be resized automatically\n     *\n     * @member {boolean}\n     */\n    this.autoResize = options.autoResize || false;\n\n    /**\n     * The value of the preserveDrawingBuffer flag affects whether or not the contents of the stencil buffer is retained after rendering.\n     *\n     * @member {boolean}\n     */\n    this.preserveDrawingBuffer = options.preserveDrawingBuffer;\n\n    /**\n     * This sets if the WebGLRenderer will clear the context texture or not before the new render pass. If true:\n     * If the renderer is NOT transparent, Pixi will clear to alpha (0, 0, 0, 0).\n     * If the renderer is transparent, Pixi will clear to the target Stage's background color.\n     * Disable this by setting this to false. For example: if your game has a canvas filling background image, you often don't need this set.\n     *\n     * @member {boolean}\n     * @default\n     */\n    this.clearBeforeRender = options.clearBeforeRender;\n\n    /**\n     * The width of the canvas view\n     *\n     * @member {number}\n     * @default 800\n     */\n    this.width = width || 800;\n\n    /**\n     * The height of the canvas view\n     *\n     * @member {number}\n     * @default 600\n     */\n    this.height = height || 600;\n\n    /**\n     * The canvas element that everything is drawn to\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.view = options.view || document.createElement( 'canvas' );\n\n    // deal with losing context..\n\n    /**\n     * @member {Function}\n     */\n    this.contextLostBound = this.handleContextLost.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.contextRestoredBound = this.handleContextRestored.bind(this);\n\n    this.view.addEventListener('webglcontextlost', this.contextLostBound, false);\n    this.view.addEventListener('webglcontextrestored', this.contextRestoredBound, false);\n\n    /**\n     * @member {object}\n     * @private\n     */\n    this._contextOptions = {\n        alpha: this.transparent,\n        antialias: options.antialias, // SPEED UP??\n        premultipliedAlpha:this.transparent && this.transparent !== 'notMultiplied',\n        stencil:true,\n        preserveDrawingBuffer: options.preserveDrawingBuffer\n    };\n\n    /**\n     * @member {Point}\n     */\n    this.projection = new math.Point();\n\n    /**\n     * @member {Point}\n     */\n    this.offset = new math.Point(0, 0);\n\n    /**\n     * Counter for the number of draws made each frame\n     *\n     * @member {number}\n     */\n    this.drawCount = 0;\n\n    // time to create the render managers! each one focuses on managing a state in webGL\n\n    /**\n     * Deals with managing the shader programs and their attribs\n     * @member {WebGLShaderManager}\n     */\n    this.shaderManager = new WebGLShaderManager(this);\n\n    /**\n     * Manages the rendering of sprites\n     * @member {WebGLSpriteBatch}\n     */\n    this.spriteBatch = new WebGLSpriteBatch(this);\n\n    /**\n     * Manages the rendering of sprites\n     * @member {WebGLFastSpriteBatch}\n     */\n    this.fastSpriteBatch = new WebGLFastSpriteBatch(this);\n\n    /**\n     * Manages the masks using the stencil buffer\n     * @member {WebGLMaskManager}\n     */\n    this.maskManager = new WebGLMaskManager(this);\n\n    /**\n     * Manages the filters\n     * @member {WebGLFilterManager}\n     */\n    this.filterManager = new WebGLFilterManager(this);\n\n    /**\n     * Manages the stencil buffer\n     * @member {WebGLStencilManager}\n     */\n    this.stencilManager = new WebGLStencilManager(this);\n\n    /**\n     * Manages the blendModes\n     * @member {WebGLBlendModeManager}\n     */\n    this.blendModeManager = new WebGLBlendModeManager(this);\n\n    this.blendModes = null;\n\n    this._boundUpdateTexture = this.updateTexture.bind(this);\n    this._boundDestroyTexture = this.destroyTexture.bind(this);\n\n    // time init the context..\n    this._initContext();\n\n    // map some webGL blend modes..\n    this._mapBlendModes();\n}\n\n// constructor\nWebGLRenderer.prototype.constructor = WebGLRenderer;\nmodule.exports = WebGLRenderer;\n\nutils.eventTarget.mixin(WebGLRenderer.prototype);\n\nObject.defineProperties(WebGLRenderer.prototype, {\n    /**\n     * The background color to fill if not transparent\n     *\n     * @member {number}\n     * @memberof WebGLRenderer#\n     */\n    backgroundColor: {\n        get: function () {\n            return this._backgroundColor;\n        },\n        set: function (val) {\n            this._backgroundColor = val;\n            utils.hex2rgb(val, this._backgroundColorRgb);\n        }\n    }\n});\n\n/**\n *\n * @private\n */\nWebGLRenderer.prototype._initContext = function () {\n    var gl = this.view.getContext('webgl', this._contextOptions) || this.view.getContext('experimental-webgl', this._contextOptions);\n    this.gl = gl;\n\n    if (!gl) {\n        // fail, not able to get a context\n        throw new Error('This browser does not support webGL. Try using the canvas renderer');\n    }\n\n    this.glContextId = WebGLRenderer.glContextId++;\n    gl.id = this.glContextId;\n    gl.renderer = this;\n\n    // set up the default pixi settings..\n    gl.disable(gl.DEPTH_TEST);\n    gl.disable(gl.CULL_FACE);\n    gl.enable(gl.BLEND);\n\n    this.emit('context', gl);\n\n    // now resize and we are good to go!\n    this.resize(this.width, this.height);\n};\n\n/**\n * Renders the object to its webGL view\n *\n * @param object {DisplayObject} the object to be rendered\n */\nWebGLRenderer.prototype.render = function (object) {\n    // no point rendering if our context has been blown up!\n    if (this.gl.isContextLost()) {\n        return;\n    }\n\n    // update the scene graph\n    object.updateTransform();\n\n    var gl = this.gl;\n\n    // -- Does this need to be set every frame? -- //\n    gl.viewport(0, 0, this.width, this.height);\n\n    // make sure we are bound to the main frame buffer\n    gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n\n    if (this.clearBeforeRender) {\n        if (this.transparent) {\n            gl.clearColor(0, 0, 0, 0);\n        }\n        else {\n            gl.clearColor(this._backgroundColorRgb[0], this._backgroundColorRgb[1], this._backgroundColorRgb[2], 1);\n        }\n\n        gl.clear(gl.COLOR_BUFFER_BIT);\n    }\n\n    this.renderDisplayObject(object, this.projection);\n};\n\n/**\n * Renders a Display Object.\n *\n * @param displayObject {DisplayObject} The DisplayObject to render\n * @param projection {Point} The projection\n * @param buffer {Array} a standard WebGL buffer\n */\nWebGLRenderer.prototype.renderDisplayObject = function (displayObject, projection, buffer) {\n    this.blendModeManager.setBlendMode(CONST.blendModes.NORMAL);\n\n    // reset the render session data..\n    this.drawCount = 0;\n\n    // make sure to flip the Y if using a render texture..\n    this.flipY = buffer ? -1 : 1;\n\n    // set the default projection\n    this.projection = projection;\n\n    //set the default offset\n    this.offset = this.offset;\n\n    // start the sprite batch\n    this.spriteBatch.begin();\n\n    // start the filter manager\n    this.filterManager.begin(buffer);\n\n    // render the scene!\n    displayObject.renderWebGL(this);\n\n    // finish the sprite batch\n    this.spriteBatch.end();\n};\n\n/**\n * Resizes the webGL view to the specified width and height.\n *\n * @param width {number} the new width of the webGL view\n * @param height {number} the new height of the webGL view\n */\nWebGLRenderer.prototype.resize = function (width, height) {\n    this.width = width * this.resolution;\n    this.height = height * this.resolution;\n\n    this.view.width = this.width;\n    this.view.height = this.height;\n\n    if (this.autoResize) {\n        this.view.style.width = this.width / this.resolution + 'px';\n        this.view.style.height = this.height / this.resolution + 'px';\n    }\n\n    this.gl.viewport(0, 0, this.width, this.height);\n\n    this.projection.x =  this.width / 2 / this.resolution;\n    this.projection.y =  -this.height / 2 / this.resolution;\n};\n\n/**\n * Updates and/or Creates a WebGL texture for the renderer's context.\n *\n * @param texture {BaseTexture|Texture} the texture to update\n */\nWebGLRenderer.prototype.updateTexture = function (texture) {\n    texture = texture.baseTexture || texture;\n\n    if (!texture.hasLoaded) {\n        return;\n    }\n\n    var gl = this.gl;\n\n    if (!texture._glTextures[gl.id]) {\n        texture._glTextures[gl.id] = gl.createTexture();\n        texture.on('update', this._boundUpdateTexture);\n        texture.on('dispose', this._boundDestroyTexture);\n    }\n\n    gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]);\n\n    gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, texture.premultipliedAlpha);\n    gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, texture.source);\n\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n\n\n    if (texture.mipmap && utils.isPowerOfTwo(texture.width, texture.height)) {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR_MIPMAP_LINEAR : gl.NEAREST_MIPMAP_NEAREST);\n        gl.generateMipmap(gl.TEXTURE_2D);\n    }\n    else {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n    }\n\n    if (!texture._powerOf2) {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n    }\n    else {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT);\n    }\n\n    return  texture._glTextures[gl.id];\n};\n\nWebGLRenderer.prototype.destroyTexture = function (texture) {\n    texture = texture.baseTexture || texture;\n\n    if (!texture.hasLoaded) {\n        return;\n    }\n\n    if (texture._glTextures[this.gl.id]) {\n        this.gl.deleteTexture(texture._glTextures[this.gl.id]);\n    }\n};\n\n/**\n * Handles a lost webgl context\n *\n * @param event {Event}\n * @private\n */\nWebGLRenderer.prototype.handleContextLost = function (event) {\n    event.preventDefault();\n};\n\n/**\n * Handles a restored webgl context\n *\n * @param event {Event}\n * @private\n */\nWebGLRenderer.prototype.handleContextRestored = function () {\n    this._initContext();\n\n    // empty all the ol gl textures as they are useless now\n    for (var key in utils.TextureCache) {\n        var texture = utils.TextureCache[key].baseTexture;\n        texture._glTextures = [];\n    }\n};\n\n/**\n * Removes everything from the renderer (event listeners, spritebatch, etc...)\n *\n * @param [removeView=false] {boolean} Removes the Canvas element from the DOM.\n */\nWebGLRenderer.prototype.destroy = function (removeView) {\n    if (removeView && this.view.parent) {\n        this.view.parent.removeChild(this.view);\n    }\n\n    // remove listeners\n    this.view.removeEventListener('webglcontextlost', this.contextLostBound);\n    this.view.removeEventListener('webglcontextrestored', this.contextRestoredBound);\n\n    // time to create the render managers! each one focuses on managine a state in webGL\n    this.shaderManager.destroy();\n    this.spriteBatch.destroy();\n    this.maskManager.destroy();\n    this.filterManager.destroy();\n\n\n    // this.uuid = utils.uuid();\n    // this.type = CONST.WEBGL_RENDERER;\n\n    // this.resolution = options.resolution;\n    // this.transparent = options.transparent;\n\n    this._backgroundColor = 0x000000;\n    this._backgroundColorRgb = null;\n\n    // this.backgroundColor = null;\n    // this.autoResize = options.autoResize || false;\n    // this.preserveDrawingBuffer = options.preserveDrawingBuffer;\n    // this.clearBeforeRender = options.clearBeforeRender;\n    // this.width = width || 800;\n    // this.height = height || 600;\n\n    this.view = null;\n\n    this.contextLostBound = null;\n    this.contextRestoredBound = null;\n\n    this._contextOptions = null;\n\n    this.projection = null;\n    this.offset = null;\n    this.drawCount = 0;\n\n    this.shaderManager = null;\n    this.spriteBatch = null;\n    this.maskManager = null;\n    this.filterManager = null;\n    this.stencilManager = null;\n    this.blendModeManager = null;\n\n    this.blendModes = null;\n\n    this.gl = null;\n    this.blendModes = null;\n};\n\n/**\n * Maps Pixi blend modes to WebGL blend modes.\n *\n * @private\n */\nWebGLRenderer.prototype._mapBlendModes = function () {\n    var gl = this.gl;\n\n    if (!this.blendModes) {\n        this.blendModes = {};\n\n        this.blendModes[CONST.blendModes.NORMAL]        = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.ADD]           = [gl.SRC_ALPHA, gl.DST_ALPHA];\n        this.blendModes[CONST.blendModes.MULTIPLY]      = [gl.DST_COLOR, gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.SCREEN]        = [gl.SRC_ALPHA, gl.ONE];\n        this.blendModes[CONST.blendModes.OVERLAY]       = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.DARKEN]        = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.LIGHTEN]       = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.COLOR_DODGE]   = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.COLOR_BURN]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.HARD_LIGHT]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.SOFT_LIGHT]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.DIFFERENCE]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.EXCLUSION]     = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.HUE]           = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.SATURATION]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.COLOR]         = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.LUMINOSITY]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n    }\n};\n\nWebGLRenderer.glContextId = 0;\n","var WebGLManager = require('./WebGLManager');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLBlendModeManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    /**\n     * @member {number}\n     */\n    this.currentBlendMode = 99999;\n}\n\nWebGLBlendModeManager.prototype = Object.create(WebGLManager.prototype);\nWebGLBlendModeManager.prototype.constructor = WebGLBlendModeManager;\nmodule.exports = WebGLBlendModeManager;\n\n/**\n * Sets-up the given blendMode from WebGL's point of view.\n *\n * @param blendMode {number} the blendMode, should be a Pixi const, such as BlendModes.ADD\n */\nWebGLBlendModeManager.prototype.setBlendMode = function (blendMode) {\n    if (this.currentBlendMode === blendMode) {\n        return false;\n    }\n\n    this.currentBlendMode = blendMode;\n\n    var mode = this.renderer.blendModes[this.currentBlendMode];\n    this.renderer.gl.blendFunc(mode[0], mode[1]);\n\n    return true;\n};\n","var WebGLManager = require('./WebGLManager'),\n    FilterTexture = require('../utils/FilterTexture'),\n    Shader = require('../shaders/Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLFilterManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    /**\n     * @member {any[]}\n     */\n    this.filterStack = [];\n\n    /**\n     * @member {any[]]}\n     */\n    this.texturePool = [];\n\n    /**\n     * @member {number}\n     */\n    this.offsetX = 0;\n\n    /**\n     * @member {number}\n     */\n    this.offsetY = 0;\n\n    // listen for context and update necessary buffers\n    var self = this;\n    this.renderer.on('context', function () {\n        self.texturePool.length = 0;\n        self.initShaderBuffers();\n    });\n}\n\nWebGLFilterManager.prototype = Object.create(WebGLManager.prototype);\nWebGLFilterManager.prototype.constructor = WebGLFilterManager;\nmodule.exports = WebGLFilterManager;\n\n/**\n * @param renderer {WebGLRenderer}\n * @param buffer {ArrayBuffer}\n */\nWebGLFilterManager.prototype.begin = function (buffer) {\n    this.defaultShader = this.renderer.shaderManager.defaultShader;\n\n    this.width = this.renderer.projection.x * 2;\n    this.height = -this.renderer.projection.y * 2;\n\n    this.buffer = buffer;\n};\n\n/**\n * Applies the filter and adds it to the current filter stack.\n *\n * @param filterBlock {object} the filter that will be pushed to the current filter stack\n */\nWebGLFilterManager.prototype.pushFilter = function (filterBlock) {\n    var gl = this.renderer.gl;\n\n    var projection = this.renderer.projection;\n    var offset = this.renderer.offset;\n\n    filterBlock._filterArea = filterBlock.target.filterArea || filterBlock.target.getBounds();\n\n    // filter program\n    // OPTIMISATION - the first filter is free if its a simple color change?\n    this.filterStack.push(filterBlock);\n\n    var filter = filterBlock.filterPasses[0];\n\n    this.offsetX += filterBlock._filterArea.x;\n    this.offsetY += filterBlock._filterArea.y;\n\n    var texture = this.texturePool.pop();\n    if (!texture) {\n        texture = new FilterTexture(this.renderer.gl, this.width, this.height);\n    }\n    else {\n        texture.resize(this.width, this.height);\n    }\n\n    gl.bindTexture(gl.TEXTURE_2D,  texture.texture);\n\n    var filterArea = filterBlock._filterArea;// filterBlock.target.getBounds();///filterBlock.target.filterArea;\n\n    var padding = filter.padding;\n    filterArea.x -= padding;\n    filterArea.y -= padding;\n    filterArea.width += padding * 2;\n    filterArea.height += padding * 2;\n\n    var localX = filterArea.x,\n        localY = filterArea.y;\n\n    if (filterArea.x < 0) {\n        filterArea.width += filterArea.x;\n        filterArea.x = 0;\n    }\n\n    if (filterArea.y < 0) {\n        filterArea.height += filterArea.y;\n        filterArea.y = 0;\n    }\n\n    if (localX + filterArea.width > this.width) {\n        filterArea.width = this.width - localX;\n    }\n\n    if (localY + filterArea.height > this.height) {\n        filterArea.height = this.height - localY;\n    }\n\n    if (filterArea.width < 0) {\n        filterArea.width = 0;\n    }\n\n    if (filterArea.height < 0) {\n        filterArea.height = 0;\n    }\n\n    //gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA,  filterArea.width, filterArea.height, 0, gl.RGBA, gl.UNSIGNED_BYTE, null);\n    gl.bindFramebuffer(gl.FRAMEBUFFER, texture.frameBuffer);\n\n    // set view port\n    gl.viewport(0, 0, filterArea.width, filterArea.height);\n\n    projection.x = filterArea.width/2;\n    projection.y = -filterArea.height/2;\n\n    offset.x = -filterArea.x;\n    offset.y = -filterArea.y;\n\n    // update projection\n    // now restore the regular shader..\n    // this.renderer.shaderManager.setShader(this.defaultShader);\n    //gl.uniform2f(this.defaultShader.projectionVector, filterArea.width/2, -filterArea.height/2);\n    //gl.uniform2f(this.defaultShader.offsetVector, -filterArea.x, -filterArea.y);\n\n    gl.colorMask(true, true, true, true);\n    gl.clearColor(0,0,0, 0);\n    gl.clear(gl.COLOR_BUFFER_BIT);\n\n    filterBlock._glFilterTexture = texture;\n\n};\n\n/**\n * Removes the last filter from the filter stack and doesn't return it.\n *\n */\nWebGLFilterManager.prototype.popFilter = function () {\n    var gl = this.renderer.gl;\n\n    var filterBlock = this.filterStack.pop();\n    var filterArea = filterBlock._filterArea;\n    var texture = filterBlock._glFilterTexture;\n    var projection = this.renderer.projection;\n    var offset = this.renderer.offset;\n\n    if (filterBlock.filterPasses.length > 1) {\n        gl.viewport(0, 0, filterArea.width, filterArea.height);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n\n        this.vertexArray[0] = 0;\n        this.vertexArray[1] = filterArea.height;\n\n        this.vertexArray[2] = filterArea.width;\n        this.vertexArray[3] = filterArea.height;\n\n        this.vertexArray[4] = 0;\n        this.vertexArray[5] = 0;\n\n        this.vertexArray[6] = filterArea.width;\n        this.vertexArray[7] = 0;\n\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n        // now set the uvs..\n        this.uvArray[2] = filterArea.width/this.width;\n        this.uvArray[5] = filterArea.height/this.height;\n        this.uvArray[6] = filterArea.width/this.width;\n        this.uvArray[7] = filterArea.height/this.height;\n\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray);\n\n        var inputTexture = texture;\n        var outputTexture = this.texturePool.pop();\n        if (!outputTexture) {\n            outputTexture = new FilterTexture(this.renderer.gl, this.width, this.height);\n        }\n        outputTexture.resize(this.width, this.height);\n\n        // need to clear this FBO as it may have some left over elements from a previous filter.\n        gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer );\n        gl.clear(gl.COLOR_BUFFER_BIT);\n\n        gl.disable(gl.BLEND);\n\n        for (var i = 0; i < filterBlock.filterPasses.length-1; i++) {\n            var filterPass = filterBlock.filterPasses[i];\n\n            gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer );\n\n            // set texture\n            gl.activeTexture(gl.TEXTURE0);\n            gl.bindTexture(gl.TEXTURE_2D, inputTexture.texture);\n\n            // draw texture..\n            //filterPass.applyFilterPass(filterArea.width, filterArea.height);\n            this.applyFilterPass(filterPass, filterArea, filterArea.width, filterArea.height);\n\n            // swap the textures..\n            var temp = inputTexture;\n            inputTexture = outputTexture;\n            outputTexture = temp;\n        }\n\n        gl.enable(gl.BLEND);\n\n        texture = inputTexture;\n        this.texturePool.push(outputTexture);\n    }\n\n    var filter = filterBlock.filterPasses[filterBlock.filterPasses.length-1];\n\n    this.offsetX -= filterArea.x;\n    this.offsetY -= filterArea.y;\n\n    var sizeX = this.width;\n    var sizeY = this.height;\n\n    var offsetX = 0;\n    var offsetY = 0;\n\n    var buffer = this.buffer;\n\n    // time to render the filters texture to the previous scene\n    if (this.filterStack.length === 0) {\n        gl.colorMask(true, true, true, true);//this.transparent);\n    }\n    else {\n        var currentFilter = this.filterStack[this.filterStack.length-1];\n        filterArea = currentFilter._filterArea;\n\n        sizeX = filterArea.width;\n        sizeY = filterArea.height;\n\n        offsetX = filterArea.x;\n        offsetY = filterArea.y;\n\n        buffer =  currentFilter._glFilterTexture.frameBuffer;\n    }\n\n    // TODO need to remove these global elements..\n    projection.x = sizeX/2;\n    projection.y = -sizeY/2;\n\n    offset.x = offsetX;\n    offset.y = offsetY;\n\n    filterArea = filterBlock._filterArea;\n\n    var x = filterArea.x-offsetX;\n    var y = filterArea.y-offsetY;\n\n    // update the buffers..\n    // make sure to flip the y!\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n\n    this.vertexArray[0] = x;\n    this.vertexArray[1] = y + filterArea.height;\n\n    this.vertexArray[2] = x + filterArea.width;\n    this.vertexArray[3] = y + filterArea.height;\n\n    this.vertexArray[4] = x;\n    this.vertexArray[5] = y;\n\n    this.vertexArray[6] = x + filterArea.width;\n    this.vertexArray[7] = y;\n\n    gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n\n    this.uvArray[2] = filterArea.width/this.width;\n    this.uvArray[5] = filterArea.height/this.height;\n    this.uvArray[6] = filterArea.width/this.width;\n    this.uvArray[7] = filterArea.height/this.height;\n\n    gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray);\n\n    gl.viewport(0, 0, sizeX, sizeY);\n\n    // bind the buffer\n    gl.bindFramebuffer(gl.FRAMEBUFFER, buffer );\n\n    // set the blend mode!\n    //gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)\n\n    // set texture\n    gl.activeTexture(gl.TEXTURE0);\n    gl.bindTexture(gl.TEXTURE_2D, texture.texture);\n\n    // apply!\n    this.applyFilterPass(filter, filterArea, sizeX, sizeY);\n\n    // now restore the regular shader.. should happen automatically now..\n    // this.renderer.shaderManager.setShader(this.defaultShader);\n    // gl.uniform2f(this.defaultShader.projectionVector, sizeX/2, -sizeY/2);\n    // gl.uniform2f(this.defaultShader.offsetVector, -offsetX, -offsetY);\n\n    // return the texture to the pool\n    this.texturePool.push(texture);\n    filterBlock._glFilterTexture = null;\n};\n\n\n/**\n * Applies the filter to the specified area.\n *\n * @param filter {AbstractFilter} the filter that needs to be applied\n * @param filterArea {Texture} TODO - might need an update\n * @param width {number} the horizontal range of the filter\n * @param height {number} the vertical range of the filter\n */\nWebGLFilterManager.prototype.applyFilterPass = function (filter, filterArea, width, height) {\n    // use program\n    var gl = this.renderer.gl;\n\n    var shader = filter.shaders[gl.id];\n\n    if (!shader) {\n        shader = new Shader(gl);\n\n        shader.fragmentSrc = filter.fragmentSrc;\n        shader.uniforms = filter.uniforms;\n        shader.init();\n\n        filter.shaders[gl.id] = shader;\n    }\n\n    // set the shader\n    this.renderer.shaderManager.setShader(shader);\n\n//    gl.useProgram(shader.program);\n\n    gl.uniform2f(shader.projectionVector, width/2, -height/2);\n    gl.uniform2f(shader.offsetVector, 0,0);\n\n    if (filter.uniforms.dimensions) {\n        filter.uniforms.dimensions.value[0] = this.width;//width;\n        filter.uniforms.dimensions.value[1] = this.height;//height;\n        filter.uniforms.dimensions.value[2] = this.vertexArray[0];\n        filter.uniforms.dimensions.value[3] = this.vertexArray[5];//filterArea.height;\n    }\n\n    shader.syncUniforms();\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n    gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer);\n    gl.vertexAttribPointer(shader.aColor, 2, gl.FLOAT, false, 0, 0);\n\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n\n    // draw the filter...\n    gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0 );\n\n    this.renderer.drawCount++;\n};\n\n/**\n * Initialises the shader buffers.\n *\n */\nWebGLFilterManager.prototype.initShaderBuffers = function () {\n    var gl = this.renderer.gl;\n\n    // create some buffers\n    this.vertexBuffer = gl.createBuffer();\n    this.uvBuffer = gl.createBuffer();\n    this.colorBuffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n\n    // bind and upload the vertexs..\n    // keep a reference to the vertexFloatData..\n    this.vertexArray = new Float32Array([0.0, 0.0,\n                                         1.0, 0.0,\n                                         0.0, 1.0,\n                                         1.0, 1.0]);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertexArray, gl.STATIC_DRAW);\n\n    // bind and upload the uv buffer\n    this.uvArray = new Float32Array([0.0, 0.0,\n                                     1.0, 0.0,\n                                     0.0, 1.0,\n                                     1.0, 1.0]);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.uvArray, gl.STATIC_DRAW);\n\n    this.colorArray = new Float32Array([1.0, 0xFFFFFF,\n                                        1.0, 0xFFFFFF,\n                                        1.0, 0xFFFFFF,\n                                        1.0, 0xFFFFFF]);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.colorArray, gl.STATIC_DRAW);\n\n    // bind and upload the index\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array([0, 1, 2, 1, 3, 2]), gl.STATIC_DRAW);\n\n};\n\n/**\n * Destroys the filter and removes it from the filter stack.\n *\n */\nWebGLFilterManager.prototype.destroy = function () {\n    var gl = this.renderer.gl;\n\n    this.filterStack = null;\n\n    this.offsetX = 0;\n    this.offsetY = 0;\n\n    // destroy textures\n    for (var i = 0; i < this.texturePool.length; i++) {\n        this.texturePool[i].destroy();\n    }\n\n    this.texturePool = null;\n\n    //destroy buffers..\n    gl.deleteBuffer(this.vertexBuffer);\n    gl.deleteBuffer(this.uvBuffer);\n    gl.deleteBuffer(this.colorBuffer);\n    gl.deleteBuffer(this.indexBuffer);\n\n    this.renderer = null;\n};\n","/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLManager(renderer) {\n    /**\n     * The renderer this manager works for.\n     *\n     * @member {WebGLRenderer}\n     */\n    this.renderer = renderer;\n}\n\nWebGLManager.prototype.constructor = WebGLManager;\nmodule.exports = WebGLManager;\n\nWebGLManager.prototype.destroy = function () {\n    this.renderer = null;\n};\n","var WebGLManager = require('./WebGLManager'),\n    WebGLGraphics = require('../utils/WebGLGraphics');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLMaskManager(renderer) {\n    WebGLManager.call(this, renderer);\n}\n\nWebGLMaskManager.prototype = Object.create(WebGLManager.prototype);\nWebGLMaskManager.prototype.constructor = WebGLMaskManager;\nmodule.exports = WebGLMaskManager;\n\n/**\n * Applies the Mask and adds it to the current filter stack.\n *\n * @param maskData {any[]}\n */\nWebGLMaskManager.prototype.pushMask = function (maskData) {\n    if (maskData.dirty) {\n        WebGLGraphics.updateGraphics(maskData, this.renderer.gl);\n    }\n\n    if (!maskData._webGL[this.renderer.gl.id].data.length) {\n        return;\n    }\n\n    this.renderer.stencilManager.pushStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer);\n};\n\n/**\n * Removes the last filter from the filter stack and doesn't return it.\n *\n * @param maskData {any[]}\n */\nWebGLMaskManager.prototype.popMask = function (maskData) {\n    this.renderer.stencilManager.popStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer);\n};\n","var WebGLManager = require('./WebGLManager'),\n    PrimitiveShader = require('../shaders/PrimitiveShader'),\n    ComplexPrimitiveShader = require('../shaders/ComplexPrimitiveShader'),\n    Shader = require('../shaders/Shader'),\n    FastShader = require('../shaders/FastShader'),\n    StripShader = require('../shaders/StripShader');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLShaderManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    /**\n     * @member {number}\n     */\n    this.maxAttibs = 10;\n\n    /**\n     * @member {any[]}\n     */\n    this.attribState = [];\n\n    /**\n     * @member {any[]}\n     */\n    this.tempAttribState = [];\n\n    for (var i = 0; i < this.maxAttibs; i++) {\n        this.attribState[i] = false;\n    }\n\n    /**\n     * @member {any[]}\n     */\n    this.stack = [];\n\n    /**\n     * @member {number}\n     * @private\n     */\n    this._currentId = -1;\n\n    /**\n     * @member {Shader}\n     * @private\n     */\n    this.currentShader = null;\n\n    // this shader is used for rendering primitives\n    this.primitiveShader = null;\n\n    // this shader is used for rendering triangle strips\n    this.complexPrimitiveShader = null;\n\n    // this shader is used for the default sprite rendering\n    this.defaultShader = null;\n\n    // this shader is used for the fast sprite rendering\n    this.fastShader = null;\n\n    // the next one is used for rendering triangle strips\n    this.stripShader = null;\n\n    // listen for context and update necessary shaders\n    var self = this;\n    this.renderer.on('context', function (event) {\n        var gl = event.data;\n\n        // this shader is used for rendering primitives\n        self.primitiveShader = new PrimitiveShader(gl);\n\n        // this shader is used for rendering triangle strips\n        self.complexPrimitiveShader = new ComplexPrimitiveShader(gl);\n\n        // this shader is used for the default sprite rendering\n        self.defaultShader = new Shader(gl);\n\n        // this shader is used for the fast sprite rendering\n        self.fastShader = new FastShader(gl);\n\n        // the next one is used for rendering triangle strips\n        self.stripShader = new StripShader(gl);\n\n        self.setShader(self.defaultShader);\n    });\n}\n\nWebGLShaderManager.prototype = Object.create(WebGLManager.prototype);\nWebGLShaderManager.prototype.constructor = WebGLShaderManager;\nmodule.exports = WebGLShaderManager;\n\n/**\n * Takes the attributes given in parameters.\n *\n * @param attribs {Array} attribs\n */\nWebGLShaderManager.prototype.setAttribs = function (attribs) {\n    // reset temp state\n    var i;\n\n    for (i = 0; i < this.tempAttribState.length; i++) {\n        this.tempAttribState[i] = false;\n    }\n\n    // set the new attribs\n    for (var a in attribs) {\n        this.tempAttribState[attribs[a]] = true;\n    }\n\n    var gl = this.renderer.gl;\n\n    for (i = 0; i < this.attribState.length; i++) {\n        if (this.attribState[i] !== this.tempAttribState[i]) {\n            this.attribState[i] = this.tempAttribState[i];\n\n            if (this.attribState[i]) {\n                gl.enableVertexAttribArray(i);\n            }\n            else {\n                gl.disableVertexAttribArray(i);\n            }\n        }\n    }\n};\n\n/**\n * Sets the current shader.\n *\n * @param shader {Any}\n */\nWebGLShaderManager.prototype.setShader = function (shader) {\n    if (this._currentId === shader.uuid) {\n        return false;\n    }\n\n    this._currentId = shader.uuid;\n\n    this.currentShader = shader;\n\n    this.renderer.gl.useProgram(shader.program);\n    this.setAttribs(shader.attributes);\n\n    return true;\n};\n\n/**\n * Destroys this object.\n *\n */\nWebGLShaderManager.prototype.destroy = function () {\n    this.attribState = null;\n\n    this.tempAttribState = null;\n\n    this.primitiveShader.destroy();\n    this.primitiveShader = null;\n\n    this.complexPrimitiveShader.destroy();\n    this.complexPrimitiveShader = null;\n\n    this.defaultShader.destroy();\n    this.defaultShader = null;\n\n    this.fastShader.destroy();\n    this.fastShader = null;\n\n    this.stripShader.destroy();\n    this.stripShader = null;\n\n    this.renderer = null;\n};\n","var WebGLManager = require('./WebGLManager'),\n    utils = require('../../../utils');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLStencilManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    this.stencilStack = [];\n    this.reverse = true;\n    this.count = 0;\n}\n\nWebGLStencilManager.prototype = Object.create(WebGLManager.prototype);\nWebGLStencilManager.prototype.constructor = WebGLStencilManager;\nmodule.exports = WebGLStencilManager;\n\n/**\n * Applies the Mask and adds it to the current filter stack.\n *\n * @param graphics {Graphics}\n * @param webGLData {any[]}\n */\nWebGLStencilManager.prototype.pushStencil = function (graphics, webGLData) {\n    var gl = this.renderer.gl;\n\n    this.bindGraphics(graphics, webGLData, this.renderer);\n\n    if (this.stencilStack.length === 0) {\n        gl.enable(gl.STENCIL_TEST);\n        gl.clear(gl.STENCIL_BUFFER_BIT);\n        this.reverse = true;\n        this.count = 0;\n    }\n\n    this.stencilStack.push(webGLData);\n\n    var level = this.count;\n\n    gl.colorMask(false, false, false, false);\n\n    gl.stencilFunc(gl.ALWAYS,0,0xFF);\n    gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT);\n\n    // draw the triangle strip!\n\n    if (webGLData.mode === 1) {\n        gl.drawElements(gl.TRIANGLE_FAN,  webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 );\n\n        if (this.reverse) {\n            gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n        }\n\n        // draw a quad to increment..\n        gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 );\n\n        if (this.reverse) {\n            gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n        }\n\n        this.reverse = !this.reverse;\n    }\n    else {\n        if (!this.reverse) {\n            gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n        }\n\n        gl.drawElements(gl.TRIANGLE_STRIP,  webGLData.indices.length, gl.UNSIGNED_SHORT, 0 );\n\n        if (!this.reverse) {\n            gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n        }\n    }\n\n    gl.colorMask(true, true, true, true);\n    gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP);\n\n    this.count++;\n};\n\n/**\n * TODO this does not belong here!\n *\n * @param graphics {Graphics}\n * @param webGLData {Array}\n */\nWebGLStencilManager.prototype.bindGraphics = function (graphics, webGLData) {\n    //if (this._currentGraphics === graphics)return;\n    this._currentGraphics = graphics;\n\n    var gl = this.renderer.gl;\n\n     // bind the graphics object..\n    var projection = this.renderer.projection,\n        offset = this.renderer.offset,\n        shader;// = this.renderer.shaderManager.primitiveShader;\n\n    if (webGLData.mode === 1) {\n        shader = this.renderer.shaderManager.complexPrimitiveShader;\n\n        this.renderer.shaderManager.setShader(shader);\n\n        gl.uniform1f(shader.flipY, this.renderer.flipY);\n\n        gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true));\n\n        gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n        gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n\n        gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint));\n        gl.uniform3fv(shader.color, webGLData.color);\n\n        gl.uniform1f(shader.alpha, graphics.worldAlpha * webGLData.alpha);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer);\n\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 2, 0);\n\n\n        // now do the rest..\n        // set the index buffer!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer);\n    }\n    else {\n        //this.renderer.shaderManager.activatePrimitiveShader();\n        shader = this.renderer.shaderManager.primitiveShader;\n        this.renderer.shaderManager.setShader( shader );\n\n        gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true));\n\n        gl.uniform1f(shader.flipY, this.renderer.flipY);\n        gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n        gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n\n        gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint));\n\n        gl.uniform1f(shader.alpha, graphics.worldAlpha);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer);\n\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0);\n        gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4);\n\n        // set the index buffer!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer);\n    }\n};\n\n/**\n * @param graphics {Graphics}\n * @param webGLData {Array}\n */\nWebGLStencilManager.prototype.popStencil = function (graphics, webGLData) {\n\tvar gl = this.renderer.gl;\n\n    this.stencilStack.pop();\n\n    this.count--;\n\n    if (this.stencilStack.length === 0) {\n        // the stack is empty!\n        gl.disable(gl.STENCIL_TEST);\n\n    }\n    else {\n\n        var level = this.count;\n\n        this.bindGraphics(graphics, webGLData, this.renderer);\n\n        gl.colorMask(false, false, false, false);\n\n        if (webGLData.mode === 1) {\n            this.reverse = !this.reverse;\n\n            if (this.reverse) {\n                gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n            }\n\n            // draw a quad to increment..\n            gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 );\n\n            gl.stencilFunc(gl.ALWAYS,0,0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT);\n\n            // draw the triangle strip!\n            gl.drawElements(gl.TRIANGLE_FAN,  webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 );\n\n            if (!this.reverse) {\n                gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            }\n\n        }\n        else {\n          //  console.log(\"<<>>\")\n            if (!this.reverse) {\n                gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n            }\n\n            gl.drawElements(gl.TRIANGLE_STRIP,  webGLData.indices.length, gl.UNSIGNED_SHORT, 0 );\n\n            if (!this.reverse) {\n                gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            }\n        }\n\n        gl.colorMask(true, true, true, true);\n        gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP);\n\n\n    }\n};\n\n/**\n * Destroys the mask stack.\n *\n */\nWebGLStencilManager.prototype.destroy = function () {\n    this.renderer = null;\n    this.stencilStack = null;\n};\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction ComplexPrimitiveShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            // 'attribute vec2 aTextureCoord;',\n            // 'attribute vec4 aColor;',\n\n            'uniform mat3 translationMatrix;',\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n\n            'uniform vec3 tint;',\n            'uniform float alpha;',\n            'uniform vec3 color;',\n            'uniform float flipY;',\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);',\n            '   v -= offsetVector.xyx;',\n            '   gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);',\n            '   vColor = vec4(color * alpha * tint, alpha);',//\" * vec4(tint * alpha, alpha);',\n            '}'\n        ].join('\\n'),\n        // fragment shader\n        [\n            'precision mediump float;',\n\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   gl_FragColor = vColor;',\n            '}'\n        ].join('\\n'),\n        // custom uniforms\n        {\n            tint:   { type: '3f', value: [0, 0, 0] },\n            flipY:  { type: '1f', value: 0 },\n            alpha:  { type: '1f', value: 0 },\n            translationMatrix: { type: 'mat3', value: new Float32Array(9) }\n        }\n    );\n}\n\nComplexPrimitiveShader.prototype = Object.create(Shader.prototype);\nComplexPrimitiveShader.prototype.constructor = ComplexPrimitiveShader;\nmodule.exports = ComplexPrimitiveShader;\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @extends Shader\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction FastShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            'attribute vec2 aTextureCoord;',\n            'attribute vec4 aColor;',\n\n            'attribute vec2 aPositionCoord;',\n            'attribute vec2 aScale;',\n            'attribute float aRotation;',\n\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n            'uniform mat3 uMatrix;',\n\n            'varying vec2 vTextureCoord;',\n            'varying vec4 vColor;',\n\n            'const vec2 center = vec2(-1.0, 1.0);',\n\n            'void main(void) {',\n            '   vec2 v;',\n            '   vec2 sv = aVertexPosition * aScale;',\n            '   v.x = (sv.x) * cos(aRotation) - (sv.y) * sin(aRotation);',\n            '   v.y = (sv.x) * sin(aRotation) + (sv.y) * cos(aRotation);',\n            '   v = ( uMatrix * vec3(v + aPositionCoord , 1.0) ).xy ;',\n            '   gl_Position = vec4( ( v / projectionVector) + center , 0.0, 1.0);',\n            '   vTextureCoord = aTextureCoord;',\n          //  '   vec3 color = mod(vec3(aColor.y/65536.0, aColor.y/256.0, aColor.y), 256.0) / 256.0;',\n            '   vColor = aColor;',\n            '}'\n        ].join('\\n'),\n        // fragment shader, use default\n        null,\n        // custom uniforms\n        {\n            uMatrix: { type: 'mat3', value: new Float32Array(9) }\n        },\n        // custom attributes\n        {\n            aPositionCoord: 0,\n            aRotation:      0,\n            aScale:         0\n        }\n    );\n}\n\nFastShader.prototype = Object.create(Shader.prototype);\nFastShader.prototype.constructor = FastShader;\nmodule.exports = FastShader;\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction PrimitiveShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            // 'attribute vec2 aTextureCoord;',\n            'attribute vec4 aColor;',\n\n            'uniform mat3 translationMatrix;',\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n            'uniform float alpha;',\n            'uniform float flipY;',\n            'uniform vec3 tint;',\n\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);',\n            '   v -= offsetVector.xyx;',\n            '   gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);',\n            '   vColor = aColor * vec4(tint * alpha, alpha);',\n            '}'\n        ].join('\\n'),\n        // fragment shader\n        [\n            'precision mediump float;',\n\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   gl_FragColor = vColor;',\n            '}'\n        ].join('\\n'),\n        // custom uniforms\n        {\n            tint:   { type: '3f', value: [0, 0, 0] },\n            flipY:  { type: '1f', value: 0 },\n            alpha:  { type: '1f', value: 0 },\n            translationMatrix: { type: 'mat3', value: new Float32Array(9) }\n        }\n    );\n}\n\nPrimitiveShader.prototype = Object.create(Shader.prototype);\nPrimitiveShader.prototype.constructor = PrimitiveShader;\nmodule.exports = PrimitiveShader;\n","var utils = require('../../../utils');\n\n/**\n * @class\n * @namespace PIXI\n * @param [fragmentSrc] {string} The source of the fragment shader.\n * @param [vertexSrc] {string} The source of the vertex shader.\n */\nfunction Shader(gl, vertexSrc, fragmentSrc, customUniforms, customAttributes) {\n    /**\n     * @member {number}\n     * @readonly\n     */\n    this.uuid = utils.uuid();\n\n    /**\n     * @member {WebGLContext}\n     * @readonly\n     */\n    this.gl = gl;\n\n    /**\n     * The WebGL program.\n     * @member {WebGLProgram}\n     * @readonly\n     */\n    this.program = null;\n\n    this.uniforms = {\n        uSampler:           { type: 'sampler2D', value: 0 },\n        projectionVector:   { type: '2f', value: { x: 0, y: 0 } },\n        offsetVector:       { type: '2f', value: { x: 0, y: 0 } },\n        dimensions:         { type: '4f', value: new Float32Array(4) }\n    };\n\n    for (var u in customUniforms) {\n        this.uniforms[u] = customUniforms[u];\n    }\n\n    this.attributes = {\n        aVertexPosition:    0,\n        aTextureCoord:      0,\n        aColor:             0\n    };\n\n    for (var a in customAttributes) {\n        this.attributes[a] = customAttributes[a];\n    }\n\n    this.textureCount = 0;\n\n    /**\n     * The vertex shader.\n     * @member {Array}\n     */\n    this.vertexSrc = vertexSrc || [\n        'attribute vec2 aVertexPosition;',\n        'attribute vec2 aTextureCoord;',\n        'attribute vec4 aColor;',\n\n        'uniform vec2 projectionVector;',\n        'uniform vec2 offsetVector;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'const vec2 center = vec2(-1.0, 1.0);',\n\n        'void main(void) {',\n        '   gl_Position = vec4( ((aVertexPosition + offsetVector) / projectionVector) + center , 0.0, 1.0);',\n        '   vTextureCoord = aTextureCoord;',\n        '   vColor = vec4(aColor.rgb * aColor.a, aColor.a);',\n        '}'\n    ].join('\\n');\n\n    /**\n     * The fragment shader.\n     * @member {Array}\n     */\n    this.fragmentSrc = fragmentSrc || [\n        'precision lowp float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord) * vColor ;',\n        '}'\n    ].join('\\n');\n\n    this.init();\n}\n\nShader.prototype.constructor = Shader;\nmodule.exports = Shader;\n\nShader.prototype.init = function () {\n    this.compile();\n\n    this.gl.useProgram(this.program);\n\n    this.cacheUniformLocations(Object.keys(this.uniforms));\n    this.cacheAttributeLocations(Object.keys(this.attributes));\n};\n\nShader.prototype.cacheUniformLocations = function (keys) {\n    for (var i = 0; i < keys.length; ++i) {\n        this.uniforms[keys[i]]._location = this.gl.getUniformLocation(this.program, keys[i]);\n    }\n};\n\nShader.prototype.cacheAttributeLocations = function (keys) {\n    for (var i = 0; i < keys.length; ++i) {\n        this.attributes[keys[i]] = this.gl.getAttribLocation(this.program, keys[i]);\n    }\n\n    // TODO: Check if this is needed anymore...\n\n    // Begin worst hack eva //\n\n    // WHY??? ONLY on my chrome pixel the line above returns -1 when using filters?\n    // maybe its something to do with the current state of the gl context.\n    // I'm convinced this is a bug in the chrome browser as there is NO reason why this should be returning -1 especially as it only manifests on my chrome pixel\n    // If theres any webGL people that know why could happen please help :)\n    // if (this.attributes.aColor === -1) {\n    //     this.attributes.aColor = 2;\n    // }\n\n    // End worst hack eva //\n};\n\nShader.prototype.compile = function () {\n    var gl = this.gl;\n\n    var glVertShader = this._glCompile(gl.VERTEX_SHADER, this.vertexSrc);\n    var glFragShader = this._glCompile(gl.FRAGMENT_SHADER, this.fragmentSrc);\n\n    var program = gl.createProgram();\n\n    gl.attachShader(program, glVertShader);\n    gl.attachShader(program, glFragShader);\n    gl.linkProgram(program);\n\n    // if linking fails, then log and cleanup\n    if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {\n        window.console.error('Pixi.js Error: Could not initialize shader.');\n        window.console.error('gl.VALIDATE_STATUS', gl.getProgramParameter(program, gl.VALIDATE_STATUS));\n        window.console.error('gl.getError()', gl.getError());\n\n        // if there is a program info log, log it\n        if (gl.getProgramInfoLog(program) !== '') {\n            window.console.warn('Pixi.js Warning: gl.getProgramInfoLog()', gl.getProgramInfoLog(program));\n        }\n\n        gl.deleteProgram(program);\n        program = null;\n    }\n\n    // clean up some shaders\n    gl.deleteShader(glVertShader);\n    gl.deleteShader(glFragShader);\n\n    return (this.program = program);\n};\n\nShader.prototype.syncUniforms = function () {\n    var gl = this.gl;\n\n    this.textureCount = 1;\n\n    for (var key in this.uniforms) {\n        var uniform = this.uniforms[key],\n            location = uniform._location,\n            value = uniform.value,\n            i, il;\n\n        switch (uniform.type) {\n            case 'i':\n            case '1i':\n                gl.uniform1i(location, value);\n                break;\n\n            case 'f':\n            case '1f':\n                gl.uniform1f(location, value);\n                break;\n\n            case '2f':\n                gl.uniform2f(location, value[0], value[1]);\n                break;\n\n            case '3f':\n                gl.uniform3f(location, value[0], value[1], value[2]);\n                break;\n\n            case '4f':\n                gl.uniform4f(location, value[0], value[1], value[2], value[3]);\n                break;\n\n            // a 2D Point object\n            case 'v2':\n                gl.uniform2f(location, value.x, value.y);\n                break;\n\n            // a 3D Point object\n            case 'v3':\n                gl.uniform3f(location, value.x, value.y, value.z);\n                break;\n\n            // a 4D Point object\n            case 'v4':\n                gl.uniform4f(location, value.x, value.y, value.z, value.w);\n                break;\n\n            case '1iv':\n                gl.uniform1iv(location, value);\n                break;\n\n            case '3iv':\n                gl.uniform3iv(location, value);\n                break;\n\n            case '1fv':\n                gl.uniform1fv(location, value);\n                break;\n\n            case '2fv':\n                gl.uniform2fv(location, value);\n                break;\n\n            case '3fv':\n                gl.uniform3fv(location, value);\n                break;\n\n            case '4fv':\n                gl.uniform4fv(location, value);\n                break;\n\n            case 'm2':\n            case 'mat2':\n            case 'Matrix2fv':\n                gl.uniformMatrix2fv(location, uniform.transpose, value);\n                break;\n\n            case 'm3':\n            case 'mat3':\n            case 'Matrix3fv':\n                gl.uniformMatrix3fv(location, uniform.transpose, value);\n                break;\n\n            case 'm4':\n            case 'mat4':\n            case 'Matrix4fv':\n                gl.uniformMatrix4fv(location, uniform.transpose, value);\n                break;\n\n            // a Color Value\n            case 'c':\n                if (typeof value === 'number') {\n                    value = utils.hex2rgb(value);\n                }\n\n                gl.uniform3f(location, value[0], value[1], value[2]);\n                break;\n\n            // flat array of integers (JS or typed array)\n            case 'iv1':\n                gl.uniform1iv(location, value);\n                break;\n\n            // flat array of integers with 3 x N size (JS or typed array)\n            case 'iv':\n                gl.uniform3iv(location, value);\n                break;\n\n            // flat array of floats (JS or typed array)\n            case 'fv1':\n                gl.uniform1fv(location, value);\n                break;\n\n            // flat array of floats with 3 x N size (JS or typed array)\n            case 'fv':\n                gl.uniform3fv(location, value);\n                break;\n\n            // array of 2D Point objects\n            case 'v2v':\n                if (!uniform._array) {\n                    uniform._array = new Float32Array(2 * value.length);\n                }\n\n                for (i = 0, il = value.length; i < il; ++i) {\n                    uniform._array[i * 2]       = value[i].x;\n                    uniform._array[i * 2 + 1]   = value[i].y;\n                }\n\n                gl.uniform2fv(location, uniform._array);\n                break;\n\n            // array of 3D Point objects\n            case 'v3v':\n                if (!uniform._array) {\n                    uniform._array = new Float32Array(3 * value.length);\n                }\n\n                for (i = 0, il = value.length; i < il; ++i) {\n                    uniform._array[i * 3]       = value[i].x;\n                    uniform._array[i * 3 + 1]   = value[i].y;\n                    uniform._array[i * 3 + 2]   = value[i].z;\n\n                }\n\n                gl.uniform3fv(location, uniform._array);\n                break;\n\n            // array of 4D Point objects\n            case 'v4v':\n                if (!uniform._array) {\n                    uniform._array = new Float32Array(4 * value.length);\n                }\n\n                for (i = 0, il = value.length; i < il; ++i) {\n                    uniform._array[i * 4]       = value[i].x;\n                    uniform._array[i * 4 + 1]   = value[i].y;\n                    uniform._array[i * 4 + 2]   = value[i].z;\n                    uniform._array[i * 4 + 3]   = value[i].w;\n\n                }\n\n                gl.uniform4fv(location, uniform._array);\n                break;\n\n            case 't':\n            case 'sampler2D':\n                if (!uniform.value || !uniform.value.baseTexture || !uniform.value.baseTexture.hasLoaded) {\n                    break;\n                }\n\n                // activate this texture\n                gl.activeTexture(gl['TEXTURE' + this.textureCount]);\n\n                // bind the texture\n                gl.bindTexture(gl.TEXTURE_2D, uniform.value.baseTexture._glTextures[gl.id]);\n\n                // set uniform to texture index\n                gl.uniform1i(uniform._location, this.textureCount);\n\n                // increment next texture id\n                this.textureCount++;\n\n                // initialize the texture if we haven't yet\n                if (!uniform._init) {\n                    this.initSampler2D(uniform);\n\n                    uniform._init = true;\n                }\n                break;\n\n            default:\n                window.console.warn('Pixi.js Shader Warning: Unknown uniform type: ' + uniform.type);\n        }\n    }\n};\n\n\n/**\n * Initialises a Sampler2D uniform (which may only be available later on after initUniforms once the texture has loaded)\n *\n */\nShader.prototype.initSampler2D = function (uniform) {\n    var gl = this.gl;\n\n    //  Extended texture data\n    if (uniform.textureData) {\n        var data = uniform.textureData;\n\n        // GLTexture = mag linear, min linear_mipmap_linear, wrap repeat + gl.generateMipmap(gl.TEXTURE_2D);\n        // GLTextureLinear = mag/min linear, wrap clamp\n        // GLTextureNearestRepeat = mag/min NEAREST, wrap repeat\n        // GLTextureNearest = mag/min nearest, wrap clamp\n        // AudioTexture = whatever + luminance + width 512, height 2, border 0\n        // KeyTexture = whatever + luminance + width 256, height 2, border 0\n\n        //  magFilter can be: gl.LINEAR, gl.LINEAR_MIPMAP_LINEAR or gl.NEAREST\n        //  wrapS/T can be: gl.CLAMP_TO_EDGE or gl.REPEAT\n\n        var magFilter = (data.magFilter) ? data.magFilter : gl.LINEAR;\n        var minFilter = (data.minFilter) ? data.minFilter : gl.LINEAR;\n        var wrapS = (data.wrapS) ? data.wrapS : gl.CLAMP_TO_EDGE;\n        var wrapT = (data.wrapT) ? data.wrapT : gl.CLAMP_TO_EDGE;\n        var format = (data.luminance) ? gl.LUMINANCE : gl.RGBA;\n\n        if (data.repeat) {\n            wrapS = gl.REPEAT;\n            wrapT = gl.REPEAT;\n        }\n\n        gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, !!data.flipY);\n\n        if (data.width) {\n            var width = (data.width) ? data.width : 512;\n            var height = (data.height) ? data.height : 2;\n            var border = (data.border) ? data.border : 0;\n\n            // void texImage2D(GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, ArrayBufferView? pixels);\n            gl.texImage2D(gl.TEXTURE_2D, 0, format, width, height, border, format, gl.UNSIGNED_BYTE, null);\n        }\n        else {\n            //  void texImage2D(GLenum target, GLint level, GLenum internalformat, GLenum format, GLenum type, ImageData? pixels);\n            gl.texImage2D(gl.TEXTURE_2D, 0, format, gl.RGBA, gl.UNSIGNED_BYTE, uniform.value.baseTexture.source);\n        }\n\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, magFilter);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrapS);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrapT);\n    }\n};\n\n/**\n * Destroys the shader.\n *\n */\nShader.prototype.destroy = function () {\n    this.gl.deleteProgram(this.program);\n\n    this.gl = null;\n    this.uniforms = null;\n    this.attributes = null;\n\n    this.vertexSrc = null;\n    this.fragmentSrc = null;\n};\n\nShader.prototype._glCompile = function (type, src) {\n    var shader = this.gl.createShader(type);\n\n    if (Array.isArray(src)) {\n        debugger;\n    }\n\n    this.gl.shaderSource(shader, src);\n    this.gl.compileShader(shader);\n\n    if (!this.gl.getShaderParameter(shader, this.gl.COMPILE_STATUS)) {\n        window.console.log(this.gl.getShaderInfoLog(shader));\n        return null;\n    }\n\n    return shader;\n};\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction StripShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            'attribute vec2 aTextureCoord;',\n            // 'attribute vec4 aColor;',\n\n            'uniform mat3 translationMatrix;',\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n\n            'varying vec2 vTextureCoord;',\n\n            'void main(void) {',\n            '   vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);',\n            '   v -= offsetVector.xyx;',\n            '   gl_Position = vec4( v.x / projectionVector.x -1.0, v.y / -projectionVector.y + 1.0 , 0.0, 1.0);',\n            '   vTextureCoord = aTextureCoord;',\n            '}'\n        ].join('\\n'),\n        // fragment shader\n        [\n            'precision mediump float;',\n\n            'uniform float alpha;',\n            'uniform sampler2D uSampler;',\n\n            'varying vec2 vTextureCoord;',\n\n            'void main(void) {',\n            '   gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * alpha;',\n            '}'\n        ].join('\\n'),\n        // custom uniforms\n        {\n            alpha:  { type: '1f', value: 0 },\n            translationMatrix: { type: 'mat3', value: new Float32Array(9) }\n        }\n    );\n}\n\nStripShader.prototype = Object.create(Shader.prototype);\nStripShader.prototype.constructor = StripShader;\nmodule.exports = StripShader;\n","var CONST = require('../../../const');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n * @param width {number} the horizontal range of the filter\n * @param height {number} the vertical range of the filter\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n */\nfunction FilterTexture(gl, width, height, scaleMode) {\n    /**\n     * @member {WebGLContext}\n     */\n    this.gl = gl;\n\n    // next time to create a frame buffer and texture\n\n    /**\n     * @member {Any}\n     */\n    this.frameBuffer = gl.createFramebuffer();\n\n    /**\n     * @member {Any}\n     */\n    this.texture = gl.createTexture();\n\n    /**\n     * @member {number}\n     */\n    scaleMode = scaleMode || CONST.scaleModes.DEFAULT;\n\n    gl.bindTexture(gl.TEXTURE_2D,  this.texture);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n    gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer );\n\n    gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer );\n    gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0);\n\n    // required for masking a mask??\n    this.renderBuffer = gl.createRenderbuffer();\n    gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer);\n    gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, this.renderBuffer);\n\n    // reset render buffer\n    gl.bindRenderbuffer(gl.RENDERBUFFER, null);\n\n    this.resize(width, height);\n}\n\nFilterTexture.prototype.constructor = FilterTexture;\nmodule.exports = FilterTexture;\n\n/**\n * Clears the filter texture.\n *\n */\nFilterTexture.prototype.clear = function () {\n    var gl = this.gl;\n\n    gl.clearColor(0,0,0, 0);\n    gl.clear(gl.COLOR_BUFFER_BIT);\n};\n\n/**\n * Resizes the texture to the specified width and height\n *\n * @param width {number} the new width of the texture\n * @param height {number} the new height of the texture\n */\nFilterTexture.prototype.resize = function (width, height) {\n    if (this.width === width && this.height === height) {\n        return;\n    }\n\n    this.width = width;\n    this.height = height;\n\n    var gl = this.gl;\n\n    gl.bindTexture(gl.TEXTURE_2D,  this.texture);\n    gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA,  width , height , 0, gl.RGBA, gl.UNSIGNED_BYTE, null);\n    // update the stencil buffer width and height\n    gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer);\n    gl.renderbufferStorage(gl.RENDERBUFFER, gl.DEPTH_STENCIL, width , height);\n\n    // reset render buffer\n    gl.bindRenderbuffer(gl.RENDERBUFFER, null);\n};\n\n/**\n * Destroys the filter texture.\n *\n */\nFilterTexture.prototype.destroy = function () {\n    var gl = this.gl;\n    gl.deleteFramebuffer( this.frameBuffer );\n    gl.deleteTexture( this.texture );\n\n    this.frameBuffer = null;\n    this.texture = null;\n};\n","/**\n * @author Mat Groves\n *\n * Big thanks to the very clever Matt DesLauriers <mattdesl> https://github.com/mattdesl/\n * for creating the original pixi version!\n *\n * Heavily inspired by LibGDX's WebGLSpriteBatch:\n * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java\n */\n\n/**\n * @class\n * @private\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this sprite batch works for.\n */\nfunction WebGLFastSpriteBatch(renderer) {\n    /**\n     * The renderer instance this sprite batch operates on.\n     *\n     * @member {WebGLRenderer}\n     */\n    this.renderer = renderer;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.vertSize = 10;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.maxSize = 6000;//Math.pow(2, 16) /  this.vertSize;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.size = this.maxSize;\n\n    //the total number of floats in our batch\n    var numVerts = this.size * 4 *  this.vertSize;\n\n    //the total number of indices in our batch\n    var numIndices = this.maxSize * 6;\n\n    /**\n     * Vertex data\n     *\n     * @member {Float32Array}\n     */\n    this.vertices = new Float32Array(numVerts);\n\n    /**\n     * Index data\n     *\n     * @member {Uint16Array}\n     */\n    this.indices = new Uint16Array(numIndices);\n\n    /**\n     *\n     *\n     * @member {object}\n     */\n    this.vertexBuffer = null;\n\n    /**\n     *\n     *\n     * @member {object}\n     */\n    this.indexBuffer = null;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.lastIndexCount = 0;\n\n    for (var i=0, j=0; i < numIndices; i += 6, j += 4) {\n        this.indices[i + 0] = j + 0;\n        this.indices[i + 1] = j + 1;\n        this.indices[i + 2] = j + 2;\n        this.indices[i + 3] = j + 0;\n        this.indices[i + 4] = j + 2;\n        this.indices[i + 5] = j + 3;\n    }\n\n    /**\n     *\n     *\n     * @member {boolean}\n     */\n    this.drawing = false;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.currentBatchSize = 0;\n\n    /**\n     *\n     *\n     * @member {BaseTexture}\n     */\n    this.currentBaseTexture = null;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.currentBlendMode = 0;\n\n    /**\n     *\n     *\n     * @member {object}\n     */\n    this.shader = null;\n\n    /**\n     *\n     *\n     * @member {Matrix}\n     */\n    this.matrix = null;\n\n    // listen for context and update necessary buffers\n    var self = this;\n    this.renderer.on('context', function () {\n        self.setupContext();\n    });\n}\n\nWebGLFastSpriteBatch.prototype.constructor = WebGLFastSpriteBatch;\nmodule.exports = WebGLFastSpriteBatch;\n\n/**\n * Sets the WebGL Context.\n *\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nWebGLFastSpriteBatch.prototype.setupContext = function () {\n    var gl = this.renderer.gl;\n\n    // create a couple of buffers\n    this.vertexBuffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n\n    // 65535 is max index, so 65535 / 6 = 10922.\n\n    //upload the index data\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW);\n};\n\n/**\n * @param spriteBatch {SpriteBatch} The SpriteBatch container to prepare for.\n */\nWebGLFastSpriteBatch.prototype.begin = function (spriteBatch) {\n    this.shader = this.renderer.shaderManager.fastShader;\n\n    this.matrix = spriteBatch.worldTransform.toArray(true);\n\n    this.start();\n};\n\n/**\n */\nWebGLFastSpriteBatch.prototype.end = function () {\n    this.flush();\n};\n\n/**\n * @param spriteBatch {SpriteBatch} The SpriteBatch container to render.\n */\nWebGLFastSpriteBatch.prototype.render = function (spriteBatch) {\n    var children = spriteBatch.children;\n    var sprite = children[0];\n\n    // if the uvs have not updated then no point rendering just yet!\n\n    // check texture.\n    if (!sprite.texture._uvs) {\n        return;\n    }\n\n    this.currentBaseTexture = sprite.texture.baseTexture;\n\n    // check blend mode\n    if (sprite.blendMode !== this.renderer.blendModeManager.currentBlendMode) {\n        this.flush();\n        this.renderer.blendModeManager.setBlendMode(sprite.blendMode);\n    }\n\n    for (var i=0,j= children.length; i<j; i++) {\n        this.renderSprite(children[i]);\n    }\n\n    this.flush();\n};\n\n/**\n * @param sprite {Sprite} The Sprite to render.\n */\nWebGLFastSpriteBatch.prototype.renderSprite = function (sprite) {\n    //sprite = children[i];\n    if (!sprite.visible) {\n        return;\n    }\n\n    // TODO trim??\n    if (sprite.texture.baseTexture !== this.currentBaseTexture) {\n        this.flush();\n        this.currentBaseTexture = sprite.texture.baseTexture;\n\n        if (!sprite.texture._uvs) {\n            return;\n        }\n    }\n\n    var uvs, vertices = this.vertices, width, height, w0, w1, h0, h1, index;\n\n    uvs = sprite.texture._uvs;\n\n    width = sprite.texture.frame.width;\n    height = sprite.texture.frame.height;\n\n    if (sprite.texture.trim) {\n        // if the sprite is trimmed then we need to add the extra space before transforming the sprite coords..\n        var trim = sprite.texture.trim;\n\n        w1 = trim.x - sprite.anchor.x * trim.width;\n        w0 = w1 + sprite.texture.crop.width;\n\n        h1 = trim.y - sprite.anchor.y * trim.height;\n        h0 = h1 + sprite.texture.crop.height;\n    }\n    else {\n        w0 = (sprite.texture.frame.width ) * (1-sprite.anchor.x);\n        w1 = (sprite.texture.frame.width ) * -sprite.anchor.x;\n\n        h0 = sprite.texture.frame.height * (1-sprite.anchor.y);\n        h1 = sprite.texture.frame.height * -sprite.anchor.y;\n    }\n\n    index = this.currentBatchSize * 4 * this.vertSize;\n\n    // xy\n    vertices[index++] = w1;\n    vertices[index++] = h1;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n    //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x0;\n    vertices[index++] = uvs.y1;\n    // color\n    vertices[index++] = sprite.alpha;\n\n\n    // xy\n    vertices[index++] = w0;\n    vertices[index++] = h1;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n     //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x1;\n    vertices[index++] = uvs.y1;\n    // color\n    vertices[index++] = sprite.alpha;\n\n\n    // xy\n    vertices[index++] = w0;\n    vertices[index++] = h0;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n     //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x2;\n    vertices[index++] = uvs.y2;\n    // color\n    vertices[index++] = sprite.alpha;\n\n\n\n\n    // xy\n    vertices[index++] = w1;\n    vertices[index++] = h0;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n     //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x3;\n    vertices[index++] = uvs.y3;\n    // color\n    vertices[index++] = sprite.alpha;\n\n    // increment the batchs\n    this.currentBatchSize++;\n\n    if (this.currentBatchSize >= this.size) {\n        this.flush();\n    }\n};\n\n/**\n *\n */\nWebGLFastSpriteBatch.prototype.flush = function () {\n    // If the batch is length 0 then return as there is nothing to draw\n    if (this.currentBatchSize === 0) {\n        return;\n    }\n\n    var gl = this.renderer.gl;\n\n    // bind the current texture\n    if (!this.currentBaseTexture._glTextures[gl.id]) {\n        this.renderer.updateTexture(this.currentBaseTexture, gl);\n    }\n    //TODO-SHOUD THIS BE ELSE??!?!?!\n    else {\n        gl.bindTexture(gl.TEXTURE_2D, this.currentBaseTexture._glTextures[gl.id]);\n    }\n\n    // upload the verts to the buffer\n\n    if (this.currentBatchSize > ( this.size * 0.5 ) ) {\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices);\n    }\n    else {\n        var view = this.vertices.subarray(0, this.currentBatchSize * 4 * this.vertSize);\n\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, view);\n    }\n\n    // now draw those suckas!\n    gl.drawElements(gl.TRIANGLES, this.currentBatchSize * 6, gl.UNSIGNED_SHORT, 0);\n\n    // then reset the batch!\n    this.currentBatchSize = 0;\n\n    // increment the draw count\n    this.renderer.drawCount++;\n};\n\n\n/**\n * Ends the batch and flushes\n *\n */\nWebGLFastSpriteBatch.prototype.stop = function () {\n    this.flush();\n};\n\n/**\n *\n */\nWebGLFastSpriteBatch.prototype.start = function () {\n    var gl = this.renderer.gl;\n\n    // bind the main texture\n    gl.activeTexture(gl.TEXTURE0);\n\n    // bind the buffers\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n\n    // set the projection\n    var projection = this.renderer.projection;\n    gl.uniform2f(this.shader.projectionVector, projection.x, projection.y);\n\n    // set the matrix\n    gl.uniformMatrix3fv(this.shader.uMatrix, false, this.matrix);\n\n    // set the pointers\n    var stride =  this.vertSize * 4;\n\n    gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0);\n    gl.vertexAttribPointer(this.shader.attributes.aPositionCoord, 2, gl.FLOAT, false, stride, 2 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aScale, 2, gl.FLOAT, false, stride, 4 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aRotation, 1, gl.FLOAT, false, stride, 6 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 7 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aColor, 1, gl.FLOAT, false, stride, 9 * 4);\n};\n","var utils = require('../../../utils'),\n    math = require('../../../math'),\n    CONST = require('../../../const'),\n    WebGLGraphicsData = require('./WebGLGraphicsData');\n\n/**\n * A set of functions used by the webGL renderer to draw the primitive graphics data\n *\n * @namespace PIXI\n * @private\n */\nvar WebGLGraphics = module.exports = {};\n\n/**\n * Renders the graphics object\n *\n * @static\n * @private\n * @param graphics {Graphics}\n * @param renderer {WebGLRenderer}\n */\nWebGLGraphics.renderGraphics = function (graphics, renderer) {//projection, offset) {\n    var gl = renderer.gl;\n\n    var projection = renderer.projection,\n        offset = renderer.offset,\n        shader = renderer.shaderManager.primitiveShader,\n        webGLData;\n\n    if (graphics.dirty) {\n        WebGLGraphics.updateGraphics(graphics, gl);\n    }\n\n    var webGL = graphics._webGL[gl.id];\n\n    // This  could be speeded up for sure!\n\n    for (var i = 0; i < webGL.data.length; i++) {\n        if (webGL.data[i].mode === 1) {\n            webGLData = webGL.data[i];\n\n            renderer.stencilManager.pushStencil(graphics, webGLData, renderer);\n\n            // render quad..\n            gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 );\n\n            renderer.stencilManager.popStencil(graphics, webGLData, renderer);\n        }\n        else {\n            webGLData = webGL.data[i];\n\n\n            renderer.shaderManager.setShader( shader );//activatePrimitiveShader();\n            shader = renderer.shaderManager.primitiveShader;\n            gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true));\n\n            gl.uniform1f(shader.flipY, 1);\n\n            gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n            gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n\n            gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint));\n\n            gl.uniform1f(shader.alpha, graphics.worldAlpha);\n\n\n            gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer);\n\n            gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0);\n            gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4);\n\n            // set the index buffer!\n            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer);\n            gl.drawElements(gl.TRIANGLE_STRIP,  webGLData.indices.length, gl.UNSIGNED_SHORT, 0 );\n        }\n    }\n};\n\n/**\n * Updates the graphics object\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object to update\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nWebGLGraphics.updateGraphics = function (graphics, gl) {\n    // get the contexts graphics object\n    var webGL = graphics._webGL[gl.id];\n\n    // if the graphics object does not exist in the webGL context time to create it!\n    if (!webGL) {\n        webGL = graphics._webGL[gl.id] = {lastIndex:0, data:[], gl:gl};\n    }\n\n    // flag the graphics as not dirty as we are about to update it...\n    graphics.dirty = false;\n\n    var i;\n\n    // if the user cleared the graphics object we will need to clear every object\n    if (graphics.clearDirty) {\n        graphics.clearDirty = false;\n\n        // lop through and return all the webGLDatas to the object pool so than can be reused later on\n        for (i = 0; i < webGL.data.length; i++) {\n            var graphicsData = webGL.data[i];\n            graphicsData.reset();\n            WebGLGraphics.graphicsDataPool.push( graphicsData );\n        }\n\n        // clear the array and reset the index..\n        webGL.data = [];\n        webGL.lastIndex = 0;\n    }\n\n    var webGLData;\n\n    // loop through the graphics datas and construct each one..\n    // if the object is a complex fill then the new stencil buffer technique will be used\n    // other wise graphics objects will be pushed into a batch..\n    for (i = webGL.lastIndex; i < graphics.graphicsData.length; i++) {\n        var data = graphics.graphicsData[i];\n\n        if (data.type === CONST.SHAPES.POLY) {\n            // need to add the points the the graphics object..\n            data.points = data.shape.points.slice();\n            if (data.shape.closed) {\n                // close the poly if the value is true!\n                if (data.points[0] !== data.points[data.points.length-2] || data.points[1] !== data.points[data.points.length-1]) {\n                    data.points.push(data.points[0], data.points[1]);\n                }\n            }\n\n            // MAKE SURE WE HAVE THE CORRECT TYPE..\n            if (data.fill) {\n                if (data.points.length >= 6) {\n                    if (data.points.length < 6 * 2) {\n                        webGLData = WebGLGraphics.switchMode(webGL, 0);\n\n                        var canDrawUsingSimple = WebGLGraphics.buildPoly(data, webGLData);\n                   //     console.log(canDrawUsingSimple);\n\n                        if (!canDrawUsingSimple) {\n                        //    console.log(\"<>>>\")\n                            webGLData = WebGLGraphics.switchMode(webGL, 1);\n                            WebGLGraphics.buildComplexPoly(data, webGLData);\n                        }\n\n                    }\n                    else {\n                        webGLData = WebGLGraphics.switchMode(webGL, 1);\n                        WebGLGraphics.buildComplexPoly(data, webGLData);\n                    }\n                }\n            }\n\n            if (data.lineWidth > 0) {\n                webGLData = WebGLGraphics.switchMode(webGL, 0);\n                WebGLGraphics.buildLine(data, webGLData);\n\n            }\n        }\n        else {\n            webGLData = WebGLGraphics.switchMode(webGL, 0);\n\n            if (data.type === CONST.SHAPES.RECT) {\n                WebGLGraphics.buildRectangle(data, webGLData);\n            }\n            else if (data.type === CONST.SHAPES.CIRC || data.type === CONST.SHAPES.ELIP) {\n                WebGLGraphics.buildCircle(data, webGLData);\n            }\n            else if (data.type === CONST.SHAPES.RREC) {\n                WebGLGraphics.buildRoundedRectangle(data, webGLData);\n            }\n        }\n\n        webGL.lastIndex++;\n    }\n\n    // upload all the dirty data...\n    for (i = 0; i < webGL.data.length; i++) {\n        webGLData = webGL.data[i];\n\n        if (webGLData.dirty) {\n            webGLData.upload();\n        }\n    }\n};\n\n/**\n * @static\n * @private\n * @param webGL {WebGLContext}\n * @param type {number}\n */\nWebGLGraphics.switchMode = function (webGL, type) {\n    var webGLData;\n\n    if (!webGL.data.length) {\n        webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl);\n        webGLData.mode = type;\n        webGL.data.push(webGLData);\n    }\n    else {\n        webGLData = webGL.data[webGL.data.length-1];\n\n        if (webGLData.mode !== type || type === 1) {\n            webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl);\n            webGLData.mode = type;\n            webGL.data.push(webGLData);\n        }\n    }\n\n    webGLData.dirty = true;\n\n    return webGLData;\n};\n\n/**\n * Builds a rectangle to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildRectangle = function (graphicsData, webGLData) {\n    // --- //\n    // need to convert points to a nice regular data\n    //\n    var rectData = graphicsData.shape;\n    var x = rectData.x;\n    var y = rectData.y;\n    var width = rectData.width;\n    var height = rectData.height;\n\n    if (graphicsData.fill) {\n        var color = utils.hex2rgb(graphicsData.fillColor);\n        var alpha = graphicsData.fillAlpha;\n\n        var r = color[0] * alpha;\n        var g = color[1] * alpha;\n        var b = color[2] * alpha;\n\n        var verts = webGLData.points;\n        var indices = webGLData.indices;\n\n        var vertPos = verts.length/6;\n\n        // start\n        verts.push(x, y);\n        verts.push(r, g, b, alpha);\n\n        verts.push(x + width, y);\n        verts.push(r, g, b, alpha);\n\n        verts.push(x , y + height);\n        verts.push(r, g, b, alpha);\n\n        verts.push(x + width, y + height);\n        verts.push(r, g, b, alpha);\n\n        // insert 2 dead triangles..\n        indices.push(vertPos, vertPos, vertPos+1, vertPos+2, vertPos+3, vertPos+3);\n    }\n\n    if (graphicsData.lineWidth) {\n        var tempPoints = graphicsData.points;\n\n        graphicsData.points = [x, y,\n                  x + width, y,\n                  x + width, y + height,\n                  x, y + height,\n                  x, y];\n\n\n        WebGLGraphics.buildLine(graphicsData, webGLData);\n\n        graphicsData.points = tempPoints;\n    }\n};\n\n/**\n * Builds a rounded rectangle to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildRoundedRectangle = function (graphicsData, webGLData) {\n    var rrectData = graphicsData.shape;\n    var x = rrectData.x;\n    var y = rrectData.y;\n    var width = rrectData.width;\n    var height = rrectData.height;\n\n    var radius = rrectData.radius;\n\n    var recPoints = [];\n    recPoints.push(x, y + radius);\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x, y + height - radius, x, y + height, x + radius, y + height));\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width - radius, y + height, x + width, y + height, x + width, y + height - radius));\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width, y + radius, x + width, y, x + width - radius, y));\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + radius, y, x, y, x, y + radius));\n\n    if (graphicsData.fill) {\n        var color = utils.hex2rgb(graphicsData.fillColor);\n        var alpha = graphicsData.fillAlpha;\n\n        var r = color[0] * alpha;\n        var g = color[1] * alpha;\n        var b = color[2] * alpha;\n\n        var verts = webGLData.points;\n        var indices = webGLData.indices;\n\n        var vecPos = verts.length/6;\n\n        var triangles = utils.PolyK.Triangulate(recPoints);\n\n        //\n\n        var i = 0;\n        for (i = 0; i < triangles.length; i+=3) {\n            indices.push(triangles[i] + vecPos);\n            indices.push(triangles[i] + vecPos);\n            indices.push(triangles[i+1] + vecPos);\n            indices.push(triangles[i+2] + vecPos);\n            indices.push(triangles[i+2] + vecPos);\n        }\n\n\n        for (i = 0; i < recPoints.length; i++) {\n            verts.push(recPoints[i], recPoints[++i], r, g, b, alpha);\n        }\n    }\n\n    if (graphicsData.lineWidth) {\n        var tempPoints = graphicsData.points;\n\n        graphicsData.points = recPoints;\n\n        WebGLGraphics.buildLine(graphicsData, webGLData);\n\n        graphicsData.points = tempPoints;\n    }\n};\n\n/**\n * Calculate the points for a quadratic bezier curve. (helper function..)\n * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c\n *\n * @static\n * @private\n * @param fromX {number} Origin point x\n * @param fromY {number} Origin point x\n * @param cpX {number} Control point x\n * @param cpY {number} Control point y\n * @param toX {number} Destination point x\n * @param toY {number} Destination point y\n * @return {number[]}\n */\nWebGLGraphics.quadraticBezierCurve = function (fromX, fromY, cpX, cpY, toX, toY) {\n\n    var xa,\n        ya,\n        xb,\n        yb,\n        x,\n        y,\n        n = 20,\n        points = [];\n\n    function getPt(n1 , n2, perc) {\n        var diff = n2 - n1;\n\n        return n1 + ( diff * perc );\n    }\n\n    var j = 0;\n    for (var i = 0; i <= n; i++ ) {\n        j = i / n;\n\n        // The Green Line\n        xa = getPt( fromX , cpX , j );\n        ya = getPt( fromY , cpY , j );\n        xb = getPt( cpX , toX , j );\n        yb = getPt( cpY , toY , j );\n\n        // The Black Dot\n        x = getPt( xa , xb , j );\n        y = getPt( ya , yb , j );\n\n        points.push(x, y);\n    }\n    return points;\n};\n\n/**\n * Builds a circle to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object to draw\n * @param webGLData {object}\n */\nWebGLGraphics.buildCircle = function (graphicsData, webGLData) {\n    // need to convert points to a nice regular data\n    var circleData = graphicsData.shape;\n    var x = circleData.x;\n    var y = circleData.y;\n    var width;\n    var height;\n\n    // TODO - bit hacky??\n    if (graphicsData.type === CONST.SHAPES.CIRC) {\n        width = circleData.radius;\n        height = circleData.radius;\n    }\n    else {\n        width = circleData.width;\n        height = circleData.height;\n    }\n\n    var totalSegs = 40;\n    var seg = (Math.PI * 2) / totalSegs ;\n\n    var i = 0;\n\n    if (graphicsData.fill) {\n        var color = utils.hex2rgb(graphicsData.fillColor);\n        var alpha = graphicsData.fillAlpha;\n\n        var r = color[0] * alpha;\n        var g = color[1] * alpha;\n        var b = color[2] * alpha;\n\n        var verts = webGLData.points;\n        var indices = webGLData.indices;\n\n        var vecPos = verts.length/6;\n\n        indices.push(vecPos);\n\n        for (i = 0; i < totalSegs + 1 ; i++) {\n            verts.push(x,y, r, g, b, alpha);\n\n            verts.push(x + Math.sin(seg * i) * width,\n                       y + Math.cos(seg * i) * height,\n                       r, g, b, alpha);\n\n            indices.push(vecPos++, vecPos++);\n        }\n\n        indices.push(vecPos-1);\n    }\n\n    if (graphicsData.lineWidth) {\n        var tempPoints = graphicsData.points;\n\n        graphicsData.points = [];\n\n        for (i = 0; i < totalSegs + 1; i++) {\n            graphicsData.points.push(x + Math.sin(seg * i) * width,\n                                     y + Math.cos(seg * i) * height);\n        }\n\n        WebGLGraphics.buildLine(graphicsData, webGLData);\n\n        graphicsData.points = tempPoints;\n    }\n};\n\n/**\n * Builds a line to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildLine = function (graphicsData, webGLData) {\n    // TODO OPTIMISE!\n    var i = 0;\n    var points = graphicsData.points;\n\n    if (points.length === 0) {\n        return;\n    }\n\n    // if the line width is an odd number add 0.5 to align to a whole pixel\n    if (graphicsData.lineWidth%2) {\n        for (i = 0; i < points.length; i++) {\n            points[i] += 0.5;\n        }\n    }\n\n    // get first and last point.. figure out the middle!\n    var firstPoint = new math.Point(points[0], points[1]);\n    var lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]);\n\n    // if the first point is the last point - gonna have issues :)\n    if (firstPoint.x === lastPoint.x && firstPoint.y === lastPoint.y) {\n        // need to clone as we are going to slightly modify the shape..\n        points = points.slice();\n\n        points.pop();\n        points.pop();\n\n        lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]);\n\n        var midPointX = lastPoint.x + (firstPoint.x - lastPoint.x) *0.5;\n        var midPointY = lastPoint.y + (firstPoint.y - lastPoint.y) *0.5;\n\n        points.unshift(midPointX, midPointY);\n        points.push(midPointX, midPointY);\n    }\n\n    var verts = webGLData.points;\n    var indices = webGLData.indices;\n    var length = points.length / 2;\n    var indexCount = points.length;\n    var indexStart = verts.length/6;\n\n    // DRAW the Line\n    var width = graphicsData.lineWidth / 2;\n\n    // sort color\n    var color = utils.hex2rgb(graphicsData.lineColor);\n    var alpha = graphicsData.lineAlpha;\n    var r = color[0] * alpha;\n    var g = color[1] * alpha;\n    var b = color[2] * alpha;\n\n    var px, py, p1x, p1y, p2x, p2y, p3x, p3y;\n    var perpx, perpy, perp2x, perp2y, perp3x, perp3y;\n    var a1, b1, c1, a2, b2, c2;\n    var denom, pdist, dist;\n\n    p1x = points[0];\n    p1y = points[1];\n\n    p2x = points[2];\n    p2y = points[3];\n\n    perpx = -(p1y - p2y);\n    perpy =  p1x - p2x;\n\n    dist = Math.sqrt(perpx*perpx + perpy*perpy);\n\n    perpx /= dist;\n    perpy /= dist;\n    perpx *= width;\n    perpy *= width;\n\n    // start\n    verts.push(p1x - perpx , p1y - perpy,\n                r, g, b, alpha);\n\n    verts.push(p1x + perpx , p1y + perpy,\n                r, g, b, alpha);\n\n    for (i = 1; i < length-1; i++) {\n        p1x = points[(i-1)*2];\n        p1y = points[(i-1)*2 + 1];\n\n        p2x = points[(i)*2];\n        p2y = points[(i)*2 + 1];\n\n        p3x = points[(i+1)*2];\n        p3y = points[(i+1)*2 + 1];\n\n        perpx = -(p1y - p2y);\n        perpy = p1x - p2x;\n\n        dist = Math.sqrt(perpx*perpx + perpy*perpy);\n        perpx /= dist;\n        perpy /= dist;\n        perpx *= width;\n        perpy *= width;\n\n        perp2x = -(p2y - p3y);\n        perp2y = p2x - p3x;\n\n        dist = Math.sqrt(perp2x*perp2x + perp2y*perp2y);\n        perp2x /= dist;\n        perp2y /= dist;\n        perp2x *= width;\n        perp2y *= width;\n\n        a1 = (-perpy + p1y) - (-perpy + p2y);\n        b1 = (-perpx + p2x) - (-perpx + p1x);\n        c1 = (-perpx + p1x) * (-perpy + p2y) - (-perpx + p2x) * (-perpy + p1y);\n        a2 = (-perp2y + p3y) - (-perp2y + p2y);\n        b2 = (-perp2x + p2x) - (-perp2x + p3x);\n        c2 = (-perp2x + p3x) * (-perp2y + p2y) - (-perp2x + p2x) * (-perp2y + p3y);\n\n        denom = a1*b2 - a2*b1;\n\n        if (Math.abs(denom) < 0.1 ) {\n\n            denom+=10.1;\n            verts.push(p2x - perpx , p2y - perpy,\n                r, g, b, alpha);\n\n            verts.push(p2x + perpx , p2y + perpy,\n                r, g, b, alpha);\n\n            continue;\n        }\n\n        px = (b1*c2 - b2*c1)/denom;\n        py = (a2*c1 - a1*c2)/denom;\n\n\n        pdist = (px -p2x) * (px -p2x) + (py -p2y) + (py -p2y);\n\n\n        if (pdist > 140 * 140) {\n            perp3x = perpx - perp2x;\n            perp3y = perpy - perp2y;\n\n            dist = Math.sqrt(perp3x*perp3x + perp3y*perp3y);\n            perp3x /= dist;\n            perp3y /= dist;\n            perp3x *= width;\n            perp3y *= width;\n\n            verts.push(p2x - perp3x, p2y -perp3y);\n            verts.push(r, g, b, alpha);\n\n            verts.push(p2x + perp3x, p2y +perp3y);\n            verts.push(r, g, b, alpha);\n\n            verts.push(p2x - perp3x, p2y -perp3y);\n            verts.push(r, g, b, alpha);\n\n            indexCount++;\n        }\n        else {\n\n            verts.push(px , py);\n            verts.push(r, g, b, alpha);\n\n            verts.push(p2x - (px-p2x), p2y - (py - p2y));\n            verts.push(r, g, b, alpha);\n        }\n    }\n\n    p1x = points[(length-2)*2];\n    p1y = points[(length-2)*2 + 1];\n\n    p2x = points[(length-1)*2];\n    p2y = points[(length-1)*2 + 1];\n\n    perpx = -(p1y - p2y);\n    perpy = p1x - p2x;\n\n    dist = Math.sqrt(perpx*perpx + perpy*perpy);\n    perpx /= dist;\n    perpy /= dist;\n    perpx *= width;\n    perpy *= width;\n\n    verts.push(p2x - perpx , p2y - perpy);\n    verts.push(r, g, b, alpha);\n\n    verts.push(p2x + perpx , p2y + perpy);\n    verts.push(r, g, b, alpha);\n\n    indices.push(indexStart);\n\n    for (i = 0; i < indexCount; i++) {\n        indices.push(indexStart++);\n    }\n\n    indices.push(indexStart-1);\n};\n\n/**\n * Builds a complex polygon to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildComplexPoly = function (graphicsData, webGLData) {\n    //TODO - no need to copy this as it gets turned into a FLoat32Array anyways..\n    var points = graphicsData.points.slice();\n\n    if (points.length < 6) {\n        return;\n    }\n\n    // get first and last point.. figure out the middle!\n    var indices = webGLData.indices;\n    webGLData.points = points;\n    webGLData.alpha = graphicsData.fillAlpha;\n    webGLData.color = utils.hex2rgb(graphicsData.fillColor);\n\n    // calclate the bounds..\n    var minX = Infinity;\n    var maxX = -Infinity;\n\n    var minY = Infinity;\n    var maxY = -Infinity;\n\n    var x,y;\n\n    // get size..\n    for (var i = 0; i < points.length; i+=2) {\n        x = points[i];\n        y = points[i+1];\n\n        minX = x < minX ? x : minX;\n        maxX = x > maxX ? x : maxX;\n\n        minY = y < minY ? y : minY;\n        maxY = y > maxY ? y : maxY;\n    }\n\n    // add a quad to the end cos there is no point making another buffer!\n    points.push(minX, minY,\n                maxX, minY,\n                maxX, maxY,\n                minX, maxY);\n\n    // push a quad onto the end..\n\n    //TODO - this aint needed!\n    var length = points.length / 2;\n    for (i = 0; i < length; i++) {\n        indices.push( i );\n    }\n\n};\n\n/**\n * Builds a polygon to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildPoly = function (graphicsData, webGLData) {\n    var points = graphicsData.points;\n\n    if (points.length < 6) {\n        return;\n    }\n    // get first and last point.. figure out the middle!\n    var verts = webGLData.points;\n    var indices = webGLData.indices;\n\n    var length = points.length / 2;\n\n    // sort color\n    var color = utils.hex2rgb(graphicsData.fillColor);\n    var alpha = graphicsData.fillAlpha;\n    var r = color[0] * alpha;\n    var g = color[1] * alpha;\n    var b = color[2] * alpha;\n\n    var triangles = utils.PolyK.Triangulate(points);\n\n    if (!triangles) {\n        return false;\n    }\n\n    var vertPos = verts.length / 6;\n\n    var i = 0;\n\n    for (i = 0; i < triangles.length; i+=3) {\n        indices.push(triangles[i] + vertPos);\n        indices.push(triangles[i] + vertPos);\n        indices.push(triangles[i+1] + vertPos);\n        indices.push(triangles[i+2] +vertPos);\n        indices.push(triangles[i+2] + vertPos);\n    }\n\n    for (i = 0; i < length; i++) {\n        verts.push(points[i * 2], points[i * 2 + 1],\n                   r, g, b, alpha);\n    }\n\n    return true;\n};\n\nWebGLGraphics.graphicsDataPool = [];\n","/**\n * @class\n * @private\n */\nfunction WebGLGraphicsData(gl) {\n    this.gl = gl;\n\n    //TODO does this need to be split before uploding??\n    this.color = [0, 0, 0]; // color split!\n    this.points = [];\n    this.indices = [];\n    this.buffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n    this.mode = 1;\n    this.alpha = 1;\n    this.dirty = true;\n}\n\nWebGLGraphicsData.prototype.constructor = WebGLGraphicsData;\nmodule.exports = WebGLGraphicsData;\n\n/**\n *\n */\nWebGLGraphicsData.prototype.reset = function () {\n    this.points.length = 0;\n    this.indices.length = 0;\n};\n\n/**\n *\n */\nWebGLGraphicsData.prototype.upload = function () {\n    var gl = this.gl;\n\n//    this.lastIndex = graphics.graphicsData.length;\n    this.glPoints = new Float32Array(this.points);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.buffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.glPoints, gl.STATIC_DRAW);\n\n    this.glIndicies = new Uint16Array(this.indices);\n\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.glIndicies, gl.STATIC_DRAW);\n\n    this.dirty = false;\n};\n","var TextureUvs = require('../../../textures/TextureUvs'),\n    Shader = require('../shaders/Shader');\n\n/**\n * @author Mat Groves\n *\n * Big thanks to the very clever Matt DesLauriers <mattdesl> https://github.com/mattdesl/\n * for creating the original pixi version!\n * Also a thanks to https://github.com/bchevalier for tweaking the tint and alpha so that they now share 4 bytes on the vertex buffer\n *\n * Heavily inspired by LibGDX's WebGLSpriteBatch:\n * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java\n */\n\n/**\n *\n * @class\n * @private\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this sprite batch works for.\n */\nfunction WebGLSpriteBatch(renderer) {\n    /**\n     *\n     *\n     * @member {WebGLRenderer}\n     */\n    this.renderer = renderer;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.vertSize = 5;\n\n    /**\n     * The number of images in the SpriteBatch before it flushes.\n     *\n     * @member {number}\n     */\n    this.size = 2000;//Math.pow(2, 16) /  this.vertSize;\n\n    // the total number of bytes in our batch\n    var numVerts = this.size * 4 * 4 * this.vertSize;\n    // the total number of indices in our batch\n    var numIndices = this.size * 6;\n\n    /**\n     * Holds the vertices\n     *\n     * @member {ArrayBuffer}\n     */\n    this.vertices = new ArrayBuffer(numVerts);\n\n    /**\n     * View on the vertices as a Float32Array\n     *\n     * @member {Float32Array}\n     */\n    this.positions = new Float32Array(this.vertices);\n\n    /**\n     * View on the vertices as a Uint32Array\n     *\n     * @member {Uint32Array}\n     */\n    this.colors = new Uint32Array(this.vertices);\n\n    /**\n     * Holds the indices\n     *\n     * @member {Uint16Array}\n     */\n    this.indices = new Uint16Array(numIndices);\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.lastIndexCount = 0;\n\n    for (var i=0, j=0; i < numIndices; i += 6, j += 4) {\n        this.indices[i + 0] = j + 0;\n        this.indices[i + 1] = j + 1;\n        this.indices[i + 2] = j + 2;\n        this.indices[i + 3] = j + 0;\n        this.indices[i + 4] = j + 2;\n        this.indices[i + 5] = j + 3;\n    }\n\n    /**\n     *\n     *\n     * @member {boolean}\n     */\n    this.drawing = false;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.currentBatchSize = 0;\n\n    /**\n     *\n     *\n     * @member {BaseTexture}\n     */\n    this.currentBaseTexture = null;\n\n    /**\n     *\n     *\n     * @member {boolean}\n     */\n    this.dirty = true;\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.textures = [];\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.blendModes = [];\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.shaders = [];\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.sprites = [];\n\n    /**\n     * The default shader that is used if a sprite doesn't have a more specific one.\n     *\n     * @member {Shader}\n     */\n    this.shader = null;\n\n    // listen for context and update necessary buffers\n    var self = this;\n    this.renderer.on('context', function () {\n        self.setupContext();\n    });\n}\n\nWebGLSpriteBatch.prototype.constructor = WebGLSpriteBatch;\nmodule.exports = WebGLSpriteBatch;\n\n/**\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nWebGLSpriteBatch.prototype.setupContext = function () {\n    var gl = this.renderer.gl;\n\n    // setup default shader\n    this.shader = new Shader(gl);\n\n    // create a couple of buffers\n    this.vertexBuffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n\n    // 65535 is max index, so 65535 / 6 = 10922.\n\n    //upload the index data\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW);\n\n    this.currentBlendMode = 99999;\n};\n\n/**\n *\n */\nWebGLSpriteBatch.prototype.begin = function () {\n    // this.shader = this.renderer.shaderManager.defaultShader;\n\n    this.start();\n};\n\n/**\n */\nWebGLSpriteBatch.prototype.end = function () {\n    this.flush();\n};\n\n/**\n * @param sprite {Sprite} the sprite to render when using this spritebatch\n */\nWebGLSpriteBatch.prototype.render = function (sprite) {\n    var texture = sprite.texture;\n\n    //TODO set blend modes..\n    // check texture..\n    if (this.currentBatchSize >= this.size) {\n        this.flush();\n        this.currentBaseTexture = texture.baseTexture;\n    }\n\n    // get the uvs for the texture\n    var uvs = texture._uvs;\n\n    // if the uvs have not updated then no point rendering just yet!\n    if (!uvs) {\n        return;\n    }\n\n    // TODO trim??\n    var aX = sprite.anchor.x;\n    var aY = sprite.anchor.y;\n\n    var w0, w1, h0, h1;\n\n    if (texture.trim) {\n        // if the sprite is trimmed then we need to add the extra space before transforming the sprite coords..\n        var trim = texture.trim;\n\n        w1 = trim.x - aX * trim.width;\n        w0 = w1 + texture.crop.width;\n\n        h1 = trim.y - aY * trim.height;\n        h0 = h1 + texture.crop.height;\n\n    }\n    else {\n        w0 = (texture.frame.width ) * (1-aX);\n        w1 = (texture.frame.width ) * -aX;\n\n        h0 = texture.frame.height * (1-aY);\n        h1 = texture.frame.height * -aY;\n    }\n\n    var index = this.currentBatchSize * 4 * this.vertSize;\n\n    var resolution = texture.baseTexture.resolution;\n\n    var worldTransform = sprite.worldTransform;\n\n    var a = worldTransform.a / resolution;\n    var b = worldTransform.b / resolution;\n    var c = worldTransform.c / resolution;\n    var d = worldTransform.d / resolution;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var colors = this.colors;\n    var positions = this.positions;\n\n    if (this.renderer.roundPixels) {\n        // xy\n        positions[index] = a * w1 + c * h1 + tx | 0;\n        positions[index+1] = d * h1 + b * w1 + ty | 0;\n\n        // xy\n        positions[index+5] = a * w0 + c * h1 + tx | 0;\n        positions[index+6] = d * h1 + b * w0 + ty | 0;\n\n         // xy\n        positions[index+10] = a * w0 + c * h0 + tx | 0;\n        positions[index+11] = d * h0 + b * w0 + ty | 0;\n\n        // xy\n        positions[index+15] = a * w1 + c * h0 + tx | 0;\n        positions[index+16] = d * h0 + b * w1 + ty | 0;\n    }\n    else {\n        // xy\n        positions[index] = a * w1 + c * h1 + tx;\n        positions[index+1] = d * h1 + b * w1 + ty;\n\n        // xy\n        positions[index+5] = a * w0 + c * h1 + tx;\n        positions[index+6] = d * h1 + b * w0 + ty;\n\n         // xy\n        positions[index+10] = a * w0 + c * h0 + tx;\n        positions[index+11] = d * h0 + b * w0 + ty;\n\n        // xy\n        positions[index+15] = a * w1 + c * h0 + tx;\n        positions[index+16] = d * h0 + b * w1 + ty;\n    }\n\n    // uv\n    positions[index+2] = uvs.x0;\n    positions[index+3] = uvs.y0;\n\n    // uv\n    positions[index+7] = uvs.x1;\n    positions[index+8] = uvs.y1;\n\n     // uv\n    positions[index+12] = uvs.x2;\n    positions[index+13] = uvs.y2;\n\n    // uv\n    positions[index+17] = uvs.x3;\n    positions[index+18] = uvs.y3;\n\n    // color and alpha\n    var tint = sprite.tint;\n    colors[index+4] = colors[index+9] = colors[index+14] = colors[index+19] = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (sprite.worldAlpha * 255 << 24);\n\n    // increment the batchsize\n    this.sprites[this.currentBatchSize++] = sprite;\n\n\n};\n\n/**\n * Renders a TilingSprite using the spriteBatch.\n *\n * @param sprite {TilingSprite} the tilingSprite to render\n */\nWebGLSpriteBatch.prototype.renderTilingSprite = function (tilingSprite) {\n    var texture = tilingSprite.tilingTexture;\n\n    // check texture..\n    if (this.currentBatchSize >= this.size) {\n        //return;\n        this.flush();\n        this.currentBaseTexture = texture.baseTexture;\n    }\n\n    // set the textures uvs temporarily\n    // TODO create a separate texture so that we can tile part of a texture\n\n    if (!tilingSprite._uvs) {\n        tilingSprite._uvs = new TextureUvs();\n    }\n\n    var uvs = tilingSprite._uvs;\n\n    tilingSprite.tilePosition.x %= texture.baseTexture.width * tilingSprite.tileScaleOffset.x;\n    tilingSprite.tilePosition.y %= texture.baseTexture.height * tilingSprite.tileScaleOffset.y;\n\n    var offsetX =  tilingSprite.tilePosition.x/(texture.baseTexture.width*tilingSprite.tileScaleOffset.x);\n    var offsetY =  tilingSprite.tilePosition.y/(texture.baseTexture.height*tilingSprite.tileScaleOffset.y);\n\n    var scaleX =  (tilingSprite.width / texture.baseTexture.width)  / (tilingSprite.tileScale.x * tilingSprite.tileScaleOffset.x);\n    var scaleY =  (tilingSprite.height / texture.baseTexture.height) / (tilingSprite.tileScale.y * tilingSprite.tileScaleOffset.y);\n\n    uvs.x0 = 0 - offsetX;\n    uvs.y0 = 0 - offsetY;\n\n    uvs.x1 = (1 * scaleX) - offsetX;\n    uvs.y1 = 0 - offsetY;\n\n    uvs.x2 = (1 * scaleX) - offsetX;\n    uvs.y2 = (1 * scaleY) - offsetY;\n\n    uvs.x3 = 0 - offsetX;\n    uvs.y3 = (1 * scaleY) - offsetY;\n\n    // get the tilingSprites current alpha and tint and combining them into a single color\n    var tint = tilingSprite.tint;\n    var color = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (tilingSprite.alpha * 255 << 24);\n\n    var positions = this.positions;\n    var colors = this.colors;\n\n    var width = tilingSprite.width;\n    var height = tilingSprite.height;\n\n    // TODO trim??\n    var aX = tilingSprite.anchor.x;\n    var aY = tilingSprite.anchor.y;\n    var w0 = width * (1-aX);\n    var w1 = width * -aX;\n\n    var h0 = height * (1-aY);\n    var h1 = height * -aY;\n\n    var index = this.currentBatchSize * 4 * this.vertSize;\n\n    var resolution = texture.baseTexture.resolution;\n\n    var worldTransform = tilingSprite.worldTransform;\n\n    var a = worldTransform.a / resolution;//[0];\n    var b = worldTransform.b / resolution;//[3];\n    var c = worldTransform.c / resolution;//[1];\n    var d = worldTransform.d / resolution;//[4];\n    var tx = worldTransform.tx;//[2];\n    var ty = worldTransform.ty;//[5];\n\n    // xy\n    positions[index++] = a * w1 + c * h1 + tx;\n    positions[index++] = d * h1 + b * w1 + ty;\n    // uv\n    positions[index++] = uvs.x0;\n    positions[index++] = uvs.y0;\n    // color\n    colors[index++] = color;\n\n    // xy\n    positions[index++] = (a * w0 + c * h1 + tx);\n    positions[index++] = d * h1 + b * w0 + ty;\n    // uv\n    positions[index++] = uvs.x1;\n    positions[index++] = uvs.y1;\n    // color\n    colors[index++] = color;\n\n    // xy\n    positions[index++] = a * w0 + c * h0 + tx;\n    positions[index++] = d * h0 + b * w0 + ty;\n    // uv\n    positions[index++] = uvs.x2;\n    positions[index++] = uvs.y2;\n    // color\n    colors[index++] = color;\n\n    // xy\n    positions[index++] = a * w1 + c * h0 + tx;\n    positions[index++] = d * h0 + b * w1 + ty;\n    // uv\n    positions[index++] = uvs.x3;\n    positions[index++] = uvs.y3;\n    // color\n    colors[index++] = color;\n\n    // increment the batchsize\n    this.sprites[this.currentBatchSize++] = tilingSprite;\n};\n\n/**\n * Renders the content and empties the current batch.\n *\n */\nWebGLSpriteBatch.prototype.flush = function () {\n    // If the batch is length 0 then return as there is nothing to draw\n    if (this.currentBatchSize === 0) {\n        return;\n    }\n\n    var gl = this.renderer.gl;\n    var shader;\n\n    if (this.dirty) {\n        this.dirty = false;\n        // bind the main texture\n        gl.activeTexture(gl.TEXTURE0);\n\n        // bind the buffers\n        gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n\n        // this is the same for each shader?\n        var stride =  this.vertSize * 4;\n        gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0);\n        gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 2 * 4);\n\n        // color attributes will be interpreted as unsigned bytes and normalized\n        gl.vertexAttribPointer(this.shader.attributes.aColor, 4, gl.UNSIGNED_BYTE, true, stride, 4 * 4);\n    }\n\n    // upload the verts to the buffer\n    if (this.currentBatchSize > ( this.size * 0.5 ) ) {\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices);\n    }\n    else {\n        var view = this.positions.subarray(0, this.currentBatchSize * 4 * this.vertSize);\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, view);\n    }\n\n    var nextTexture, nextBlendMode, nextShader;\n    var batchSize = 0;\n    var start = 0;\n\n    var currentBaseTexture = null;\n    var currentBlendMode = this.renderer.blendModeManager.currentBlendMode;\n    var currentShader = null;\n\n    var blendSwap = false;\n    var shaderSwap = false;\n    var sprite;\n\n    for (var i = 0, j = this.currentBatchSize; i < j; i++) {\n\n        sprite = this.sprites[i];\n\n        nextTexture = sprite.texture.baseTexture;\n        nextBlendMode = sprite.blendMode;\n        nextShader = sprite.shader || this.shader;\n\n        blendSwap = currentBlendMode !== nextBlendMode;\n        shaderSwap = currentShader !== nextShader; // should I use uuidS???\n\n        if (currentBaseTexture !== nextTexture || blendSwap || shaderSwap) {\n            this.renderBatch(currentBaseTexture, batchSize, start);\n\n            start = i;\n            batchSize = 0;\n            currentBaseTexture = nextTexture;\n\n            if (blendSwap) {\n                currentBlendMode = nextBlendMode;\n                this.renderer.blendModeManager.setBlendMode( currentBlendMode );\n            }\n\n            if (shaderSwap) {\n                currentShader = nextShader;\n\n                shader = currentShader.shaders ? currentShader.shaders[gl.id] : currentShader;\n\n                if (!shader) {\n                    shader = new Shader(gl, null, currentShader.fragmentSrc, currentShader.uniforms);\n                    currentShader.shaders[gl.id] = shader;\n                }\n\n                // set shader function???\n                this.renderer.shaderManager.setShader(shader);\n\n                if (shader.dirty) {\n                    shader.syncUniforms();\n                }\n\n                // both thease only need to be set if they are changing..\n                // set the projection\n                var projection = this.renderer.projection;\n                gl.uniform2f(shader.uniforms.projectionVector._location, projection.x, projection.y);\n\n                // TODO - this is temprorary!\n                var offsetVector = this.renderer.offset;\n                gl.uniform2f(shader.uniforms.offsetVector._location, offsetVector.x, offsetVector.y);\n\n                // set the pointers\n            }\n        }\n\n        batchSize++;\n    }\n\n    this.renderBatch(currentBaseTexture, batchSize, start);\n\n    // then reset the batch!\n    this.currentBatchSize = 0;\n};\n\n/**\n * @param texture {Texture}\n * @param size {number}\n * @param startIndex {number}\n */\nWebGLSpriteBatch.prototype.renderBatch = function (texture, size, startIndex) {\n    if (size === 0) {\n        return;\n    }\n\n    var gl = this.renderer.gl;\n\n    if (!texture._glTextures[gl.id]) {\n        this.renderer.updateTexture(texture);\n    }\n    else {\n        // bind the current texture\n        gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]);\n    }\n\n    // now draw those suckas!\n    gl.drawElements(gl.TRIANGLES, size * 6, gl.UNSIGNED_SHORT, startIndex * 6 * 2);\n\n    // increment the draw count\n    this.renderer.drawCount++;\n};\n\n/**\n *\n */\nWebGLSpriteBatch.prototype.stop = function () {\n    this.flush();\n    this.dirty = true;\n};\n\n/**\n *\n */\nWebGLSpriteBatch.prototype.start = function () {\n    this.dirty = true;\n};\n\n/**\n * Destroys the SpriteBatch.\n *\n */\nWebGLSpriteBatch.prototype.destroy = function () {\n    this.renderer.gl.deleteBuffer(this.vertexBuffer);\n    this.renderer.gl.deleteBuffer(this.indexBuffer);\n\n    this.vertices = null;\n    this.indices = null;\n\n    this.vertexBuffer = null;\n    this.indexBuffer = null;\n\n    this.currentBaseTexture = null;\n\n    this.renderer = null;\n};\n","var utils = require('../utils'),\n    CONST = require('../const');\n\n/**\n * A texture stores the information that represents an image. All textures have a base texture.\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param source {Image|Canvas} the source object of the texture.\n * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values\n */\nfunction BaseTexture(source, scaleMode) {\n    this.uuid = utils.uuid();\n\n    /**\n     * The Resolution of the texture.\n     *\n     * @member {number}\n     */\n    this.resolution = 1;\n\n    /**\n     * The width of the base texture set when the image has loaded\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.width = 100;\n\n    /**\n     * The height of the base texture set when the image has loaded\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.height = 100;\n\n    /**\n     * The scale mode to apply when scaling this texture\n     *\n     * @member {{number}}\n     * @default scaleModes.LINEAR\n     */\n    this.scaleMode = scaleMode || CONST.scaleModes.DEFAULT;\n\n    /**\n     * Set to true once the base texture has successfully loaded.\n     *\n     * This is never true if the underlying source fails to load or has no texture data.\n     *\n     * @member {boolean}\n     * @readOnly\n     */\n    this.hasLoaded = false;\n\n    /**\n     * Set to true if the source is currently loading.\n     *\n     * If an Image source is loading the 'loaded' or 'error' event will be\n     * dispatched when the operation ends. An underyling source that is\n     * immediately-available bypasses loading entirely.\n     *\n     * @member {boolean}\n     * @readonly\n     */\n    this.isLoading = false;\n\n    /**\n     * The image source that is used to create the texture.\n     *\n     * TODO: Make this a setter that calls loadSource();\n     *\n     * @member {Image|Canvas}\n     * @readonly\n     */\n    this.source = null; // set in loadSource, if at all\n\n    /**\n     * Controls if RGB channels should be pre-multiplied by Alpha  (WebGL only)\n     *\n     * @member {boolean}\n     * @default true\n     */\n    this.premultipliedAlpha = true;\n\n    /**\n     * @member {string}\n     */\n    this.imageUrl = null;\n\n    /**\n     * @member {boolean}\n     * @private\n     */\n    this._powerOf2 = false;\n\n    // used for webGL\n\n    /**\n     *\n     * Set this to true if a mipmap of this texture needs to be generated. This value needs to be set before the texture is used\n     * Also the texture must be a power of two size to work\n     *\n     * @member {boolean}\n     */\n    this.mipmap = false;\n\n    /**\n     * A map of renderer IDs to webgl textures\n     *\n     * @member {object<number, WebGLTexture>}\n     * @private\n     */\n    this._glTextures = {};\n\n    /**\n     * Does the texture on the GPU need to be updated?\n     *\n     * @member {boolean}\n     * @private\n     */\n    this._needsUpdate = false;\n\n    // if no source passed don't try to load\n    if (source) {\n        this.loadSource(source);\n    }\n\n    /**\n     * Fired when a not-immediately-available source finishes loading.\n     *\n     * @event loaded\n     * @protected\n     */\n\n    /**\n     * Fired when a not-immediately-available source fails to load.\n     *\n     * @event error\n     * @protected\n     */\n}\n\nBaseTexture.prototype.constructor = BaseTexture;\nmodule.exports = BaseTexture;\n\nutils.eventTarget.mixin(BaseTexture.prototype);\n\nObject.defineProperties(BaseTexture.prototype, {\n    needsUpdate: {\n        get: function () {\n            return this._needsUpdate;\n        },\n        set: function (val) {\n            this._needsUpdate = val;\n\n            if (val) {\n                this.emit('update', this);\n            }\n        }\n    }\n});\n\n/**\n * Load a source.\n *\n * If the source is not-immediately-available, such as an image that needs to be\n * downloaded, then the 'loaded' or 'error' event will be dispatched in the future\n * and `hasLoaded` will remain false after this call.\n *\n * The logic state after calling `loadSource` directly or indirectly (eg. `fromImage`, `new BaseTexture`) is:\n *\n *     if (texture.hasLoaded) {\n *        // texture ready for use\n *     } else if (texture.isLoading) {\n *        // listen to 'loaded' and/or 'error' events on texture\n *     } else {\n *        // not loading, not going to load UNLESS the source is reloaded\n *        // (it may still make sense to listen to the events)\n *     }\n *\n * @protected\n * @param source {Image|Canvas} the source object of the texture.\n */\nBaseTexture.prototype.loadSource = function (source) {\n    var wasLoading = this.isLoading;\n    this.hasLoaded = false;\n    this.isLoading = false;\n\n    if (wasLoading && this.source) {\n        this.source.onload = null;\n        this.source.onerror = null;\n    }\n\n    this.source = source;\n\n    // Apply source if loaded. Otherwise setup appropriate loading monitors.\n    if ((this.source.complete || this.source.getContext) && this.source.width && this.source.height) {\n        this._sourceLoaded();\n    }\n    else  if(!source.getContext) {\n        // Image fail / not ready\n        this.isLoading = true;\n\n        var scope = this;\n\n        source.onload = function () {\n            source.onload = null;\n            source.onerror = null;\n\n            if(!scope.isLoading) {\n                return;\n            }\n\n            scope.isLoading = false;\n            scope._sourceLoaded();\n\n            scope.emit('loaded', scope);\n        };\n\n        source.onerror = function () {\n            source.onload = null;\n            source.onerror = null;\n\n            if(!scope.isLoading) {\n                return;\n            }\n\n            scope.isLoading = false;\n            scope.emit('error', scope);\n        };\n\n        // Per http://www.w3.org/TR/html5/embedded-content-0.html#the-img-element\n        //   \"The value of `complete` can thus change while a script is executing.\"\n        // So complete needs to be re-checked after the callbacks have been added..\n        if (source.complete) {\n            this.isLoading = false;\n\n            // ..and if we're complete now, no need for callbacks\n            source.onload = null;\n            source.onerror = null;\n\n            if (source.width && source.height) {\n                this._sourceLoaded();\n\n                // If any previous subscribers possible\n                if (wasLoading) {\n                    this.emit('loaded', this);\n                }\n            }\n            else {\n                // If any previous subscribers possible\n                if (wasLoading) {\n                    this.emit('error', this);\n                }\n            }\n        }\n    }\n};\n\n/**\n * Used internally to update the width, height, and some other tracking vars once\n * a source has successfully loaded.\n *\n * @private\n */\nBaseTexture.prototype._sourceLoaded = function () {\n    this.hasLoaded = true;\n\n    this.width = this.source.naturalWidth || this.source.width;\n    this.height = this.source.naturalHeight || this.source.height;\n\n    this.needsUpdate = true;\n};\n\n/**\n * Destroys this base texture\n *\n */\nBaseTexture.prototype.destroy = function () {\n    if (this.imageUrl) {\n        delete utils.BaseTextureCache[this.imageUrl];\n        delete utils.TextureCache[this.imageUrl];\n        this.imageUrl = null;\n        if (!navigator.isCocoonJS) {\n            this.source.src = '';\n        }\n    }\n    else if (this.source && this.source._pixiId) {\n        delete utils.BaseTextureCache[this.source._pixiId];\n    }\n    this.source = null;\n\n    this.dispose();\n};\n\n/**\n * Frees the texture from WebGL memory without destroying this texture object.\n * This means you can still use the texture later which will upload it to GPU\n * memory again.\n *\n */\nBaseTexture.prototype.dispose = function () {\n    this.emit('dispose', this);\n};\n\n/**\n * Changes the source image of the texture.\n * The original source must be an Image element.\n *\n * @param newSrc {string} the path of the image\n */\nBaseTexture.prototype.updateSourceImage = function (newSrc) {\n    this.source.src = newSrc;\n    this.loadSource(this.source);\n};\n\n/**\n * Helper function that creates a base texture from the given image url.\n * If the image is not in the base texture cache it will be created and loaded.\n *\n * @static\n * @param imageUrl {string} The image url of the texture\n * @param [crossorigin=(auto)] {boolean} Should use anonymouse CORS? Defaults to true if the URL is not a data-URI.\n * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values\n * @return BaseTexture\n */\nBaseTexture.fromImage = function (imageUrl, crossorigin, scaleMode) {\n    var baseTexture = utils.BaseTextureCache[imageUrl];\n\n    if (crossorigin === undefined && imageUrl.indexOf('data:') !== 0) {\n        crossorigin = true;\n    }\n\n    if (!baseTexture) {\n        // new Image() breaks tex loading in some versions of Chrome.\n        // See https://code.google.com/p/chromium/issues/detail?id=238071\n        var image = new Image();//document.createElement('img');\n        if (crossorigin) {\n            image.crossOrigin = '';\n        }\n\n        image.src = imageUrl;\n        baseTexture = new BaseTexture(image, scaleMode);\n        baseTexture.imageUrl = imageUrl;\n        utils.BaseTextureCache[imageUrl] = baseTexture;\n\n        // if there is an @2x at the end of the url we are going to assume its a highres image\n        if ( imageUrl.indexOf(CONST.RETINA_PREFIX + '.') !== -1) {\n            baseTexture.resolution = 2;\n        }\n    }\n\n    return baseTexture;\n};\n\n/**\n * Helper function that creates a base texture from the given canvas element.\n *\n * @static\n * @param canvas {Canvas} The canvas element source of the texture\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return BaseTexture\n */\nBaseTexture.fromCanvas = function (canvas, scaleMode) {\n    if (!canvas._pixiId) {\n        canvas._pixiId = 'canvas_' + utils.uuid();\n    }\n\n    var baseTexture = utils.BaseTextureCache[canvas._pixiId];\n\n    if (!baseTexture) {\n        baseTexture = new BaseTexture(canvas, scaleMode);\n        utils.BaseTextureCache[canvas._pixiId] = baseTexture;\n    }\n\n    return baseTexture;\n};\n","var BaseTexture = require('./BaseTexture'),\n    Texture = require('./Texture'),\n    FilterTexture = require('../renderers/webgl/utils/FilterTexture'),\n    CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'),\n    math = require('../math'),\n    CONST = require('../const');\n\n/**\n * A RenderTexture is a special texture that allows any Pixi display object to be rendered to it.\n *\n * __Hint__: All DisplayObjects (i.e. Sprites) that render to a RenderTexture should be preloaded\n * otherwise black rectangles will be drawn instead.\n *\n * A RenderTexture takes a snapshot of any Display Object given to its render method. The position\n * and rotation of the given Display Objects is ignored. For example:\n *\n * ```js\n * var renderTexture = new PIXI.RenderTexture(800, 600);\n * var sprite = PIXI.Sprite.fromImage(\"spinObj_01.png\");\n *\n * sprite.position.x = 800/2;\n * sprite.position.y = 600/2;\n * sprite.anchor.x = 0.5;\n * sprite.anchor.y = 0.5;\n *\n * renderTexture.render(sprite);\n * ```\n *\n * The Sprite in this case will be rendered to a position of 0,0. To render this sprite at its actual\n * position a DisplayObjectContainer should be used:\n *\n * ```js\n * var doc = new DisplayObjectContainer();\n *\n * doc.addChild(sprite);\n *\n * renderTexture.render(doc);  // Renders to center of renderTexture\n * ```\n *\n * @class\n * @extends Texture\n * @namespace PIXI\n * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used for this RenderTexture\n * @param [width=100] {number} The width of the render texture\n * @param [height=100] {number} The height of the render texture\n * @param [scaleMode] {number} See {@link scaleModes} for possible values\n * @param [resolution=1] {number} The resolution of the texture being generated\n */\nfunction RenderTexture(renderer, width, height, scaleMode, resolution) {\n    if (!renderer) {\n        throw new Error('Unable to create RenderTexture, you must pass a renderer into the constructor.');\n    }\n\n    /**\n     * The with of the render texture\n     *\n     * @member {number}\n     */\n    this.width = width || 100;\n\n    /**\n     * The height of the render texture\n     *\n     * @member {number}\n     */\n    this.height = height || 100;\n\n    /**\n     * The Resolution of the texture.\n     *\n     * @member {number}\n     */\n    this.resolution = resolution || 1;\n\n    /**\n     * The framing rectangle of the render texture\n     *\n     * @member {Rectangle}\n     */\n    this.frame = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution);\n\n    /**\n     * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering,\n     * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases)\n     *\n     * @member {Rectangle}\n     */\n    this.crop = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution);\n\n    /**\n     * Draw/render the given DisplayObject onto the texture.\n     *\n     * The displayObject and descendents are transformed during this operation.\n     * If `restoreWorldTransform` is true then the transformations will be restored before the\n     * method returns. Otherwise it is up to the calling code to correctly use or reset\n     * the transformed display objects.\n     *\n     * The display object is always rendered with a worldAlpha value of 1.\n     *\n     * @method\n     * @param displayObject {DisplayObject} The display object to render this texture on\n     * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering.\n     * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn\n     * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children\n     *  transformations will be restored. Not restoring this information will be a little faster.\n     */\n    this.render = null;\n\n    /**\n     * The base texture object that this texture uses\n     *\n     * @member {BaseTexture}\n     */\n    this.baseTexture = new BaseTexture();\n    this.baseTexture.width = this.width * this.resolution;\n    this.baseTexture.height = this.height * this.resolution;\n    this.baseTexture._glTextures = [];\n    this.baseTexture.resolution = this.resolution;\n\n    this.baseTexture.scaleMode = scaleMode || CONST.scaleModes.DEFAULT;\n\n    this.baseTexture.hasLoaded = true;\n\n    Texture.call(this,\n        this.baseTexture,\n        new math.Rectangle(0, 0, this.width, this.height)\n    );\n\n    /**\n     * The renderer this RenderTexture uses. A RenderTexture can only belong to one renderer at the moment if its webGL.\n     *\n     * @member {CanvasRenderer|WebGLRenderer}\n     */\n    this.renderer = renderer;\n\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        var gl = this.renderer.gl;\n\n        this.textureBuffer = new FilterTexture(gl, this.width * this.resolution, this.height * this.resolution, this.baseTexture.scaleMode);\n        this.baseTexture._glTextures[gl.id] =  this.textureBuffer.texture;\n\n        this.render = this.renderWebGL;\n        this.projection = new math.Point(this.width*0.5, -this.height*0.5);\n    }\n    else {\n        this.render = this.renderCanvas;\n        this.textureBuffer = new CanvasBuffer(this.width* this.resolution, this.height* this.resolution);\n        this.baseTexture.source = this.textureBuffer.canvas;\n    }\n\n    /**\n     * @member {boolean}\n     */\n    this.valid = true;\n\n    this._updateUvs();\n}\n\nRenderTexture.prototype = Object.create(Texture.prototype);\nRenderTexture.prototype.constructor = RenderTexture;\nmodule.exports = RenderTexture;\n\n/**\n * Resizes the RenderTexture.\n *\n * @param width {number} The width to resize to.\n * @param height {number} The height to resize to.\n * @param updateBase {boolean} Should the baseTexture.width and height values be resized as well?\n */\nRenderTexture.prototype.resize = function (width, height, updateBase) {\n    if (width === this.width && height === this.height) {\n        return;\n    }\n\n    this.valid = (width > 0 && height > 0);\n\n    this.width = this.frame.width = this.crop.width = width;\n    this.height =  this.frame.height = this.crop.height = height;\n\n    if (updateBase) {\n        this.baseTexture.width = this.width;\n        this.baseTexture.height = this.height;\n    }\n\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        this.projection.x = this.width / 2;\n        this.projection.y = -this.height / 2;\n    }\n\n    if (!this.valid) {\n        return;\n    }\n\n    this.textureBuffer.resize(this.width * this.resolution, this.height * this.resolution);\n};\n\n/**\n * Clears the RenderTexture.\n *\n */\nRenderTexture.prototype.clear = function () {\n    if (!this.valid) {\n        return;\n    }\n\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        this.renderer.gl.bindFramebuffer(this.renderer.gl.FRAMEBUFFER, this.textureBuffer.frameBuffer);\n    }\n\n    this.textureBuffer.clear();\n};\n\n/**\n * Internal method assigned to the `render` property if using a CanvasRenderer.\n *\n * @private\n * @param displayObject {DisplayObject} The display object to render this texture on\n * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering.\n * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn\n * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children\n *  transformations will be restored. Not restoring this information will be a little faster.\n */\nRenderTexture.prototype.renderWebGL = function (displayObject, matrix, clear, restoreWorldTransform) {\n    if (!this.valid) {\n        return;\n    }\n\n    if (typeof restoreWorldTransform === 'undefined') {\n        restoreWorldTransform = true;\n    }\n\n    var tempAlpha,\n        tempTransform;\n\n    if (restoreWorldTransform) {\n        tempAlpha = displayObject.worldAlpha;\n        tempTransform = displayObject.worldTransform.toArray();\n    }\n\n    //TOOD replace position with matrix..\n\n    //Lets create a nice matrix to apply to our display object. Frame buffers come in upside down so we need to flip the matrix\n    var wt = displayObject.worldTransform;\n\n    wt.identity();\n    wt.translate(0, this.projection.y * 2);\n\n    if (matrix) {\n        wt.append(matrix);\n    }\n\n    wt.scale(1,-1);\n\n    // setWorld Alpha to ensure that the object is renderer at full opacity\n    displayObject.worldAlpha = 1;\n\n    // Time to update all the children of the displayObject with the new matrix..\n    var children = displayObject.children;\n    var i, j;\n\n    for (i = 0, j = children.length; i < j; ++i) {\n        children[i].updateTransform();\n    }\n\n    // time for the webGL fun stuff!\n    var gl = this.renderer.gl;\n\n    gl.viewport(0, 0, this.width * this.resolution, this.height * this.resolution);\n\n    gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer );\n\n    if (clear) {\n        this.textureBuffer.clear();\n    }\n\n    this.renderer.spriteBatch.dirty = true;\n\n    this.renderer.renderDisplayObject(displayObject, this.projection, this.textureBuffer.frameBuffer);\n\n    this.renderer.spriteBatch.dirty = true;\n\n    if (restoreWorldTransform) {\n        displayObject.worldAlpha = tempAlpha;\n        displayObject.worldTransform.fromArray(tempTransform);\n\n        for (i = 0, j = children.length; i < j; ++i) {\n            children[i].updateTransform();\n        }\n    }\n};\n\n\n/**\n * Internal method assigned to the `render` property if using a CanvasRenderer.\n *\n * @private\n * @param displayObject {DisplayObject} The display object to render this texture on\n * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering.\n * @param [clear] {boolean} If true the texture will be cleared before the displayObject is drawn\n * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children\n *  transformations will be restored. Not restoring this information will be a little faster.\n */\nRenderTexture.prototype.renderCanvas = function (displayObject, matrix, clear, restoreWorldTransform) {\n    if (!this.valid) {\n        return;\n    }\n\n    if (typeof restoreWorldTransform === 'undefined') {\n        restoreWorldTransform = true;\n    }\n\n    var tempAlpha,\n        tempTransform;\n\n    if (restoreWorldTransform) {\n        tempAlpha = displayObject.worldAlpha;\n        tempTransform = displayObject.worldTransform.toArray();\n    }\n\n    var wt = displayObject.worldTransform;\n    wt.identity();\n    if (matrix) {\n        wt.append(matrix);\n    }\n\n    // setWorld Alpha to ensure that the object is renderer at full opacity\n    displayObject.worldAlpha = 1;\n\n    // Time to update all the children of the displayObject with the new matrix..\n    var children = displayObject.children;\n    var i, j;\n\n    for (i = 0, j = children.length; i < j; ++i) {\n        children[i].updateTransform();\n    }\n\n    if (clear) {\n        this.textureBuffer.clear();\n    }\n\n    var context = this.textureBuffer.context;\n\n    var realResolution = this.renderer.resolution;\n\n    this.renderer.resolution = this.resolution;\n\n    this.renderer.renderDisplayObject(displayObject, context);\n\n    this.renderer.resolution = realResolution;\n\n    if (restoreWorldTransform) {\n        displayObject.worldAlpha = tempAlpha;\n        displayObject.worldTransform.fromArray(tempTransform);\n\n        for (i = 0, j = children.length; i < j; ++i) {\n            children[i].updateTransform();\n        }\n    }\n};\n\n/**\n * Will return a HTML Image of the texture\n *\n * @return {Image}\n */\nRenderTexture.prototype.getImage = function () {\n    var image = new Image();\n    image.src = this.getBase64();\n    return image;\n};\n\n/**\n * Will return a a base64 encoded string of this texture. It works by calling RenderTexture.getCanvas and then running toDataURL on that.\n *\n * @return {string} A base64 encoded string of the texture.\n */\nRenderTexture.prototype.getBase64 = function () {\n    return this.getCanvas().toDataURL();\n};\n\n/**\n * Creates a Canvas element, renders this RenderTexture to it and then returns it.\n *\n * @return {HTMLCanvasElement} A Canvas element with the texture rendered on.\n */\nRenderTexture.prototype.getCanvas = function () {\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        var gl =  this.renderer.gl;\n        var width = this.textureBuffer.width;\n        var height = this.textureBuffer.height;\n\n        var webGLPixels = new Uint8Array(4 * width * height);\n\n        gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer);\n        gl.readPixels(0, 0, width, height, gl.RGBA, gl.UNSIGNED_BYTE, webGLPixels);\n        gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n\n        var tempCanvas = new CanvasBuffer(width, height);\n        var canvasData = tempCanvas.context.getImageData(0, 0, width, height);\n        canvasData.data.set(webGLPixels);\n\n        tempCanvas.context.putImageData(canvasData, 0, 0);\n\n        return tempCanvas.canvas;\n    }\n    else {\n        return this.textureBuffer.canvas;\n    }\n};\n","var BaseTexture = require('./BaseTexture'),\n    VideoBaseTexture = require('./VideoBaseTexture'),\n    TextureUvs = require('./TextureUvs'),\n    eventTarget = require('../utils/eventTarget'),\n    math = require('../math'),\n    utils = require('../utils');\n\n/**\n * A texture stores the information that represents an image or part of an image. It cannot be added\n * to the display list directly. Instead use it as the texture for a Sprite. If no frame is provided then the whole image is used.\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param baseTexture {BaseTexture} The base texture source to create the texture from\n * @param [frame] {Rectangle} The rectangle frame of the texture to show\n * @param [crop] {Rectangle} The area of original texture\n * @param [trim] {Rectangle} Trimmed texture rectangle\n */\nfunction Texture(baseTexture, frame, crop, trim) {\n    /**\n     * Does this Texture have any frame data assigned to it?\n     *\n     * @member {boolean}\n     */\n    this.noFrame = false;\n\n    if (!frame) {\n        this.noFrame = true;\n        frame = new math.Rectangle(0, 0, 1, 1);\n    }\n\n    if (baseTexture instanceof Texture) {\n        baseTexture = baseTexture.baseTexture;\n    }\n\n    /**\n     * The base texture that this texture uses.\n     *\n     * @member {BaseTexture}\n     */\n    this.baseTexture = baseTexture;\n\n    /**\n     * The frame specifies the region of the base texture that this texture uses\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._frame = frame;\n\n    /**\n     * The texture trim data.\n     *\n     * @member {Rectangle}\n     */\n    this.trim = trim;\n\n    /**\n     * This will let the renderer know if the texture is valid. If it's not then it cannot be rendered.\n     *\n     * @member {boolean}\n     */\n    this.valid = false;\n\n    /**\n     * This will let a renderer know that a texture has been updated (used mainly for webGL uv updates)\n     *\n     * @member {boolean}\n     */\n    this.requiresUpdate = false;\n\n    /**\n     * The WebGL UV data cache.\n     *\n     * @member {object}\n     * @private\n     */\n    this._uvs = null;\n\n    /**\n     * The width of the Texture in pixels.\n     *\n     * @member {number}\n     */\n    this.width = 0;\n\n    /**\n     * The height of the Texture in pixels.\n     *\n     * @member {number}\n     */\n    this.height = 0;\n\n    /**\n     * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering,\n     * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases)\n     *\n     * @member {Rectangle}\n     */\n    this.crop = crop || new math.Rectangle(0, 0, 1, 1);\n\n    if (baseTexture.hasLoaded) {\n        if (this.noFrame) {\n            frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height);\n        }\n        this.frame = frame;\n    }\n    else {\n        baseTexture.addEventListener('loaded', this.onBaseTextureLoaded.bind(this));\n    }\n}\n\nTexture.prototype.constructor = Texture;\nmodule.exports = Texture;\n\neventTarget.mixin(Texture.prototype);\n\nObject.defineProperties(Texture.prototype, {\n    needsUpdate: {\n        get: function () {\n            return this.baseTexture.needsUpdate;\n        },\n        set: function (val) {\n            this.baseTexture.needsUpdate = val;\n        }\n    },\n\n    frame: {\n        get: function () {\n            return this._frame;\n        },\n        set: function (frame) {\n            this._frame = frame;\n\n            this.noFrame = false;\n\n            this.width = frame.width;\n            this.height = frame.height;\n\n            this.crop.x = frame.x;\n            this.crop.y = frame.y;\n            this.crop.width = frame.width;\n            this.crop.height = frame.height;\n\n            if (!this.trim && (frame.x + frame.width > this.baseTexture.width || frame.y + frame.height > this.baseTexture.height)) {\n                throw new Error('Texture Error: frame does not fit inside the base Texture dimensions ' + this);\n            }\n\n            this.valid = frame && frame.width && frame.height && this.baseTexture.source && this.baseTexture.hasLoaded;\n\n            if (this.trim) {\n                this.width = this.trim.width;\n                this.height = this.trim.height;\n                this._frame.width = this.trim.width;\n                this._frame.height = this.trim.height;\n            }\n\n            if (this.valid) {\n                this._updateUvs();\n            }\n        }\n    }\n});\n\n/**\n * Called when the base texture is loaded\n *\n * @private\n */\nTexture.prototype.onBaseTextureLoaded = function () {\n    var baseTexture = this.baseTexture;\n    baseTexture.removeEventListener('loaded', this.onLoaded);\n\n    if (this.noFrame) {\n        this.frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height);\n    }\n\n    this.dispatchEvent( { type: 'update', content: this } );\n};\n\n/**\n * Destroys this texture\n *\n * @param destroyBase {boolean} Whether to destroy the base texture as well\n */\nTexture.prototype.destroy = function (destroyBase) {\n    if (destroyBase) {\n        this.baseTexture.destroy();\n    }\n\n    this.valid = false;\n};\n\n/**\n * Updates the internal WebGL UV cache.\n *\n * @private\n */\nTexture.prototype._updateUvs = function () {\n    if (!this._uvs) {\n        this._uvs = new TextureUvs();\n    }\n\n    var frame = this.crop;\n    var tw = this.baseTexture.width;\n    var th = this.baseTexture.height;\n\n    this._uvs.x0 = frame.x / tw;\n    this._uvs.y0 = frame.y / th;\n\n    this._uvs.x1 = (frame.x + frame.width) / tw;\n    this._uvs.y1 = frame.y / th;\n\n    this._uvs.x2 = (frame.x + frame.width) / tw;\n    this._uvs.y2 = (frame.y + frame.height) / th;\n\n    this._uvs.x3 = frame.x / tw;\n    this._uvs.y3 = (frame.y + frame.height) / th;\n};\n\n/**\n * Helper function that creates a Texture object from the given image url.\n * If the image is not in the texture cache it will be  created and loaded.\n *\n * @static\n * @param imageUrl {string} The image url of the texture\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return Texture\n */\nTexture.fromImage = function (imageUrl, crossorigin, scaleMode) {\n    var texture = utils.TextureCache[imageUrl];\n\n    if (!texture) {\n        texture = new Texture(BaseTexture.fromImage(imageUrl, crossorigin, scaleMode));\n        utils.TextureCache[imageUrl] = texture;\n    }\n\n    return texture;\n};\n\n/**\n * Helper function that returns a Texture objected based on the given frame id.\n * If the frame id is not in the texture cache an error will be thrown.\n *\n * @static\n * @param frameId {string} The frame id of the texture\n * @return Texture\n */\nTexture.fromFrame = function (frameId) {\n    var texture = utils.TextureCache[frameId];\n    if (!texture) {\n        throw new Error('The frameId \"' + frameId + '\" does not exist in the texture cache ');\n    }\n    return texture;\n};\n\n/**\n * Helper function that creates a new Texture based on the given canvas element.\n *\n * @static\n * @param canvas {Canvas} The canvas element source of the texture\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return {Texture}\n */\nTexture.fromCanvas = function (canvas, scaleMode) {\n    return new Texture(BaseTexture.fromCanvas(canvas, scaleMode));\n};\n\n/**\n * Helper function that creates a new Texture based on the given video element.\n *\n * @static\n * @param video {HTMLVideoElement}\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return {Texture} A Texture\n */\nTexture.fromVideo = function (video, scaleMode) {\n    return new Texture(VideoBaseTexture.baseTextureFromVideo(video, scaleMode));\n};\n\n/**\n * Adds a texture to the global utils.TextureCache. This cache is shared across the whole PIXI object.\n *\n * @static\n * @param texture {Texture} The Texture to add to the cache.\n * @param id {string} The id that the texture will be stored against.\n */\nTexture.addTextureToCache = function (texture, id) {\n    utils.TextureCache[id] = texture;\n};\n\n/**\n * Remove a texture from the global utils.TextureCache.\n *\n * @static\n * @param id {string} The id of the texture to be removed\n * @return {Texture} The texture that was removed\n */\nTexture.removeTextureFromCache = function (id) {\n    var texture = utils.TextureCache[id];\n\n    delete utils.TextureCache[id];\n    delete utils.BaseTextureCache[id];\n\n    return texture;\n};\n\nTexture.emptyTexture = new Texture(new BaseTexture());\n","function TextureUvs() {\n    this.x0 = 0;\n    this.y0 = 0;\n\n    this.x1 = 0;\n    this.y1 = 0;\n\n    this.x2 = 0;\n    this.y2 = 0;\n\n    this.x3 = 0;\n    this.y3 = 0;\n}\n\nmodule.exports = TextureUvs;\n","var BaseTexture = require('./BaseTexture'),\n    utils = require('../utils');\n\n/**\n * A texture of a [playing] Video.\n *\n * See the [\"deus\" demo](http://www.goodboydigital.com/pixijs/examples/deus/).\n *\n * @class\n * @extends BaseTexture\n * @namespace PIXI\n * @param source {HTMLVideoElement}\n * @param [scaleMode] {number} See {@link scaleModes} for possible values\n */\nfunction VideoBaseTexture(source, scaleMode) {\n    if (!source){\n        throw new Error('No video source element specified.');\n    }\n\n    // hook in here to check if video is already available.\n    // BaseTexture looks for a source.complete boolean, plus width & height.\n\n    if ((source.readyState === source.HAVE_ENOUGH_DATA || source.readyState === source.HAVE_FUTURE_DATA) && source.width && source.height) {\n        source.complete = true;\n    }\n\n    BaseTexture.call(this, source, scaleMode);\n\n    this.autoUpdate = false;\n\n    this._boundOnUpdate = this._onUpdate.bind(this);\n    this._boundOnCanPlay = this._onCanPlay.bind(this);\n\n    if (!source.complete) {\n        source.addEventListener('canplay', this._boundOnCanPlay);\n        source.addEventListener('canplaythrough', this._boundOnCanPlay);\n\n        // started playing..\n        source.addEventListener('play', this._onPlayStart.bind(this));\n        source.addEventListener('pause', this._onPlayStop.bind(this));\n    }\n\n    this.__loaded = false;\n}\n\nVideoBaseTexture.prototype = Object.create(BaseTexture.prototype);\nVideoBaseTexture.prototype.constructor = VideoBaseTexture;\nmodule.exports = VideoBaseTexture;\n\nVideoBaseTexture.prototype._onUpdate = function () {\n    if (this.autoUpdate) {\n        window.requestAnimationFrame(this._boundOnUpdate);\n        this.needsUpdate = true;\n    }\n};\n\nVideoBaseTexture.prototype._onPlayStart = function () {\n    if (!this.autoUpdate) {\n        window.requestAnimationFrame(this._boundOnUpdate);\n        this.autoUpdate = true;\n    }\n};\n\nVideoBaseTexture.prototype._onPlayStop = function () {\n    this.autoUpdate = false;\n};\n\nVideoBaseTexture.prototype._onCanPlay = function () {\n    this.hasLoaded = true;\n\n    if (this.source) {\n        this.source.removeEventListener('canplay', this._boundOnCanPlay);\n        this.source.removeEventListener('canplaythrough', this._boundOnCanPlay);\n\n        this.width = this.source.videoWidth;\n        this.height = this.source.videoHeight;\n\n        this.source.play();\n\n        // prevent multiple loaded dispatches..\n        if(!this.__loaded){\n            this.__loaded = true;\n            this.emit('loaded', this);\n        }\n    }\n};\n\nVideoBaseTexture.prototype.destroy = function () {\n    if (this.source && this.source._pixiId) {\n        utils.BaseTextureCache[ this.source._pixiId ] = null;\n        delete utils.BaseTextureCache[ this.source._pixiId ];\n\n        this.source._pixiId = null;\n        delete this.source._pixiId;\n    }\n\n    BaseTexture.prototype.destroy.call(this);\n};\n\n/**\n * Mimic Pixi BaseTexture.from.... method.\n *\n * @static\n * @param video {HTMLVideoElement}\n * @param scaleMode {number} See {@link scaleModes} for possible values\n * @return {VideoBaseTexture}\n */\nVideoBaseTexture.fromVideo = function (video, scaleMode) {\n    if (!video._pixiId) {\n        video._pixiId = 'video_' + utils.uuid();\n    }\n\n    var baseTexture = utils.BaseTextureCache[video._pixiId];\n\n    if (!baseTexture) {\n        baseTexture = new VideoBaseTexture(video, scaleMode);\n        utils.BaseTextureCache[ video._pixiId ] = baseTexture;\n    }\n\n    return baseTexture;\n};\n\n/**\n * Mimic Pixi BaseTexture.from.... method.\n *\n * This can be used in a couple ways, such as:\n *\n * ```js\n * var texture = PIXI.VideoBaseTexture.fromUrl('http://mydomain.com/video.mp4');\n *\n * var texture = PIXI.VideoBaseTexture.fromUrl({ src: 'http://mydomain.com/video.mp4', mime: 'video/mp4' });\n *\n * var texture = PIXI.VideoBaseTexture.fromUrls(['/video.webm', '/video.mp4']);\n *\n * var texture = PIXI.VideoBaseTexture.fromUrls([\n *     { src: '/video.webm', mime: 'video/webm' },\n *     { src: '/video.mp4', mime: 'video/mp4' }\n * ]);\n * ```\n *\n * @alias fromUrls\n * @static\n * @param videoSrc {string|object|string[]|object[]} The URL(s) for the video.\n * @param [videoSrc.src] {string} One of the source urls for the video\n * @param [videoSrc.mime] {string} The mimetype of the video (e.g. 'video/mp4'). If not specified\n *  the url's extension will be used as the second part of the mime type.\n * @param scaleMode {number} See {@link scaleModes} for possible values\n * @return {VideoBaseTexture}\n */\nVideoBaseTexture.fromUrl = function (videoSrc, scaleMode) {\n    var video = document.createElement('video');\n\n    // array of objects or strings\n    if (Array.isArray(videoSrc)) {\n        for (var i = 0; i < videoSrc.length; ++i) {\n            video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime));\n        }\n    }\n    // single object or string\n    else {\n        video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime));\n    }\n\n    video.load();\n    video.play();\n\n    return VideoBaseTexture.textureFromVideo(video, scaleMode);\n};\n\nVideoBaseTexture.fromUrls = VideoBaseTexture.fromUrl;\n\nfunction createSource(path, type) {\n    if (!type) {\n        type = 'video/' + path.substr(path.lastIndexOf('.') + 1);\n    }\n\n    var source = document.createElement('source');\n\n    source.src = path;\n    source.type = type;\n\n    return source;\n}\n","/**\n * Creates an homogenous object for tracking events so users can know what to expect.\n *\n * @class\n * @namespace PIXI\n * @param target {object} The target object that the event is called on\n * @param name {string} The string name of the event that was triggered\n * @param data {object} Arbitrary event data to pass along\n */\nfunction EventData(target, name, data) {\n    // for duck typing in the \".on()\" function\n    this.__isEventObject = true;\n\n    /**\n     * Tracks the state of bubbling propagation. Do not\n     * set this directly, instead use `event.stopPropagation()`\n     *\n     * @member {boolean}\n     * @private\n     * @readonly\n     */\n    this.stopped = false;\n\n    /**\n     * Tracks the state of sibling listener propagation. Do not\n     * set this directly, instead use `event.stopImmediatePropagation()`\n     *\n     * @member {boolean}\n     * @private\n     * @readonly\n     */\n    this.stoppedImmediate = false;\n\n    /**\n     * The original target the event triggered on.\n     *\n     * @member {object}\n     * @readonly\n     */\n    this.target = target;\n\n    /**\n     * The string name of the event that this represents.\n     *\n     * @member {string}\n     * @readonly\n     */\n    this.type = name;\n\n    /**\n     * The data that was passed in with this event.\n     *\n     * @member {object}\n     * @readonly\n     */\n    this.data = data;\n\n    /**\n     * The timestamp when the event occurred.\n     *\n     * @member {number}\n     * @readonly\n     */\n    this.timeStamp = Date.now();\n}\n\nEventData.prototype.constructor = EventData;\nmodule.exports = EventData;\n\n/**\n * Stops the propagation of events up the scene graph (prevents bubbling).\n *\n */\nEventData.prototype.stopPropagation = function stopPropagation() {\n    this.stopped = true;\n};\n\n/**\n * Stops the propagation of events to sibling listeners (no longer calls any listeners).\n *\n */\nEventData.prototype.stopImmediatePropagation = function stopImmediatePropagation() {\n    this.stoppedImmediate = true;\n};\n","//TODO: Have Graphics use https://github.com/mattdesl/shape2d\n// and https://github.com/mattdesl/shape2d-triangulate instead of custom code.\n\n/*\n    PolyK library\n    url: http://polyk.ivank.net\n    Released under MIT licence.\n\n    Copyright (c) 2012 Ivan Kuckir\n\n    Permission is hereby granted, free of charge, to any person\n    obtaining a copy of this software and associated documentation\n    files (the \"Software\"), to deal in the Software without\n    restriction, including without limitation the rights to use,\n    copy, modify, merge, publish, distribute, sublicense, and/or sell\n    copies of the Software, and to permit persons to whom the\n    Software is furnished to do so, subject to the following\n    conditions:\n\n    The above copyright notice and this permission notice shall be\n    included in all copies or substantial portions of the Software.\n\n    THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND,\n    EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES\n    OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND\n    NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT\n    HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,\n    WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n    FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR\n    OTHER DEALINGS IN THE SOFTWARE.\n\n    This is an amazing lib!\n\n    Slightly modified by Mat Groves (matgroves.com);\n*/\n\n/**\n * Based on the Polyk library http://polyk.ivank.net released under MIT licence.\n * This is an amazing lib!\n * Slightly modified by Mat Groves (matgroves.com);\n *\n * @namespace PIXI\n */\nvar PolyK = module.exports = {};\n\n/**\n * Triangulates shapes for webGL graphic fills.\n *\n */\nPolyK.Triangulate = function (p) {\n    var sign = true;\n\n    var n = p.length >> 1;\n    if (n < 3) return [];\n\n    var tgs = [];\n    var avl = [];\n    for (var i = 0; i < n; i++) avl.push(i);\n\n    i = 0;\n    var al = n;\n    while(al > 3) {\n        var i0 = avl[(i+0)%al];\n        var i1 = avl[(i+1)%al];\n        var i2 = avl[(i+2)%al];\n\n        var ax = p[2*i0],  ay = p[2*i0+1];\n        var bx = p[2*i1],  by = p[2*i1+1];\n        var cx = p[2*i2],  cy = p[2*i2+1];\n\n        var earFound = false;\n        if (PolyK._convex(ax, ay, bx, by, cx, cy, sign)) {\n            earFound = true;\n            for (var j = 0; j < al; j++) {\n                var vi = avl[j];\n                if (vi === i0 || vi === i1 || vi === i2) continue;\n\n                if (PolyK._PointInTriangle(p[2*vi], p[2*vi+1], ax, ay, bx, by, cx, cy)) {\n                    earFound = false;\n                    break;\n                }\n            }\n        }\n\n        if (earFound) {\n            tgs.push(i0, i1, i2);\n            avl.splice((i+1)%al, 1);\n            al--;\n            i = 0;\n        }\n        else if (i++ > 3*al) {\n            // need to flip flip reverse it!\n            // reset!\n            if (sign) {\n                tgs = [];\n                avl = [];\n                for (i = 0; i < n; i++) avl.push(i);\n\n                i = 0;\n                al = n;\n\n                sign = false;\n            }\n            else {\n             //   window.console.log(\"PIXI Warning: shape too complex to fill\");\n                return null;\n            }\n        }\n    }\n\n    tgs.push(avl[0], avl[1], avl[2]);\n    return tgs;\n};\n\n/**\n * Checks whether a point is within a triangle\n *\n * @param px {number} x coordinate of the point to test\n * @param py {number} y coordinate of the point to test\n * @param ax {number} x coordinate of the a point of the triangle\n * @param ay {number} y coordinate of the a point of the triangle\n * @param bx {number} x coordinate of the b point of the triangle\n * @param by {number} y coordinate of the b point of the triangle\n * @param cx {number} x coordinate of the c point of the triangle\n * @param cy {number} y coordinate of the c point of the triangle\n * @private\n * @return {boolean}\n */\nPolyK._PointInTriangle = function (px, py, ax, ay, bx, by, cx, cy) {\n    var v0x = cx-ax;\n    var v0y = cy-ay;\n    var v1x = bx-ax;\n    var v1y = by-ay;\n    var v2x = px-ax;\n    var v2y = py-ay;\n\n    var dot00 = v0x*v0x+v0y*v0y;\n    var dot01 = v0x*v1x+v0y*v1y;\n    var dot02 = v0x*v2x+v0y*v2y;\n    var dot11 = v1x*v1x+v1y*v1y;\n    var dot12 = v1x*v2x+v1y*v2y;\n\n    var invDenom = 1 / (dot00 * dot11 - dot01 * dot01);\n    var u = (dot11 * dot02 - dot01 * dot12) * invDenom;\n    var v = (dot00 * dot12 - dot01 * dot02) * invDenom;\n\n    // Check if point is in triangle\n    return (u >= 0) && (v >= 0) && (u + v < 1);\n};\n\n/**\n * Checks whether a shape is convex\n *\n * @private\n * @return {boolean}\n */\nPolyK._convex = function (ax, ay, bx, by, cx, cy, sign) {\n    return ((ay-by)*(cx-bx) + (bx-ax)*(cy-by) >= 0) === sign;\n};\n","var EventData = require('./EventData');\n\n/**\n * Mixins event emitter functionality to an object.\n *\n * @mixin\n * @namespace PIXI\n * @example\n *      function MyEmitter() {}\n *\n *      eventTarget.mixin(MyEmitter.prototype);\n *\n *      var em = new MyEmitter();\n *      em.emit('eventName', 'some data', 'some more data', {}, null, ...);\n */\nfunction eventTarget(obj) {\n    /**\n     * Return a list of assigned event listeners.\n     *\n     * @param eventName {string} The events that should be listed.\n     * @return {Array} An array of listener functions\n     */\n    obj.listeners = function listeners(eventName) {\n        this._listeners = this._listeners || {};\n\n        return this._listeners[eventName] ? this._listeners[eventName].slice() : [];\n    };\n\n    /**\n     * Emit an event to all registered event listeners.\n     *\n     * @alias dispatchEvent\n     * @param eventName {string} The name of the event.\n     * @return {boolean} Indication if we've emitted an event.\n     */\n    obj.emit = obj.dispatchEvent = function emit(eventName, data) {\n        this._listeners = this._listeners || {};\n\n        // fast return when there are no listeners\n        if (!this._listeners[eventName]) {\n            return;\n        }\n\n        //backwards compat with old method \".emit({ type: 'something' })\"\n        if (typeof eventName === 'object') {\n            data = eventName;\n            eventName = eventName.type;\n        }\n\n        //ensure we are using a real pixi event\n        if (!data || data.__isEventObject !== true) {\n            data = new EventData(this, eventName, data);\n        }\n\n        //iterate the listeners\n        var listeners = this._listeners[eventName].slice(0),\n            length = listeners.length,\n            fn = listeners[0],\n            i;\n\n        for (i = 0; i < length; fn = listeners[++i]) {\n            //call the event listener\n            fn.call(this, data);\n\n            //if \"stopImmediatePropagation\" is called, stop calling sibling events\n            if (data.stoppedImmediate) {\n                return this;\n            }\n        }\n\n        //if \"stopPropagation\" is called then don't bubble the event\n        if (data.stopped) {\n            return this;\n        }\n\n        //bubble this event up the scene graph\n        if (this.parent && this.parent.emit) {\n            this.parent.emit.call(this.parent, eventName, data);\n        }\n\n        return this;\n    };\n\n    /**\n     * Register a new EventListener for the given event.\n     *\n     * @alias addEventListener\n     * @param eventName {string} Name of the event.\n     * @param callback {Functon} fn Callback function.\n     */\n    obj.on = obj.addEventListener = function on(eventName, fn) {\n        this._listeners = this._listeners || {};\n\n        (this._listeners[eventName] = this._listeners[eventName] || [])\n            .push(fn);\n\n        return this;\n    };\n\n    /**\n     * Add an EventListener that's only called once.\n     *\n     * @param eventName {string} Name of the event.\n     * @param callback {Function} Callback function.\n     */\n    obj.once = function once(eventName, fn) {\n        this._listeners = this._listeners || {};\n\n        var self = this;\n        function onceHandlerWrapper() {\n            fn.apply(self.off(eventName, onceHandlerWrapper), arguments);\n        }\n        onceHandlerWrapper._originalHandler = fn;\n\n        return this.on(eventName, onceHandlerWrapper);\n    };\n\n    /**\n     * Remove event listeners.\n     *\n     * @alias removeEventListener\n     * @param eventName {string} The event we want to remove.\n     * @param callback {Function} The listener that we need to find.\n     */\n    obj.off = obj.removeEventListener = function off(eventName, fn) {\n        this._listeners = this._listeners || {};\n\n        if (!this._listeners[eventName]) {\n            return this;\n        }\n\n        var list = this._listeners[eventName],\n            i = fn ? list.length : 0;\n\n        while(i-- > 0) {\n            if (list[i] === fn || list[i]._originalHandler === fn) {\n                list.splice(i, 1);\n            }\n        }\n\n        if (list.length === 0) {\n            delete this._listeners[eventName];\n        }\n\n        return this;\n    };\n\n    /**\n     * Remove all listeners or only the listeners for the specified event.\n     *\n     * @param eventName {string} The event you want to remove all listeners for.\n     */\n    obj.removeAllListeners = function removeAllListeners(eventName) {\n        this._listeners = this._listeners || {};\n\n        if (!this._listeners[eventName]) {\n            return this;\n        }\n\n        delete this._listeners[eventName];\n\n        return this;\n    };\n}\n\nmodule.exports = {\n    /**\n     * Mixes in the properties of the eventTarget prototype onto another object\n     *\n     * @param object {object} The obj to mix into\n     */\n    mixin: function mixin(obj) {\n        eventTarget(obj);\n    }\n};\n","var CONST = require('../const');\n\n/**\n * @namespace PIXI\n */\nvar utils = module.exports = {\n    _uid: 0,\n    _saidHello: false,\n\n    PolyK:      require('./PolyK'),\n    EventData:  require('./EventData'),\n    eventTarget: require('./eventTarget'),\n\n    /**\n     * Gets the next uuid\n     *\n     * @return {number} The next uuid to use.\n     */\n    uuid: function () {\n        return ++utils._uid;\n    },\n\n    /**\n     * Converts a hex color number to an [R, G, B] array\n     *\n     * @param hex {number}\n     * @return {number[]} An array representing the [R, G, B] of the color.\n     */\n    hex2rgb: function (hex, out) {\n        out = out || [];\n\n        out[0] = (hex >> 16 & 0xFF) / 255;\n        out[1] = (hex >> 8 & 0xFF) / 255;\n        out[2] = (hex & 0xFF) / 255;\n\n        return out;\n    },\n\n    /**\n     * Converts a hex color number to a string.\n     *\n     * @param hex {number}\n     * @return {string} The string color.\n     */\n    hex2string: function (hex) {\n        hex = hex.toString(16);\n        hex = '000000'.substr(0, 6 - hex.length) + hex;\n\n        return '#' + hex;\n    },\n\n    /**\n     * Converts a color as an [R, G, B] array to a hex number\n     *\n     * @param rgb {number[]}\n     * @return {number} The color number\n     */\n    rgb2hex: function (rgb) {\n        return ((rgb[0]*255 << 16) + (rgb[1]*255 << 8) + rgb[2]*255);\n    },\n\n    /**\n     * Checks whether the Canvas BlendModes are supported by the current browser\n     *\n     * @return {boolean} whether they are supported\n     */\n    canUseNewCanvasBlendModes: function () {\n        if (typeof document === 'undefined') {\n            return false;\n        }\n\n        var canvas = document.createElement('canvas'),\n            context = canvas.getContext('2d');\n\n        canvas.width = 1;\n        canvas.height = 1;\n\n        context.fillStyle = '#000';\n        context.fillRect(0, 0, 1, 1);\n\n        context.globalCompositeOperation = 'multiply';\n\n        context.fillStyle = '#fff';\n        context.fillRect(0, 0, 1, 1);\n\n        return context.getImageData(0,0,1,1).data[0] === 0;\n    },\n\n    /**\n     * Given a number, this function returns the closest number that is a power of two\n     * this function is taken from Starling Framework as its pretty neat ;)\n     *\n     * @param number {number}\n     * @return {number} the closest number that is a power of two\n     */\n    getNextPowerOfTwo: function (number) {\n        // see: http://en.wikipedia.org/wiki/Power_of_two#Fast_algorithm_to_check_if_a_positive_number_is_a_power_of_two\n        if (number > 0 && (number & (number - 1)) === 0) {\n            return number;\n        }\n        else {\n            var result = 1;\n\n            while (result < number) {\n                result <<= 1;\n            }\n\n            return result;\n        }\n    },\n\n    /**\n     * checks if the given width and height make a power of two rectangle\n     *\n     * @param width {number}\n     * @param height {number}\n     * @return {boolean}\n     */\n    isPowerOfTwo: function (width, height) {\n        return (width > 0 && (width & (width - 1)) === 0 && height > 0 && (height & (height - 1)) === 0);\n    },\n\n    /**\n     * Logs out the version and renderer information for this running instance of PIXI.\n     * If you don't want to see this message you can set `PIXI.utils._saidHello = true;`\n     * so the library thinks it already said it. Keep in mind that doing that will forever\n     * makes you a jerk face.\n     *\n     * @param {string} type - The string renderer type to log.\n     * @constant\n     * @static\n     */\n    sayHello: function (type) {\n        if (utils._saidHello) {\n            return;\n        }\n\n        if (navigator.userAgent.toLowerCase().indexOf('chrome') > -1) {\n            var args = [\n                '%c %c %c Pixi.js ' + CONST.VERSION + ' - ' + type + '  %c ' + ' %c ' + ' http://www.pixijs.com/  %c %c ♥%c♥%c♥ ',\n                'background: #ff66a5',\n                'background: #ff66a5',\n                'color: #ff66a5; background: #030307;',\n                'background: #ff66a5',\n                'background: #ffc3dc',\n                'background: #ff66a5',\n                'color: #ff2424; background: #fff',\n                'color: #ff2424; background: #fff',\n                'color: #ff2424; background: #fff'\n            ];\n\n            console.log.apply(console, args); //jshint ignore:line\n        }\n        else if (window.console) {\n            console.log('Pixi.js ' + CONST.VERSION + ' - ' + type + ' - http://www.pixijs.com/'); //jshint ignore:line\n        }\n\n        utils._saidHello = true;\n    },\n\n    /**\n     * A wrapper for ajax requests to be handled cross browser\n     *\n     * TODO: Replace this wil superagent\n     *\n     * @class\n     * @namespace PIXI\n     */\n    AjaxRequest: function () {\n        var activexmodes = ['Msxml2.XMLHTTP.6.0', 'Msxml2.XMLHTTP.3.0', 'Microsoft.XMLHTTP']; //activeX versions to check for in IE\n\n        if (window.ActiveXObject) { //Test for support for ActiveXObject in IE first (as XMLHttpRequest in IE7 is broken)\n            for (var i=0; i<activexmodes.length; i++) {\n                try{\n                    return new window.ActiveXObject(activexmodes[i]);\n                }\n                catch(e) {\n                    //suppress error\n                }\n            }\n        }\n        else if (window.XMLHttpRequest) // if Mozilla, Safari etc\n        {\n            return new window.XMLHttpRequest();\n        }\n        else {\n            return false;\n        }\n    },\n\n    // TODO: refactor out this\n    AnimCache: {},\n    FrameCache: {},\n    TextureCache: {},\n    BaseTextureCache: {}\n};\n","var core = require('../core');\n\n/**\n * A MovieClip is a simple way to display an animation depicted by a list of textures.\n *\n * @class\n * @extends Sprite\n * @namespace PIXI\n * @param textures {Texture[]} an array of {Texture} objects that make up the animation\n */\nfunction MovieClip(textures) {\n    core.Sprite.call(this, textures[0]);\n\n    /**\n     * The array of textures that make up the animation\n     *\n     * @member Texture[]\n     */\n    this.textures = textures;\n\n    /**\n     * The speed that the MovieClip will play at. Higher is faster, lower is slower\n     *\n     * @member number\n     * @default 1\n     */\n    this.animationSpeed = 1;\n\n    /**\n     * Whether or not the movie clip repeats after playing.\n     *\n     * @member boolean\n     * @default true\n     */\n    this.loop = true;\n\n    /**\n     * Function to call when a MovieClip finishes playing\n     *\n     * @method\n     * @memberof MovieClip#\n     */\n    this.onComplete = null;\n\n    /**\n     * The MovieClips current frame index (this may not have to be a whole number)\n     *\n     * @member number\n     * @default 0\n     * @readonly\n     */\n    this.currentFrame = 0;\n\n    /**\n     * Indicates if the MovieClip is currently playing\n     *\n     * @member boolean\n     * @readonly\n     */\n    this.playing = false;\n}\n\n// constructor\nMovieClip.prototype = Object.create(core.Sprite.prototype);\nMovieClip.prototype.constructor = MovieClip;\n\nObject.defineProperties(MovieClip.prototype, {\n    /**\n     * totalFrames is the total number of frames in the MovieClip. This is the same as number of textures\n     * assigned to the MovieClip.\n     *\n     * @member\n     * @memberof MovieClip#\n     * @default 0\n     * @readonly\n     */\n    totalFrames: {\n        get: function() {\n            return this.textures.length;\n        }\n    }\n});\n\n/**\n * Stops the MovieClip\n *\n */\nMovieClip.prototype.stop = function () {\n    this.playing = false;\n};\n\n/**\n * Plays the MovieClip\n *\n */\nMovieClip.prototype.play = function () {\n    this.playing = true;\n};\n\n/**\n * Stops the MovieClip and goes to a specific frame\n *\n * @param frameNumber {number} frame index to stop at\n */\nMovieClip.prototype.gotoAndStop = function (frameNumber) {\n    this.playing = false;\n    this.currentFrame = frameNumber;\n\n    var round = Math.round(this.currentFrame);\n    this.setTexture(this.textures[round % this.textures.length]);\n};\n\n/**\n * Goes to a specific frame and begins playing the MovieClip\n *\n * @param frameNumber {number} frame index to start at\n */\nMovieClip.prototype.gotoAndPlay = function (frameNumber) {\n    this.currentFrame = frameNumber;\n    this.playing = true;\n};\n\n/*\n * Updates the object transform for rendering\n *\n * @private\n */\nMovieClip.prototype.updateTransform = function () {\n    this.displayObjectContainerUpdateTransform();\n\n    if (!this.playing) {\n        return;\n    }\n\n    this.currentFrame += this.animationSpeed;\n\n    var round = Math.round(this.currentFrame);\n\n    if (round < 0) {\n        if (this.loop) {\n            this.currentFrame += this.textures.length;\n            this.texture = this.textures[this.currentFrame];\n        }\n        else {\n            this.gotoAndStop(0);\n\n            if (this.onComplete) {\n                this.onComplete();\n            }\n        }\n    }\n    else if (this.loop || round < this.textures.length) {\n        this.texture = this.textures[round % this.textures.length];\n    }\n    else if (round >= this.textures.length) {\n        this.gotoAndStop(this.textures.length - 1);\n\n        if (this.onComplete) {\n            this.onComplete();\n        }\n    }\n};\n\n/**\n * A short hand way of creating a movieclip from an array of frame ids\n *\n * @static\n * @param frames {string[]} the array of frames ids the movieclip will use as its texture frames\n */\nMovieClip.fromFrames = function (frames) {\n    var textures = [];\n\n    for (var i = 0; i < frames.length; ++i) {\n        textures.push(new core.Texture.fromFrame(frames[i]));\n    }\n\n    return new MovieClip(textures);\n};\n\n/**\n * A short hand way of creating a movieclip from an array of image ids\n *\n * @static\n * @param images {string[]} the array of image urls the movieclip will use as its texture frames\n */\nMovieClip.fromImages = function (images) {\n    var textures = [];\n\n    for (var i = 0; i < images.length; ++i) {\n        textures.push(new core.Texture.fromImage(images[i]));\n    }\n\n    return new MovieClip(textures);\n};\n","var Strip = require('./Strip');\n\n/**\n *\n * @class\n * @namespace PIXI\n * @extends Strip\n * @param {Texture} texture - The texture to use on the rope.\n * @param {Array} points - An array of {Point}.\n *\n */\nfunction Rope(texture, points) {\n    Strip.call(this, texture);\n    this.points = points;\n\n    this.vertices = new Float32Array(points.length * 4);\n    this.uvs = new Float32Array(points.length * 4);\n    this.colors = new Float32Array(points.length * 2);\n    this.indices = new Uint16Array(points.length * 2);\n\n    this.refresh();\n}\n\n\n// constructor\nRope.prototype = Object.create(Strip.prototype);\nRope.prototype.constructor = Rope;\nmodule.exports = Rope;\n\n/**\n * Refreshes\n *\n */\nRope.prototype.refresh = function () {\n    var points = this.points;\n\n    if (points.length < 1) {\n        return;\n    }\n\n    var uvs = this.uvs;\n\n    var indices = this.indices;\n    var colors = this.colors;\n\n    // this.count -= 0.2;\n\n    uvs[0] = 0;\n    uvs[1] = 0;\n    uvs[2] = 0;\n    uvs[3] = 1;\n\n    colors[0] = 1;\n    colors[1] = 1;\n\n    indices[0] = 0;\n    indices[1] = 1;\n\n    var total = points.length,\n        point, index, amount;\n\n    for (var i = 1; i < total; i++) {\n        point = points[i];\n        index = i * 4;\n        // time to do some smart drawing!\n        amount = i / (total-1);\n\n        if (i%2) {\n            uvs[index] = amount;\n            uvs[index+1] = 0;\n\n            uvs[index+2] = amount;\n            uvs[index+3] = 1;\n        }\n        else {\n            uvs[index] = amount;\n            uvs[index+1] = 0;\n\n            uvs[index+2] = amount;\n            uvs[index+3] = 1;\n        }\n\n        index = i * 2;\n        colors[index] = 1;\n        colors[index+1] = 1;\n\n        index = i * 2;\n        indices[index] = index;\n        indices[index + 1] = index + 1;\n    }\n};\n\n/*\n * Updates the object transform for rendering\n *\n * @private\n */\nRope.prototype.updateTransform = function () {\n    var points = this.points;\n\n    if (points.length < 1) {\n        return;\n    }\n\n    var lastPoint = points[0];\n    var nextPoint;\n    var perpX = 0;\n    var perpY = 0;\n\n    // this.count -= 0.2;\n\n    var vertices = this.vertices;\n    var total = points.length,\n        point, index, ratio, perpLength, num;\n\n    for (var i = 0; i < total; i++) {\n        point = points[i];\n        index = i * 4;\n\n        if (i < points.length-1) {\n            nextPoint = points[i+1];\n        }\n        else {\n            nextPoint = point;\n        }\n\n        perpY = -(nextPoint.x - lastPoint.x);\n        perpX = nextPoint.y - lastPoint.y;\n\n        ratio = (1 - (i / (total-1))) * 10;\n\n        if (ratio > 1) {\n            ratio = 1;\n        }\n\n        perpLength = Math.sqrt(perpX * perpX + perpY * perpY);\n        num = this.texture.height / 2; //(20 + Math.abs(Math.sin((i + this.count) * 0.3) * 50) )* ratio;\n        perpX /= perpLength;\n        perpY /= perpLength;\n\n        perpX *= num;\n        perpY *= num;\n\n        vertices[index] = point.x + perpX;\n        vertices[index+1] = point.y + perpY;\n        vertices[index+2] = point.x - perpX;\n        vertices[index+3] = point.y - perpY;\n\n        lastPoint = point;\n    }\n\n    this.displayObjectContainerUpdateTransform();\n};\n\n/**\n * Sets the texture that the Rope will use\n *\n * @param texture {Texture} the texture that will be used\n */\nRope.prototype.setTexture = function (texture) {\n    // stop current texture\n    this.texture = texture;\n    //this.updateFrame = true;\n};\n","var core = require('../core');\n\n/**\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param texture {Texture} The texture to use\n * @param width {number} the width\n * @param height {number} the height\n *\n */\nfunction Strip(texture) {\n    core.DisplayObjectContainer.call(this);\n\n    /**\n     * The texture of the strip\n     *\n     * @member {Texture}\n     */\n    this.texture = texture;\n\n    // set up the main bits..\n    this.uvs = new Float32Array([0, 1,\n                                 1, 1,\n                                 1, 0,\n                                 0, 1]);\n\n    this.vertices = new Float32Array([0, 0,\n                                      100, 0,\n                                      100, 100,\n                                      0, 100]);\n\n    this.colors = new Float32Array([1, 1, 1, 1]);\n\n    this.indices = new Uint16Array([0, 1, 2, 3]);\n\n    /**\n     * Whether the strip is dirty or not\n     *\n     * @member {boolean}\n     */\n    this.dirty = true;\n\n    /**\n     * The blend mode to be applied to the sprite. Set to blendModes.NORMAL to remove any blend mode.\n     *\n     * @member {number}\n     * @default CONST.blendModes.NORMAL;\n     */\n    this.blendMode = core.CONST.blendModes.NORMAL;\n\n    /**\n     * Triangles in canvas mode are automatically antialiased, use this value to force triangles to overlap a bit with each other.\n     *\n     * @member {number}\n     */\n    this.canvasPadding = 0;\n\n    this.drawMode = Strip.DrawModes.TRIANGLE_STRIP;\n}\n\n// constructor\nStrip.prototype = Object.create(core.DisplayObjectContainer.prototype);\nStrip.prototype.constructor = Strip;\nmodule.exports = Strip;\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer}\n */\nStrip.prototype.renderWebGL = function (renderer) {\n    // if the sprite is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    // render triangle strip..\n\n    renderer.spriteBatch.stop();\n\n    // init! init!\n    if (!this._vertexBuffer) {\n        this._initWebGL(renderer);\n    }\n\n    renderer.shaderManager.setShader(renderer.shaderManager.stripShader);\n\n    this._renderStrip(renderer);\n\n    ///renderer.shaderManager.activateDefaultShader();\n\n    renderer.spriteBatch.start();\n\n    //TODO check culling\n};\n\nStrip.prototype._initWebGL = function (renderer) {\n    // build the strip!\n    var gl = renderer.gl;\n\n    this._vertexBuffer = gl.createBuffer();\n    this._indexBuffer = gl.createBuffer();\n    this._uvBuffer = gl.createBuffer();\n    this._colorBuffer = gl.createBuffer();\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER,  this.uvs, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this._colorBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.colors, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n};\n\nStrip.prototype._renderStrip = function (renderer) {\n    var gl = renderer.gl;\n    var projection = renderer.projection,\n        offset = renderer.offset,\n        shader = renderer.shaderManager.stripShader;\n\n    var drawMode = this.drawMode === Strip.DrawModes.TRIANGLE_STRIP ? gl.TRIANGLE_STRIP : gl.TRIANGLES;\n\n    // gl.uniformMatrix4fv(shaderProgram.mvMatrixUniform, false, mat4Real);\n\n    renderer.blendModeManager.setBlendMode(this.blendMode);\n\n\n    // set uniforms\n    gl.uniformMatrix3fv(shader.translationMatrix, false, this.worldTransform.toArray(true));\n    gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n    gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n    gl.uniform1f(shader.alpha, this.worldAlpha);\n\n    if (!this.dirty) {\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer);\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices);\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0);\n\n        // update the uvs\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer);\n        gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0);\n\n        gl.activeTexture(gl.TEXTURE0);\n\n        // check if a texture is dirty..\n        if (this.texture.baseTexture._dirty[gl.id]) {\n            renderer.updateTexture(this.texture.baseTexture);\n        }\n        else {\n            // bind the current texture\n            gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]);\n        }\n\n        // dont need to upload!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer);\n\n\n    }\n    else {\n\n        this.dirty = false;\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer);\n        gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.STATIC_DRAW);\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0);\n\n        // update the uvs\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer);\n        gl.bufferData(gl.ARRAY_BUFFER, this.uvs, gl.STATIC_DRAW);\n        gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0);\n\n        gl.activeTexture(gl.TEXTURE0);\n\n        // check if a texture is dirty..\n        if (this.texture.baseTexture._dirty[gl.id]) {\n            renderer.updateTexture(this.texture.baseTexture);\n        }\n        else {\n            gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]);\n        }\n\n        // dont need to upload!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer);\n        gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n\n    }\n    //console.log(gl.TRIANGLE_STRIP)\n    //\n    //\n    gl.drawElements(drawMode, this.indices.length, gl.UNSIGNED_SHORT, 0);\n\n\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer}\n */\nStrip.prototype.renderCanvas = function (renderer) {\n    var context = renderer.context;\n\n    var transform = this.worldTransform;\n\n    if (renderer.roundPixels) {\n        context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx | 0, transform.ty | 0);\n    }\n    else {\n        context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx, transform.ty);\n    }\n\n    if (this.drawMode === Strip.DrawModes.TRIANGLE_STRIP) {\n        this._renderCanvasTriangleStrip(context);\n    }\n    else {\n        this._renderCanvasTriangles(context);\n    }\n};\n\nStrip.prototype._renderCanvasTriangleStrip = function (context) {\n    // draw triangles!!\n    var vertices = this.vertices;\n    var uvs = this.uvs;\n\n    var length = vertices.length / 2;\n    // this.count++;\n\n    for (var i = 0; i < length - 2; i++) {\n        // draw some triangles!\n        var index = i * 2;\n        this._renderCanvasDrawTriangle(context, vertices, uvs, index, (index + 2), (index + 4));\n    }\n};\n\nStrip.prototype._renderCanvasTriangles = function (context) {\n    // draw triangles!!\n    var vertices = this.vertices;\n    var uvs = this.uvs;\n    var indices = this.indices;\n\n    var length = indices.length;\n    // this.count++;\n\n    for (var i = 0; i < length; i += 3) {\n        // draw some triangles!\n        var index0 = indices[i] * 2, index1 = indices[i + 1] * 2, index2 = indices[i + 2] * 2;\n        this._renderCanvasDrawTriangle(context, vertices, uvs, index0, index1, index2);\n    }\n};\n\nStrip.prototype._renderCanvasDrawTriangle = function (context, vertices, uvs, index0, index1, index2) {\n    var textureSource = this.texture.baseTexture.source;\n    var textureWidth = this.texture.width;\n    var textureHeight = this.texture.height;\n\n    var x0 = vertices[index0], x1 = vertices[index1], x2 = vertices[index2];\n    var y0 = vertices[index0 + 1], y1 = vertices[index1 + 1], y2 = vertices[index2 + 1];\n\n    var u0 = uvs[index0] * textureWidth, u1 = uvs[index1] * textureWidth, u2 = uvs[index2] * textureWidth;\n    var v0 = uvs[index0 + 1] * textureHeight, v1 = uvs[index1 + 1] * textureHeight, v2 = uvs[index2 + 1] * textureHeight;\n\n    if (this.canvasPadding > 0) {\n        var paddingX = this.canvasPadding / this.worldTransform.a;\n        var paddingY = this.canvasPadding / this.worldTransform.d;\n        var centerX = (x0 + x1 + x2) / 3;\n        var centerY = (y0 + y1 + y2) / 3;\n\n        var normX = x0 - centerX;\n        var normY = y0 - centerY;\n\n        var dist = Math.sqrt(normX * normX + normY * normY);\n        x0 = centerX + (normX / dist) * (dist + paddingX);\n        y0 = centerY + (normY / dist) * (dist + paddingY);\n\n        //\n\n        normX = x1 - centerX;\n        normY = y1 - centerY;\n\n        dist = Math.sqrt(normX * normX + normY * normY);\n        x1 = centerX + (normX / dist) * (dist + paddingX);\n        y1 = centerY + (normY / dist) * (dist + paddingY);\n\n        normX = x2 - centerX;\n        normY = y2 - centerY;\n\n        dist = Math.sqrt(normX * normX + normY * normY);\n        x2 = centerX + (normX / dist) * (dist + paddingX);\n        y2 = centerY + (normY / dist) * (dist + paddingY);\n    }\n\n    context.save();\n    context.beginPath();\n\n\n    context.moveTo(x0, y0);\n    context.lineTo(x1, y1);\n    context.lineTo(x2, y2);\n\n    context.closePath();\n\n    context.clip();\n\n    // Compute matrix transform\n    var delta =  (u0 * v1)      + (v0 * u2)      + (u1 * v2)      - (v1 * u2)      - (v0 * u1)      - (u0 * v2);\n    var deltaA = (x0 * v1)      + (v0 * x2)      + (x1 * v2)      - (v1 * x2)      - (v0 * x1)      - (x0 * v2);\n    var deltaB = (u0 * x1)      + (x0 * u2)      + (u1 * x2)      - (x1 * u2)      - (x0 * u1)      - (u0 * x2);\n    var deltaC = (u0 * v1 * x2) + (v0 * x1 * u2) + (x0 * u1 * v2) - (x0 * v1 * u2) - (v0 * u1 * x2) - (u0 * x1 * v2);\n    var deltaD = (y0 * v1)      + (v0 * y2)      + (y1 * v2)      - (v1 * y2)      - (v0 * y1)      - (y0 * v2);\n    var deltaE = (u0 * y1)      + (y0 * u2)      + (u1 * y2)      - (y1 * u2)      - (y0 * u1)      - (u0 * y2);\n    var deltaF = (u0 * v1 * y2) + (v0 * y1 * u2) + (y0 * u1 * v2) - (y0 * v1 * u2) - (v0 * u1 * y2) - (u0 * y1 * v2);\n\n    context.transform(deltaA / delta, deltaD / delta,\n        deltaB / delta, deltaE / delta,\n        deltaC / delta, deltaF / delta);\n\n    context.drawImage(textureSource, 0, 0);\n    context.restore();\n};\n\n\n\n/**\n * Renders a flat strip\n *\n * @param strip {Strip} The Strip to render\n * @private\n */\nStrip.prototype.renderStripFlat = function (strip) {\n    var context = this.context;\n    var vertices = strip.vertices;\n\n    var length = vertices.length/2;\n    // this.count++;\n\n    context.beginPath();\n    for (var i=1; i < length-2; i++) {\n        // draw some triangles!\n        var index = i*2;\n\n        var x0 = vertices[index],   x1 = vertices[index+2], x2 = vertices[index+4];\n        var y0 = vertices[index+1], y1 = vertices[index+3], y2 = vertices[index+5];\n\n        context.moveTo(x0, y0);\n        context.lineTo(x1, y1);\n        context.lineTo(x2, y2);\n    }\n\n    context.fillStyle = '#FF0000';\n    context.fill();\n    context.closePath();\n};\n\n/*\nStrip.prototype.setTexture = function (texture) {\n    //TODO SET THE TEXTURES\n    //TODO VISIBILITY\n\n    // stop current texture\n    this.texture = texture;\n    this.width   = texture.frame.width;\n    this.height  = texture.frame.height;\n    this.updateFrame = true;\n};\n */\n\n/**\n * When the texture is updated, this event will fire to update the scale and frame\n *\n * @param event\n * @private\n */\n\nStrip.prototype.onTextureUpdate = function () {\n    this.updateFrame = true;\n};\n\n/**\n * Returns the bounds of the mesh as a rectangle. The bounds calculation takes the worldTransform into account.\n *\n * @param matrix {Matrix} the transformation matrix of the sprite\n * @return {Rectangle} the framing rectangle\n */\nStrip.prototype.getBounds = function (matrix) {\n    var worldTransform = matrix || this.worldTransform;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var maxX = -Infinity;\n    var maxY = -Infinity;\n\n    var minX = Infinity;\n    var minY = Infinity;\n\n    var vertices = this.vertices;\n    for (var i = 0, n = vertices.length; i < n; i += 2) {\n        var rawX = vertices[i], rawY = vertices[i + 1];\n        var x = (a * rawX) + (c * rawY) + tx;\n        var y = (d * rawY) + (b * rawX) + ty;\n\n        minX = x < minX ? x : minX;\n        minY = y < minY ? y : minY;\n\n        maxX = x > maxX ? x : maxX;\n        maxY = y > maxY ? y : maxY;\n    }\n\n    if (minX === -Infinity || maxY === Infinity) {\n        return core.math.Rectangle.EMPTY;\n    }\n\n    var bounds = this._bounds;\n\n    bounds.x = minX;\n    bounds.width = maxX - minX;\n\n    bounds.y = minY;\n    bounds.height = maxY - minY;\n\n    // store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    this._currentBounds = bounds;\n\n    return bounds;\n};\n\n/**\n * Different drawing buffer modes supported\n *\n * @static\n * @constant\n * @property {object} DrawModes\n * @property {number} DrawModes.TRIANGLE_STRIP\n * @property {number} DrawModes.TRIANGLES\n */\nStrip.DrawModes = {\n    TRIANGLE_STRIP: 0,\n    TRIANGLES: 1\n};\n","var core = require('../core');\n\n/**\n * A tiling sprite is a fast way of rendering a tiling image\n *\n * @class\n * @extends Sprite\n * @namespace PIXI\n * @param texture {Texture} the texture of the tiling sprite\n * @param width {number}  the width of the tiling sprite\n * @param height {number} the height of the tiling sprite\n */\nfunction TilingSprite(texture, width, height) {\n    core.Sprite.call( this, texture);\n\n    /**\n     * The with of the tiling sprite\n     *\n     * @member {number}\n     */\n    this._width = width || 100;\n\n    /**\n     * The height of the tiling sprite\n     *\n     * @member {number}\n     */\n    this._height = height || 100;\n\n    /**\n     * The scaling of the image that is being tiled\n     *\n     * @member {Point}\n     */\n    this.tileScale = new core.math.Point(1,1);\n\n    /**\n     * A point that represents the scale of the texture object\n     *\n     * @member {Point}\n     */\n    this.tileScaleOffset = new core.math.Point(1,1);\n\n    /**\n     * The offset position of the image that is being tiled\n     *\n     * @member {Point}\n     */\n    this.tilePosition = new core.math.Point(0,0);\n\n    /**\n     * Whether this sprite is renderable or not\n     *\n     * @member {boolean}\n     * @default true\n     */\n    this.renderable = true;\n\n    /**\n     * The tint applied to the sprite. This is a hex value\n     *\n     * @member {number}\n     * @default 0xFFFFFF\n     */\n    this.tint = 0xFFFFFF;\n\n    /**\n     * The blend mode to be applied to the sprite\n     *\n     * @member {number}\n     * @default blendModes.NORMAL;\n     */\n    this.blendMode = core.CONST.blendModes.NORMAL;\n}\n\nTilingSprite.prototype = Object.create(core.Sprite.prototype);\nTilingSprite.prototype.constructor = TilingSprite;\nmodule.exports = TilingSprite;\n\n\nObject.defineProperties(TilingSprite.prototype, {\n    /**\n     * The width of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof TilingSprite#\n     */\n    width: {\n        get: function () {\n            return this._width;\n        },\n        set: function (value) {\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the TilingSprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof TilingSprite#\n     */\n    height: {\n        get: function () {\n            return this._height;\n        },\n        set: function (value) {\n            this._height = value;\n        }\n    },\n\n    texture: {\n        get: function () {\n            return this._texture;\n        },\n        set: function (value) {\n            if (this._texture === value) {\n                return;\n            }\n\n            this._texture = value;\n            this.refreshTexture = true;\n            this.cachedTint = 0xFFFFFF;\n        }\n    }\n});\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer}\n */\nTilingSprite.prototype.renderWebGL = function (renderer) {\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    var i, j;\n\n    if (this._mask) {\n        renderer.spriteBatch.stop();\n        renderer.maskManager.pushMask(this.mask, renderer);\n        renderer.spriteBatch.start();\n    }\n\n    if (this._filters) {\n        renderer.spriteBatch.flush();\n        renderer.filterManager.pushFilter(this._filterBlock);\n    }\n\n\n\n    if (!this.tilingTexture || this.refreshTexture) {\n        this.generateTilingTexture(true);\n\n        if (this.tilingTexture && this.tilingTexture.needsUpdate) {\n            //TODO - tweaking\n            renderer.updateTexture(this.tilingTexture.baseTexture);\n            this.tilingTexture.needsUpdate = false;\n           // this.tilingTexture._uvs = null;\n        }\n    }\n    else {\n        renderer.spriteBatch.renderTilingSprite(this);\n    }\n    // simple render children!\n    for (i=0,j=this.children.length; i<j; i++) {\n        this.children[i].renderWebGL(renderer);\n    }\n\n    renderer.spriteBatch.stop();\n\n    if (this._filters) {\n        renderer.filterManager.popFilter();\n    }\n\n    if (this._mask) {\n        renderer.maskManager.popMask(this._mask, renderer);\n    }\n\n    renderer.spriteBatch.start();\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer}\n */\nTilingSprite.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    var context = renderer.context;\n\n    if (this._mask) {\n        renderer.maskManager.pushMask(this._mask, context);\n    }\n\n    context.globalAlpha = this.worldAlpha;\n\n    var transform = this.worldTransform;\n\n    var i,j;\n\n    var resolution = renderer.resolution;\n\n    context.setTransform(transform.a * resolution,\n                         transform.b * resolution,\n                         transform.c * resolution,\n                         transform.d * resolution,\n                         transform.tx * resolution,\n                         transform.ty * resolution);\n\n    if (!this.__tilePattern ||  this.refreshTexture) {\n        this.generateTilingTexture(false);\n\n        if (this.tilingTexture) {\n            this.__tilePattern = context.createPattern(this.tilingTexture.baseTexture.source, 'repeat');\n        }\n        else {\n            return;\n        }\n    }\n\n    // check blend mode\n    if (this.blendMode !== renderer.currentBlendMode) {\n        renderer.currentBlendMode = this.blendMode;\n        context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode];\n    }\n\n    var tilePosition = this.tilePosition;\n    var tileScale = this.tileScale;\n\n    tilePosition.x %= this.tilingTexture.baseTexture.width;\n    tilePosition.y %= this.tilingTexture.baseTexture.height;\n\n    // offset - make sure to account for the anchor point..\n    context.scale(tileScale.x,tileScale.y);\n    context.translate(tilePosition.x + (this.anchor.x * -this._width), tilePosition.y + (this.anchor.y * -this._height));\n\n    context.fillStyle = this.__tilePattern;\n\n    context.fillRect(-tilePosition.x,\n                    -tilePosition.y,\n                    this._width / tileScale.x,\n                    this._height / tileScale.y);\n\n    context.translate(-tilePosition.x + (this.anchor.x * this._width), -tilePosition.y + (this.anchor.y * this._height));\n    context.scale(1 / tileScale.x, 1 / tileScale.y);\n\n    if (this._mask) {\n        renderer.maskManager.popMask(renderer.context);\n    }\n\n    for (i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].renderCanvas(renderer);\n    }\n};\n\n\n/**\n * Returns the framing rectangle of the sprite as a Rectangle object\n*\n * @return {Rectangle} the framing rectangle\n */\nTilingSprite.prototype.getBounds = function () {\n    var width = this._width;\n    var height = this._height;\n\n    var w0 = width * (1-this.anchor.x);\n    var w1 = width * -this.anchor.x;\n\n    var h0 = height * (1-this.anchor.y);\n    var h1 = height * -this.anchor.y;\n\n    var worldTransform = this.worldTransform;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var x1 = a * w1 + c * h1 + tx;\n    var y1 = d * h1 + b * w1 + ty;\n\n    var x2 = a * w0 + c * h1 + tx;\n    var y2 = d * h1 + b * w0 + ty;\n\n    var x3 = a * w0 + c * h0 + tx;\n    var y3 = d * h0 + b * w0 + ty;\n\n    var x4 =  a * w1 + c * h0 + tx;\n    var y4 =  d * h0 + b * w1 + ty;\n\n    var minX,\n        maxX,\n        minY,\n        maxY;\n\n    minX = x1;\n    minX = x2 < minX ? x2 : minX;\n    minX = x3 < minX ? x3 : minX;\n    minX = x4 < minX ? x4 : minX;\n\n    minY = y1;\n    minY = y2 < minY ? y2 : minY;\n    minY = y3 < minY ? y3 : minY;\n    minY = y4 < minY ? y4 : minY;\n\n    maxX = x1;\n    maxX = x2 > maxX ? x2 : maxX;\n    maxX = x3 > maxX ? x3 : maxX;\n    maxX = x4 > maxX ? x4 : maxX;\n\n    maxY = y1;\n    maxY = y2 > maxY ? y2 : maxY;\n    maxY = y3 > maxY ? y3 : maxY;\n    maxY = y4 > maxY ? y4 : maxY;\n\n    var bounds = this._bounds;\n\n    bounds.x = minX;\n    bounds.width = maxX - minX;\n\n    bounds.y = minY;\n    bounds.height = maxY - minY;\n\n    // store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    this._currentBounds = bounds;\n\n    return bounds;\n};\n\n/**\n * When the texture is updated, this event will fire to update the scale and frame\n *\n * @param event\n * @private\n */\nTilingSprite.prototype.onTextureUpdate = function () {\n   // overriding the sprite version of this!\n};\n\n/**\n *\n * @param forcePowerOfTwo {boolean} Whether we want to force the texture to be a power of two\n */\nTilingSprite.prototype.generateTilingTexture = function (forcePowerOfTwo) {\n    if (!this.texture.baseTexture.hasLoaded) {\n        return;\n    }\n\n    var texture = this.originalTexture || this.texture;\n    var frame = texture.frame;\n    var targetWidth, targetHeight;\n\n    //  Check that the frame is the same size as the base texture.\n    var isFrame = frame.width !== texture.baseTexture.width || frame.height !== texture.baseTexture.height;\n\n    var newTextureRequired = false;\n\n    if (!forcePowerOfTwo) {\n        if (isFrame) {\n            targetWidth = frame.width;\n            targetHeight = frame.height;\n\n            newTextureRequired = true;\n        }\n    }\n    else {\n        targetWidth = core.utils.getNextPowerOfTwo(frame.width);\n        targetHeight = core.utils.getNextPowerOfTwo(frame.height);\n\n        //  If the BaseTexture dimensions don't match the texture frame then we need a new texture anyway because it's part of a texture atlas\n        if (frame.width !== targetWidth || frame.height !== targetHeight || texture.baseTexture.width !== targetWidth || texture.baseTexture.height || targetHeight) {\n            newTextureRequired = true;\n        }\n    }\n\n    if (newTextureRequired) {\n        var canvasBuffer;\n\n        if (this.tilingTexture && this.tilingTexture.isTiling) {\n            canvasBuffer = this.tilingTexture.canvasBuffer;\n            canvasBuffer.resize(targetWidth, targetHeight);\n            this.tilingTexture.baseTexture.width = targetWidth;\n            this.tilingTexture.baseTexture.height = targetHeight;\n            this.tilingTexture.needsUpdate = true;\n        }\n        else {\n            canvasBuffer = new core.CanvasBuffer(targetWidth, targetHeight);\n\n            this.tilingTexture = core.Texture.fromCanvas(canvasBuffer.canvas);\n            this.tilingTexture.canvasBuffer = canvasBuffer;\n            this.tilingTexture.isTiling = true;\n        }\n\n        canvasBuffer.context.drawImage(texture.baseTexture.source,\n                               texture.crop.x,\n                               texture.crop.y,\n                               texture.crop.width,\n                               texture.crop.height,\n                               0,\n                               0,\n                               targetWidth,\n                               targetHeight);\n\n        this.tileScaleOffset.x = frame.width / targetWidth;\n        this.tileScaleOffset.y = frame.height / targetHeight;\n    }\n    else {\n        //  TODO - switching?\n        if (this.tilingTexture && this.tilingTexture.isTiling) {\n            // destroy the tiling texture!\n            // TODO could store this somewhere?\n            this.tilingTexture.destroy(true);\n        }\n\n        this.tileScaleOffset.x = 1;\n        this.tileScaleOffset.y = 1;\n        this.tilingTexture = texture;\n    }\n\n    this.refreshTexture = false;\n\n    this.originalTexture = this.texture;\n    this.texture = this.tilingTexture;\n\n    this.tilingTexture.baseTexture._powerOf2 = true;\n};\n","/**\n * @file        Main export of the PIXI extras library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    MovieClip:      require('./MovieClip'),\n    Rope:           require('./Rope'),\n    Strip:          require('./Strip'),\n    TilingSprite:   require('./TilingSprite')\n};\n","/**\n * This is the base class for creating a PIXI filter. Currently only WebGL supports filters.\n * If you want to make a custom filter this should be your base class.\n *\n * @class\n * @namespace PIXI\n * @param fragmentSrc {string|string[]} The fragment source in an array of strings.\n * @param uniforms {object} An object containing the uniforms for this filter.\n */\nfunction AbstractFilter(fragmentSrc, uniforms) {\n    /**\n     * An array of passes - some filters contain a few steps this array simply stores the steps in a liniear fashion.\n     * For example the blur filter has two passes blurX and blurY.\n     *\n     * @member {AbstractFilter[]}\n     * @private\n     */\n    this.passes = [this];\n\n    /**\n     * @member {Shader[]}\n     * @private\n     */\n    this.shaders = [];\n\n    /**\n     * @member {number}\n     */\n    this.padding = 0;\n\n    /**\n     * @member {object}\n     * @private\n     */\n    this.uniforms = uniforms || {};\n\n    /**\n     * @member {string[]}\n     * @private\n     */\n    this.fragmentSrc = typeof fragmentSrc === 'string' ? fragmentSrc.split('') : (fragmentSrc || []);\n}\n\nAbstractFilter.prototype.constructor = AbstractFilter;\nmodule.exports = AbstractFilter;\n\n/**\n * Syncs the uniforms between the class object and the shaders.\n *\n */\nAbstractFilter.prototype.syncUniforms = function () {\n    for (var i = 0, j = this.shaders.length; i < j; ++i) {\n        this.shaders[i].dirty = true;\n    }\n};\n\n/*\nAbstractFilter.prototype.apply = function (frameBuffer) {\n    // TODO :)\n};\n*/\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The AlphaMaskFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object.\n * You can use this filter to apply all manor of crazy warping effects\n * Currently the r property of the texture is used to offset the x and the g property of the texture is used to offset the y.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment\n */\nfunction AlphaMaskFilter(texture) {\n    AbstractFilter.call(this);\n\n    texture.baseTexture._powerOf2 = true;\n\n    // set the uniforms\n    this.uniforms = {\n        mask:           { type: 'sampler2D',    value: texture },\n        mapDimensions:  { type: '2f',           value: { x: 1, y: 5112 } },\n        dimensions:     { type: '4fv',          value: [0, 0, 0, 0] },\n        offset:         { type: '2f',           value: { x: 0, y: 0 } }\n    };\n\n    if (texture.baseTexture.hasLoaded) {\n        this.uniforms.mask.value.x = texture.width;\n        this.uniforms.mask.value.y = texture.height;\n    }\n    else {\n        this.boundLoadedFunction = this.onTextureLoaded.bind(this);\n\n        texture.baseTexture.on('loaded', this.boundLoadedFunction);\n    }\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D uSampler;',\n        'uniform sampler2D mask;',\n        'uniform vec2 mapDimensions;',\n        'uniform vec4 dimensions;',\n        'uniform vec2 offset;',\n\n        'void main() {',\n        '   vec2 mapCords = vTextureCoord.xy;',\n        '   mapCords += (dimensions.zw + offset)/ dimensions.xy ;',\n        '   mapCords.y *= -1.0;',\n        '   mapCords.y += 1.0;',\n        '   mapCords *= dimensions.xy / mapDimensions;',\n\n        '   vec4 original =  texture2D(uSampler, vTextureCoord);',\n        '   float maskAlpha =  texture2D(mask, mapCords).r;',\n        '   original *= maskAlpha;',\n        //'   original.rgb *= maskAlpha;',\n        '   gl_FragColor =  original;',\n        //'   gl_FragColor = gl_FragColor;',\n        '}'\n    ];\n}\n\nAlphaMaskFilter.prototype = Object.create(AbstractFilter.prototype);\nAlphaMaskFilter.prototype.constructor = AlphaMaskFilter;\nmodule.exports = AlphaMaskFilter;\n\n/**\n * Sets the map dimensions uniforms when the texture becomes available.\n *\n */\nAlphaMaskFilter.prototype.onTextureLoaded = function () {\n    this.uniforms.mapDimensions.value.x = this.uniforms.mask.value.width;\n    this.uniforms.mapDimensions.value.y = this.uniforms.mask.value.height;\n\n    this.uniforms.mask.value.baseTexture.off('loaded', this.boundLoadedFunction);\n};\n\nObject.defineProperties(AlphaMaskFilter.prototype, {\n    /**\n     * The texture used for the displacement map. Must be power of 2 sized texture.\n     *\n     * @member {Texture}\n     * @memberof AlphaMaskFilter#\n     */\n    map: {\n        get: function () {\n            return this.uniforms.mask.value;\n        },\n        set: function (value) {\n            this.uniforms.mask.value = value;\n        }\n    },\n\n    /**\n     * The offset used to move the displacement map.\n     *\n     * @member {Point}\n     * @memberof AlphaMaskFilter#\n     */\n    offset: {\n        get: function() {\n            return this.uniforms.offset.value;\n        },\n        set: function(value) {\n            this.uniforms.offset.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original shader : https://www.shadertoy.com/view/lssGDj by @movAX13h\n */\n\n/**\n * An ASCII filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction AsciiFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        dimensions: { type: '4fv', value: new Float32Array([10000, 100, 10, 10]) },\n        pixelSize:  { type: '1f', value: 8}\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'uniform vec4 dimensions;',\n        'uniform float pixelSize;',\n        'uniform sampler2D uSampler;',\n\n        'float character(float n, vec2 p)',\n        '{',\n        '    p = floor(p*vec2(4.0, -4.0) + 2.5);',\n        '    if (clamp(p.x, 0.0, 4.0) == p.x && clamp(p.y, 0.0, 4.0) == p.y)',\n        '    {',\n        '        if (int(mod(n/exp2(p.x + 5.0*p.y), 2.0)) == 1) return 1.0;',\n        '    }',\n        '    return 0.0;',\n        '}',\n\n        'void main()',\n        '{',\n        '    vec2 uv = gl_FragCoord.xy;',\n        '    vec3 col = texture2D(uSampler, floor( uv / pixelSize ) * pixelSize / dimensions.xy).rgb;',\n\n        '    #ifdef HAS_GREENSCREEN',\n        '    float gray = (col.r + col.b)/2.0;',\n        '    #else',\n        '    float gray = (col.r + col.g + col.b)/3.0;',\n        '    #endif',\n\n        '    float n =  65536.0;             // .',\n        '    if (gray > 0.2) n = 65600.0;    // :',\n        '    if (gray > 0.3) n = 332772.0;   // *',\n        '    if (gray > 0.4) n = 15255086.0; // o',\n        '    if (gray > 0.5) n = 23385164.0; // &',\n        '    if (gray > 0.6) n = 15252014.0; // 8',\n        '    if (gray > 0.7) n = 13199452.0; // @',\n        '    if (gray > 0.8) n = 11512810.0; // #',\n\n        '    vec2 p = mod( uv / ( pixelSize * 0.5 ), 2.0) - vec2(1.0);',\n        '    col = col * character(n, p);',\n\n        '    gl_FragColor = vec4(col, 1.0);',\n        '}'\n    ];\n}\n\nAsciiFilter.prototype = Object.create(AbstractFilter.prototype);\nAsciiFilter.prototype.constructor = AsciiFilter;\nmodule.exports = AsciiFilter;\n\nObject.defineProperties(AsciiFilter.prototype, {\n    /**\n     * The pixel size used by the filter.\n     *\n     * @member {number}\n     * @memberof AsciiFilter#\n     */\n    size: {\n        get: function () {\n            return this.uniforms.pixelSize.value;\n        },\n        set: function (value) {\n            this.uniforms.pixelSize.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter'),\n    BlurXFilter = require('./BlurXFilter'),\n    BlurYFilter = require('./BlurYFilter');\n\n/**\n * The BlurFilter applies a Gaussian blur to an object.\n * The strength of the blur can be set for x- and y-axis separately.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction BlurFilter() {\n    AbstractFilter.call(this);\n\n    this.blurXFilter = new BlurXFilter();\n    this.blurYFilter = new BlurYFilter();\n\n    this.passes = [this.blurXFilter, this.blurYFilter];\n}\n\nBlurFilter.prototype = Object.create(AbstractFilter.prototype);\nBlurFilter.prototype.constructor = BlurFilter;\nmodule.exports = BlurFilter;\n\nObject.defineProperties(BlurFilter.prototype, {\n    /**\n     * Sets the strength of both the blurX and blurY properties simultaneously\n     *\n     * @member {number}\n     * @memberOf BlurFilter#\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.blurXFilter.blur;\n        },\n        set: function (value) {\n            this.blurXFilter.blur = this.blurYFilter.blur = value;\n        }\n    },\n\n    /**\n     * Sets the strength of the blurX property\n     *\n     * @member {number}\n     * @memberOf BlurFilter#\n     * @default 2\n     */\n    blurX: {\n        get: function () {\n            return this.blurXFilter.blur;\n        },\n        set: function (value) {\n            this.blurXFilter.blur = value;\n        }\n    },\n\n    /**\n     * Sets the strength of the blurY property\n     *\n     * @member {number}\n     * @memberOf BlurFilter#\n     * @default 2\n     */\n    blurY: {\n        get: function () {\n            return this.blurYFilter.blur;\n        },\n        set: function (value) {\n            this.blurYFilter.blur = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter'),\n    blurFactor = 1 / 7000;\n\n/**\n * The BlurXFilter applies a horizontal Gaussian blur to an object.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction BlurXFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur: { type: '1f', value: 1 / 512 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float blur;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec4 sum = vec4(0.0);',\n\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - 4.0*blur, vTextureCoord.y)) * 0.05;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - 3.0*blur, vTextureCoord.y)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - 2.0*blur, vTextureCoord.y)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - blur, vTextureCoord.y)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + blur, vTextureCoord.y)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + 2.0*blur, vTextureCoord.y)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + 3.0*blur, vTextureCoord.y)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + 4.0*blur, vTextureCoord.y)) * 0.05;',\n\n        '   gl_FragColor = sum;',\n        '}'\n    ];\n}\n\nBlurXFilter.prototype = Object.create(AbstractFilter.prototype);\nBlurXFilter.prototype.constructor = BlurXFilter;\nmodule.exports = BlurXFilter;\n\nObject.defineProperties(BlurXFilter.prototype, {\n    /**\n     * Sets the strength of both the blur.\n     *\n     * @member {number}\n     * @memberof BlurXFilter#\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value / blurFactor;\n        },\n        set: function (value) {\n            this.uniforms.blur.value = blurFactor * value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter'),\n    blurFactor = 1 / 7000;\n\n/**\n * The BlurYFilter applies a vertical Gaussian blur to an object.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction BlurYFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur: { type: '1f', value: 1 / 512 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float blur;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec4 sum = vec4(0.0);',\n\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 4.0*blur)) * 0.05;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 3.0*blur)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 2.0*blur)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - blur)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + blur)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 2.0*blur)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 3.0*blur)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 4.0*blur)) * 0.05;',\n\n        '   gl_FragColor = sum;',\n        '}'\n    ];\n}\n\nBlurYFilter.prototype = Object.create(AbstractFilter.prototype);\nBlurYFilter.prototype.constructor = BlurYFilter;\nmodule.exports = BlurYFilter;\n\nObject.defineProperties(BlurYFilter.prototype, {\n    /**\n     * Sets the strength of both the blur.\n     *\n     * @member {number}\n     * @memberof BlurYFilter\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value / blurFactor;\n        },\n        set: function (value) {\n            //this.padding = value;\n            this.uniforms.blur.value = blurFactor * value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The ColorMatrixFilter class lets you apply a 4x4 matrix transformation on the RGBA\n * color and alpha values of every pixel on your displayObject to produce a result\n * with a new set of RGBA color and alpha values. It's pretty powerful!\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction ColorMatrixFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        matrix: { type: 'mat4', value: [1, 0, 0, 0,\n                                        0, 1, 0, 0,\n                                        0, 0, 1, 0,\n                                        0, 0, 0, 1] }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float invert;',\n        'uniform mat4 matrix;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord) * matrix;',\n        '}'\n    ];\n}\n\nColorMatrixFilter.prototype = Object.create(AbstractFilter.prototype);\nColorMatrixFilter.prototype.constructor = ColorMatrixFilter;\nmodule.exports = ColorMatrixFilter;\n\nObject.defineProperties(ColorMatrixFilter.prototype, {\n    /**\n     * Sets the matrix of the color matrix filter\n     *\n     * @member {number[]}\n     * @memberof ColorMatrixFilter#\n     * @default [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]\n     */\n    matrix: {\n        get: function () {\n            return this.uniforms.matrix.value;\n        },\n        set: function (value) {\n            this.uniforms.matrix.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This lowers the color depth of your image by the given amount, producing an image with a smaller palette.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction ColorStepFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        step: { type: '1f', value: 5 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n        'uniform sampler2D uSampler;',\n        'uniform float step;',\n\n        'void main(void) {',\n        '   vec4 color = texture2D(uSampler, vTextureCoord);',\n        '   color = floor(color * step) / step;',\n        '   gl_FragColor = color;',\n        '}'\n    ];\n}\n\nColorStepFilter.prototype = Object.create(AbstractFilter.prototype);\nColorStepFilter.prototype.constructor = ColorStepFilter;\nmodule.exports = ColorStepFilter;\n\nObject.defineProperties(ColorStepFilter.prototype, {\n    /**\n     * The number of steps to reduce the palette by.\n     *\n     * @member {number}\n     * @memberof ColorStepFilter#\n     */\n    step: {\n        get: function () {\n            return this.uniforms.step.value;\n        },\n        set: function (value) {\n            this.uniforms.step.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The ConvolutionFilter class applies a matrix convolution filter effect.\n * A convolution combines pixels in the input image with neighboring pixels to produce a new image.\n * A wide variety of image effects can be achieved through convolutions, including blurring, edge\n * detection, sharpening, embossing, and beveling. The matrix should be specified as a 9 point Array.\n * See http://docs.gimp.org/en/plug-in-convmatrix.html for more info.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param matrix {number[]} An array of values used for matrix transformation. Specified as a 9 point Array.\n * @param width {number} Width of the object you are transforming\n * @param height {number} Height of the object you are transforming\n */\nfunction ConvolutionFilter(matrix, width, height) {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        matrix:     { type: '1fv', value: new Float32Array(matrix) },\n        texelSizeX: { type: '1f', value: 1 / width },\n        texelSizeY: { type: '1f', value: 1 / height }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying mediump vec2 vTextureCoord;',\n\n        'uniform sampler2D texture;',\n        'uniform float texelSizeX;',\n        'uniform float texelSizeY;',\n        'uniform float matrix[9];',\n\n        'vec2 px = vec2(texelSizeX, texelSizeY);',\n\n        'void main(void) {',\n        '   vec4 c11 = texture2D(texture, vTextureCoord - px);', // top left\n        '   vec4 c12 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y - px.y));', // top center\n        '   vec4 c13 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y - px.y));', // top right\n\n        '   vec4 c21 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y) );', // mid left\n        '   vec4 c22 = texture2D(texture, vTextureCoord);', // mid center\n        '   vec4 c23 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y) );', // mid right\n\n        '   vec4 c31 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y + px.y) );', // bottom left\n        '   vec4 c32 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y + px.y) );', // bottom center\n        '   vec4 c33 = texture2D(texture, vTextureCoord + px );', // bottom right\n\n        '   gl_FragColor = ',\n        '       c11 * matrix[0] + c12 * matrix[1] + c13 * matrix[2] +',\n        '       c21 * matrix[3] + c22 * matrix[4] + c23 * matrix[5] +',\n        '       c31 * matrix[6] + c32 * matrix[7] + c33 * matrix[8];',\n        '   gl_FragColor.a = c22.a;',\n        '}'\n    ];\n}\n\nConvolutionFilter.prototype = Object.create(AbstractFilter.prototype);\nConvolutionFilter.prototype.constructor = ConvolutionFilter;\nmodule.exports = ConvolutionFilter;\n\nObject.defineProperties(ConvolutionFilter.prototype, {\n    /**\n     * An array of values used for matrix transformation. Specified as a 9 point Array.\n     *\n     * @member {number[]}\n     * @memberof ConvolutionFilter#\n     */\n    matrix: {\n        get: function () {\n            return this.uniforms.matrix.value;\n        },\n        set: function (value) {\n            this.uniforms.matrix.value = new Float32Array(value);\n        }\n    },\n\n    /**\n     * Width of the object you are transforming\n     *\n     * @member {number}\n     * @memberof ConvolutionFilter#\n     */\n    width: {\n        get: function () {\n            return this.uniforms.texelSizeX.value;\n        },\n        set: function (value) {\n            this.uniforms.texelSizeX.value = 1/value;\n        }\n    },\n\n    /**\n     * Height of the object you are transforming\n     *\n     * @member {number}\n     * @memberof ConvolutionFilter#\n     */\n    height: {\n        get: function () {\n            return this.uniforms.texelSizeY.value;\n        },\n        set: function (value) {\n            this.uniforms.texelSizeY.value = 1/value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * A Cross Hatch effect filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction CrossHatchFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur: { type: '1f', value: 1 / 512 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float blur;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '    float lum = length(texture2D(uSampler, vTextureCoord.xy).rgb);',\n\n        '    gl_FragColor = vec4(1.0, 1.0, 1.0, 1.0);',\n\n        '    if (lum < 1.00) {',\n        '        if (mod(gl_FragCoord.x + gl_FragCoord.y, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n\n        '    if (lum < 0.75) {',\n        '        if (mod(gl_FragCoord.x - gl_FragCoord.y, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n\n        '    if (lum < 0.50) {',\n        '        if (mod(gl_FragCoord.x + gl_FragCoord.y - 5.0, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n\n        '    if (lum < 0.3) {',\n        '        if (mod(gl_FragCoord.x - gl_FragCoord.y - 5.0, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n        '}'\n    ];\n}\n\nCrossHatchFilter.prototype = Object.create(AbstractFilter.prototype);\nCrossHatchFilter.prototype.constructor = CrossHatchFilter;\nmodule.exports = CrossHatchFilter;\n\nObject.defineProperties(CrossHatchFilter.prototype, {\n    /**\n     * Sets the strength of both the blur.\n     *\n     * @member {number}\n     * @memberof CrossHatchFilter#\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value / (1/7000);\n        },\n        set: function (value) {\n            //this.padding = value;\n            this.uniforms.blur.value = (1/7000) * value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The DisplacementFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object.\n * You can use this filter to apply all manor of crazy warping effects\n * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment\n */\nfunction DisplacementFilter(texture) {\n    AbstractFilter.call(this);\n\n    texture.baseTexture._powerOf2 = true;\n\n    // set the uniforms\n    this.uniforms = {\n        displacementMap: { type: 'sampler2D', value: texture },\n        scale:           { type: '2f',  value: { x: 30, y: 30 } },\n        offset:          { type: '2f',  value: { x: 0,  y: 0 } },\n        mapDimensions:   { type: '2f',  value: { x: 1,  y: 5112 } },\n        dimensions:      { type: '4fv', value: [0, 0, 0, 0] }\n    };\n\n    if (texture.baseTexture.hasLoaded) {\n        this.onTextureLoaded();\n    }\n    else {\n        this.boundLoadedFunction = this.onTextureLoaded.bind(this);\n\n        texture.baseTexture.on('loaded', this.boundLoadedFunction);\n    }\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D displacementMap;',\n        'uniform sampler2D uSampler;',\n        'uniform vec2 scale;',\n        'uniform vec2 offset;',\n        'uniform vec4 dimensions;',\n        'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);',\n        // 'const vec2 textureDimensions = vec2(750.0, 750.0);',\n\n        'void main(void) {',\n        '   vec2 mapCords = vTextureCoord.xy;',\n        '   mapCords += (dimensions.zw + offset)/ dimensions.xy ;',\n        '   mapCords.y *= -1.0;',\n        '   mapCords.y += 1.0;',\n\n        '   vec2 matSample = texture2D(displacementMap, mapCords).xy;',\n        '   matSample -= 0.5;',\n        '   matSample *= scale;',\n        '   matSample /= mapDimensions;',\n\n        '   gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x + matSample.x, vTextureCoord.y + matSample.y));',\n\n        //TODO: Is this needed?\n        '   gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb, 1.0);',\n        '}'\n    ];\n}\n\nDisplacementFilter.prototype = Object.create(AbstractFilter.prototype);\nDisplacementFilter.prototype.constructor = DisplacementFilter;\nmodule.exports = DisplacementFilter;\n\n/**\n * Sets the map dimensions uniforms when the texture becomes available.\n *\n * @private\n */\nDisplacementFilter.prototype.onTextureLoaded = function () {\n    this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width;\n    this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height;\n\n    this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction);\n};\n\nObject.defineProperties(DisplacementFilter.prototype, {\n    /**\n     * The texture used for the displacement map. Must be power of 2 texture.\n     *\n     * @member {Texture}\n     * @memberof DisplacementFilter#\n     */\n    map: {\n        get: function () {\n            return this.uniforms.displacementMap.value;\n        },\n        set: function (value) {\n            this.uniforms.displacementMap.value = value;\n        }\n    },\n\n    /**\n     * The multiplier used to scale the displacement result from the map calculation.\n     *\n     * @member {Point}\n     * @memberof DisplacementFilter#\n     */\n    scale: {\n        get: function () {\n            return this.uniforms.scale.value;\n        },\n        set: function (value) {\n            this.uniforms.scale.value = value;\n        }\n    },\n\n    /**\n     * The offset used to move the displacement map.\n     *\n     * @member {Point}\n     * @memberof DisplacementFilter#\n     */\n    offset: {\n        get: function () {\n            return this.uniforms.offset.value;\n        },\n        set: function (value) {\n            this.uniforms.offset.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Mat Groves http://matgroves.com/ @Doormat23\n * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/fun/dotscreen.js\n */\n\n/**\n * This filter applies a dotscreen effect making display objects appear to be made out of\n * black and white halftone dots like an old printer.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction DotScreenFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        scale:      { type: '1f', value: 1 },\n        angle:      { type: '1f', value: 5 },\n        dimensions: { type: '4fv', value: [0, 0, 0, 0] }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform vec4 dimensions;',\n        'uniform sampler2D uSampler;',\n\n        'uniform float angle;',\n        'uniform float scale;',\n\n        'float pattern() {',\n        '   float s = sin(angle), c = cos(angle);',\n        '   vec2 tex = vTextureCoord * dimensions.xy;',\n        '   vec2 point = vec2(',\n        '       c * tex.x - s * tex.y,',\n        '       s * tex.x + c * tex.y',\n        '   ) * scale;',\n        '   return (sin(point.x) * sin(point.y)) * 4.0;',\n        '}',\n\n        'void main() {',\n        '   vec4 color = texture2D(uSampler, vTextureCoord);',\n        '   float average = (color.r + color.g + color.b) / 3.0;',\n        '   gl_FragColor = vec4(vec3(average * 10.0 - 5.0 + pattern()), color.a);',\n        '}'\n    ];\n}\n\nDotScreenFilter.prototype = Object.create(AbstractFilter.prototype);\nDotScreenFilter.prototype.constructor = DotScreenFilter;\nmodule.exports = DotScreenFilter;\n\nObject.defineProperties(DotScreenFilter.prototype, {\n    /**\n     * The scale of the effect.\n     * @member {number}\n     * @memberof DotScreenFilter#\n     */\n    scale: {\n        get: function () {\n            return this.uniforms.scale.value;\n        },\n        set: function (value) {\n            this.uniforms.scale.value = value;\n        }\n    },\n\n    /**\n     * The radius of the effect.\n     * @member {number}\n     * @memberof DotScreenFilter#\n     */\n    angle: {\n        get: function () {\n            return this.uniforms.angle.value;\n        },\n        set: function (value) {\n            this.uniforms.angle.value = value;\n        }\n    }\n});\n","/**\n * A target and pass info object for filters.\n *\n * @class\n * @namespace PIXI\n */\nfunction FilterBlock() {\n    /**\n     * The visible state of this FilterBlock.\n     *\n     * @member {boolean}\n     */\n    this.visible = true;\n\n    /**\n     * The renderable state of this FilterBlock.\n     *\n     * @member {boolean}\n     */\n    this.renderable = true;\n}\n\nFilterBlock.prototype.constructor = FilterBlock;\nmodule.exports = FilterBlock;\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This greyscales the palette of your Display Objects.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction GrayFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        gray: { type: '1f', value: 1 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D uSampler;',\n        'uniform float gray;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord);',\n        '   gl_FragColor.rgb = mix(gl_FragColor.rgb, vec3(0.2126*gl_FragColor.r + 0.7152*gl_FragColor.g + 0.0722*gl_FragColor.b), gray);',\n     //   '   gl_FragColor = gl_FragColor;',\n        '}'\n    ];\n}\n\nGrayFilter.prototype = Object.create(AbstractFilter.prototype);\nGrayFilter.prototype.constructor = GrayFilter;\nmodule.exports = GrayFilter;\n\nObject.defineProperties(GrayFilter.prototype, {\n    /**\n     * The strength of the gray. 1 will make the object black and white, 0 will make the object its normal color.\n     *\n     * @member {number}\n     * @memberof GrayFilter#\n     */\n    gray: {\n        get: function () {\n            return this.uniforms.gray.value;\n        },\n        set: function (value) {\n            this.uniforms.gray.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This inverts your Display Objects colors.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction InvertFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        invert: { type: '1f', value: 1 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float invert;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord);',\n        '   gl_FragColor.rgb = mix( (vec3(1)-gl_FragColor.rgb) * gl_FragColor.a, gl_FragColor.rgb, 1.0 - invert);',\n        //'   gl_FragColor.rgb = gl_FragColor.rgb  * gl_FragColor.a;',\n      //  '   gl_FragColor = gl_FragColor * vColor;',\n        '}'\n    ];\n}\n\nInvertFilter.prototype = Object.create(AbstractFilter.prototype);\nInvertFilter.prototype.constructor = InvertFilter;\nmodule.exports = InvertFilter;\n\nObject.defineProperties(InvertFilter.prototype, {\n    /**\n     * The strength of the invert. 1 will fully invert the colors, 0 will make the object its normal color\n     *\n     * @member {number}\n     * @memberof InvertFilter#\n     */\n    invert: {\n        get: function () {\n            return this.uniforms.invert.value;\n        },\n        set: function (value) {\n            this.uniforms.invert.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/adjust/noise.js\n */\n\n/**\n * A Noise effect filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction NoiseFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        noise: { type: '1f', value: 0.5 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float noise;',\n        'uniform sampler2D uSampler;',\n\n        'float rand(vec2 co) {',\n        '    return fract(sin(dot(co.xy ,vec2(12.9898,78.233))) * 43758.5453);',\n        '}',\n\n        'void main() {',\n        '    vec4 color = texture2D(uSampler, vTextureCoord);',\n\n        '    float diff = (rand(vTextureCoord) - 0.5) * noise;',\n        '    color.r += diff;',\n        '    color.g += diff;',\n        '    color.b += diff;',\n\n        '    gl_FragColor = color;',\n        '}'\n    ];\n}\n\nNoiseFilter.prototype = Object.create(AbstractFilter.prototype);\nNoiseFilter.prototype.constructor = NoiseFilter;\nmodule.exports = NoiseFilter;\n\nObject.defineProperties(NoiseFilter.prototype, {\n    /**\n     * The amount of noise to apply.\n     *\n     * @member {number}\n     * @memberof NoiseFilter#\n     * @default 0.5\n     */\n    noise: {\n        get: function () {\n            return this.uniforms.noise.value;\n        },\n        set: function (value) {\n            this.dirty = true;\n            this.uniforms.noise.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The NormalMapFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object.\n * You can use this filter to apply all manor of crazy warping effects\n * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param texture {Texture} The texture used for the normal map, must be power of 2 texture at the moment\n */\nfunction NormalMapFilter(texture) {\n    AbstractFilter.call(this);\n\n    texture.baseTexture._powerOf2 = true;\n\n    // set the uniforms\n    this.uniforms = {\n        displacementMap:  { type: 'sampler2D', value: texture },\n        scale:            { type: '2f', value: { x: 15, y: 15 } },\n        offset:           { type: '2f', value: { x: 0,  y: 0 } },\n        mapDimensions:    { type: '2f', value: { x: 1,  y: 1 } },\n        dimensions:       { type: '4f', value: [0, 0, 0, 0] },\n        // LightDir:         { type: 'f3', value: [0, 1, 0] },\n        LightPos:         { type: '3f', value: [0, 1, 0] }\n    };\n\n    if (texture.baseTexture.hasLoaded) {\n        this.onTextureLoaded();\n    }\n    else {\n        this.boundLoadedFunction = this.onTextureLoaded.bind(this);\n\n        texture.baseTexture.on('loaded', this.boundLoadedFunction);\n    }\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying float vColor;',\n\n        'uniform sampler2D displacementMap;',\n        'uniform sampler2D uSampler;',\n\n        'uniform vec4 dimensions;',\n\n        'const vec2 Resolution = vec2(1.0,1.0);',      //resolution of screen\n        'uniform vec3 LightPos;',    //light position, normalized\n        'const vec4 LightColor = vec4(1.0, 1.0, 1.0, 1.0);',      //light RGBA -- alpha is intensity\n        'const vec4 AmbientColor = vec4(1.0, 1.0, 1.0, 0.5);',    //ambient RGBA -- alpha is intensity\n        'const vec3 Falloff = vec3(0.0, 1.0, 0.2);',         //attenuation coefficients\n\n        'uniform vec3 LightDir;',//' = vec3(1.0, 0.0, 1.0);',\n\n\n        'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);',\n\n\n        'void main(void) {',\n        '   vec2 mapCords = vTextureCoord.xy;',\n\n        '   vec4 color = texture2D(uSampler, vTextureCoord.st);',\n        '   vec3 nColor = texture2D(displacementMap, vTextureCoord.st).rgb;',\n\n\n        '   mapCords *= vec2(dimensions.x/512.0, dimensions.y/512.0);',\n\n        '   mapCords.y *= -1.0;',\n        '   mapCords.y += 1.0;',\n\n        //RGBA of our diffuse color\n        '   vec4 DiffuseColor = texture2D(uSampler, vTextureCoord);',\n\n        //RGB of our normal map\n        '   vec3 NormalMap = texture2D(displacementMap, mapCords).rgb;',\n\n        //The delta position of light\n        //'vec3 LightDir = vec3(LightPos.xy - (gl_FragCoord.xy / Resolution.xy), LightPos.z);',\n        '   vec3 LightDir = vec3(LightPos.xy - (mapCords.xy), LightPos.z);',\n        //Correct for aspect ratio\n        // '   LightDir.x *= Resolution.x / Resolution.y;',\n\n        //Determine distance (used for attenuation) BEFORE we normalize our LightDir\n        '   float D = length(LightDir);',\n\n        //normalize our vectors\n        '   vec3 N = normalize(NormalMap * 2.0 - 1.0);',\n        '   vec3 L = normalize(LightDir);',\n\n        //Pre-multiply light color with intensity\n        //Then perform 'N dot L' to determine our diffuse term\n        '   vec3 Diffuse = (LightColor.rgb * LightColor.a) * max(dot(N, L), 0.0);',\n\n        //pre-multiply ambient color with intensity\n        '   vec3 Ambient = AmbientColor.rgb * AmbientColor.a;',\n\n        //calculate attenuation\n        '   float Attenuation = 1.0 / ( Falloff.x + (Falloff.y*D) + (Falloff.z*D*D) );',\n\n        //the calculation which brings it all together\n        '   vec3 Intensity = Ambient + Diffuse * Attenuation;',\n        '   vec3 FinalColor = DiffuseColor.rgb * Intensity;',\n        '   gl_FragColor = vColor * vec4(FinalColor, DiffuseColor.a);',\n        // '   gl_FragColor = vec4(1.0, 0.0, 0.0, Attenuation);',//vColor * vec4(FinalColor, DiffuseColor.a);',\n\n        /*// normalise color\n        '   vec3 normal = normalize(nColor * 2.0 - 1.0);',\n\n        '   vec3 deltaPos = vec3( (light.xy - gl_FragCoord.xy) / resolution.xy, light.z );',\n\n        '   float lambert = clamp(dot(normal, lightDir), 0.0, 1.0);',\n\n        '   float d = sqrt(dot(deltaPos, deltaPos));',\n        '   float att = 1.0 / ( attenuation.x + (attenuation.y*d) + (attenuation.z*d*d) );',\n\n        '   vec3 result = (ambientColor * ambientIntensity) + (lightColor.rgb * lambert) * att;',\n        '   result *= color.rgb;',\n\n        '   gl_FragColor = vec4(result, 1.0);',*/\n        '}'\n    ];\n}\n\nNormalMapFilter.prototype = Object.create(AbstractFilter.prototype);\nNormalMapFilter.prototype.constructor = NormalMapFilter;\nmodule.exports = NormalMapFilter;\n\n/**\n * Sets the map dimensions uniforms when the texture becomes available.\n *\n */\nNormalMapFilter.prototype.onTextureLoaded = function () {\n    this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width;\n    this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height;\n\n    this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction);\n};\n\nObject.defineProperties(NormalMapFilter.prototype, {\n    /**\n     * The texture used for the displacement map. Must be power of 2 texture.\n     *\n     * @member {Texture}\n     * @memberof NormalMapFilter#\n     */\n    map: {\n        get: function () {\n            return this.uniforms.displacementMap.value;\n        },\n        set: function (value) {\n            this.uniforms.displacementMap.value = value;\n        }\n    },\n\n    /**\n     * The multiplier used to scale the displacement result from the map calculation.\n     *\n     * @member {Point}\n     * @memberof NormalMapFilter#\n     */\n    scale: {\n        get: function () {\n            return this.uniforms.scale.value;\n        },\n        set: function (value) {\n            this.uniforms.scale.value = value;\n        }\n    },\n\n    /**\n     * The offset used to move the displacement map.\n     *\n     * @member {Point}\n     * @memberof NormalMapFilter#\n     */\n    offset: {\n        get: function () {\n            return this.uniforms.offset.value;\n        },\n        set: function (value) {\n            this.uniforms.offset.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This filter applies a pixelate effect making display objects appear 'blocky'.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction PixelateFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        invert:     { type: '1f',   value: 0 },\n        dimensions: { type: '4fv',  value: new Float32Array([10000, 100, 10, 10]) },\n        pixelSize:  { type: '2f',   value: { x: 10, y: 10 } }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform vec2 testDim;',\n        'uniform vec4 dimensions;',\n        'uniform vec2 pixelSize;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec2 coord = vTextureCoord;',\n\n        '   vec2 size = dimensions.xy/pixelSize;',\n\n        '   vec2 color = floor( ( vTextureCoord * size ) ) / size + pixelSize/dimensions.xy * 0.5;',\n        '   gl_FragColor = texture2D(uSampler, color);',\n        '}'\n    ];\n}\n\nPixelateFilter.prototype = Object.create(AbstractFilter.prototype);\nPixelateFilter.prototype.constructor = PixelateFilter;\nmodule.exports = PixelateFilter;\n\nObject.defineProperties(PixelateFilter.prototype, {\n    /**\n     * This a point that describes the size of the blocks. x is the width of the block and y is the height.\n     *\n     * @member {Point}\n     * @memberof PixelateFilter#\n     */\n    size: {\n        get: function () {\n            return this.uniforms.pixelSize.value;\n        },\n        set: function (value) {\n            this.uniforms.pixelSize.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * An RGB Split Filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction RGBSplitFilter() {\n    AbstractFilter.call(this);\n\n    this.passes = [this];\n\n    // set the uniforms\n    this.uniforms = {\n        red:        { type: '2f', value: { x: 20, y: 20 } },\n        green:      { type: '2f', value: { x: -20, y: 20 } },\n        blue:       { type: '2f', value: { x: 20, y: -20 } },\n        dimensions: { type: '4fv', value: [0, 0, 0, 0] }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform vec2 red;',\n        'uniform vec2 green;',\n        'uniform vec2 blue;',\n        'uniform vec4 dimensions;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor.r = texture2D(uSampler, vTextureCoord + red/dimensions.xy).r;',\n        '   gl_FragColor.g = texture2D(uSampler, vTextureCoord + green/dimensions.xy).g;',\n        '   gl_FragColor.b = texture2D(uSampler, vTextureCoord + blue/dimensions.xy).b;',\n        '   gl_FragColor.a = texture2D(uSampler, vTextureCoord).a;',\n        '}'\n    ];\n}\n\nRGBSplitFilter.prototype = Object.create(AbstractFilter.prototype);\nRGBSplitFilter.prototype.constructor = RGBSplitFilter;\nmodule.exports = RGBSplitFilter;\n\nObject.defineProperties(RGBSplitFilter.prototype, {\n    /**\n     * Red channel offset.\n     *\n     * @member {Point}\n     * @memberof RGBSplitFilter#\n     */\n    red: {\n        get: function () {\n            return this.uniforms.red.value;\n        },\n        set: function (value) {\n            this.uniforms.red.value = value;\n        }\n    },\n\n    /**\n     * Green channel offset.\n     *\n     * @member {Point}\n     * @memberof RGBSplitFilter#\n     */\n    green: {\n        get: function () {\n            return this.uniforms.green.value;\n        },\n        set: function (value) {\n            this.uniforms.green.value = value;\n        }\n    },\n\n    /**\n     * Blue offset.\n     *\n     * @member {Point}\n     * @memberof RGBSplitFilter#\n     */\n    blue: {\n        get: function () {\n            return this.uniforms.blue.value;\n        },\n        set: function (value) {\n            this.uniforms.blue.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This applies a sepia effect to your Display Objects.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction SepiaFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        sepia: { type: '1f', value: 1 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float sepia;',\n        'uniform sampler2D uSampler;',\n\n        'const mat3 sepiaMatrix = mat3(0.3588, 0.7044, 0.1368, 0.2990, 0.5870, 0.1140, 0.2392, 0.4696, 0.0912);',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord);',\n        '   gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb * sepiaMatrix, sepia);',\n        '}'\n    ];\n}\n\nSepiaFilter.prototype = Object.create(AbstractFilter.prototype);\nSepiaFilter.prototype.constructor = SepiaFilter;\nmodule.exports = SepiaFilter;\n\nObject.defineProperties(SepiaFilter.prototype, {\n    /**\n     * The strength of the sepia. 1 will apply the full sepia effect, 0 will make the object its normal color.\n     *\n     * @member {number}\n     * @memberof SepiaFilter#\n     */\n    sepia: {\n        get: function () {\n            return this.uniforms.sepia.value;\n        },\n        set: function (value) {\n            this.uniforms.sepia.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * A Smart Blur Filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction SmartBlurFilter() {\n    AbstractFilter.call(this);\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n\n        'uniform sampler2D uSampler;',\n        'const vec2 delta = vec2(1.0/10.0, 0.0);',\n\n        'float random(vec3 scale, float seed) {',\n        '   return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);',\n        '}',\n\n\n        'void main(void) {',\n        '   vec4 color = vec4(0.0);',\n        '   float total = 0.0;',\n\n        '   float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);',\n\n        '   for (float t = -30.0; t <= 30.0; t++) {',\n        '       float percent = (t + offset - 0.5) / 30.0;',\n        '       float weight = 1.0 - abs(percent);',\n        '       vec4 sample = texture2D(uSampler, vTextureCoord + delta * percent);',\n        '       sample.rgb *= sample.a;',\n        '       color += sample * weight;',\n        '       total += weight;',\n        '   }',\n\n        '   gl_FragColor = color / total;',\n        '   gl_FragColor.rgb /= gl_FragColor.a + 0.00001;',\n        '}'\n    ];\n}\n\nSmartBlurFilter.prototype = Object.create(AbstractFilter.prototype);\nSmartBlurFilter.prototype.constructor = SmartBlurFilter;\nmodule.exports = SmartBlurFilter;\n","var AbstractFilter = require('./AbstractFilter'),\n    TiltShiftXFilter = require('./TiltShiftXFilter'),\n    TiltShiftYFilter = require('./TiltShiftYFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/\n */\n\n/**\n * A TiltShift Filter. Manages the pass of both a TiltShiftXFilter and TiltShiftYFilter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TiltShiftFilter() {\n    AbstractFilter.call(this);\n\n    this.tiltShiftXFilter = new TiltShiftXFilter();\n    this.tiltShiftYFilter = new TiltShiftYFilter();\n\n    this.tiltShiftXFilter.updateDelta();\n    this.tiltShiftXFilter.updateDelta();\n\n    this.passes = [this.tiltShiftXFilter, this.tiltShiftYFilter];\n}\n\nTiltShiftFilter.prototype = Object.create(AbstractFilter.prototype);\nTiltShiftFilter.prototype.constructor = TiltShiftFilter;\nmodule.exports = TiltShiftFilter;\n\nObject.defineProperties(TiltShiftFilter.prototype, {\n    /**\n     * The strength of the blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    blur: {\n        get: function () {\n            return this.tiltShiftXFilter.blur;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.blur = this.tiltShiftYFilter.blur = value;\n        }\n    },\n\n    /**\n     * The strength of the gradient blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    gradientBlur: {\n        get: function () {\n            return this.tiltShiftXFilter.gradientBlur;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.gradientBlur = this.tiltShiftYFilter.gradientBlur = value;\n        }\n    },\n\n    /**\n     * The Y value to start the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    start: {\n        get: function () {\n            return this.tiltShiftXFilter.start;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.start = this.tiltShiftYFilter.start = value;\n        }\n    },\n\n    /**\n     * The Y value to end the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    end: {\n        get: function () {\n            return this.tiltShiftXFilter.end;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.end = this.tiltShiftYFilter.end = value;\n        }\n    },\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/\n */\n\n/**\n * A TiltShiftXFilter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TiltShiftXFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur:           { type: '1f', value: 100 },\n        gradientBlur:   { type: '1f', value: 600 },\n        start:          { type: '2f', value: { x: 0,    y: window.screenHeight / 2 } },\n        end:            { type: '2f', value: { x: 600,  y: window.screenHeight / 2 } },\n        delta:          { type: '2f', value: { x: 30,   y: 30 } },\n        texSize:        { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } }\n    };\n\n    this.updateDelta();\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n\n        'uniform sampler2D uSampler;',\n        'uniform float blur;',\n        'uniform float gradientBlur;',\n        'uniform vec2 start;',\n        'uniform vec2 end;',\n        'uniform vec2 delta;',\n        'uniform vec2 texSize;',\n\n        'float random(vec3 scale, float seed) {',\n        '   return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);',\n        '}',\n\n        'void main(void) {',\n        '    vec4 color = vec4(0.0);',\n        '    float total = 0.0;',\n\n        '    float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);',\n        '    vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));',\n        '    float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;',\n\n        '    for (float t = -30.0; t <= 30.0; t++) {',\n        '        float percent = (t + offset - 0.5) / 30.0;',\n        '        float weight = 1.0 - abs(percent);',\n        '        vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);',\n        '        sample.rgb *= sample.a;',\n        '        color += sample * weight;',\n        '        total += weight;',\n        '    }',\n\n        '    gl_FragColor = color / total;',\n        '    gl_FragColor.rgb /= gl_FragColor.a + 0.00001;',\n        '}'\n    ];\n}\n\nTiltShiftXFilter.prototype = Object.create(AbstractFilter.prototype);\nTiltShiftXFilter.prototype.constructor = TiltShiftXFilter;\nmodule.exports = TiltShiftXFilter;\n\n/**\n * Updates the filter delta values.\n *\n */\nTiltShiftXFilter.prototype.updateDelta = function () {\n    var dx = this.uniforms.end.value.x - this.uniforms.start.value.x;\n    var dy = this.uniforms.end.value.y - this.uniforms.start.value.y;\n    var d = Math.sqrt(dx * dx + dy * dy);\n\n    this.uniforms.delta.value.x = dx / d;\n    this.uniforms.delta.value.y = dy / d;\n};\n\n\nObject.defineProperties(TiltShiftXFilter.prototype, {\n    /**\n     * The strength of the blur.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value;\n        },\n        set: function (value) {\n            this.uniforms.blur.value = value;\n        }\n    },\n\n    /**\n     * The strength of the gradient blur.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    gradientBlur: {\n        get: function () {\n            return this.uniforms.gradientBlur.value;\n        },\n        set: function (value) {\n            this.uniforms.gradientBlur.value = value;\n        }\n    },\n\n    /**\n     * The X value to start the effect at.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    start: {\n        get: function () {\n            return this.uniforms.start.value;\n        },\n        set: function (value) {\n            this.uniforms.start.value = value;\n            this.updateDelta();\n        }\n    },\n\n    /**\n     * The X value to end the effect at.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    end: {\n        get: function () {\n            return this.uniforms.end.value;\n        },\n        set: function (value) {\n            this.uniforms.end.value = value;\n            this.updateDelta();\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/\n */\n\n/**\n * A TiltShiftYFilter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TiltShiftYFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur:           { type: '1f', value: 100 },\n        gradientBlur:   { type: '1f', value: 600 },\n        start:          { type: '2f', value: { x: 0,    y: window.screenHeight / 2 } },\n        end:            { type: '2f', value: { x: 600,  y: window.screenHeight / 2 } },\n        delta:          { type: '2f', value: { x: 30,   y: 30 } },\n        texSize:        { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } }\n    };\n\n    this.updateDelta();\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n\n        'uniform sampler2D uSampler;',\n        'uniform float blur;',\n        'uniform float gradientBlur;',\n        'uniform vec2 start;',\n        'uniform vec2 end;',\n        'uniform vec2 delta;',\n        'uniform vec2 texSize;',\n\n        'float random(vec3 scale, float seed) {',\n        '   return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);',\n        '}',\n\n        'void main(void) {',\n        '    vec4 color = vec4(0.0);',\n        '    float total = 0.0;',\n\n        '    float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);',\n        '    vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));',\n        '    float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;',\n\n        '    for (float t = -30.0; t <= 30.0; t++) {',\n        '        float percent = (t + offset - 0.5) / 30.0;',\n        '        float weight = 1.0 - abs(percent);',\n        '        vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);',\n        '        sample.rgb *= sample.a;',\n        '        color += sample * weight;',\n        '        total += weight;',\n        '    }',\n\n        '    gl_FragColor = color / total;',\n        '    gl_FragColor.rgb /= gl_FragColor.a + 0.00001;',\n        '}'\n    ];\n}\n\nTiltShiftYFilter.prototype = Object.create(AbstractFilter.prototype);\nTiltShiftYFilter.prototype.constructor = TiltShiftYFilter;\nmodule.exports = TiltShiftYFilter;\n\n/**\n * Updates the filter delta values.\n *\n */\nTiltShiftYFilter.prototype.updateDelta = function (){\n    var dx = this.uniforms.end.value.x - this.uniforms.start.value.x;\n    var dy = this.uniforms.end.value.y - this.uniforms.start.value.y;\n    var d = Math.sqrt(dx * dx + dy * dy);\n    this.uniforms.delta.value.x = -dy / d;\n    this.uniforms.delta.value.y = dx / d;\n};\n\nObject.defineProperties(TiltShiftYFilter.prototype, {\n    /**\n     * The strength of the blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value;\n        },\n        set: function (value) {\n            this.uniforms.blur.value = value;\n        }\n    },\n\n    /**\n     * The strength of the gradient blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    gradientBlur: {\n        get: function () {\n            return this.uniforms.gradientBlur.value;\n        },\n        set: function (value) {\n            this.uniforms.gradientBlur.value = value;\n        }\n    },\n\n    /**\n     * The Y value to start the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    start: {\n        get: function () {\n            return this.uniforms.start.value;\n        },\n        set: function (value) {\n            this.uniforms.start.value = value;\n            this.updateDelta();\n        }\n    },\n\n    /**\n     * The Y value to end the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    end: {\n        get: function () {\n            return this.uniforms.end.value;\n        },\n        set: function (value) {\n            this.uniforms.end.value = value;\n            this.updateDelta();\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This filter applies a twist effect making display objects appear twisted in the given direction.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TwistFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        radius:     { type: '1f', value: 0.5},\n        angle:      { type: '1f', value: 5},\n        offset:     { type: '2f', value: { x: 0.5, y: 0.5 } }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float radius;',\n        'uniform float angle;',\n        'uniform vec2 offset;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec2 coord = vTextureCoord - offset;',\n        '   float distance = length(coord);',\n\n        '   if (distance < radius) {',\n        '       float ratio = (radius - distance) / radius;',\n        '       float angleMod = ratio * ratio * angle;',\n        '       float s = sin(angleMod);',\n        '       float c = cos(angleMod);',\n        '       coord = vec2(coord.x * c - coord.y * s, coord.x * s + coord.y * c);',\n        '   }',\n\n        '   gl_FragColor = texture2D(uSampler, coord+offset);',\n        '}'\n    ];\n}\n\nTwistFilter.prototype = Object.create(AbstractFilter.prototype);\nTwistFilter.prototype.constructor = TwistFilter;\nmodule.exports = TwistFilter;\n\nObject.defineProperties(TwistFilter.prototype, {\n    /**\n     * This point describes the the offset of the twist.\n     *\n     * @member {Point}\n     * @memberof TwistFilter#\n     */\n    offset: {\n        get: function () {\n            return this.uniforms.offset.value;\n        },\n        set: function (value) {\n            this.uniforms.offset.value = value;\n        }\n    },\n\n    /**\n     * This radius of the twist.\n     *\n     * @member {number}\n     * @memberof TwistFilter#\n     */\n    radius: {\n        get: function () {\n            return this.uniforms.radius.value;\n        },\n        set: function (value) {\n            this.uniforms.radius.value = value;\n        }\n    },\n\n    /**\n     * This angle of the twist.\n     *\n     * @member {number}\n     * @memberof TwistFilter#\n     */\n    angle: {\n        get: function () {\n            return this.uniforms.angle.value;\n        },\n        set: function (value) {\n            this.uniforms.angle.value = value;\n        }\n    }\n});\n","/**\n * @file        Main export of the PIXI filters library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    AbstractFilter:     require('./AbstractFilter'),\n    AlphaMaskFilter:    require('./AlphaMaskFilter'),\n    AsciiFilter:        require('./AsciiFilter'),\n    BlurFilter:         require('./BlurFilter'),\n    BlurXFilter:        require('./BlurXFilter'),\n    BlurYFilter:        require('./BlurYFilter'),\n    ColorMatrixFilter:  require('./ColorMatrixFilter'),\n    ColorStepFilter:    require('./ColorStepFilter'),\n    ConvolutionFilter:  require('./ConvolutionFilter'),\n    CrossHatchFilter:   require('./CrossHatchFilter'),\n    DisplacementFilter: require('./DisplacementFilter'),\n    DotScreenFilter:    require('./DotScreenFilter'),\n    FilterBlock:        require('./FilterBlock'),\n    GrayFilter:         require('./GrayFilter'),\n    InvertFilter:       require('./InvertFilter'),\n    NoiseFilter:        require('./NoiseFilter'),\n    NormalMapFilter:    require('./NormalMapFilter'),\n    PixelateFilter:     require('./PixelateFilter'),\n    RGBSplitFilter:     require('./RGBSplitFilter'),\n    SepiaFilter:        require('./SepiaFilter'),\n    SmartBlurFilter:    require('./SmartBlurFilter'),\n    TiltShiftFilter:    require('./TiltShiftFilter'),\n    TiltShiftXFilter:   require('./TiltShiftXFilter'),\n    TiltShiftYFilter:   require('./TiltShiftYFilter'),\n    TwistFilter:        require('./TwistFilter')\n};\n","var core = require('../core');\n\n/**\n * Holds all information related to an Interaction event\n *\n * @class\n * @namespace PIXI\n */\nfunction InteractionData() {\n    /**\n     * This point stores the global coords of where the touch/mouse event happened\n     *\n     * @member {Point}\n     */\n    this.global = new core.math.Point();\n\n    /**\n     * The target Sprite that was interacted with\n     *\n     * @member {Sprite}\n     */\n    this.target = null;\n\n    /**\n     * When passed to an event handler, this will be the original DOM Event that was captured\n     *\n     * @member {Event}\n     */\n    this.originalEvent = null;\n}\n\nInteractionData.prototype.constructor = InteractionData;\nmodule.exports = InteractionData;\n\n/**\n * This will return the local coordinates of the specified displayObject for this InteractionData\n *\n * @param displayObject {DisplayObject} The DisplayObject that you would like the local coords off\n * @param [point] {Point} A Point object in which to store the value, optional (otherwise will create a new point)\n * @return {Point} A point containing the coordinates of the InteractionData position relative to the DisplayObject\n */\nInteractionData.prototype.getLocalPosition = function (displayObject, point) {\n    var worldTransform = displayObject.worldTransform;\n    var global = this.global;\n\n    // do a cheeky transform to get the mouse coords;\n    var a00 = worldTransform.a, a01 = worldTransform.c, a02 = worldTransform.tx,\n        a10 = worldTransform.b, a11 = worldTransform.d, a12 = worldTransform.ty,\n        id = 1 / (a00 * a11 + a01 * -a10);\n\n    point = point || new core.math.Point();\n\n    point.x = a11 * id * global.x + -a01 * id * global.y + (a12 * a01 - a02 * a11) * id;\n    point.y = a00 * id * global.y + -a10 * id * global.x + (-a12 * a00 + a02 * a10) * id;\n\n    // set the mouse coords...\n    return point;\n};\n","var core = require('../core'),\n    InteractionData = require('./InteractionData');\n\n// TODO: Obviously rewrite this...\nvar INTERACTION_FREQUENCY = 30;\nvar AUTO_PREVENT_DEFAULT = true;\n\n/**\n * The interaction manager deals with mouse and touch events. Any DisplayObject can be interactive\n * if its interactive parameter is set to true\n * This manager also supports multitouch.\n *\n * @class\n * @namespace PIXI\n * @param stage {Stage} The stage to handle interactions\n */\nfunction InteractionManager(stage) {\n    /**\n     * A reference to the stage\n     *\n     * @member {Stage}\n     */\n    this.stage = stage;\n\n    /**\n     * The mouse data\n     *\n     * @member {InteractionData}\n     */\n    this.mouse = new InteractionData();\n\n    /**\n     * An object that stores current touches (InteractionData) by id reference\n     *\n     * @member {object}\n     */\n    this.touches = {};\n\n    /**\n     * @member {Point}\n     * @private\n     */\n    this.tempPoint = new core.math.Point();\n\n    /**\n     * @member {boolean}\n     * @default\n     */\n    this.mouseoverEnabled = true;\n\n    /**\n     * Tiny little interactiveData pool !\n     *\n     * @member {Array}\n     */\n    this.pool = [];\n\n    /**\n     * An array containing all the iterative items from the our interactive tree\n     *\n     * @member {Array}\n     * @private\n     */\n    this.interactiveItems = [];\n\n    /**\n     * The DOM element to bind to.\n     *\n     * @member {HTMLElement}\n     * @private\n     */\n    this.interactionDOMElement = null;\n\n    /**\n     * Have events been attached to the dom element?\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.eventsAdded = false;\n\n    //this will make it so that you don't have to call bind all the time\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseMove = this.onMouseMove.bind( this );\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseDown = this.onMouseDown.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseOut = this.onMouseOut.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseUp = this.onMouseUp.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onTouchStart = this.onTouchStart.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onTouchEnd = this.onTouchEnd.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onTouchMove = this.onTouchMove.bind(this);\n\n    /**\n     * @member {number}\n     */\n    this.last = 0;\n\n    /**\n     * The css style of the cursor that is being used\n     * @member {string}\n     */\n    this.currentCursorStyle = 'inherit';\n\n    /**\n     * Is set to true when the mouse is moved out of the canvas\n     * @member {boolean}\n     */\n    this.mouseOut = false;\n\n    /**\n     * @member {number}\n     */\n    this.resolution = 1;\n\n    // used for hit testing\n    this._tempPoint = new core.math.Point();\n}\n\nInteractionManager.prototype.constructor = InteractionManager;\nmodule.exports = InteractionManager;\n\n/**\n * Collects an interactive sprite recursively to have their interactions managed\n *\n * @param displayObject {DisplayObject} the displayObject to collect\n * @param iParent {DisplayObject} the display object's parent\n * @private\n */\nInteractionManager.prototype.collectInteractiveSprite = function (displayObject, iParent) {\n    var children = displayObject.children;\n    var length = children.length;\n\n    // make an interaction tree... {item.__interactiveParent}\n    for (var i = length - 1; i >= 0; i--) {\n        var child = children[i];\n\n        // push all interactive bits\n        if (child._interactive) {\n            iParent.interactiveChildren = true;\n            //child.__iParent = iParent;\n            this.interactiveItems.push(child);\n\n            if (child.children.length > 0) {\n                this.collectInteractiveSprite(child, child);\n            }\n        }\n        else {\n            child.__iParent = null;\n            if (child.children.length > 0) {\n                this.collectInteractiveSprite(child, iParent);\n            }\n        }\n\n    }\n};\n\n/**\n * Sets the DOM element which will receive mouse/touch events. This is useful for when you have\n * other DOM elements on top of the renderers Canvas element. With this you'll be bale to deletegate\n * another DOM element to receive those events.\n *\n * @param element {HTMLElement} the DOM element which will receive mouse and touch events.\n * @param [resolution=1] {number} THe resolution of the new element (relative to the canvas).\n * @private\n */\nInteractionManager.prototype.setTargetElement = function (element, resolution) {\n    this.removeEvents();\n\n    this.interactionDOMElement = element;\n\n    this.resolution = resolution || 1;\n\n    this.addEvents();\n};\n\n/**\n *\n * @private\n */\nInteractionManager.prototype.addEvents = function () {\n    if (!this.interactionDOMElement) {\n        return;\n    }\n\n    if (window.navigator.msPointerEnabled) {\n        this.interactionDOMElement.style['-ms-content-zooming'] = 'none';\n        this.interactionDOMElement.style['-ms-touch-action'] = 'none';\n    }\n\n    this.interactionDOMElement.addEventListener('mousemove',    this.onMouseMove, true);\n    this.interactionDOMElement.addEventListener('mousedown',    this.onMouseDown, true);\n    this.interactionDOMElement.addEventListener('mouseout',     this.onMouseOut, true);\n\n    this.interactionDOMElement.addEventListener('touchstart',   this.onTouchStart, true);\n    this.interactionDOMElement.addEventListener('touchend',     this.onTouchEnd, true);\n    this.interactionDOMElement.addEventListener('touchmove',    this.onTouchMove, true);\n\n    window.addEventListener('mouseup',  this.onMouseUp, true);\n\n    this.eventsAdded = true;\n};\n\n/**\n *\n * @private\n */\nInteractionManager.prototype.removeEvents = function () {\n    if (!this.interactionDOMElement) {\n        return;\n    }\n\n    if (window.navigator.msPointerEnabled) {\n        this.interactionDOMElement.style['-ms-content-zooming'] = '';\n        this.interactionDOMElement.style['-ms-touch-action'] = '';\n    }\n\n    this.interactionDOMElement.removeEventListener('mousemove', this.onMouseMove, true);\n    this.interactionDOMElement.removeEventListener('mousedown', this.onMouseDown, true);\n    this.interactionDOMElement.removeEventListener('mouseout',  this.onMouseOut, true);\n\n    this.interactionDOMElement.removeEventListener('touchstart', this.onTouchStart, true);\n    this.interactionDOMElement.removeEventListener('touchend',  this.onTouchEnd, true);\n    this.interactionDOMElement.removeEventListener('touchmove', this.onTouchMove, true);\n\n    this.interactionDOMElement = null;\n\n    window.removeEventListener('mouseup',  this.onMouseUp, true);\n\n    this.eventsAdded = false;\n};\n\n/**\n * updates the state of interactive objects\n *\n * @private\n */\nInteractionManager.prototype.update = function () {\n    if (!this.interactionDOMElement) {\n        return;\n    }\n\n    // frequency of 30fps??\n    var now = Date.now();\n    var diff = now - this.last;\n    diff = (diff * INTERACTION_FREQUENCY ) / 1000;\n    if (diff < 1) {\n        return;\n    }\n\n    this.last = now;\n\n    var i = 0;\n\n    // ok.. so mouse events??\n    // yes for now :)\n    // OPTIMISE - how often to check??\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    // loop through interactive objects!\n    var length = this.interactiveItems.length;\n    var cursor = 'inherit';\n    var over = false;\n\n    for (i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        // OPTIMISATION - only calculate every time if the mousemove function exists..\n        // OK so.. does the object have any other interactive functions?\n        // hit-test the clip!\n       // if (item.mouseover || item.mouseout || item.buttonMode)\n       // {\n        // ok so there are some functions so lets hit test it..\n        item.__hit = this.hitTest(item, this.mouse);\n        this.mouse.target = item;\n        // ok so deal with interactions..\n        // looks like there was a hit!\n        if (item.__hit && !over) {\n            if (item.buttonMode) {\n                cursor = item.defaultCursor;\n            }\n\n            if (!item.interactiveChildren) {\n                over = true;\n            }\n\n            if (!item.__isOver) {\n                if (item.mouseover) {\n                    item.mouseover (this.mouse);\n                }\n                item.__isOver = true;\n            }\n        }\n        else {\n            if (item.__isOver) {\n                // roll out!\n                if (item.mouseout) {\n                    item.mouseout (this.mouse);\n                }\n                item.__isOver = false;\n            }\n        }\n    }\n\n    if (this.currentCursorStyle !== cursor) {\n        this.currentCursorStyle = cursor;\n        this.interactionDOMElement.style.cursor = cursor;\n    }\n};\n\n/**\n * @private\n */\nInteractionManager.prototype.rebuildInteractiveGraph = function () {\n    this.dirty = false;\n\n    var len = this.interactiveItems.length;\n\n    for (var i = 0; i < len; i++) {\n        this.interactiveItems[i].interactiveChildren = false;\n    }\n\n    this.interactiveItems.length = 0;\n\n    if (this.stage.interactive) {\n        this.interactiveItems.push(this.stage);\n    }\n\n    // Go through and collect all the objects that are interactive..\n    this.collectInteractiveSprite(this.stage, this.stage);\n};\n\n/**\n * Is called when the mouse moves across the renderer element\n *\n * @param event {Event} The DOM event of the mouse moving\n * @private\n */\nInteractionManager.prototype.onMouseMove = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    // TODO optimize by not check EVERY TIME! maybe half as often? //\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n\n    this.mouse.global.x = (event.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) / this.resolution;\n    this.mouse.global.y = (event.clientY - rect.top) * ( this.interactionDOMElement.height / rect.height) / this.resolution;\n\n    var length = this.interactiveItems.length;\n\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        // Call the function!\n        if (item.mousemove) {\n            item.mousemove(this.mouse);\n        }\n    }\n};\n\n/**\n * Is called when the mouse button is pressed down on the renderer element\n *\n * @param event {Event} The DOM event of a mouse button being pressed down\n * @private\n */\nInteractionManager.prototype.onMouseDown = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    if (AUTO_PREVENT_DEFAULT) {\n        this.mouse.originalEvent.preventDefault();\n    }\n\n    // loop through interaction tree...\n    // hit test each item! ->\n    // get interactive items under point??\n    //stage.__i\n    var length = this.interactiveItems.length;\n\n    var e = this.mouse.originalEvent;\n    var isRightButton = e.button === 2 || e.which === 3;\n    var downFunction = isRightButton ? 'rightdown' : 'mousedown';\n    var clickFunction = isRightButton ? 'rightclick' : 'click';\n    var buttonIsDown = isRightButton ? '__rightIsDown' : '__mouseIsDown';\n    var isDown = isRightButton ? '__isRightDown' : '__isDown';\n\n    // while\n    // hit test\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        if (item[downFunction] || item[clickFunction]) {\n            item[buttonIsDown] = true;\n            item.__hit = this.hitTest(item, this.mouse);\n\n            if (item.__hit) {\n                //call the function!\n                if (item[downFunction]) {\n                    item[downFunction](this.mouse);\n                }\n                item[isDown] = true;\n\n                // just the one!\n                if (!item.interactiveChildren) {\n                    break;\n                }\n            }\n        }\n    }\n};\n\n/**\n * Is called when the mouse is moved out of the renderer element\n *\n * @param event {Event} The DOM event of a mouse being moved out\n * @private\n */\nInteractionManager.prototype.onMouseOut = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    var length = this.interactiveItems.length;\n\n    this.interactionDOMElement.style.cursor = 'inherit';\n\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n        if (item.__isOver) {\n            this.mouse.target = item;\n            if (item.mouseout) {\n                item.mouseout(this.mouse);\n            }\n            item.__isOver = false;\n        }\n    }\n\n    this.mouseOut = true;\n\n    // move the mouse to an impossible position\n    this.mouse.global.x = -10000;\n    this.mouse.global.y = -10000;\n};\n\n/**\n * Is called when the mouse button is released on the renderer element\n *\n * @param event {Event} The DOM event of a mouse button being released\n * @private\n */\nInteractionManager.prototype.onMouseUp = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    var length = this.interactiveItems.length;\n    var up = false;\n\n    var e = this.mouse.originalEvent;\n    var isRightButton = e.button === 2 || e.which === 3;\n\n    var upFunction = isRightButton ? 'rightup' : 'mouseup';\n    var clickFunction = isRightButton ? 'rightclick' : 'click';\n    var upOutsideFunction = isRightButton ? 'rightupoutside' : 'mouseupoutside';\n    var isDown = isRightButton ? '__isRightDown' : '__isDown';\n\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        if (item[clickFunction] || item[upFunction] || item[upOutsideFunction]) {\n            item.__hit = this.hitTest(item, this.mouse);\n\n            if (item.__hit && !up) {\n                //call the function!\n                if (item[upFunction]) {\n                    item[upFunction](this.mouse);\n                }\n                if (item[isDown]) {\n                    if (item[clickFunction]) {\n                        item[clickFunction](this.mouse);\n                    }\n                }\n\n                if (!item.interactiveChildren) {\n                    up = true;\n                }\n            }\n            else {\n                if (item[isDown]) {\n                    if (item[upOutsideFunction]) {\n                        item[upOutsideFunction](this.mouse);\n                    }\n                }\n            }\n\n            item[isDown] = false;\n        }\n    }\n};\n\n/**\n * Tests if the current mouse coordinates hit a sprite\n *\n * @param item {DisplayObject} The displayObject to test for a hit\n * @param interactionData {InteractionData} The interactionData object to update in the case there is a hit\n * @private\n */\nInteractionManager.prototype.hitTest = function (item, interactionData) {\n    var global = interactionData.global;\n\n    if (!item.worldVisible) {\n        return false;\n    }\n\n    // map the global point to local space.\n    item.worldTransform.applyInverse(global,  this._tempPoint);\n\n    var x = this._tempPoint.x,\n        y = this._tempPoint.y,\n        i;\n\n    interactionData.target = item;\n\n    //a sprite or display object with a hit area defined\n    if (item.hitArea && item.hitArea.contains) {\n        return item.hitArea.contains(x, y);\n    }\n    // a sprite with no hitarea defined\n    else if (item instanceof core.Sprite) {\n        var width = item.texture.frame.width;\n        var height = item.texture.frame.height;\n        var x1 = -width * item.anchor.x;\n        var y1;\n\n        if (x > x1 && x < x1 + width) {\n            y1 = -height * item.anchor.y;\n\n            if (y > y1 && y < y1 + height) {\n                // set the target property if a hit is true!\n                return true;\n            }\n        }\n    }\n    else if (item instanceof core.Graphics) {\n        var graphicsData = item.graphicsData;\n        for (i = 0; i < graphicsData.length; i++) {\n            var data = graphicsData[i];\n\n            if (!data.fill) {\n                continue;\n            }\n\n            // only deal with fills..\n            if (data.shape) {\n                if (data.shape.contains(x, y)) {\n                    //interactionData.target = item;\n                    return true;\n                }\n            }\n        }\n    }\n\n    var length = item.children.length;\n\n    for (i = 0; i < length; i++) {\n        var tempItem = item.children[i];\n        var hit = this.hitTest(tempItem, interactionData);\n        if (hit) {\n            // hmm.. TODO SET CORRECT TARGET?\n            interactionData.target = item;\n            return true;\n        }\n    }\n    return false;\n};\n\n/**\n * Is called when a touch is moved across the renderer element\n *\n * @param event {Event} The DOM event of a touch moving across the renderer view\n * @private\n */\nInteractionManager.prototype.onTouchMove = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n    var changedTouches = event.changedTouches;\n    var touchData;\n    var i = 0;\n\n    for (i = 0; i < changedTouches.length; i++) {\n        var touchEvent = changedTouches[i];\n        touchData = this.touches[touchEvent.identifier];\n        touchData.originalEvent = event;\n\n        // update the touch position\n        touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution;\n        touchData.global.y = ( (touchEvent.clientY - rect.top)  * (this.interactionDOMElement.height / rect.height) )  / this.resolution;\n        if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) {\n            //Support for CocoonJS fullscreen scale modes\n            touchData.global.x = touchEvent.clientX;\n            touchData.global.y = touchEvent.clientY;\n        }\n\n        for (var j = 0; j < this.interactiveItems.length; j++) {\n            var item = this.interactiveItems[j];\n            if (item.touchmove && item.__touchData && item.__touchData[touchEvent.identifier]) {\n                item.touchmove(touchData);\n            }\n        }\n    }\n};\n\n/**\n * Is called when a touch is started on the renderer element\n *\n * @param event {Event} The DOM event of a touch starting on the renderer view\n * @private\n */\nInteractionManager.prototype.onTouchStart = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n\n    if (AUTO_PREVENT_DEFAULT) {\n        event.preventDefault();\n    }\n\n    var changedTouches = event.changedTouches;\n    for (var i=0; i < changedTouches.length; i++) {\n        var touchEvent = changedTouches[i];\n\n        var touchData = this.pool.pop();\n        if (!touchData) {\n            touchData = new InteractionData();\n        }\n\n        touchData.originalEvent = event;\n\n        this.touches[touchEvent.identifier] = touchData;\n        touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution;\n        touchData.global.y = ( (touchEvent.clientY - rect.top)  * (this.interactionDOMElement.height / rect.height) ) / this.resolution;\n        if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) {\n            //Support for CocoonJS fullscreen scale modes\n            touchData.global.x = touchEvent.clientX;\n            touchData.global.y = touchEvent.clientY;\n        }\n\n        var length = this.interactiveItems.length;\n\n        for (var j = 0; j < length; j++) {\n            var item = this.interactiveItems[j];\n\n            if (item.touchstart || item.tap) {\n                item.__hit = this.hitTest(item, touchData);\n\n                if (item.__hit) {\n                    //call the function!\n                    if (item.touchstart) {\n                        item.touchstart(touchData);\n                    }\n\n                    item.__isDown = true;\n                    item.__touchData = item.__touchData || {};\n                    item.__touchData[touchEvent.identifier] = touchData;\n\n                    if (!item.interactiveChildren) {\n                        break;\n                    }\n                }\n            }\n        }\n    }\n};\n\n/**\n * Is called when a touch is ended on the renderer element\n *\n * @param event {Event} The DOM event of a touch ending on the renderer view\n * @private\n */\nInteractionManager.prototype.onTouchEnd = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n    var changedTouches = event.changedTouches;\n\n    for (var i=0; i < changedTouches.length; i++) {\n        var touchEvent = changedTouches[i];\n        var touchData = this.touches[touchEvent.identifier];\n        var up = false;\n        touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution;\n        touchData.global.y = ( (touchEvent.clientY - rect.top)  * (this.interactionDOMElement.height / rect.height) ) / this.resolution;\n        if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) {\n            //Support for CocoonJS fullscreen scale modes\n            touchData.global.x = touchEvent.clientX;\n            touchData.global.y = touchEvent.clientY;\n        }\n\n        var length = this.interactiveItems.length;\n        for (var j = 0; j < length; j++) {\n            var item = this.interactiveItems[j];\n\n            if (item.__touchData && item.__touchData[touchEvent.identifier]) {\n\n                item.__hit = this.hitTest(item, item.__touchData[touchEvent.identifier]);\n\n                // so this one WAS down...\n                touchData.originalEvent = event;\n                // hitTest??\n\n                if (item.touchend || item.tap) {\n                    if (item.__hit && !up) {\n                        if (item.touchend) {\n                            item.touchend(touchData);\n                        }\n                        if (item.__isDown && item.tap) {\n                            item.tap(touchData);\n                        }\n                        if (!item.interactiveChildren) {\n                            up = true;\n                        }\n                    }\n                    else {\n                        if (item.__isDown && item.touchendoutside) {\n                            item.touchendoutside(touchData);\n                        }\n                    }\n\n                    item.__isDown = false;\n                }\n\n                item.__touchData[touchEvent.identifier] = null;\n            }\n        }\n        // remove the touch..\n        this.pool.push(touchData);\n        this.touches[touchEvent.identifier] = null;\n    }\n};\n","/**\n * @file        Main export of the PIXI interactions library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    InteractionData:    require('./InteractionData'),\n    InteractionManager: require('./InteractionManager')\n};\n","var core = require('../core'),\n    ImageLoader = require('./ImageLoader');\n\n/**\n * The atlas file loader is used to load in Texture Atlas data and parse it. When loaded this class will dispatch a 'loaded' event. If loading fails this class will dispatch an 'error' event.\n *\n * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format.\n *\n * It is highly recommended to use texture atlases (also know as 'sprite sheets') as it allowed sprites to be batched and drawn together for highly increased rendering speed.\n * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId()\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction AtlasLoader(url, crossorigin) {\n    this.url = url;\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n    this.crossorigin = crossorigin;\n    this.loaded = false;\n}\n\nAtlasLoader.prototype.constructor = AtlasLoader;\nmodule.exports = AtlasLoader;\n\ncore.utils.eventTarget.mixin(AtlasLoader.prototype);\n\n /**\n * Starts loading the JSON file\n *\n */\nAtlasLoader.prototype.load = function () {\n    this.ajaxRequest = new core.utils.AjaxRequest();\n    this.ajaxRequest.onreadystatechange = this.onAtlasLoaded.bind(this);\n\n    this.ajaxRequest.open('GET', this.url, true);\n\n    if (this.ajaxRequest.overrideMimeType) {\n        this.ajaxRequest.overrideMimeType('application/json');\n    }\n\n    this.ajaxRequest.send(null);\n};\n\n/**\n * Invoked when the Atlas has fully loaded. Parses the JSON and builds the texture frames.\n *\n * @private\n */\nAtlasLoader.prototype.onAtlasLoaded = function () {\n    if (this.ajaxRequest.readyState === 4) {\n        if (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1) {\n            this.atlas = {\n                meta : {\n                    image : []\n                },\n                frames : []\n            };\n            var result = this.ajaxRequest.responseText.split(/\\r?\\n/);\n            var lineCount = -3;\n\n            var currentImageId = 0;\n            var currentFrame = null;\n            var nameInNextLine = false;\n\n            var i = 0,\n                j = 0,\n                selfOnLoaded = this.onLoaded.bind(this);\n\n            // parser without rotation support yet!\n            for (i = 0; i < result.length; i++) {\n                result[i] = result[i].replace(/^\\s+|\\s+$/g, '');\n\n                if (result[i] === '') {\n                    nameInNextLine = i+1;\n                }\n\n                if (result[i].length > 0) {\n                    if (nameInNextLine === i) {\n                        this.atlas.meta.image.push(result[i]);\n                        currentImageId = this.atlas.meta.image.length - 1;\n                        this.atlas.frames.push({});\n                        lineCount = -3;\n                    } else if (lineCount > 0) {\n                        if (lineCount % 7 === 1) { // frame name\n                            if (currentFrame != null) { //jshint ignore:line\n                                this.atlas.frames[currentImageId][currentFrame.name] = currentFrame;\n                            }\n                            currentFrame = { name: result[i], frame : {} };\n                        } else {\n                            var text = result[i].split(' ');\n                            if (lineCount % 7 === 3) { // position\n                                currentFrame.frame.x = Number(text[1].replace(',', ''));\n                                currentFrame.frame.y = Number(text[2]);\n                            } else if (lineCount % 7 === 4) { // size\n                                currentFrame.frame.w = Number(text[1].replace(',', ''));\n                                currentFrame.frame.h = Number(text[2]);\n                            } else if (lineCount % 7 === 5) { // real size\n                                var realSize = {\n                                    x : 0,\n                                    y : 0,\n                                    w : Number(text[1].replace(',', '')),\n                                    h : Number(text[2])\n                                };\n\n                                if (realSize.w > currentFrame.frame.w || realSize.h > currentFrame.frame.h) {\n                                    currentFrame.trimmed = true;\n                                    currentFrame.realSize = realSize;\n                                } else {\n                                    currentFrame.trimmed = false;\n                                }\n                            }\n                        }\n                    }\n                    lineCount++;\n                }\n            }\n\n            if (currentFrame != null) { //jshint ignore:line\n                this.atlas.frames[currentImageId][currentFrame.name] = currentFrame;\n            }\n\n            if (this.atlas.meta.image.length > 0) {\n                this.images = [];\n                for (j = 0; j < this.atlas.meta.image.length; j++) {\n                    // sprite sheet\n                    var textureUrl = this.baseUrl + this.atlas.meta.image[j];\n                    var frameData = this.atlas.frames[j];\n                    this.images.push(new ImageLoader(textureUrl, this.crossorigin));\n\n                    for (i in frameData) {\n                        var rect = frameData[i].frame;\n                        if (rect) {\n                            core.utils.TextureCache[i] = new core.Texture(this.images[j].texture.baseTexture, {\n                                x: rect.x,\n                                y: rect.y,\n                                width: rect.w,\n                                height: rect.h\n                            });\n                            if (frameData[i].trimmed) {\n                                core.utils.TextureCache[i].realSize = frameData[i].realSize;\n                                // trim in pixi not supported yet, todo update trim properties if it is done ...\n                                core.utils.TextureCache[i].trim.x = 0;\n                                core.utils.TextureCache[i].trim.y = 0;\n                            }\n                        }\n                    }\n                }\n\n                this.currentImageId = 0;\n                for (j = 0; j < this.images.length; j++) {\n                    this.images[j].on('loaded', selfOnLoaded);\n                }\n                this.images[this.currentImageId].load();\n\n            } else {\n                this.onLoaded();\n            }\n\n        } else {\n            this.onError();\n        }\n    }\n};\n\n/**\n * Invoked when json file has loaded.\n *\n * @private\n */\nAtlasLoader.prototype.onLoaded = function () {\n    if (this.images.length - 1 > this.currentImageId) {\n        this.currentImageId++;\n        this.images[this.currentImageId].load();\n    } else {\n        this.loaded = true;\n        this.emit('loaded', { content: this });\n    }\n};\n\n/**\n * Invoked when an error occurs.\n *\n * @private\n */\nAtlasLoader.prototype.onError = function () {\n    this.emit('error', { content: this });\n};\n","var core = require('../core'),\n    ImageLoader = require('./ImageLoader');\n\n/**\n * The xml loader is used to load in XML bitmap font data ('xml' or 'fnt')\n * To generate the data you can use http://www.angelcode.com/products/bmfont/\n * This loader will also load the image file as the data.\n * When loaded this class will dispatch a 'loaded' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the sprite sheet JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction BitmapFontLoader(url, crossorigin) {\n    /**\n     * The url of the bitmap font data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * The base url of the bitmap font data\n     *\n     * @member {String}\n     * @readOnly\n     */\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n\n    /**\n     * The texture of the bitmap font\n     *\n     * @member {Texture}\n     */\n    this.texture = null;\n}\n\n// constructor\nBitmapFontLoader.prototype.constructor = BitmapFontLoader;\nmodule.exports = BitmapFontLoader;\n\ncore.utils.eventTarget.mixin(BitmapFontLoader.prototype);\n\n/**\n * Loads the XML font data\n *\n */\nBitmapFontLoader.prototype.load = function () {\n    this.ajaxRequest = new core.utils.AjaxRequest();\n    this.ajaxRequest.onreadystatechange = this.onXMLLoaded.bind(this);\n\n    this.ajaxRequest.open('GET', this.url, true);\n\n    if (this.ajaxRequest.overrideMimeType) {\n        this.ajaxRequest.overrideMimeType('application/xml');\n    }\n\n    this.ajaxRequest.send(null);\n};\n\n/**\n * Invoked when the XML file is loaded, parses the data.\n *\n * @private\n */\nBitmapFontLoader.prototype.onXMLLoaded = function () {\n    if (this.ajaxRequest.readyState === 4) {\n        if (this.ajaxRequest.status === 200 || window.location.protocol.indexOf('http') === -1) {\n            var responseXML = this.ajaxRequest.responseXML;\n            if (!responseXML || /MSIE 9/i.test(navigator.userAgent) || navigator.isCocoonJS) {\n                if (typeof(window.DOMParser) === 'function') {\n                    var domparser = new DOMParser();\n                    responseXML = domparser.parseFromString(this.ajaxRequest.responseText, 'text/xml');\n                } else {\n                    var div = document.createElement('div');\n                    div.innerHTML = this.ajaxRequest.responseText;\n                    responseXML = div;\n                }\n            }\n\n            var textureUrl = this.baseUrl + responseXML.getElementsByTagName('page')[0].getAttribute('file');\n            var image = new ImageLoader(textureUrl, this.crossorigin);\n            this.texture = image.texture.baseTexture;\n\n            var data = {};\n            var info = responseXML.getElementsByTagName('info')[0];\n            var common = responseXML.getElementsByTagName('common')[0];\n            data.font = info.getAttribute('face');\n            data.size = parseInt(info.getAttribute('size'), 10);\n            data.lineHeight = parseInt(common.getAttribute('lineHeight'), 10);\n            data.chars = {};\n\n            //parse letters\n            var letters = responseXML.getElementsByTagName('char');\n\n            for (var i = 0; i < letters.length; i++) {\n                var charCode = parseInt(letters[i].getAttribute('id'), 10);\n\n                var textureRect = new core.math.Rectangle(\n                    parseInt(letters[i].getAttribute('x'), 10),\n                    parseInt(letters[i].getAttribute('y'), 10),\n                    parseInt(letters[i].getAttribute('width'), 10),\n                    parseInt(letters[i].getAttribute('height'), 10)\n                );\n\n                data.chars[charCode] = {\n                    xOffset: parseInt(letters[i].getAttribute('xoffset'), 10),\n                    yOffset: parseInt(letters[i].getAttribute('yoffset'), 10),\n                    xAdvance: parseInt(letters[i].getAttribute('xadvance'), 10),\n                    kerning: {},\n                    texture: core.utils.TextureCache[charCode] = new core.Texture(this.texture, textureRect)\n\n                };\n            }\n\n            //parse kernings\n            var kernings = responseXML.getElementsByTagName('kerning');\n            for (i = 0; i < kernings.length; i++) {\n                var first = parseInt(kernings[i].getAttribute('first'), 10);\n                var second = parseInt(kernings[i].getAttribute('second'), 10);\n                var amount = parseInt(kernings[i].getAttribute('amount'), 10);\n\n                data.chars[second].kerning[first] = amount;\n\n            }\n\n            core.BitmapText.fonts[data.font] = data;\n\n            image.addEventListener('loaded', this.onLoaded.bind(this));\n            image.load();\n        }\n    }\n};\n\n/**\n * Invoked when all files are loaded (xml/fnt and texture)\n *\n * @private\n */\nBitmapFontLoader.prototype.onLoaded = function () {\n    this.emit('loaded', { content: this });\n};\n","var core = require('../core');\n\n/**\n * The image loader class is responsible for loading images file formats ('jpeg', 'jpg', 'png' and 'gif')\n * Once the image has been loaded it is stored in the PIXI texture cache and can be accessed though Texture.fromFrame() and Sprite.fromFrame()\n * When loaded this class will dispatch a 'loaded' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the image\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction ImageLoader(url, crossorigin) {\n    /**\n     * The texture being loaded\n     *\n     * @member {Texture}\n     */\n    this.texture = core.Texture.fromImage(url, crossorigin);\n\n    /**\n     * if the image is loaded with loadFramedSpriteSheet\n     * frames will contain the sprite sheet frames\n     *\n     * @member {Array}\n     * @readOnly\n     */\n    this.frames = [];\n}\n\n// constructor\nImageLoader.prototype.constructor = ImageLoader;\nmodule.exports = ImageLoader;\n\ncore.utils.eventTarget.mixin(ImageLoader.prototype);\n\n/**\n * Loads image or takes it from cache\n *\n */\nImageLoader.prototype.load = function () {\n    if (!this.texture.baseTexture.hasLoaded) {\n        this.texture.baseTexture.on('loaded', this.onLoaded.bind(this));\n        this.texture.baseTexture.on('error', this.onError.bind(this));\n    }\n    else {\n        this.onLoaded();\n    }\n};\n\n/**\n * Invoked when image file is loaded or it is already cached and ready to use\n *\n * @private\n */\nImageLoader.prototype.onLoaded = function () {\n    this.emit('loaded', { content: this });\n};\n\n/**\n * Invoked when image file failed loading\n *\n * @method onError\n * @private\n */\nImageLoader.prototype.onError = function () {\n    this.emit('error', { content: this });\n};\n\n/**\n * Loads image and split it to uniform sized frames\n *\n * @param frameWidth {number} width of each frame\n * @param frameHeight {number} height of each frame\n * @param textureName {String} if given, the frames will be cached in <textureName>-<ord> format\n */\nImageLoader.prototype.loadFramedSpriteSheet = function (frameWidth, frameHeight, textureName) {\n    this.frames = [];\n\n    var cols = Math.floor(this.texture.width / frameWidth);\n    var rows = Math.floor(this.texture.height / frameHeight);\n\n    var i=0;\n    for (var y = 0; y < rows; ++y) {\n        for (var x = 0; x < cols; ++x, ++i) {\n            var texture = new core.Texture(\n                this.texture.baseTexture,\n                new core.math.Rectangle(\n                    x * frameWidth,\n                    y * frameHeight,\n                    frameWidth,\n                    frameHeight\n                )\n            );\n\n            this.frames.push(texture);\n\n            if (textureName) {\n                core.utils.TextureCache[textureName + '-' + i] = texture;\n            }\n        }\n    }\n\n\tthis.load();\n};\n","var core = require('../core'),\n    spine = require('../spine/SpineRuntime'),\n    ImageLoader = require('./ImageLoader'),\n    SpineTextureLoader = require('./SpineTextureLoader');\n\n/**\n * The json file loader is used to load in JSON data and parse it\n * When loaded this class will dispatch a 'loaded' event\n * If loading fails this class will dispatch an 'error' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction JsonLoader(url, crossorigin) {\n    /**\n     * The url of the bitmap font data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * The base url of the bitmap font data\n     *\n     * @member {String}\n     * @readOnly\n     */\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n\n    /**\n     * Whether the data has loaded yet\n     *\n     * @member {boolean}\n     * @readOnly\n     */\n     this.loaded = false;\n}\n\n// constructor\nJsonLoader.prototype.constructor = JsonLoader;\nmodule.exports = JsonLoader;\n\ncore.utils.eventTarget.mixin(JsonLoader.prototype);\n\n/**\n * Loads the JSON data\n *\n */\nJsonLoader.prototype.load = function () {\n    if (window.XDomainRequest && this.crossorigin) {\n        this.ajaxRequest = new window.XDomainRequest();\n\n        // XDomainRequest has a few quirks. Occasionally it will abort requests\n        // A way to avoid this is to make sure ALL callbacks are set even if not used\n        // More info here: http://stackoverflow.com/questions/15786966/xdomainrequest-aborts-post-on-ie-9\n        this.ajaxRequest.timeout = 3000;\n\n        this.ajaxRequest.onerror = this.onError.bind(this);\n        this.ajaxRequest.ontimeout = this.onError.bind(this);\n\n        this.ajaxRequest.onprogress = function () {};\n\n        this.ajaxRequest.onload = this.onJSONLoaded.bind(this);\n    }\n    else {\n        if (window.XMLHttpRequest) {\n            this.ajaxRequest = new window.XMLHttpRequest();\n        }\n        else {\n            this.ajaxRequest = new window.ActiveXObject('Microsoft.XMLHTTP');\n        }\n\n        this.ajaxRequest.onreadystatechange = this.onReadyStateChanged.bind(this);\n    }\n\n    this.ajaxRequest.open('GET',this.url,true);\n\n    this.ajaxRequest.send();\n};\n\n/**\n * Bridge function to be able to use the more reliable onreadystatechange in XMLHttpRequest.\n *\n * @private\n */\nJsonLoader.prototype.onReadyStateChanged = function () {\n    if (this.ajaxRequest.readyState === 4 && (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1)) {\n        this.onJSONLoaded();\n    }\n};\n\n/**\n * Invoke when JSON file is loaded\n *\n * @private\n */\nJsonLoader.prototype.onJSONLoaded = function () {\n    if (!this.ajaxRequest.responseText) {\n        this.onError();\n        return;\n    }\n\n    this.json = JSON.parse(this.ajaxRequest.responseText);\n\n    if (this.json.frames) {\n        // sprite sheet\n        var textureUrl = this.baseUrl + this.json.meta.image;\n        var image = new ImageLoader(textureUrl, this.crossorigin);\n        var frameData = this.json.frames;\n\n        this.texture = image.texture.baseTexture;\n        image.addEventListener('loaded', this.onLoaded.bind(this));\n        image.addEventListener('error', this.onError.bind(this));\n\n        for (var i in frameData) {\n            var rect = frameData[i].frame;\n\n            if (rect) {\n                var textureSize = new core.math.Rectangle(rect.x, rect.y, rect.w, rect.h);\n                var crop = textureSize.clone();\n                var trim = null;\n\n                //  Check to see if the sprite is trimmed\n                if (frameData[i].trimmed) {\n                    var actualSize = frameData[i].sourceSize;\n                    var realSize = frameData[i].spriteSourceSize;\n                    trim = new core.math.Rectangle(realSize.x, realSize.y, actualSize.w, actualSize.h);\n                }\n                core.utils.TextureCache[i] = new core.Texture(this.texture, textureSize, crop, trim);\n            }\n        }\n\n        image.load();\n\n    }\n    else if (this.json.bones) {\n\t\t// check if the json was loaded before\n\t\tif (core.utils.AnimCache[this.url]) {\n\t\t\tthis.onLoaded();\n\t\t}\n\t\telse {\n\t\t\t/**\n             * use a bit of hackery to load the atlas file, here we assume that the .json, .atlas and .png files\n\t\t\t * that correspond to the spine file are in the same base URL and that the .json and .atlas files\n\t\t\t * have the same name\n\t\t\t */\n\t\t\tvar atlasPath = this.url.substr(0, this.url.lastIndexOf('.')) + '.atlas';\n\t\t\tvar atlasLoader = new JsonLoader(atlasPath, this.crossorigin);\n\t\t\t// save a copy of the current object for future reference //\n\t\t\tvar originalLoader = this;\n\t\t\t// before loading the file, replace the \"onJSONLoaded\" function for our own //\n\t\t\tatlasLoader.onJSONLoaded = function () {\n\t\t\t\t// at this point \"this\" points at the atlasLoader (JsonLoader) instance //\n\t\t\t\tif (!this.ajaxRequest.responseText) {\n\t\t\t\t\tthis.onError(); // FIXME: hmm, this is funny because we are not responding to errors yet\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t\t// create a new instance of a spine texture loader for this spine object //\n\t\t\t\tvar textureLoader = new SpineTextureLoader(this.url.substring(0, this.url.lastIndexOf('/')));\n\t\t\t\t// create a spine atlas using the loaded text and a spine texture loader instance //\n\t\t\t\tvar spineAtlas = new spine.Atlas(this.ajaxRequest.responseText, textureLoader);\n\t\t\t\t// now we use an atlas attachment loader //\n\t\t\t\tvar attachmentLoader = new spine.AtlasAttachmentLoader(spineAtlas);\n\t\t\t\t// spine animation\n\t\t\t\tvar spineJsonParser = new spine.SkeletonJson(attachmentLoader);\n\t\t\t\tvar skeletonData = spineJsonParser.readSkeletonData(originalLoader.json);\n\t\t\t\tcore.utils.AnimCache[originalLoader.url] = skeletonData;\n\t\t\t\toriginalLoader.spine = skeletonData;\n\t\t\t\toriginalLoader.spineAtlas = spineAtlas;\n\t\t\t\toriginalLoader.spineAtlasLoader = atlasLoader;\n\t\t\t\t// wait for textures to finish loading if needed\n\t\t\t\tif (textureLoader.loadingCount > 0) {\n\t\t\t\t\ttextureLoader.addEventListener('loadedBaseTexture', function (evt){\n\t\t\t\t\t\tif (evt.content.content.loadingCount <= 0) {\n\t\t\t\t\t\t\toriginalLoader.onLoaded();\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\toriginalLoader.onLoaded();\n\t\t\t\t}\n\t\t\t};\n\t\t\t// start the loading //\n\t\t\tatlasLoader.load();\n\t\t}\n    }\n    else {\n        this.onLoaded();\n    }\n};\n\n/**\n * Invoke when json file loaded\n *\n * @private\n */\nJsonLoader.prototype.onLoaded = function () {\n    this.loaded = true;\n    this.dispatchEvent({\n        type: 'loaded',\n        content: this\n    });\n};\n\n/**\n * Invoke when error occured\n *\n * @private\n */\nJsonLoader.prototype.onError = function () {\n\n    this.dispatchEvent({\n        type: 'error',\n        content: this\n    });\n};\n","var core = require('../core'),\n    JsonLoader = require('./JsonLoader');\n\n/**\n * @author Mat Groves http://matgroves.com/ @Doormat23\n * based on pixi impact spine implementation made by Eemeli Kelokorpi (@ekelokorpi) https://github.com/ekelokorpi\n *\n * Awesome JS run time provided by EsotericSoftware\n * https://github.com/EsotericSoftware/spine-runtimes\n *\n */\n\n/**\n * The Spine loader is used to load in JSON spine data\n * To generate the data you need to use http://esotericsoftware.com/ and export in the \"JSON\" format\n * Due to a clash of names  You will need to change the extension of the spine file from *.json to *.anim for it to load\n * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source\n * You will need to generate a sprite sheet to accompany the spine data\n * When loaded this class will dispatch a \"loaded\" event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction SpineLoader(url, crossorigin) {\n    /**\n     * The url of the bitmap font data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * Whether the data has loaded yet\n     *\n     * @member {boolean}\n     * @readOnly\n     */\n    this.loaded = false;\n}\n\nSpineLoader.prototype.constructor = SpineLoader;\nmodule.exports = SpineLoader;\n\ncore.utils.eventTarget.mixin(SpineLoader.prototype);\n\n/**\n * Loads the JSON data\n *\n */\nSpineLoader.prototype.load = function () {\n    var scope = this;\n    var jsonLoader = new JsonLoader(this.url, this.crossorigin);\n\n    jsonLoader.on('loaded', function (event) {\n        scope.json = event.data.content.json;\n        scope.onLoaded();\n    });\n\n    jsonLoader.load();\n};\n\n/**\n * Invoked when JSON file is loaded.\n *\n * @private\n */\nSpineLoader.prototype.onLoaded = function () {\n    this.loaded = true;\n    this.emit('loaded', { content: this });\n};\n","var core = require('../core');\n\n/**\n * Supporting class to load images from spine atlases as per spine spec.\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param basePath {string} Tha base path where to look for the images to be loaded\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction SpineTextureLoader(basePath, crossorigin) {\n    this.basePath = basePath;\n    this.crossorigin = crossorigin;\n    this.loadingCount = 0;\n}\n\nSpineTextureLoader.prototype.constructor = SpineTextureLoader;\nmodule.exports = SpineTextureLoader;\n\ncore.utils.eventTarget.mixin(SpineTextureLoader.prototype);\n\n/**\n * Starts loading a base texture as per spine specification\n *\n * @param page {spine.AtlasPage} Atlas page to which texture belongs\n * @param file {string} The file to load, this is just the file path relative to the base path configured in the constructor\n */\nSpineTextureLoader.prototype.load = function (page, file) {\n    page.rendererObject = core.BaseTexture.fromImage(this.basePath + '/' + file, this.crossorigin);\n    if (!page.rendererObject.hasLoaded) {\n        var scope = this;\n        ++scope.loadingCount;\n        page.rendererObject.addEventListener('loaded', function (){\n            --scope.loadingCount;\n            scope.dispatchEvent({\n                type: 'loadedBaseTexture',\n                content: scope\n            });\n        });\n    }\n};\n\n/**\n * Unloads a previously loaded texture as per spine specification\n *\n * @param texture {BaseTexture} Texture object to destroy\n */\nSpineTextureLoader.prototype.unload = function (texture) {\n    texture.destroy(true);\n};\n","var core = require('../core'),\n    JsonLoader = require('./JsonLoader');\n\n/**\n * The sprite sheet loader is used to load in JSON sprite sheet data\n * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format\n * There is a free version so thats nice, although the paid version is great value for money.\n * It is highly recommended to use Sprite sheets (also know as a 'texture atlas') as it means sprites can be batched and drawn together for highly increased rendering speed.\n * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId()\n * This loader will load the image file that the Spritesheet points to as well as the data.\n * When loaded this class will dispatch a 'loaded' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the sprite sheet JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction SpriteSheetLoader(url, crossorigin) {\n\n    /**\n     * The url of the atlas data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * The base url of the bitmap font data\n     *\n     * @member {String}\n     * @readOnly\n     */\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n\n    /**\n     * The texture being loaded\n     *\n     * @member {Texture}\n     */\n    this.texture = null;\n\n    /**\n     * The frames of the sprite sheet\n     *\n     * @member {object}\n     */\n    this.frames = {};\n}\n\n// constructor\nSpriteSheetLoader.prototype.constructor = SpriteSheetLoader;\nmodule.exports = SpriteSheetLoader;\n\ncore.utils.eventTarget.mixin(SpriteSheetLoader.prototype);\n\n/**\n * This will begin loading the JSON file\n *\n */\nSpriteSheetLoader.prototype.load = function () {\n    var scope = this;\n    var jsonLoader = new JsonLoader(this.url, this.crossorigin);\n\n    jsonLoader.on('loaded', function (event) {\n        scope.json = event.data.content.json;\n        scope.onLoaded();\n    });\n\n    jsonLoader.load();\n};\n\n/**\n * Invoke when all files are loaded (json and texture)\n *\n * @private\n */\nSpriteSheetLoader.prototype.onLoaded = function () {\n    this.emit('loaded', {\n        content: this\n    });\n};\n","/**\n * @file        Main export of the PIXI loaders library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    AtlasLoader:        require('./AtlasLoader'),\n    BitmapFontLoader:   require('./BitmapFontLoader'),\n    ImageLoader:        require('./ImageLoader'),\n    JsonLoader:         require('./JsonLoader'),\n    SpineLoader:        require('./SpineLoader'),\n    SpriteSheetLoader:  require('./SpriteSheetLoader')\n};\n","var core = require('../core'),\n    spine = require('./SpineRuntime');\n\n/* Esoteric Software SPINE wrapper for pixi.js */\n\nspine.Bone.yDown = true;\n\n/**\n * A class that enables the you to import and run your spine animations in pixi.\n * Spine animation data needs to be loaded using the AssetLoader or SpineLoader before it can be used by this class\n * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param url {string} The url of the spine anim file to be used\n */\nfunction Spine(url) {\n    core.DisplayObjectContainer.call(this);\n\n    this.spineData = core.utils.AnimCache[url];\n\n    if (!this.spineData) {\n        throw new Error('Spine data must be preloaded using SpineLoader or AssetLoader: ' + url);\n    }\n\n    this.skeleton = new spine.Skeleton(this.spineData);\n    this.skeleton.updateWorldTransform();\n\n    this.stateData = new spine.AnimationStateData(this.spineData);\n    this.state = new spine.AnimationState(this.stateData);\n\n    this.slotContainers = [];\n\n    for (var i = 0, n = this.skeleton.drawOrder.length; i < n; i++) {\n        var slot = this.skeleton.drawOrder[i];\n        var attachment = slot.attachment;\n        var slotContainer = new core.DisplayObjectContainer();\n        this.slotContainers.push(slotContainer);\n        this.addChild(slotContainer);\n\n        if (attachment instanceof spine.RegionAttachment) {\n            var spriteName = attachment.rendererObject.name;\n            var sprite = this.createSprite(slot, attachment);\n            slot.currentSprite = sprite;\n            slot.currentSpriteName = spriteName;\n            slotContainer.addChild(sprite);\n        }\n        else if (attachment instanceof spine.MeshAttachment) {\n            var mesh = this.createMesh(slot, attachment);\n            slot.currentMesh = mesh;\n            slot.currentMeshName = attachment.name;\n            slotContainer.addChild(mesh);\n        }\n        else {\n            continue;\n        }\n\n    }\n\n    this.autoUpdate = true;\n}\n\nSpine.prototype = Object.create(core.DisplayObjectContainer.prototype);\nSpine.prototype.constructor = Spine;\nmodule.exports = Spine;\n\nObject.defineProperties(Spine.prototype, {\n    /**\n     * If this flag is set to true, the spine animation will be autoupdated every time\n     * the object id drawn. The down side of this approach is that the delta time is\n     * automatically calculated and you could miss out on cool effects like slow motion,\n     * pause, skip ahead and the sorts. Most of these effects can be achieved even with\n     * autoupdate enabled but are harder to achieve.\n     *\n     * @member {boolean}\n     * @memberof Spine#\n     * @default true\n     */\n    autoUpdate: {\n        get: function () {\n            return (this.updateTransform === Spine.prototype.autoUpdateTransform);\n        },\n\n        set: function (value) {\n            this.updateTransform = value ? Spine.prototype.autoUpdateTransform : core.DisplayObjectContainer.prototype.updateTransform;\n        }\n    }\n});\n\n/**\n * Update the spine skeleton and its animations by delta time (dt)\n *\n * @param dt {number} Delta time. Time by which the animation should be updated\n */\nSpine.prototype.update = function (dt) {\n    this.state.update(dt);\n    this.state.apply(this.skeleton);\n    this.skeleton.updateWorldTransform();\n\n    var drawOrder = this.skeleton.drawOrder;\n    for (var i = 0, n = drawOrder.length; i < n; i++) {\n        var slot = drawOrder[i];\n        var attachment = slot.attachment;\n        var slotContainer = this.slotContainers[i];\n\n        if (!attachment) {\n            slotContainer.visible = false;\n            continue;\n        }\n\n        var type = attachment.type;\n        if (type === spine.AttachmentType.region) {\n            if (attachment.rendererObject) {\n                if (!slot.currentSpriteName || slot.currentSpriteName !== attachment.rendererObject.name) {\n                    var spriteName = attachment.rendererObject.name;\n                    if (slot.currentSprite !== undefined) {\n                        slot.currentSprite.visible = false;\n                    }\n                    slot.sprites = slot.sprites || {};\n                    if (slot.sprites[spriteName] !== undefined) {\n                        slot.sprites[spriteName].visible = true;\n                    }\n                    else {\n                        var sprite = this.createSprite(slot, attachment);\n                        slotContainer.addChild(sprite);\n                    }\n                    slot.currentSprite = slot.sprites[spriteName];\n                    slot.currentSpriteName = spriteName;\n                }\n            }\n\n            var bone = slot.bone;\n\n            slotContainer.position.x = bone.worldX + attachment.x * bone.m00 + attachment.y * bone.m01;\n            slotContainer.position.y = bone.worldY + attachment.x * bone.m10 + attachment.y * bone.m11;\n            slotContainer.scale.x = bone.worldScaleX;\n            slotContainer.scale.y = bone.worldScaleY;\n\n            slotContainer.rotation = -(slot.bone.worldRotation * spine.degRad);\n\n            slot.currentSprite.tint = core.utils.rgb2hex([slot.r,slot.g,slot.b]);\n        }\n        else if (type === spine.AttachmentType.skinnedmesh) {\n            if (!slot.currentMeshName || slot.currentMeshName !== attachment.name) {\n                var meshName = attachment.name;\n                if (slot.currentMesh !== undefined) {\n                    slot.currentMesh.visible = false;\n                }\n\n                slot.meshes = slot.meshes || {};\n\n                if (slot.meshes[meshName] !== undefined) {\n                    slot.meshes[meshName].visible = true;\n                }\n                else {\n                    var mesh = this.createMesh(slot, attachment);\n                    slotContainer.addChild(mesh);\n                }\n\n                slot.currentMesh = slot.meshes[meshName];\n                slot.currentMeshName = meshName;\n            }\n\n            attachment.computeWorldVertices(slot.bone.skeleton.x, slot.bone.skeleton.y, slot, slot.currentMesh.vertices);\n\n        }\n        else {\n            slotContainer.visible = false;\n            continue;\n        }\n        slotContainer.visible = true;\n\n        slotContainer.alpha = slot.a;\n    }\n};\n\n/**\n * When autoupdate is set to yes this function is used as pixi's updateTransform function\n *\n * @private\n */\nSpine.prototype.autoUpdateTransform = function () {\n    this.lastTime = this.lastTime || Date.now();\n    var timeDelta = (Date.now() - this.lastTime) * 0.001;\n    this.lastTime = Date.now();\n\n    this.update(timeDelta);\n\n    core.DisplayObjectContainer.prototype.updateTransform.call(this);\n};\n\n/**\n * Create a new sprite to be used with spine.RegionAttachment\n *\n * @param slot {spine.Slot} The slot to which the attachment is parented\n * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent\n * @private\n */\nSpine.prototype.createSprite = function (slot, attachment) {\n    var descriptor = attachment.rendererObject;\n    var baseTexture = descriptor.page.rendererObject;\n    var spriteRect = new core.math.Rectangle(descriptor.x,\n                                        descriptor.y,\n                                        descriptor.rotate ? descriptor.height : descriptor.width,\n                                        descriptor.rotate ? descriptor.width : descriptor.height);\n    var spriteTexture = new core.Texture(baseTexture, spriteRect);\n    var sprite = new core.Sprite(spriteTexture);\n\n    var baseRotation = descriptor.rotate ? Math.PI * 0.5 : 0.0;\n    sprite.scale.set(descriptor.width / descriptor.originalWidth, descriptor.height / descriptor.originalHeight);\n    sprite.rotation = baseRotation - (attachment.rotation * spine.degRad);\n    sprite.anchor.x = sprite.anchor.y = 0.5;\n\n    slot.sprites = slot.sprites || {};\n    slot.sprites[descriptor.name] = sprite;\n    return sprite;\n};\n\n/**\n *\n * @param slot {spine.Slot} The slot to which the attachment is parented\n * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent\n * @private\n */\nSpine.prototype.createMesh = function (slot, attachment) {\n    var descriptor = attachment.rendererObject;\n    var baseTexture = descriptor.page.rendererObject;\n    var texture = new core.Texture(baseTexture);\n\n    var strip = new core.Strip(texture);\n    strip.drawMode = core.Strip.DrawModes.TRIANGLES;\n    strip.canvasPadding = 1.5;\n\n    strip.vertices = new Float32Array(attachment.uvs.length);\n    strip.uvs = attachment.uvs;\n    strip.indices = attachment.triangles;\n\n    slot.meshes = slot.meshes || {};\n    slot.meshes[attachment.name] = strip;\n\n    return strip;\n};\n","/******************************************************************************\n * Spine Runtimes Software License\n * Version 2.1\n *\n * Copyright (c) 2013, Esoteric Software\n * All rights reserved.\n *\n * You are granted a perpetual, non-exclusive, non-sublicensable and\n * non-transferable license to install, execute and perform the Spine Runtimes\n * Software (the \"Software\") solely for internal use. Without the written\n * permission of Esoteric Software (typically granted by licensing Spine), you\n * may not (a) modify, translate, adapt or otherwise create derivative works,\n * improvements of the Software or develop new applications using the Software\n * or (b) remove, delete, alter or obscure any trademarks or any copyright,\n * trademark, patent or other intellectual property or proprietary rights\n * notices on or in the Software, including any copy thereof. Redistributions\n * in binary or source form must include this license and terms.\n *\n * THIS SOFTWARE IS PROVIDED BY ESOTERIC SOFTWARE \"AS IS\" AND ANY EXPRESS OR\n * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF\n * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO\n * EVENT SHALL ESOTERIC SOFTARE BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,\n * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,\n * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;\n * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,\n * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR\n * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF\n * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n *****************************************************************************/\n\nvar spine = module.exports = {\n\tradDeg: 180 / Math.PI,\n\tdegRad: Math.PI / 180,\n\ttemp: [],\n    Float32Array: (typeof(Float32Array) === 'undefined') ? Array : Float32Array,\n    Uint16Array: (typeof(Uint16Array) === 'undefined') ? Array : Uint16Array\n};\n\nspine.BoneData = function (name, parent) {\n\tthis.name = name;\n\tthis.parent = parent;\n};\nspine.BoneData.prototype = {\n\tlength: 0,\n\tx: 0, y: 0,\n\trotation: 0,\n\tscaleX: 1, scaleY: 1,\n\tinheritScale: true,\n\tinheritRotation: true,\n\tflipX: false, flipY: false\n};\n\nspine.SlotData = function (name, boneData) {\n\tthis.name = name;\n\tthis.boneData = boneData;\n};\nspine.SlotData.prototype = {\n\tr: 1, g: 1, b: 1, a: 1,\n\tattachmentName: null,\n\tadditiveBlending: false\n};\n\nspine.IkConstraintData = function (name) {\n\tthis.name = name;\n\tthis.bones = [];\n};\nspine.IkConstraintData.prototype = {\n\ttarget: null,\n\tbendDirection: 1,\n\tmix: 1\n};\n\nspine.Bone = function (boneData, skeleton, parent) {\n\tthis.data = boneData;\n\tthis.skeleton = skeleton;\n\tthis.parent = parent;\n\tthis.setToSetupPose();\n};\nspine.Bone.yDown = false;\nspine.Bone.prototype = {\n\tx: 0, y: 0,\n\trotation: 0, rotationIK: 0,\n\tscaleX: 1, scaleY: 1,\n\tflipX: false, flipY: false,\n\tm00: 0, m01: 0, worldX: 0, // a b x\n\tm10: 0, m11: 0, worldY: 0, // c d y\n\tworldRotation: 0,\n\tworldScaleX: 1, worldScaleY: 1,\n\tworldFlipX: false, worldFlipY: false,\n\tupdateWorldTransform: function () {\n\t\tvar parent = this.parent;\n\t\tif (parent) {\n\t\t\tthis.worldX = this.x * parent.m00 + this.y * parent.m01 + parent.worldX;\n\t\t\tthis.worldY = this.x * parent.m10 + this.y * parent.m11 + parent.worldY;\n\t\t\tif (this.data.inheritScale) {\n\t\t\t\tthis.worldScaleX = parent.worldScaleX * this.scaleX;\n\t\t\t\tthis.worldScaleY = parent.worldScaleY * this.scaleY;\n\t\t\t} else {\n\t\t\t\tthis.worldScaleX = this.scaleX;\n\t\t\t\tthis.worldScaleY = this.scaleY;\n\t\t\t}\n\t\t\tthis.worldRotation = this.data.inheritRotation ? (parent.worldRotation + this.rotationIK) : this.rotationIK;\n\t\t\tthis.worldFlipX = parent.worldFlipX != this.flipX;\n\t\t\tthis.worldFlipY = parent.worldFlipY != this.flipY;\n\t\t} else {\n\t\t\tvar skeletonFlipX = this.skeleton.flipX, skeletonFlipY = this.skeleton.flipY;\n\t\t\tthis.worldX = skeletonFlipX ? -this.x : this.x;\n\t\t\tthis.worldY = (skeletonFlipY != spine.Bone.yDown) ? -this.y : this.y;\n\t\t\tthis.worldScaleX = this.scaleX;\n\t\t\tthis.worldScaleY = this.scaleY;\n\t\t\tthis.worldRotation = this.rotationIK;\n\t\t\tthis.worldFlipX = skeletonFlipX != this.flipX;\n\t\t\tthis.worldFlipY = skeletonFlipY != this.flipY;\n\t\t}\n\t\tvar radians = this.worldRotation * spine.degRad;\n\t\tvar cos = Math.cos(radians);\n\t\tvar sin = Math.sin(radians);\n\t\tif (this.worldFlipX) {\n\t\t\tthis.m00 = -cos * this.worldScaleX;\n\t\t\tthis.m01 = sin * this.worldScaleY;\n\t\t} else {\n\t\t\tthis.m00 = cos * this.worldScaleX;\n\t\t\tthis.m01 = -sin * this.worldScaleY;\n\t\t}\n\t\tif (this.worldFlipY != spine.Bone.yDown) {\n\t\t\tthis.m10 = -sin * this.worldScaleX;\n\t\t\tthis.m11 = -cos * this.worldScaleY;\n\t\t} else {\n\t\t\tthis.m10 = sin * this.worldScaleX;\n\t\t\tthis.m11 = cos * this.worldScaleY;\n\t\t}\n\t},\n\tsetToSetupPose: function () {\n\t\tvar data = this.data;\n\t\tthis.x = data.x;\n\t\tthis.y = data.y;\n\t\tthis.rotation = data.rotation;\n\t\tthis.rotationIK = this.rotation;\n\t\tthis.scaleX = data.scaleX;\n\t\tthis.scaleY = data.scaleY;\n\t\tthis.flipX = data.flipX;\n\t\tthis.flipY = data.flipY;\n\t},\n\tworldToLocal: function (world) {\n\t\tvar dx = world[0] - this.worldX, dy = world[1] - this.worldY;\n\t\tvar m00 = this.m00, m10 = this.m10, m01 = this.m01, m11 = this.m11;\n\t\tif (this.worldFlipX != (this.worldFlipY != spine.Bone.yDown)) {\n\t\t\tm00 = -m00;\n\t\t\tm11 = -m11;\n\t\t}\n\t\tvar invDet = 1 / (m00 * m11 - m01 * m10);\n\t\tworld[0] = dx * m00 * invDet - dy * m01 * invDet;\n\t\tworld[1] = dy * m11 * invDet - dx * m10 * invDet;\n\t},\n\tlocalToWorld: function (local) {\n\t\tvar localX = local[0], localY = local[1];\n\t\tlocal[0] = localX * this.m00 + localY * this.m01 + this.worldX;\n\t\tlocal[1] = localX * this.m10 + localY * this.m11 + this.worldY;\n\t}\n};\n\nspine.Slot = function (slotData, bone) {\n\tthis.data = slotData;\n\tthis.bone = bone;\n\tthis.setToSetupPose();\n};\nspine.Slot.prototype = {\n\tr: 1, g: 1, b: 1, a: 1,\n\t_attachmentTime: 0,\n\tattachment: null,\n\tattachmentVertices: [],\n\tsetAttachment: function (attachment) {\n\t\tthis.attachment = attachment;\n\t\tthis._attachmentTime = this.bone.skeleton.time;\n\t\tthis.attachmentVertices.length = 0;\n\t},\n\tsetAttachmentTime: function (time) {\n\t\tthis._attachmentTime = this.bone.skeleton.time - time;\n\t},\n\tgetAttachmentTime: function () {\n\t\treturn this.bone.skeleton.time - this._attachmentTime;\n\t},\n\tsetToSetupPose: function () {\n\t\tvar data = this.data;\n\t\tthis.r = data.r;\n\t\tthis.g = data.g;\n\t\tthis.b = data.b;\n\t\tthis.a = data.a;\n\n\t\tvar slotDatas = this.bone.skeleton.data.slots;\n\t\tfor (var i = 0, n = slotDatas.length; i < n; i++) {\n\t\t\tif (slotDatas[i] == data) {\n\t\t\t\tthis.setAttachment(!data.attachmentName ? null : this.bone.skeleton.getAttachmentBySlotIndex(i, data.attachmentName));\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.IkConstraint = function (data, skeleton) {\n\tthis.data = data;\n\tthis.mix = data.mix;\n\tthis.bendDirection = data.bendDirection;\n\n\tthis.bones = [];\n\tfor (var i = 0, n = data.bones.length; i < n; i++)\n\t\tthis.bones.push(skeleton.findBone(data.bones[i].name));\n\tthis.target = skeleton.findBone(data.target.name);\n};\nspine.IkConstraint.prototype = {\n\tapply: function () {\n\t\tvar target = this.target;\n\t\tvar bones = this.bones;\n\t\tswitch (bones.length) {\n\t\tcase 1:\n\t\t\tspine.IkConstraint.apply1(bones[0], target.worldX, target.worldY, this.mix);\n\t\t\tbreak;\n\t\tcase 2:\n\t\t\tspine.IkConstraint.apply2(bones[0], bones[1], target.worldX, target.worldY, this.bendDirection, this.mix);\n\t\t\tbreak;\n\t\t}\n\t}\n};\n/** Adjusts the bone rotation so the tip is as close to the target position as possible. The target is specified in the world\n * coordinate system. */\nspine.IkConstraint.apply1 = function (bone, targetX, targetY, alpha) {\n\tvar parentRotation = (!bone.data.inheritRotation || !bone.parent) ? 0 : bone.parent.worldRotation;\n\tvar rotation = bone.rotation;\n\tvar rotationIK = Math.atan2(targetY - bone.worldY, targetX - bone.worldX) * spine.radDeg - parentRotation;\n\tbone.rotationIK = rotation + (rotationIK - rotation) * alpha;\n};\n/** Adjusts the parent and child bone rotations so the tip of the child is as close to the target position as possible. The\n * target is specified in the world coordinate system.\n * @param child Any descendant bone of the parent. */\nspine.IkConstraint.apply2 = function (parent, child, targetX, targetY, bendDirection, alpha) {\n\tvar childRotation = child.rotation, parentRotation = parent.rotation;\n\tif (!alpha) {\n\t\tchild.rotationIK = childRotation;\n\t\tparent.rotationIK = parentRotation;\n\t\treturn;\n\t}\n\tvar positionX, positionY, tempPosition = spine.temp;\n\tvar parentParent = parent.parent;\n\tif (parentParent) {\n\t\ttempPosition[0] = targetX;\n\t\ttempPosition[1] = targetY;\n\t\tparentParent.worldToLocal(tempPosition);\n\t\ttargetX = (tempPosition[0] - parent.x) * parentParent.worldScaleX;\n\t\ttargetY = (tempPosition[1] - parent.y) * parentParent.worldScaleY;\n\t} else {\n\t\ttargetX -= parent.x;\n\t\ttargetY -= parent.y;\n\t}\n\tif (child.parent == parent) {\n\t\tpositionX = child.x;\n\t\tpositionY = child.y;\n\t} else {\n\t\ttempPosition[0] = child.x;\n\t\ttempPosition[1] = child.y;\n\t\tchild.parent.localToWorld(tempPosition);\n\t\tparent.worldToLocal(tempPosition);\n\t\tpositionX = tempPosition[0];\n\t\tpositionY = tempPosition[1];\n\t}\n\tvar childX = positionX * parent.worldScaleX, childY = positionY * parent.worldScaleY;\n\tvar offset = Math.atan2(childY, childX);\n\tvar len1 = Math.sqrt(childX * childX + childY * childY), len2 = child.data.length * child.worldScaleX;\n\t// Based on code by Ryan Juckett with permission: Copyright (c) 2008-2009 Ryan Juckett, http://www.ryanjuckett.com/\n\tvar cosDenom = 2 * len1 * len2;\n\tif (cosDenom < 0.0001) {\n\t\tchild.rotationIK = childRotation + (Math.atan2(targetY, targetX) * spine.radDeg - parentRotation - childRotation) * alpha;\n\t\treturn;\n\t}\n\tvar cos = (targetX * targetX + targetY * targetY - len1 * len1 - len2 * len2) / cosDenom;\n\tif (cos < -1)\n\t\tcos = -1;\n\telse if (cos > 1)\n\t\tcos = 1;\n\tvar childAngle = Math.acos(cos) * bendDirection;\n\tvar adjacent = len1 + len2 * cos, opposite = len2 * Math.sin(childAngle);\n\tvar parentAngle = Math.atan2(targetY * adjacent - targetX * opposite, targetX * adjacent + targetY * opposite);\n\tvar rotation = (parentAngle - offset) * spine.radDeg - parentRotation;\n\tif (rotation > 180)\n\t\trotation -= 360;\n\telse if (rotation < -180) //\n\t\trotation += 360;\n\tparent.rotationIK = parentRotation + rotation * alpha;\n\trotation = (childAngle + offset) * spine.radDeg - childRotation;\n\tif (rotation > 180)\n\t\trotation -= 360;\n\telse if (rotation < -180) //\n\t\trotation += 360;\n\tchild.rotationIK = childRotation + (rotation + parent.worldRotation - child.parent.worldRotation) * alpha;\n};\n\nspine.Skin = function (name) {\n\tthis.name = name;\n\tthis.attachments = {};\n};\nspine.Skin.prototype = {\n\taddAttachment: function (slotIndex, name, attachment) {\n\t\tthis.attachments[slotIndex + \":\" + name] = attachment;\n\t},\n\tgetAttachment: function (slotIndex, name) {\n\t\treturn this.attachments[slotIndex + \":\" + name];\n\t},\n\t_attachAll: function (skeleton, oldSkin) {\n\t\tfor (var key in oldSkin.attachments) {\n\t\t\tvar colon = key.indexOf(\":\");\n\t\t\tvar slotIndex = parseInt(key.substring(0, colon));\n\t\t\tvar name = key.substring(colon + 1);\n\t\t\tvar slot = skeleton.slots[slotIndex];\n\t\t\tif (slot.attachment && slot.attachment.name == name) {\n\t\t\t\tvar attachment = this.getAttachment(slotIndex, name);\n\t\t\t\tif (attachment) slot.setAttachment(attachment);\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.Animation = function (name, timelines, duration) {\n\tthis.name = name;\n\tthis.timelines = timelines;\n\tthis.duration = duration;\n};\nspine.Animation.prototype = {\n\tapply: function (skeleton, lastTime, time, loop, events) {\n\t\tif (loop && this.duration != 0) {\n\t\t\ttime %= this.duration;\n\t\t\tlastTime %= this.duration;\n\t\t}\n\t\tvar timelines = this.timelines;\n\t\tfor (var i = 0, n = timelines.length; i < n; i++)\n\t\t\ttimelines[i].apply(skeleton, lastTime, time, events, 1);\n\t},\n\tmix: function (skeleton, lastTime, time, loop, events, alpha) {\n\t\tif (loop && this.duration != 0) {\n\t\t\ttime %= this.duration;\n\t\t\tlastTime %= this.duration;\n\t\t}\n\t\tvar timelines = this.timelines;\n\t\tfor (var i = 0, n = timelines.length; i < n; i++)\n\t\t\ttimelines[i].apply(skeleton, lastTime, time, events, alpha);\n\t}\n};\nspine.Animation.binarySearch = function (values, target, step) {\n\tvar low = 0;\n\tvar high = Math.floor(values.length / step) - 2;\n\tif (!high) return step;\n\tvar current = high >>> 1;\n\twhile (true) {\n\t\tif (values[(current + 1) * step] <= target)\n\t\t\tlow = current + 1;\n\t\telse\n\t\t\thigh = current;\n\t\tif (low == high) return (low + 1) * step;\n\t\tcurrent = (low + high) >>> 1;\n\t}\n};\nspine.Animation.binarySearch1 = function (values, target) {\n\tvar low = 0;\n\tvar high = values.length - 2;\n\tif (!high) return 1;\n\tvar current = high >>> 1;\n\twhile (true) {\n\t\tif (values[current + 1] <= target)\n\t\t\tlow = current + 1;\n\t\telse\n\t\t\thigh = current;\n\t\tif (low == high) return low + 1;\n\t\tcurrent = (low + high) >>> 1;\n\t}\n};\nspine.Animation.linearSearch = function (values, target, step) {\n\tfor (var i = 0, last = values.length - step; i <= last; i += step)\n\t\tif (values[i] > target) return i;\n\treturn -1;\n};\n\nspine.Curves = function (frameCount) {\n\tthis.curves = []; // type, x, y, ...\n\t//this.curves.length = (frameCount - 1) * 19/*BEZIER_SIZE*/;\n};\nspine.Curves.prototype = {\n\tsetLinear: function (frameIndex) {\n\t\tthis.curves[frameIndex * 19/*BEZIER_SIZE*/] = 0/*LINEAR*/;\n\t},\n\tsetStepped: function (frameIndex) {\n\t\tthis.curves[frameIndex * 19/*BEZIER_SIZE*/] = 1/*STEPPED*/;\n\t},\n\t/** Sets the control handle positions for an interpolation bezier curve used to transition from this keyframe to the next.\n\t * cx1 and cx2 are from 0 to 1, representing the percent of time between the two keyframes. cy1 and cy2 are the percent of\n\t * the difference between the keyframe's values. */\n\tsetCurve: function (frameIndex, cx1, cy1, cx2, cy2) {\n\t\tvar subdiv1 = 1 / 10/*BEZIER_SEGMENTS*/, subdiv2 = subdiv1 * subdiv1, subdiv3 = subdiv2 * subdiv1;\n\t\tvar pre1 = 3 * subdiv1, pre2 = 3 * subdiv2, pre4 = 6 * subdiv2, pre5 = 6 * subdiv3;\n\t\tvar tmp1x = -cx1 * 2 + cx2, tmp1y = -cy1 * 2 + cy2, tmp2x = (cx1 - cx2) * 3 + 1, tmp2y = (cy1 - cy2) * 3 + 1;\n\t\tvar dfx = cx1 * pre1 + tmp1x * pre2 + tmp2x * subdiv3, dfy = cy1 * pre1 + tmp1y * pre2 + tmp2y * subdiv3;\n\t\tvar ddfx = tmp1x * pre4 + tmp2x * pre5, ddfy = tmp1y * pre4 + tmp2y * pre5;\n\t\tvar dddfx = tmp2x * pre5, dddfy = tmp2y * pre5;\n\n\t\tvar i = frameIndex * 19/*BEZIER_SIZE*/;\n\t\tvar curves = this.curves;\n\t\tcurves[i++] = 2/*BEZIER*/;\n\n\t\tvar x = dfx, y = dfy;\n\t\tfor (var n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) {\n\t\t\tcurves[i] = x;\n\t\t\tcurves[i + 1] = y;\n\t\t\tdfx += ddfx;\n\t\t\tdfy += ddfy;\n\t\t\tddfx += dddfx;\n\t\t\tddfy += dddfy;\n\t\t\tx += dfx;\n\t\t\ty += dfy;\n\t\t}\n\t},\n\tgetCurvePercent: function (frameIndex, percent) {\n\t\tpercent = percent < 0 ? 0 : (percent > 1 ? 1 : percent);\n\t\tvar curves = this.curves;\n\t\tvar i = frameIndex * 19/*BEZIER_SIZE*/;\n\t\tvar type = curves[i];\n\t\tif (type === 0/*LINEAR*/) return percent;\n\t\tif (type == 1/*STEPPED*/) return 0;\n\t\ti++;\n\t\tvar x = 0;\n\t\tfor (var start = i, n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) {\n\t\t\tx = curves[i];\n\t\t\tif (x >= percent) {\n\t\t\t\tvar prevX, prevY;\n\t\t\t\tif (i == start) {\n\t\t\t\t\tprevX = 0;\n\t\t\t\t\tprevY = 0;\n\t\t\t\t} else {\n\t\t\t\t\tprevX = curves[i - 2];\n\t\t\t\t\tprevY = curves[i - 1];\n\t\t\t\t}\n\t\t\t\treturn prevY + (curves[i + 1] - prevY) * (percent - prevX) / (x - prevX);\n\t\t\t}\n\t\t}\n\t\tvar y = curves[i - 1];\n\t\treturn y + (1 - y) * (percent - x) / (1 - x); // Last point is 1,1.\n\t}\n};\n\nspine.RotateTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, angle, ...\n\tthis.frames.length = frameCount * 2;\n};\nspine.RotateTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 2;\n\t},\n\tsetFrame: function (frameIndex, time, angle) {\n\t\tframeIndex *= 2;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = angle;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar bone = skeleton.bones[this.boneIndex];\n\n\t\tif (time >= frames[frames.length - 2]) { // Time is after last frame.\n\t\t\tvar amount = bone.data.rotation + frames[frames.length - 1] - bone.rotation;\n\t\t\twhile (amount > 180)\n\t\t\t\tamount -= 360;\n\t\t\twhile (amount < -180)\n\t\t\t\tamount += 360;\n\t\t\tbone.rotation += amount * alpha;\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 2);\n\t\tvar prevFrameValue = frames[frameIndex - 1];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex - 2/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 2 - 1, percent);\n\n\t\tvar amount = frames[frameIndex + 1/*FRAME_VALUE*/] - prevFrameValue;\n\t\twhile (amount > 180)\n\t\t\tamount -= 360;\n\t\twhile (amount < -180)\n\t\t\tamount += 360;\n\t\tamount = bone.data.rotation + (prevFrameValue + amount * percent) - bone.rotation;\n\t\twhile (amount > 180)\n\t\t\tamount -= 360;\n\t\twhile (amount < -180)\n\t\t\tamount += 360;\n\t\tbone.rotation += amount * alpha;\n\t}\n};\n\nspine.TranslateTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, x, y, ...\n\tthis.frames.length = frameCount * 3;\n};\nspine.TranslateTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 3;\n\t},\n\tsetFrame: function (frameIndex, time, x, y) {\n\t\tframeIndex *= 3;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = x;\n\t\tthis.frames[frameIndex + 2] = y;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar bone = skeleton.bones[this.boneIndex];\n\n\t\tif (time >= frames[frames.length - 3]) { // Time is after last frame.\n\t\t\tbone.x += (bone.data.x + frames[frames.length - 2] - bone.x) * alpha;\n\t\t\tbone.y += (bone.data.y + frames[frames.length - 1] - bone.y) * alpha;\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 3);\n\t\tvar prevFrameX = frames[frameIndex - 2];\n\t\tvar prevFrameY = frames[frameIndex - 1];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent);\n\n\t\tbone.x += (bone.data.x + prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent - bone.x) * alpha;\n\t\tbone.y += (bone.data.y + prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent - bone.y) * alpha;\n\t}\n};\n\nspine.ScaleTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, x, y, ...\n\tthis.frames.length = frameCount * 3;\n};\nspine.ScaleTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 3;\n\t},\n\tsetFrame: function (frameIndex, time, x, y) {\n\t\tframeIndex *= 3;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = x;\n\t\tthis.frames[frameIndex + 2] = y;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar bone = skeleton.bones[this.boneIndex];\n\n\t\tif (time >= frames[frames.length - 3]) { // Time is after last frame.\n\t\t\tbone.scaleX += (bone.data.scaleX * frames[frames.length - 2] - bone.scaleX) * alpha;\n\t\t\tbone.scaleY += (bone.data.scaleY * frames[frames.length - 1] - bone.scaleY) * alpha;\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 3);\n\t\tvar prevFrameX = frames[frameIndex - 2];\n\t\tvar prevFrameY = frames[frameIndex - 1];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent);\n\n\t\tbone.scaleX += (bone.data.scaleX * (prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent) - bone.scaleX) * alpha;\n\t\tbone.scaleY += (bone.data.scaleY * (prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent) - bone.scaleY) * alpha;\n\t}\n};\n\nspine.ColorTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, r, g, b, a, ...\n\tthis.frames.length = frameCount * 5;\n};\nspine.ColorTimeline.prototype = {\n\tslotIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 5;\n\t},\n\tsetFrame: function (frameIndex, time, r, g, b, a) {\n\t\tframeIndex *= 5;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = r;\n\t\tthis.frames[frameIndex + 2] = g;\n\t\tthis.frames[frameIndex + 3] = b;\n\t\tthis.frames[frameIndex + 4] = a;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar r, g, b, a;\n\t\tif (time >= frames[frames.length - 5]) {\n\t\t\t// Time is after last frame.\n\t\t\tvar i = frames.length - 1;\n\t\t\tr = frames[i - 3];\n\t\t\tg = frames[i - 2];\n\t\t\tb = frames[i - 1];\n\t\t\ta = frames[i];\n\t\t} else {\n\t\t\t// Interpolate between the previous frame and the current frame.\n\t\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 5);\n\t\t\tvar prevFrameR = frames[frameIndex - 4];\n\t\t\tvar prevFrameG = frames[frameIndex - 3];\n\t\t\tvar prevFrameB = frames[frameIndex - 2];\n\t\t\tvar prevFrameA = frames[frameIndex - 1];\n\t\t\tvar frameTime = frames[frameIndex];\n\t\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex - 5/*PREV_FRAME_TIME*/] - frameTime);\n\t\t\tpercent = this.curves.getCurvePercent(frameIndex / 5 - 1, percent);\n\n\t\t\tr = prevFrameR + (frames[frameIndex + 1/*FRAME_R*/] - prevFrameR) * percent;\n\t\t\tg = prevFrameG + (frames[frameIndex + 2/*FRAME_G*/] - prevFrameG) * percent;\n\t\t\tb = prevFrameB + (frames[frameIndex + 3/*FRAME_B*/] - prevFrameB) * percent;\n\t\t\ta = prevFrameA + (frames[frameIndex + 4/*FRAME_A*/] - prevFrameA) * percent;\n\t\t}\n\t\tvar slot = skeleton.slots[this.slotIndex];\n\t\tif (alpha < 1) {\n\t\t\tslot.r += (r - slot.r) * alpha;\n\t\t\tslot.g += (g - slot.g) * alpha;\n\t\t\tslot.b += (b - slot.b) * alpha;\n\t\t\tslot.a += (a - slot.a) * alpha;\n\t\t} else {\n\t\t\tslot.r = r;\n\t\t\tslot.g = g;\n\t\t\tslot.b = b;\n\t\t\tslot.a = a;\n\t\t}\n\t}\n};\n\nspine.AttachmentTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, ...\n\tthis.frames.length = frameCount;\n\tthis.attachmentNames = [];\n\tthis.attachmentNames.length = frameCount;\n};\nspine.AttachmentTimeline.prototype = {\n\tslotIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, attachmentName) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.attachmentNames[frameIndex] = attachmentName;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) {\n\t\t\tif (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0);\n\t\t\treturn;\n\t\t} else if (lastTime > time) //\n\t\t\tlastTime = -1;\n\n\t\tvar frameIndex = time >= frames[frames.length - 1] ? frames.length - 1 : spine.Animation.binarySearch1(frames, time) - 1;\n\t\tif (frames[frameIndex] < lastTime) return;\n\n\t\tvar attachmentName = this.attachmentNames[frameIndex];\n\t\tskeleton.slots[this.slotIndex].setAttachment(\n\t\t\t!attachmentName ? null : skeleton.getAttachmentBySlotIndex(this.slotIndex, attachmentName));\n\t}\n};\n\nspine.EventTimeline = function (frameCount) {\n\tthis.frames = []; // time, ...\n\tthis.frames.length = frameCount;\n\tthis.events = [];\n\tthis.events.length = frameCount;\n};\nspine.EventTimeline.prototype = {\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, event) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.events[frameIndex] = event;\n\t},\n\t/** Fires events for frames > lastTime and <= time. */\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tif (!firedEvents) return;\n\n\t\tvar frames = this.frames;\n\t\tvar frameCount = frames.length;\n\n\t\tif (lastTime > time) { // Fire events after last time for looped animations.\n\t\t\tthis.apply(skeleton, lastTime, Number.MAX_VALUE, firedEvents, alpha);\n\t\t\tlastTime = -1;\n\t\t} else if (lastTime >= frames[frameCount - 1]) // Last time is after last frame.\n\t\t\treturn;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar frameIndex;\n\t\tif (lastTime < frames[0])\n\t\t\tframeIndex = 0;\n\t\telse {\n\t\t\tframeIndex = spine.Animation.binarySearch1(frames, lastTime);\n\t\t\tvar frame = frames[frameIndex];\n\t\t\twhile (frameIndex > 0) { // Fire multiple events with the same frame.\n\t\t\t\tif (frames[frameIndex - 1] != frame) break;\n\t\t\t\tframeIndex--;\n\t\t\t}\n\t\t}\n\t\tvar events = this.events;\n\t\tfor (; frameIndex < frameCount && time >= frames[frameIndex]; frameIndex++)\n\t\t\tfiredEvents.push(events[frameIndex]);\n\t}\n};\n\nspine.DrawOrderTimeline = function (frameCount) {\n\tthis.frames = []; // time, ...\n\tthis.frames.length = frameCount;\n\tthis.drawOrders = [];\n\tthis.drawOrders.length = frameCount;\n};\nspine.DrawOrderTimeline.prototype = {\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, drawOrder) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.drawOrders[frameIndex] = drawOrder;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar frameIndex;\n\t\tif (time >= frames[frames.length - 1]) // Time is after last frame.\n\t\t\tframeIndex = frames.length - 1;\n\t\telse\n\t\t\tframeIndex = spine.Animation.binarySearch1(frames, time) - 1;\n\n\t\tvar drawOrder = skeleton.drawOrder;\n\t\tvar slots = skeleton.slots;\n\t\tvar drawOrderToSetupIndex = this.drawOrders[frameIndex];\n\t\tif (!drawOrderToSetupIndex) {\n\t\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\t\tdrawOrder[i] = slots[i];\n\t\t} else {\n\t\t\tfor (var i = 0, n = drawOrderToSetupIndex.length; i < n; i++)\n\t\t\t\tdrawOrder[i] = skeleton.slots[drawOrderToSetupIndex[i]];\n\t\t}\n\n\t}\n};\n\nspine.FfdTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = [];\n\tthis.frames.length = frameCount;\n\tthis.frameVertices = [];\n\tthis.frameVertices.length = frameCount;\n};\nspine.FfdTimeline.prototype = {\n\tslotIndex: 0,\n\tattachment: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, vertices) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frameVertices[frameIndex] = vertices;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar slot = skeleton.slots[this.slotIndex];\n\t\tif (slot.attachment != this.attachment) return;\n\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar frameVertices = this.frameVertices;\n\t\tvar vertexCount = frameVertices[0].length;\n\n\t\tvar vertices = slot.attachmentVertices;\n\t\tif (vertices.length != vertexCount) alpha = 1;\n\t\tvertices.length = vertexCount;\n\n\t\tif (time >= frames[frames.length - 1]) { // Time is after last frame.\n\t\t\tvar lastVertices = frameVertices[frames.length - 1];\n\t\t\tif (alpha < 1) {\n\t\t\t\tfor (var i = 0; i < vertexCount; i++)\n\t\t\t\t\tvertices[i] += (lastVertices[i] - vertices[i]) * alpha;\n\t\t\t} else {\n\t\t\t\tfor (var i = 0; i < vertexCount; i++)\n\t\t\t\t\tvertices[i] = lastVertices[i];\n\t\t\t}\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch1(frames, time);\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex - 1] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex - 1, percent < 0 ? 0 : (percent > 1 ? 1 : percent));\n\n\t\tvar prevVertices = frameVertices[frameIndex - 1];\n\t\tvar nextVertices = frameVertices[frameIndex];\n\n\t\tif (alpha < 1) {\n\t\t\tfor (var i = 0; i < vertexCount; i++) {\n\t\t\t\tvar prev = prevVertices[i];\n\t\t\t\tvertices[i] += (prev + (nextVertices[i] - prev) * percent - vertices[i]) * alpha;\n\t\t\t}\n\t\t} else {\n\t\t\tfor (var i = 0; i < vertexCount; i++) {\n\t\t\t\tvar prev = prevVertices[i];\n\t\t\t\tvertices[i] = prev + (nextVertices[i] - prev) * percent;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.IkConstraintTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, mix, bendDirection, ...\n\tthis.frames.length = frameCount * 3;\n};\nspine.IkConstraintTimeline.prototype = {\n\tikConstraintIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 3;\n\t},\n\tsetFrame: function (frameIndex, time, mix, bendDirection) {\n\t\tframeIndex *= 3;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = mix;\n\t\tthis.frames[frameIndex + 2] = bendDirection;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar ikConstraint = skeleton.ikConstraints[this.ikConstraintIndex];\n\n\t\tif (time >= frames[frames.length - 3]) { // Time is after last frame.\n\t\t\tikConstraint.mix += (frames[frames.length - 2] - ikConstraint.mix) * alpha;\n\t\t\tikConstraint.bendDirection = frames[frames.length - 1];\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 3);\n\t\tvar prevFrameMix = frames[frameIndex + -2/*PREV_FRAME_MIX*/];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent);\n\n\t\tvar mix = prevFrameMix + (frames[frameIndex + 1/*FRAME_MIX*/] - prevFrameMix) * percent;\n\t\tikConstraint.mix += (mix - ikConstraint.mix) * alpha;\n\t\tikConstraint.bendDirection = frames[frameIndex + -1/*PREV_FRAME_BEND_DIRECTION*/];\n\t}\n};\n\nspine.FlipXTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, flip, ...\n\tthis.frames.length = frameCount * 2;\n};\nspine.FlipXTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 2;\n\t},\n\tsetFrame: function (frameIndex, time, flip) {\n\t\tframeIndex *= 2;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = flip ? 1 : 0;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) {\n\t\t\tif (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0);\n\t\t\treturn;\n\t\t} else if (lastTime > time) //\n\t\t\tlastTime = -1;\n\t\tvar frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2;\n\t\tif (frames[frameIndex] < lastTime) return;\n\t\tskeleton.bones[boneIndex].flipX = frames[frameIndex + 1] != 0;\n\t}\n};\n\nspine.FlipYTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, flip, ...\n\tthis.frames.length = frameCount * 2;\n};\nspine.FlipYTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 2;\n\t},\n\tsetFrame: function (frameIndex, time, flip) {\n\t\tframeIndex *= 2;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = flip ? 1 : 0;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) {\n\t\t\tif (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0);\n\t\t\treturn;\n\t\t} else if (lastTime > time) //\n\t\t\tlastTime = -1;\n\t\tvar frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2;\n\t\tif (frames[frameIndex] < lastTime) return;\n\t\tskeleton.bones[boneIndex].flipY = frames[frameIndex + 1] != 0;\n\t}\n};\n\nspine.SkeletonData = function () {\n\tthis.bones = [];\n\tthis.slots = [];\n\tthis.skins = [];\n\tthis.events = [];\n\tthis.animations = [];\n\tthis.ikConstraints = [];\n};\nspine.SkeletonData.prototype = {\n\tname: null,\n\tdefaultSkin: null,\n\twidth: 0, height: 0,\n\tversion: null, hash: null,\n\t/** @return May be null. */\n\tfindBone: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].name == boneName) return bones[i];\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindBoneIndex: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].name == boneName) return i;\n\t\treturn -1;\n\t},\n\t/** @return May be null. */\n\tfindSlot: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tif (slots[i].name == slotName) return slot[i];\n\t\t}\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindSlotIndex: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\tif (slots[i].name == slotName) return i;\n\t\treturn -1;\n\t},\n\t/** @return May be null. */\n\tfindSkin: function (skinName) {\n\t\tvar skins = this.skins;\n\t\tfor (var i = 0, n = skins.length; i < n; i++)\n\t\t\tif (skins[i].name == skinName) return skins[i];\n\t\treturn null;\n\t},\n\t/** @return May be null. */\n\tfindEvent: function (eventName) {\n\t\tvar events = this.events;\n\t\tfor (var i = 0, n = events.length; i < n; i++)\n\t\t\tif (events[i].name == eventName) return events[i];\n\t\treturn null;\n\t},\n\t/** @return May be null. */\n\tfindAnimation: function (animationName) {\n\t\tvar animations = this.animations;\n\t\tfor (var i = 0, n = animations.length; i < n; i++)\n\t\t\tif (animations[i].name == animationName) return animations[i];\n\t\treturn null;\n\t},\n\t/** @return May be null. */\n\tfindIkConstraint: function (ikConstraintName) {\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tfor (var i = 0, n = ikConstraints.length; i < n; i++)\n\t\t\tif (ikConstraints[i].name == ikConstraintName) return ikConstraints[i];\n\t\treturn null;\n\t}\n};\n\nspine.Skeleton = function (skeletonData) {\n\tthis.data = skeletonData;\n\n\tthis.bones = [];\n\tfor (var i = 0, n = skeletonData.bones.length; i < n; i++) {\n\t\tvar boneData = skeletonData.bones[i];\n\t\tvar parent = !boneData.parent ? null : this.bones[skeletonData.bones.indexOf(boneData.parent)];\n\t\tthis.bones.push(new spine.Bone(boneData, this, parent));\n\t}\n\n\tthis.slots = [];\n\tthis.drawOrder = [];\n\tfor (var i = 0, n = skeletonData.slots.length; i < n; i++) {\n\t\tvar slotData = skeletonData.slots[i];\n\t\tvar bone = this.bones[skeletonData.bones.indexOf(slotData.boneData)];\n\t\tvar slot = new spine.Slot(slotData, bone);\n\t\tthis.slots.push(slot);\n\t\tthis.drawOrder.push(slot);\n\t}\n\n\tthis.ikConstraints = [];\n\tfor (var i = 0, n = skeletonData.ikConstraints.length; i < n; i++)\n\t\tthis.ikConstraints.push(new spine.IkConstraint(skeletonData.ikConstraints[i], this));\n\n\tthis.boneCache = [];\n\tthis.updateCache();\n};\nspine.Skeleton.prototype = {\n\tx: 0, y: 0,\n\tskin: null,\n\tr: 1, g: 1, b: 1, a: 1,\n\ttime: 0,\n\tflipX: false, flipY: false,\n\t/** Caches information about bones and IK constraints. Must be called if bones or IK constraints are added or removed. */\n\tupdateCache: function () {\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tvar ikConstraintsCount = ikConstraints.length;\n\n\t\tvar arrayCount = ikConstraintsCount + 1;\n\t\tvar boneCache = this.boneCache;\n\t\tif (boneCache.length > arrayCount) boneCache.length = arrayCount;\n\t\tfor (var i = 0, n = boneCache.length; i < n; i++)\n\t\t\tboneCache[i].length = 0;\n\t\twhile (boneCache.length < arrayCount)\n\t\t\tboneCache[boneCache.length] = [];\n\n\t\tvar nonIkBones = boneCache[0];\n\t\tvar bones = this.bones;\n\n\t\touter:\n\t\tfor (var i = 0, n = bones.length; i < n; i++) {\n\t\t\tvar bone = bones[i];\n\t\t\tvar current = bone;\n\t\t\tdo {\n\t\t\t\tfor (var ii = 0; ii < ikConstraintsCount; ii++) {\n\t\t\t\t\tvar ikConstraint = ikConstraints[ii];\n\t\t\t\t\tvar parent = ikConstraint.bones[0];\n\t\t\t\t\tvar child= ikConstraint.bones[ikConstraint.bones.length - 1];\n\t\t\t\t\twhile (true) {\n\t\t\t\t\t\tif (current == child) {\n\t\t\t\t\t\t\tboneCache[ii].push(bone);\n\t\t\t\t\t\t\tboneCache[ii + 1].push(bone);\n\t\t\t\t\t\t\tcontinue outer;\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (child == parent) break;\n\t\t\t\t\t\tchild = child.parent;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\tcurrent = current.parent;\n\t\t\t} while (current);\n\t\t\tnonIkBones[nonIkBones.length] = bone;\n\t\t}\n\t},\n\t/** Updates the world transform for each bone. */\n\tupdateWorldTransform: function () {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++) {\n\t\t\tvar bone = bones[i];\n\t\t\tbone.rotationIK = bone.rotation;\n\t\t}\n\t\tvar i = 0, last = this.boneCache.length - 1;\n\t\twhile (true) {\n\t\t\tvar cacheBones = this.boneCache[i];\n\t\t\tfor (var ii = 0, nn = cacheBones.length; ii < nn; ii++)\n\t\t\t\tcacheBones[ii].updateWorldTransform();\n\t\t\tif (i == last) break;\n\t\t\tthis.ikConstraints[i].apply();\n\t\t\ti++;\n\t\t}\n\t},\n\t/** Sets the bones and slots to their setup pose values. */\n\tsetToSetupPose: function () {\n\t\tthis.setBonesToSetupPose();\n\t\tthis.setSlotsToSetupPose();\n\t},\n\tsetBonesToSetupPose: function () {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tbones[i].setToSetupPose();\n\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tfor (var i = 0, n = ikConstraints.length; i < n; i++) {\n\t\t\tvar ikConstraint = ikConstraints[i];\n\t\t\tikConstraint.bendDirection = ikConstraint.data.bendDirection;\n\t\t\tikConstraint.mix = ikConstraint.data.mix;\n\t\t}\n\t},\n\tsetSlotsToSetupPose: function () {\n\t\tvar slots = this.slots;\n\t\tvar drawOrder = this.drawOrder;\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tdrawOrder[i] = slots[i];\n\t\t\tslots[i].setToSetupPose(i);\n\t\t}\n\t},\n\t/** @return May return null. */\n\tgetRootBone: function () {\n\t\treturn this.bones.length ? this.bones[0] : null;\n\t},\n\t/** @return May be null. */\n\tfindBone: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].data.name == boneName) return bones[i];\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindBoneIndex: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].data.name == boneName) return i;\n\t\treturn -1;\n\t},\n\t/** @return May be null. */\n\tfindSlot: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\tif (slots[i].data.name == slotName) return slots[i];\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindSlotIndex: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\tif (slots[i].data.name == slotName) return i;\n\t\treturn -1;\n\t},\n\tsetSkinByName: function (skinName) {\n\t\tvar skin = this.data.findSkin(skinName);\n\t\tif (!skin) throw \"Skin not found: \" + skinName;\n\t\tthis.setSkin(skin);\n\t},\n\t/** Sets the skin used to look up attachments before looking in the {@link SkeletonData#getDefaultSkin() default skin}.\n\t * Attachments from the new skin are attached if the corresponding attachment from the old skin was attached. If there was\n\t * no old skin, each slot's setup mode attachment is attached from the new skin.\n\t * @param newSkin May be null. */\n\tsetSkin: function (newSkin) {\n\t\tif (newSkin) {\n\t\t\tif (this.skin)\n\t\t\t\tnewSkin._attachAll(this, this.skin);\n\t\t\telse {\n\t\t\t\tvar slots = this.slots;\n\t\t\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\t\t\tvar slot = slots[i];\n\t\t\t\t\tvar name = slot.data.attachmentName;\n\t\t\t\t\tif (name) {\n\t\t\t\t\t\tvar attachment = newSkin.getAttachment(i, name);\n\t\t\t\t\t\tif (attachment) slot.setAttachment(attachment);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\tthis.skin = newSkin;\n\t},\n\t/** @return May be null. */\n\tgetAttachmentBySlotName: function (slotName, attachmentName) {\n\t\treturn this.getAttachmentBySlotIndex(this.data.findSlotIndex(slotName), attachmentName);\n\t},\n\t/** @return May be null. */\n\tgetAttachmentBySlotIndex: function (slotIndex, attachmentName) {\n\t\tif (this.skin) {\n\t\t\tvar attachment = this.skin.getAttachment(slotIndex, attachmentName);\n\t\t\tif (attachment) return attachment;\n\t\t}\n\t\tif (this.data.defaultSkin) return this.data.defaultSkin.getAttachment(slotIndex, attachmentName);\n\t\treturn null;\n\t},\n\t/** @param attachmentName May be null. */\n\tsetAttachment: function (slotName, attachmentName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tvar slot = slots[i];\n\t\t\tif (slot.data.name == slotName) {\n\t\t\t\tvar attachment = null;\n\t\t\t\tif (attachmentName) {\n\t\t\t\t\tattachment = this.getAttachmentBySlotIndex(i, attachmentName);\n\t\t\t\t\tif (!attachment) throw \"Attachment not found: \" + attachmentName + \", for slot: \" + slotName;\n\t\t\t\t}\n\t\t\t\tslot.setAttachment(attachment);\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\t\tthrow \"Slot not found: \" + slotName;\n\t},\n\t/** @return May be null. */\n\tfindIkConstraint: function (ikConstraintName) {\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tfor (var i = 0, n = ikConstraints.length; i < n; i++)\n\t\t\tif (ikConstraints[i].data.name == ikConstraintName) return ikConstraints[i];\n\t\treturn null;\n\t},\n\tupdate: function (delta) {\n\t\tthis.time += delta;\n\t}\n};\n\nspine.EventData = function (name) {\n\tthis.name = name;\n};\nspine.EventData.prototype = {\n\tintValue: 0,\n\tfloatValue: 0,\n\tstringValue: null\n};\n\nspine.Event = function (data) {\n\tthis.data = data;\n};\nspine.Event.prototype = {\n\tintValue: 0,\n\tfloatValue: 0,\n\tstringValue: null\n};\n\nspine.AttachmentType = {\n\tregion: 0,\n\tboundingbox: 1,\n\tmesh: 2,\n\tskinnedmesh: 3\n};\n\nspine.RegionAttachment = function (name) {\n\tthis.name = name;\n\tthis.offset = [];\n\tthis.offset.length = 8;\n\tthis.uvs = [];\n\tthis.uvs.length = 8;\n};\nspine.RegionAttachment.prototype = {\n\ttype: spine.AttachmentType.region,\n\tx: 0, y: 0,\n\trotation: 0,\n\tscaleX: 1, scaleY: 1,\n\twidth: 0, height: 0,\n\tr: 1, g: 1, b: 1, a: 1,\n\tpath: null,\n\trendererObject: null,\n\tregionOffsetX: 0, regionOffsetY: 0,\n\tregionWidth: 0, regionHeight: 0,\n\tregionOriginalWidth: 0, regionOriginalHeight: 0,\n\tsetUVs: function (u, v, u2, v2, rotate) {\n\t\tvar uvs = this.uvs;\n\t\tif (rotate) {\n\t\t\tuvs[2/*X2*/] = u;\n\t\t\tuvs[3/*Y2*/] = v2;\n\t\t\tuvs[4/*X3*/] = u;\n\t\t\tuvs[5/*Y3*/] = v;\n\t\t\tuvs[6/*X4*/] = u2;\n\t\t\tuvs[7/*Y4*/] = v;\n\t\t\tuvs[0/*X1*/] = u2;\n\t\t\tuvs[1/*Y1*/] = v2;\n\t\t} else {\n\t\t\tuvs[0/*X1*/] = u;\n\t\t\tuvs[1/*Y1*/] = v2;\n\t\t\tuvs[2/*X2*/] = u;\n\t\t\tuvs[3/*Y2*/] = v;\n\t\t\tuvs[4/*X3*/] = u2;\n\t\t\tuvs[5/*Y3*/] = v;\n\t\t\tuvs[6/*X4*/] = u2;\n\t\t\tuvs[7/*Y4*/] = v2;\n\t\t}\n\t},\n\tupdateOffset: function () {\n\t\tvar regionScaleX = this.width / this.regionOriginalWidth * this.scaleX;\n\t\tvar regionScaleY = this.height / this.regionOriginalHeight * this.scaleY;\n\t\tvar localX = -this.width / 2 * this.scaleX + this.regionOffsetX * regionScaleX;\n\t\tvar localY = -this.height / 2 * this.scaleY + this.regionOffsetY * regionScaleY;\n\t\tvar localX2 = localX + this.regionWidth * regionScaleX;\n\t\tvar localY2 = localY + this.regionHeight * regionScaleY;\n\t\tvar radians = this.rotation * spine.degRad;\n\t\tvar cos = Math.cos(radians);\n\t\tvar sin = Math.sin(radians);\n\t\tvar localXCos = localX * cos + this.x;\n\t\tvar localXSin = localX * sin;\n\t\tvar localYCos = localY * cos + this.y;\n\t\tvar localYSin = localY * sin;\n\t\tvar localX2Cos = localX2 * cos + this.x;\n\t\tvar localX2Sin = localX2 * sin;\n\t\tvar localY2Cos = localY2 * cos + this.y;\n\t\tvar localY2Sin = localY2 * sin;\n\t\tvar offset = this.offset;\n\t\toffset[0/*X1*/] = localXCos - localYSin;\n\t\toffset[1/*Y1*/] = localYCos + localXSin;\n\t\toffset[2/*X2*/] = localXCos - localY2Sin;\n\t\toffset[3/*Y2*/] = localY2Cos + localXSin;\n\t\toffset[4/*X3*/] = localX2Cos - localY2Sin;\n\t\toffset[5/*Y3*/] = localY2Cos + localX2Sin;\n\t\toffset[6/*X4*/] = localX2Cos - localYSin;\n\t\toffset[7/*Y4*/] = localYCos + localX2Sin;\n\t},\n\tcomputeVertices: function (x, y, bone, vertices) {\n\t\tx += bone.worldX;\n\t\ty += bone.worldY;\n\t\tvar m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11;\n\t\tvar offset = this.offset;\n\t\tvertices[0/*X1*/] = offset[0/*X1*/] * m00 + offset[1/*Y1*/] * m01 + x;\n\t\tvertices[1/*Y1*/] = offset[0/*X1*/] * m10 + offset[1/*Y1*/] * m11 + y;\n\t\tvertices[2/*X2*/] = offset[2/*X2*/] * m00 + offset[3/*Y2*/] * m01 + x;\n\t\tvertices[3/*Y2*/] = offset[2/*X2*/] * m10 + offset[3/*Y2*/] * m11 + y;\n\t\tvertices[4/*X3*/] = offset[4/*X3*/] * m00 + offset[5/*X3*/] * m01 + x;\n\t\tvertices[5/*X3*/] = offset[4/*X3*/] * m10 + offset[5/*X3*/] * m11 + y;\n\t\tvertices[6/*X4*/] = offset[6/*X4*/] * m00 + offset[7/*Y4*/] * m01 + x;\n\t\tvertices[7/*Y4*/] = offset[6/*X4*/] * m10 + offset[7/*Y4*/] * m11 + y;\n\t}\n};\n\nspine.MeshAttachment = function (name) {\n\tthis.name = name;\n};\nspine.MeshAttachment.prototype = {\n\ttype: spine.AttachmentType.mesh,\n\tvertices: null,\n\tuvs: null,\n\tregionUVs: null,\n\ttriangles: null,\n\thullLength: 0,\n\tr: 1, g: 1, b: 1, a: 1,\n\tpath: null,\n\trendererObject: null,\n\tregionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false,\n\tregionOffsetX: 0, regionOffsetY: 0,\n\tregionWidth: 0, regionHeight: 0,\n\tregionOriginalWidth: 0, regionOriginalHeight: 0,\n\tedges: null,\n\twidth: 0, height: 0,\n\tupdateUVs: function () {\n\t\tvar width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV;\n\t\tvar n = this.regionUVs.length;\n\t\tif (!this.uvs || this.uvs.length != n) {\n            this.uvs = new spine.Float32Array(n);\n\t\t}\n\t\tif (this.regionRotate) {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width;\n                this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height;\n\t\t\t}\n\t\t} else {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i] * width;\n                this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height;\n\t\t\t}\n\t\t}\n\t},\n\tcomputeWorldVertices: function (x, y, slot, worldVertices) {\n\t\tvar bone = slot.bone;\n\t\tx += bone.worldX;\n\t\ty += bone.worldY;\n\t\tvar m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11;\n\t\tvar vertices = this.vertices;\n\t\tvar verticesCount = vertices.length;\n\t\tif (slot.attachmentVertices.length == verticesCount) vertices = slot.attachmentVertices;\n\t\tfor (var i = 0; i < verticesCount; i += 2) {\n\t\t\tvar vx = vertices[i];\n\t\t\tvar vy = vertices[i + 1];\n\t\t\tworldVertices[i] = vx * m00 + vy * m01 + x;\n\t\t\tworldVertices[i + 1] = vx * m10 + vy * m11 + y;\n\t\t}\n\t}\n};\n\nspine.SkinnedMeshAttachment = function (name) {\n\tthis.name = name;\n};\nspine.SkinnedMeshAttachment.prototype = {\n\ttype: spine.AttachmentType.skinnedmesh,\n\tbones: null,\n\tweights: null,\n\tuvs: null,\n\tregionUVs: null,\n\ttriangles: null,\n\thullLength: 0,\n\tr: 1, g: 1, b: 1, a: 1,\n\tpath: null,\n\trendererObject: null,\n\tregionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false,\n\tregionOffsetX: 0, regionOffsetY: 0,\n\tregionWidth: 0, regionHeight: 0,\n\tregionOriginalWidth: 0, regionOriginalHeight: 0,\n\tedges: null,\n\twidth: 0, height: 0,\n\tupdateUVs: function (u, v, u2, v2, rotate) {\n\t\tvar width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV;\n\t\tvar n = this.regionUVs.length;\n\t\tif (!this.uvs || this.uvs.length != n) {\n            this.uvs = new spine.Float32Array(n);\n\t\t}\n\t\tif (this.regionRotate) {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width;\n                this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height;\n\t\t\t}\n\t\t} else {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i] * width;\n                this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height;\n\t\t\t}\n\t\t}\n\t},\n\tcomputeWorldVertices: function (x, y, slot, worldVertices) {\n\t\tvar skeletonBones = slot.bone.skeleton.bones;\n\t\tvar weights = this.weights;\n\t\tvar bones = this.bones;\n\n\t\tvar w = 0, v = 0, b = 0, f = 0, n = bones.length, nn;\n\t\tvar wx, wy, bone, vx, vy, weight;\n\t\tif (!slot.attachmentVertices.length) {\n\t\t\tfor (; v < n; w += 2) {\n\t\t\t\twx = 0;\n\t\t\t\twy = 0;\n\t\t\t\tnn = bones[v++] + v;\n\t\t\t\tfor (; v < nn; v++, b += 3) {\n\t\t\t\t\tbone = skeletonBones[bones[v]];\n\t\t\t\t\tvx = weights[b];\n\t\t\t\t\tvy = weights[b + 1];\n\t\t\t\t\tweight = weights[b + 2];\n\t\t\t\t\twx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight;\n\t\t\t\t\twy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight;\n\t\t\t\t}\n\t\t\t\tworldVertices[w] = wx + x;\n\t\t\t\tworldVertices[w + 1] = wy + y;\n\t\t\t}\n\t\t} else {\n\t\t\tvar ffd = slot.attachmentVertices;\n\t\t\tfor (; v < n; w += 2) {\n\t\t\t\twx = 0;\n\t\t\t\twy = 0;\n\t\t\t\tnn = bones[v++] + v;\n\t\t\t\tfor (; v < nn; v++, b += 3, f += 2) {\n\t\t\t\t\tbone = skeletonBones[bones[v]];\n\t\t\t\t\tvx = weights[b] + ffd[f];\n\t\t\t\t\tvy = weights[b + 1] + ffd[f + 1];\n\t\t\t\t\tweight = weights[b + 2];\n\t\t\t\t\twx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight;\n\t\t\t\t\twy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight;\n\t\t\t\t}\n\t\t\t\tworldVertices[w] = wx + x;\n\t\t\t\tworldVertices[w + 1] = wy + y;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.BoundingBoxAttachment = function (name) {\n\tthis.name = name;\n\tthis.vertices = [];\n};\nspine.BoundingBoxAttachment.prototype = {\n\ttype: spine.AttachmentType.boundingbox,\n\tcomputeWorldVertices: function (x, y, bone, worldVertices) {\n\t\tx += bone.worldX;\n\t\ty += bone.worldY;\n\t\tvar m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11;\n\t\tvar vertices = this.vertices;\n\t\tfor (var i = 0, n = vertices.length; i < n; i += 2) {\n\t\t\tvar px = vertices[i];\n\t\t\tvar py = vertices[i + 1];\n\t\t\tworldVertices[i] = px * m00 + py * m01 + x;\n\t\t\tworldVertices[i + 1] = px * m10 + py * m11 + y;\n\t\t}\n\t}\n};\n\nspine.AnimationStateData = function (skeletonData) {\n\tthis.skeletonData = skeletonData;\n\tthis.animationToMixTime = {};\n};\nspine.AnimationStateData.prototype = {\n\tdefaultMix: 0,\n\tsetMixByName: function (fromName, toName, duration) {\n\t\tvar from = this.skeletonData.findAnimation(fromName);\n\t\tif (!from) throw \"Animation not found: \" + fromName;\n\t\tvar to = this.skeletonData.findAnimation(toName);\n\t\tif (!to) throw \"Animation not found: \" + toName;\n\t\tthis.setMix(from, to, duration);\n\t},\n\tsetMix: function (from, to, duration) {\n\t\tthis.animationToMixTime[from.name + \":\" + to.name] = duration;\n\t},\n\tgetMix: function (from, to) {\n\t\tvar key = from.name + \":\" + to.name;\n\t\treturn this.animationToMixTime.hasOwnProperty(key) ? this.animationToMixTime[key] : this.defaultMix;\n\t}\n};\n\nspine.TrackEntry = function () {};\nspine.TrackEntry.prototype = {\n\tnext: null, previous: null,\n\tanimation: null,\n\tloop: false,\n\tdelay: 0, time: 0, lastTime: -1, endTime: 0,\n\ttimeScale: 1,\n\tmixTime: 0, mixDuration: 0, mix: 1,\n\tonStart: null, onEnd: null, onComplete: null, onEvent: null\n};\n\nspine.AnimationState = function (stateData) {\n\tthis.data = stateData;\n\tthis.tracks = [];\n\tthis.events = [];\n};\nspine.AnimationState.prototype = {\n\tonStart: null,\n\tonEnd: null,\n\tonComplete: null,\n\tonEvent: null,\n\ttimeScale: 1,\n\tupdate: function (delta) {\n\t\tdelta *= this.timeScale;\n\t\tfor (var i = 0; i < this.tracks.length; i++) {\n\t\t\tvar current = this.tracks[i];\n\t\t\tif (!current) continue;\n\n\t\t\tcurrent.time += delta * current.timeScale;\n\t\t\tif (current.previous) {\n\t\t\t\tvar previousDelta = delta * current.previous.timeScale;\n\t\t\t\tcurrent.previous.time += previousDelta;\n\t\t\t\tcurrent.mixTime += previousDelta;\n\t\t\t}\n\n\t\t\tvar next = current.next;\n\t\t\tif (next) {\n\t\t\t\tnext.time = current.lastTime - next.delay;\n\t\t\t\tif (next.time >= 0) this.setCurrent(i, next);\n\t\t\t} else {\n\t\t\t\t// End non-looping animation when it reaches its end time and there is no next entry.\n\t\t\t\tif (!current.loop && current.lastTime >= current.endTime) this.clearTrack(i);\n\t\t\t}\n\t\t}\n\t},\n\tapply: function (skeleton) {\n\t\tfor (var i = 0; i < this.tracks.length; i++) {\n\t\t\tvar current = this.tracks[i];\n\t\t\tif (!current) continue;\n\n\t\t\tthis.events.length = 0;\n\n\t\t\tvar time = current.time;\n\t\t\tvar lastTime = current.lastTime;\n\t\t\tvar endTime = current.endTime;\n\t\t\tvar loop = current.loop;\n\t\t\tif (!loop && time > endTime) time = endTime;\n\n\t\t\tvar previous = current.previous;\n\t\t\tif (!previous) {\n\t\t\t\tif (current.mix == 1)\n\t\t\t\t\tcurrent.animation.apply(skeleton, current.lastTime, time, loop, this.events);\n\t\t\t\telse\n\t\t\t\t\tcurrent.animation.mix(skeleton, current.lastTime, time, loop, this.events, current.mix);\n\t\t\t} else {\n\t\t\t\tvar previousTime = previous.time;\n\t\t\t\tif (!previous.loop && previousTime > previous.endTime) previousTime = previous.endTime;\n\t\t\t\tprevious.animation.apply(skeleton, previousTime, previousTime, previous.loop, null);\n\n\t\t\t\tvar alpha = current.mixTime / current.mixDuration * current.mix;\n\t\t\t\tif (alpha >= 1) {\n\t\t\t\t\talpha = 1;\n\t\t\t\t\tcurrent.previous = null;\n\t\t\t\t}\n\t\t\t\tcurrent.animation.mix(skeleton, current.lastTime, time, loop, this.events, alpha);\n\t\t\t}\n\n\t\t\tfor (var ii = 0, nn = this.events.length; ii < nn; ii++) {\n\t\t\t\tvar event = this.events[ii];\n\t\t\t\tif (current.onEvent) current.onEvent(i, event);\n\t\t\t\tif (this.onEvent) this.onEvent(i, event);\n\t\t\t}\n\n\t\t\t// Check if completed the animation or a loop iteration.\n\t\t\tif (loop ? (lastTime % endTime > time % endTime) : (lastTime < endTime && time >= endTime)) {\n\t\t\t\tvar count = Math.floor(time / endTime);\n\t\t\t\tif (current.onComplete) current.onComplete(i, count);\n\t\t\t\tif (this.onComplete) this.onComplete(i, count);\n\t\t\t}\n\n\t\t\tcurrent.lastTime = current.time;\n\t\t}\n\t},\n\tclearTracks: function () {\n\t\tfor (var i = 0, n = this.tracks.length; i < n; i++)\n\t\t\tthis.clearTrack(i);\n\t\tthis.tracks.length = 0;\n\t},\n\tclearTrack: function (trackIndex) {\n\t\tif (trackIndex >= this.tracks.length) return;\n\t\tvar current = this.tracks[trackIndex];\n\t\tif (!current) return;\n\n\t\tif (current.onEnd) current.onEnd(trackIndex);\n\t\tif (this.onEnd) this.onEnd(trackIndex);\n\n\t\tthis.tracks[trackIndex] = null;\n\t},\n\t_expandToIndex: function (index) {\n\t\tif (index < this.tracks.length) return this.tracks[index];\n\t\twhile (index >= this.tracks.length)\n\t\t\tthis.tracks.push(null);\n\t\treturn null;\n\t},\n\tsetCurrent: function (index, entry) {\n\t\tvar current = this._expandToIndex(index);\n\t\tif (current) {\n\t\t\tvar previous = current.previous;\n\t\t\tcurrent.previous = null;\n\n\t\t\tif (current.onEnd) current.onEnd(index);\n\t\t\tif (this.onEnd) this.onEnd(index);\n\n\t\t\tentry.mixDuration = this.data.getMix(current.animation, entry.animation);\n\t\t\tif (entry.mixDuration > 0) {\n\t\t\t\tentry.mixTime = 0;\n\t\t\t\t// If a mix is in progress, mix from the closest animation.\n\t\t\t\tif (previous && current.mixTime / current.mixDuration < 0.5)\n\t\t\t\t\tentry.previous = previous;\n\t\t\t\telse\n\t\t\t\t\tentry.previous = current;\n\t\t\t}\n\t\t}\n\n\t\tthis.tracks[index] = entry;\n\n\t\tif (entry.onStart) entry.onStart(index);\n\t\tif (this.onStart) this.onStart(index);\n\t},\n\tsetAnimationByName: function (trackIndex, animationName, loop) {\n\t\tvar animation = this.data.skeletonData.findAnimation(animationName);\n\t\tif (!animation) throw \"Animation not found: \" + animationName;\n\t\treturn this.setAnimation(trackIndex, animation, loop);\n\t},\n\t/** Set the current animation. Any queued animations are cleared. */\n\tsetAnimation: function (trackIndex, animation, loop) {\n\t\tvar entry = new spine.TrackEntry();\n\t\tentry.animation = animation;\n\t\tentry.loop = loop;\n\t\tentry.endTime = animation.duration;\n\t\tthis.setCurrent(trackIndex, entry);\n\t\treturn entry;\n\t},\n\taddAnimationByName: function (trackIndex, animationName, loop, delay) {\n\t\tvar animation = this.data.skeletonData.findAnimation(animationName);\n\t\tif (!animation) throw \"Animation not found: \" + animationName;\n\t\treturn this.addAnimation(trackIndex, animation, loop, delay);\n\t},\n\t/** Adds an animation to be played delay seconds after the current or last queued animation.\n\t * @param delay May be <= 0 to use duration of previous animation minus any mix duration plus the negative delay. */\n\taddAnimation: function (trackIndex, animation, loop, delay) {\n\t\tvar entry = new spine.TrackEntry();\n\t\tentry.animation = animation;\n\t\tentry.loop = loop;\n\t\tentry.endTime = animation.duration;\n\n\t\tvar last = this._expandToIndex(trackIndex);\n\t\tif (last) {\n\t\t\twhile (last.next)\n\t\t\t\tlast = last.next;\n\t\t\tlast.next = entry;\n\t\t} else\n\t\t\tthis.tracks[trackIndex] = entry;\n\n\t\tif (delay <= 0) {\n\t\t\tif (last)\n\t\t\t\tdelay += last.endTime - this.data.getMix(last.animation, animation);\n\t\t\telse\n\t\t\t\tdelay = 0;\n\t\t}\n\t\tentry.delay = delay;\n\n\t\treturn entry;\n\t},\n\t/** May be null. */\n\tgetCurrent: function (trackIndex) {\n\t\tif (trackIndex >= this.tracks.length) return null;\n\t\treturn this.tracks[trackIndex];\n\t}\n};\n\nspine.SkeletonJson = function (attachmentLoader) {\n\tthis.attachmentLoader = attachmentLoader;\n};\nspine.SkeletonJson.prototype = {\n\tscale: 1,\n\treadSkeletonData: function (root, name) {\n\t\tvar skeletonData = new spine.SkeletonData();\n\t\tskeletonData.name = name;\n\n\t\t// Skeleton.\n\t\tvar skeletonMap = root[\"skeleton\"];\n\t\tif (skeletonMap) {\n\t\t\tskeletonData.hash = skeletonMap[\"hash\"];\n\t\t\tskeletonData.version = skeletonMap[\"spine\"];\n\t\t\tskeletonData.width = skeletonMap[\"width\"] || 0;\n\t\t\tskeletonData.height = skeletonMap[\"height\"] || 0;\n\t\t}\n\n\t\t// Bones.\n\t\tvar bones = root[\"bones\"];\n\t\tfor (var i = 0, n = bones.length; i < n; i++) {\n\t\t\tvar boneMap = bones[i];\n\t\t\tvar parent = null;\n\t\t\tif (boneMap[\"parent\"]) {\n\t\t\t\tparent = skeletonData.findBone(boneMap[\"parent\"]);\n\t\t\t\tif (!parent) throw \"Parent bone not found: \" + boneMap[\"parent\"];\n\t\t\t}\n\t\t\tvar boneData = new spine.BoneData(boneMap[\"name\"], parent);\n\t\t\tboneData.length = (boneMap[\"length\"] || 0) * this.scale;\n\t\t\tboneData.x = (boneMap[\"x\"] || 0) * this.scale;\n\t\t\tboneData.y = (boneMap[\"y\"] || 0) * this.scale;\n\t\t\tboneData.rotation = (boneMap[\"rotation\"] || 0);\n\t\t\tboneData.scaleX = boneMap.hasOwnProperty(\"scaleX\") ? boneMap[\"scaleX\"] : 1;\n\t\t\tboneData.scaleY = boneMap.hasOwnProperty(\"scaleY\") ? boneMap[\"scaleY\"] : 1;\n\t\t\tboneData.inheritScale = boneMap.hasOwnProperty(\"inheritScale\") ? boneMap[\"inheritScale\"] : true;\n\t\t\tboneData.inheritRotation = boneMap.hasOwnProperty(\"inheritRotation\") ? boneMap[\"inheritRotation\"] : true;\n\t\t\tskeletonData.bones.push(boneData);\n\t\t}\n\n\t\t// IK constraints.\n\t\tvar ik = root[\"ik\"];\n\t\tif (ik) {\n\t\t\tfor (var i = 0, n = ik.length; i < n; i++) {\n\t\t\t\tvar ikMap = ik[i];\n\t\t\t\tvar ikConstraintData = new spine.IkConstraintData(ikMap[\"name\"]);\n\n\t\t\t\tvar bones = ikMap[\"bones\"];\n\t\t\t\tfor (var ii = 0, nn = bones.length; ii < nn; ii++) {\n\t\t\t\t\tvar bone = skeletonData.findBone(bones[ii]);\n\t\t\t\t\tif (!bone) throw \"IK bone not found: \" + bones[ii];\n\t\t\t\t\tikConstraintData.bones.push(bone);\n\t\t\t\t}\n\n\t\t\t\tikConstraintData.target = skeletonData.findBone(ikMap[\"target\"]);\n\t\t\t\tif (!ikConstraintData.target) throw \"Target bone not found: \" + ikMap[\"target\"];\n\n\t\t\t\tikConstraintData.bendDirection = (!ikMap.hasOwnProperty(\"bendPositive\") || ikMap[\"bendPositive\"]) ? 1 : -1;\n\t\t\t\tikConstraintData.mix = ikMap.hasOwnProperty(\"mix\") ? ikMap[\"mix\"] : 1;\n\n\t\t\t\tskeletonData.ikConstraints.push(ikConstraintData);\n\t\t\t}\n\t\t}\n\n\t\t// Slots.\n\t\tvar slots = root[\"slots\"];\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tvar slotMap = slots[i];\n\t\t\tvar boneData = skeletonData.findBone(slotMap[\"bone\"]);\n\t\t\tif (!boneData) throw \"Slot bone not found: \" + slotMap[\"bone\"];\n\t\t\tvar slotData = new spine.SlotData(slotMap[\"name\"], boneData);\n\n\t\t\tvar color = slotMap[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tslotData.r = this.toColor(color, 0);\n\t\t\t\tslotData.g = this.toColor(color, 1);\n\t\t\t\tslotData.b = this.toColor(color, 2);\n\t\t\t\tslotData.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tslotData.attachmentName = slotMap[\"attachment\"];\n\t\t\tslotData.additiveBlending = slotMap[\"additive\"] && slotMap[\"additive\"] == \"true\";\n\n\t\t\tskeletonData.slots.push(slotData);\n\t\t}\n\n\t\t// Skins.\n\t\tvar skins = root[\"skins\"];\n\t\tfor (var skinName in skins) {\n\t\t\tif (!skins.hasOwnProperty(skinName)) continue;\n\t\t\tvar skinMap = skins[skinName];\n\t\t\tvar skin = new spine.Skin(skinName);\n\t\t\tfor (var slotName in skinMap) {\n\t\t\t\tif (!skinMap.hasOwnProperty(slotName)) continue;\n\t\t\t\tvar slotIndex = skeletonData.findSlotIndex(slotName);\n\t\t\t\tvar slotEntry = skinMap[slotName];\n\t\t\t\tfor (var attachmentName in slotEntry) {\n\t\t\t\t\tif (!slotEntry.hasOwnProperty(attachmentName)) continue;\n\t\t\t\t\tvar attachment = this.readAttachment(skin, attachmentName, slotEntry[attachmentName]);\n\t\t\t\t\tif (attachment) skin.addAttachment(slotIndex, attachmentName, attachment);\n\t\t\t\t}\n\t\t\t}\n\t\t\tskeletonData.skins.push(skin);\n\t\t\tif (skin.name == \"default\") skeletonData.defaultSkin = skin;\n\t\t}\n\n\t\t// Events.\n\t\tvar events = root[\"events\"];\n\t\tfor (var eventName in events) {\n\t\t\tif (!events.hasOwnProperty(eventName)) continue;\n\t\t\tvar eventMap = events[eventName];\n\t\t\tvar eventData = new spine.EventData(eventName);\n\t\t\teventData.intValue = eventMap[\"int\"] || 0;\n\t\t\teventData.floatValue = eventMap[\"float\"] || 0;\n\t\t\teventData.stringValue = eventMap[\"string\"] || null;\n\t\t\tskeletonData.events.push(eventData);\n\t\t}\n\n\t\t// Animations.\n\t\tvar animations = root[\"animations\"];\n\t\tfor (var animationName in animations) {\n\t\t\tif (!animations.hasOwnProperty(animationName)) continue;\n\t\t\tthis.readAnimation(animationName, animations[animationName], skeletonData);\n\t\t}\n\n\t\treturn skeletonData;\n\t},\n\treadAttachment: function (skin, name, map) {\n\t\tname = map[\"name\"] || name;\n\n\t\tvar type = spine.AttachmentType[map[\"type\"] || \"region\"];\n\t\tvar path = map[\"path\"] || name;\n\n\t\tvar scale = this.scale;\n\t\tif (type == spine.AttachmentType.region) {\n\t\t\tvar region = this.attachmentLoader.newRegionAttachment(skin, name, path);\n\t\t\tif (!region) return null;\n\t\t\tregion.path = path;\n\t\t\tregion.x = (map[\"x\"] || 0) * scale;\n\t\t\tregion.y = (map[\"y\"] || 0) * scale;\n\t\t\tregion.scaleX = map.hasOwnProperty(\"scaleX\") ? map[\"scaleX\"] : 1;\n\t\t\tregion.scaleY = map.hasOwnProperty(\"scaleY\") ? map[\"scaleY\"] : 1;\n\t\t\tregion.rotation = map[\"rotation\"] || 0;\n\t\t\tregion.width = (map[\"width\"] || 0) * scale;\n\t\t\tregion.height = (map[\"height\"] || 0) * scale;\n\n\t\t\tvar color = map[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tregion.r = this.toColor(color, 0);\n\t\t\t\tregion.g = this.toColor(color, 1);\n\t\t\t\tregion.b = this.toColor(color, 2);\n\t\t\t\tregion.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tregion.updateOffset();\n\t\t\treturn region;\n\t\t} else if (type == spine.AttachmentType.mesh) {\n\t\t\tvar mesh = this.attachmentLoader.newMeshAttachment(skin, name, path);\n\t\t\tif (!mesh) return null;\n\t\t\tmesh.path = path;\n\t\t\tmesh.vertices = this.getFloatArray(map, \"vertices\", scale);\n\t\t\tmesh.triangles = this.getIntArray(map, \"triangles\");\n\t\t\tmesh.regionUVs = this.getFloatArray(map, \"uvs\", 1);\n\t\t\tmesh.updateUVs();\n\n\t\t\tcolor = map[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tmesh.r = this.toColor(color, 0);\n\t\t\t\tmesh.g = this.toColor(color, 1);\n\t\t\t\tmesh.b = this.toColor(color, 2);\n\t\t\t\tmesh.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tmesh.hullLength = (map[\"hull\"] || 0) * 2;\n\t\t\tif (map[\"edges\"]) mesh.edges = this.getIntArray(map, \"edges\");\n\t\t\tmesh.width = (map[\"width\"] || 0) * scale;\n\t\t\tmesh.height = (map[\"height\"] || 0) * scale;\n\t\t\treturn mesh;\n\t\t} else if (type == spine.AttachmentType.skinnedmesh) {\n\t\t\tvar mesh = this.attachmentLoader.newSkinnedMeshAttachment(skin, name, path);\n\t\t\tif (!mesh) return null;\n\t\t\tmesh.path = path;\n\n\t\t\tvar uvs = this.getFloatArray(map, \"uvs\", 1);\n\t\t\tvar vertices = this.getFloatArray(map, \"vertices\", 1);\n\t\t\tvar weights = [];\n\t\t\tvar bones = [];\n\t\t\tfor (var i = 0, n = vertices.length; i < n; ) {\n\t\t\t\tvar boneCount = vertices[i++] | 0;\n\t\t\t\tbones[bones.length] = boneCount;\n\t\t\t\tfor (var nn = i + boneCount * 4; i < nn; ) {\n\t\t\t\t\tbones[bones.length] = vertices[i];\n\t\t\t\t\tweights[weights.length] = vertices[i + 1] * scale;\n\t\t\t\t\tweights[weights.length] = vertices[i + 2] * scale;\n\t\t\t\t\tweights[weights.length] = vertices[i + 3];\n\t\t\t\t\ti += 4;\n\t\t\t\t}\n\t\t\t}\n\t\t\tmesh.bones = bones;\n\t\t\tmesh.weights = weights;\n\t\t\tmesh.triangles = this.getIntArray(map, \"triangles\");\n\t\t\tmesh.regionUVs = uvs;\n\t\t\tmesh.updateUVs();\n\n\t\t\tcolor = map[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tmesh.r = this.toColor(color, 0);\n\t\t\t\tmesh.g = this.toColor(color, 1);\n\t\t\t\tmesh.b = this.toColor(color, 2);\n\t\t\t\tmesh.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tmesh.hullLength = (map[\"hull\"] || 0) * 2;\n\t\t\tif (map[\"edges\"]) mesh.edges = this.getIntArray(map, \"edges\");\n\t\t\tmesh.width = (map[\"width\"] || 0) * scale;\n\t\t\tmesh.height = (map[\"height\"] || 0) * scale;\n\t\t\treturn mesh;\n\t\t} else if (type == spine.AttachmentType.boundingbox) {\n\t\t\tvar attachment = this.attachmentLoader.newBoundingBoxAttachment(skin, name);\n\t\t\tvar vertices = map[\"vertices\"];\n\t\t\tfor (var i = 0, n = vertices.length; i < n; i++)\n\t\t\t\tattachment.vertices.push(vertices[i] * scale);\n\t\t\treturn attachment;\n\t\t}\n\t\tthrow \"Unknown attachment type: \" + type;\n\t},\n\treadAnimation: function (name, map, skeletonData) {\n\t\tvar timelines = [];\n\t\tvar duration = 0;\n\n\t\tvar slots = map[\"slots\"];\n\t\tfor (var slotName in slots) {\n\t\t\tif (!slots.hasOwnProperty(slotName)) continue;\n\t\t\tvar slotMap = slots[slotName];\n\t\t\tvar slotIndex = skeletonData.findSlotIndex(slotName);\n\n\t\t\tfor (var timelineName in slotMap) {\n\t\t\t\tif (!slotMap.hasOwnProperty(timelineName)) continue;\n\t\t\t\tvar values = slotMap[timelineName];\n\t\t\t\tif (timelineName == \"color\") {\n\t\t\t\t\tvar timeline = new spine.ColorTimeline(values.length);\n\t\t\t\t\ttimeline.slotIndex = slotIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\tvar color = valueMap[\"color\"];\n\t\t\t\t\t\tvar r = this.toColor(color, 0);\n\t\t\t\t\t\tvar g = this.toColor(color, 1);\n\t\t\t\t\t\tvar b = this.toColor(color, 2);\n\t\t\t\t\t\tvar a = this.toColor(color, 3);\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], r, g, b, a);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 5 - 5]);\n\n\t\t\t\t} else if (timelineName == \"attachment\") {\n\t\t\t\t\tvar timeline = new spine.AttachmentTimeline(values.length);\n\t\t\t\t\ttimeline.slotIndex = slotIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\ttimeline.setFrame(frameIndex++, valueMap[\"time\"], valueMap[\"name\"]);\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]);\n\n\t\t\t\t} else\n\t\t\t\t\tthrow \"Invalid timeline type for a slot: \" + timelineName + \" (\" + slotName + \")\";\n\t\t\t}\n\t\t}\n\n\t\tvar bones = map[\"bones\"];\n\t\tfor (var boneName in bones) {\n\t\t\tif (!bones.hasOwnProperty(boneName)) continue;\n\t\t\tvar boneIndex = skeletonData.findBoneIndex(boneName);\n\t\t\tif (boneIndex == -1) throw \"Bone not found: \" + boneName;\n\t\t\tvar boneMap = bones[boneName];\n\n\t\t\tfor (var timelineName in boneMap) {\n\t\t\t\tif (!boneMap.hasOwnProperty(timelineName)) continue;\n\t\t\t\tvar values = boneMap[timelineName];\n\t\t\t\tif (timelineName == \"rotate\") {\n\t\t\t\t\tvar timeline = new spine.RotateTimeline(values.length);\n\t\t\t\t\ttimeline.boneIndex = boneIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], valueMap[\"angle\"]);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]);\n\n\t\t\t\t} else if (timelineName == \"translate\" || timelineName == \"scale\") {\n\t\t\t\t\tvar timeline;\n\t\t\t\t\tvar timelineScale = 1;\n\t\t\t\t\tif (timelineName == \"scale\")\n\t\t\t\t\t\ttimeline = new spine.ScaleTimeline(values.length);\n\t\t\t\t\telse {\n\t\t\t\t\t\ttimeline = new spine.TranslateTimeline(values.length);\n\t\t\t\t\t\ttimelineScale = this.scale;\n\t\t\t\t\t}\n\t\t\t\t\ttimeline.boneIndex = boneIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\tvar x = (valueMap[\"x\"] || 0) * timelineScale;\n\t\t\t\t\t\tvar y = (valueMap[\"y\"] || 0) * timelineScale;\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], x, y);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 3 - 3]);\n\n\t\t\t\t} else if (timelineName == \"flipX\" || timelineName == \"flipY\") {\n\t\t\t\t\tvar x = timelineName == \"flipX\";\n\t\t\t\t\tvar timeline = x ? new spine.FlipXTimeline(values.length) : new spine.FlipYTimeline(values.length);\n\t\t\t\t\ttimeline.boneIndex = boneIndex;\n\n\t\t\t\t\tvar field = x ? \"x\" : \"y\";\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], valueMap[field] || false);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]);\n\t\t\t\t} else\n\t\t\t\t\tthrow \"Invalid timeline type for a bone: \" + timelineName + \" (\" + boneName + \")\";\n\t\t\t}\n\t\t}\n\n\t\tvar ikMap = map[\"ik\"];\n\t\tfor (var ikConstraintName in ikMap) {\n\t\t\tif (!ikMap.hasOwnProperty(ikConstraintName)) continue;\n\t\t\tvar ikConstraint = skeletonData.findIkConstraint(ikConstraintName);\n\t\t\tvar values = ikMap[ikConstraintName];\n\t\t\tvar timeline = new spine.IkConstraintTimeline(values.length);\n\t\t\ttimeline.ikConstraintIndex = skeletonData.ikConstraints.indexOf(ikConstraint);\n\t\t\tvar frameIndex = 0;\n\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\tvar valueMap = values[i];\n\t\t\t\tvar mix = valueMap.hasOwnProperty(\"mix\") ? valueMap[\"mix\"] : 1;\n\t\t\t\tvar bendDirection = (!valueMap.hasOwnProperty(\"bendPositive\") || valueMap[\"bendPositive\"]) ? 1 : -1;\n\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], mix, bendDirection);\n\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\tframeIndex++;\n\t\t\t}\n\t\t\ttimelines.push(timeline);\n\t\t\tduration = Math.max(duration, timeline.frames[timeline.frameCount * 3 - 3]);\n\t\t}\n\n\t\tvar ffd = map[\"ffd\"];\n\t\tfor (var skinName in ffd) {\n\t\t\tvar skin = skeletonData.findSkin(skinName);\n\t\t\tvar slotMap = ffd[skinName];\n\t\t\tfor (slotName in slotMap) {\n\t\t\t\tvar slotIndex = skeletonData.findSlotIndex(slotName);\n\t\t\t\tvar meshMap = slotMap[slotName];\n\t\t\t\tfor (var meshName in meshMap) {\n\t\t\t\t\tvar values = meshMap[meshName];\n\t\t\t\t\tvar timeline = new spine.FfdTimeline(values.length);\n\t\t\t\t\tvar attachment = skin.getAttachment(slotIndex, meshName);\n\t\t\t\t\tif (!attachment) throw \"FFD attachment not found: \" + meshName;\n\t\t\t\t\ttimeline.slotIndex = slotIndex;\n\t\t\t\t\ttimeline.attachment = attachment;\n\n\t\t\t\t\tvar isMesh = attachment.type == spine.AttachmentType.mesh;\n\t\t\t\t\tvar vertexCount;\n\t\t\t\t\tif (isMesh)\n\t\t\t\t\t\tvertexCount = attachment.vertices.length;\n\t\t\t\t\telse\n\t\t\t\t\t\tvertexCount = attachment.weights.length / 3 * 2;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\tvar vertices;\n\t\t\t\t\t\tif (!valueMap[\"vertices\"]) {\n\t\t\t\t\t\t\tif (isMesh)\n\t\t\t\t\t\t\t\tvertices = attachment.vertices;\n\t\t\t\t\t\t\telse {\n\t\t\t\t\t\t\t\tvertices = [];\n\t\t\t\t\t\t\t\tvertices.length = vertexCount;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tvar verticesValue = valueMap[\"vertices\"];\n\t\t\t\t\t\t\tvar vertices = [];\n\t\t\t\t\t\t\tvertices.length = vertexCount;\n\t\t\t\t\t\t\tvar start = valueMap[\"offset\"] || 0;\n\t\t\t\t\t\t\tvar nn = verticesValue.length;\n\t\t\t\t\t\t\tif (this.scale == 1) {\n\t\t\t\t\t\t\t\tfor (var ii = 0; ii < nn; ii++)\n\t\t\t\t\t\t\t\t\tvertices[ii + start] = verticesValue[ii];\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tfor (var ii = 0; ii < nn; ii++)\n\t\t\t\t\t\t\t\t\tvertices[ii + start] = verticesValue[ii] * this.scale;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tif (isMesh) {\n\t\t\t\t\t\t\t\tvar meshVertices = attachment.vertices;\n\t\t\t\t\t\t\t\tfor (var ii = 0, nn = vertices.length; ii < nn; ii++)\n\t\t\t\t\t\t\t\t\tvertices[ii] += meshVertices[ii];\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], vertices);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines[timelines.length] = timeline;\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.frameCount - 1]);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tvar drawOrderValues = map[\"drawOrder\"];\n\t\tif (!drawOrderValues) drawOrderValues = map[\"draworder\"];\n\t\tif (drawOrderValues) {\n\t\t\tvar timeline = new spine.DrawOrderTimeline(drawOrderValues.length);\n\t\t\tvar slotCount = skeletonData.slots.length;\n\t\t\tvar frameIndex = 0;\n\t\t\tfor (var i = 0, n = drawOrderValues.length; i < n; i++) {\n\t\t\t\tvar drawOrderMap = drawOrderValues[i];\n\t\t\t\tvar drawOrder = null;\n\t\t\t\tif (drawOrderMap[\"offsets\"]) {\n\t\t\t\t\tdrawOrder = [];\n\t\t\t\t\tdrawOrder.length = slotCount;\n\t\t\t\t\tfor (var ii = slotCount - 1; ii >= 0; ii--)\n\t\t\t\t\t\tdrawOrder[ii] = -1;\n\t\t\t\t\tvar offsets = drawOrderMap[\"offsets\"];\n\t\t\t\t\tvar unchanged = [];\n\t\t\t\t\tunchanged.length = slotCount - offsets.length;\n\t\t\t\t\tvar originalIndex = 0, unchangedIndex = 0;\n\t\t\t\t\tfor (var ii = 0, nn = offsets.length; ii < nn; ii++) {\n\t\t\t\t\t\tvar offsetMap = offsets[ii];\n\t\t\t\t\t\tvar slotIndex = skeletonData.findSlotIndex(offsetMap[\"slot\"]);\n\t\t\t\t\t\tif (slotIndex == -1) throw \"Slot not found: \" + offsetMap[\"slot\"];\n\t\t\t\t\t\t// Collect unchanged items.\n\t\t\t\t\t\twhile (originalIndex != slotIndex)\n\t\t\t\t\t\t\tunchanged[unchangedIndex++] = originalIndex++;\n\t\t\t\t\t\t// Set changed items.\n\t\t\t\t\t\tdrawOrder[originalIndex + offsetMap[\"offset\"]] = originalIndex++;\n\t\t\t\t\t}\n\t\t\t\t\t// Collect remaining unchanged items.\n\t\t\t\t\twhile (originalIndex < slotCount)\n\t\t\t\t\t\tunchanged[unchangedIndex++] = originalIndex++;\n\t\t\t\t\t// Fill in unchanged items.\n\t\t\t\t\tfor (var ii = slotCount - 1; ii >= 0; ii--)\n\t\t\t\t\t\tif (drawOrder[ii] == -1) drawOrder[ii] = unchanged[--unchangedIndex];\n\t\t\t\t}\n\t\t\t\ttimeline.setFrame(frameIndex++, drawOrderMap[\"time\"], drawOrder);\n\t\t\t}\n\t\t\ttimelines.push(timeline);\n\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]);\n\t\t}\n\n\t\tvar events = map[\"events\"];\n\t\tif (events) {\n\t\t\tvar timeline = new spine.EventTimeline(events.length);\n\t\t\tvar frameIndex = 0;\n\t\t\tfor (var i = 0, n = events.length; i < n; i++) {\n\t\t\t\tvar eventMap = events[i];\n\t\t\t\tvar eventData = skeletonData.findEvent(eventMap[\"name\"]);\n\t\t\t\tif (!eventData) throw \"Event not found: \" + eventMap[\"name\"];\n\t\t\t\tvar event = new spine.Event(eventData);\n\t\t\t\tevent.intValue = eventMap.hasOwnProperty(\"int\") ? eventMap[\"int\"] : eventData.intValue;\n\t\t\t\tevent.floatValue = eventMap.hasOwnProperty(\"float\") ? eventMap[\"float\"] : eventData.floatValue;\n\t\t\t\tevent.stringValue = eventMap.hasOwnProperty(\"string\") ? eventMap[\"string\"] : eventData.stringValue;\n\t\t\t\ttimeline.setFrame(frameIndex++, eventMap[\"time\"], event);\n\t\t\t}\n\t\t\ttimelines.push(timeline);\n\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]);\n\t\t}\n\n\t\tskeletonData.animations.push(new spine.Animation(name, timelines, duration));\n\t},\n\treadCurve: function (timeline, frameIndex, valueMap) {\n\t\tvar curve = valueMap[\"curve\"];\n\t\tif (!curve)\n\t\t\ttimeline.curves.setLinear(frameIndex);\n\t\telse if (curve == \"stepped\")\n\t\t\ttimeline.curves.setStepped(frameIndex);\n\t\telse if (curve instanceof Array)\n\t\t\ttimeline.curves.setCurve(frameIndex, curve[0], curve[1], curve[2], curve[3]);\n\t},\n\ttoColor: function (hexString, colorIndex) {\n\t\tif (hexString.length != 8) throw \"Color hexidecimal length must be 8, recieved: \" + hexString;\n\t\treturn parseInt(hexString.substring(colorIndex * 2, (colorIndex * 2) + 2), 16) / 255;\n\t},\n\tgetFloatArray: function (map, name, scale) {\n\t\tvar list = map[name];\n\t\tvar values = new spine.Float32Array(list.length);\n\t\tvar i = 0, n = list.length;\n\t\tif (scale == 1) {\n\t\t\tfor (; i < n; i++)\n\t\t\t\tvalues[i] = list[i];\n\t\t} else {\n\t\t\tfor (; i < n; i++)\n\t\t\t\tvalues[i] = list[i] * scale;\n\t\t}\n\t\treturn values;\n\t},\n\tgetIntArray: function (map, name) {\n\t\tvar list = map[name];\n\t\tvar values = new spine.Uint16Array(list.length);\n\t\tfor (var i = 0, n = list.length; i < n; i++)\n\t\t\tvalues[i] = list[i] | 0;\n\t\treturn values;\n\t}\n};\n\nspine.Atlas = function (atlasText, textureLoader) {\n\tthis.textureLoader = textureLoader;\n\tthis.pages = [];\n\tthis.regions = [];\n\n\tvar reader = new spine.AtlasReader(atlasText);\n\tvar tuple = [];\n\ttuple.length = 4;\n\tvar page = null;\n\twhile (true) {\n\t\tvar line = reader.readLine();\n\t\tif (line === null) break;\n\t\tline = reader.trim(line);\n\t\tif (!line.length)\n\t\t\tpage = null;\n\t\telse if (!page) {\n\t\t\tpage = new spine.AtlasPage();\n\t\t\tpage.name = line;\n\n\t\t\tif (reader.readTuple(tuple) == 2) { // size is only optional for an atlas packed with an old TexturePacker.\n\t\t\t\tpage.width = parseInt(tuple[0]);\n\t\t\t\tpage.height = parseInt(tuple[1]);\n\t\t\t\treader.readTuple(tuple);\n\t\t\t}\n\t\t\tpage.format = spine.Atlas.Format[tuple[0]];\n\n\t\t\treader.readTuple(tuple);\n\t\t\tpage.minFilter = spine.Atlas.TextureFilter[tuple[0]];\n\t\t\tpage.magFilter = spine.Atlas.TextureFilter[tuple[1]];\n\n\t\t\tvar direction = reader.readValue();\n\t\t\tpage.uWrap = spine.Atlas.TextureWrap.clampToEdge;\n\t\t\tpage.vWrap = spine.Atlas.TextureWrap.clampToEdge;\n\t\t\tif (direction == \"x\")\n\t\t\t\tpage.uWrap = spine.Atlas.TextureWrap.repeat;\n\t\t\telse if (direction == \"y\")\n\t\t\t\tpage.vWrap = spine.Atlas.TextureWrap.repeat;\n\t\t\telse if (direction == \"xy\")\n\t\t\t\tpage.uWrap = page.vWrap = spine.Atlas.TextureWrap.repeat;\n\n\t\t\ttextureLoader.load(page, line, this);\n\n\t\t\tthis.pages.push(page);\n\n\t\t} else {\n\t\t\tvar region = new spine.AtlasRegion();\n\t\t\tregion.name = line;\n\t\t\tregion.page = page;\n\n\t\t\tregion.rotate = reader.readValue() == \"true\";\n\n\t\t\treader.readTuple(tuple);\n\t\t\tvar x = parseInt(tuple[0]);\n\t\t\tvar y = parseInt(tuple[1]);\n\n\t\t\treader.readTuple(tuple);\n\t\t\tvar width = parseInt(tuple[0]);\n\t\t\tvar height = parseInt(tuple[1]);\n\n\t\t\tregion.u = x / page.width;\n\t\t\tregion.v = y / page.height;\n\t\t\tif (region.rotate) {\n\t\t\t\tregion.u2 = (x + height) / page.width;\n\t\t\t\tregion.v2 = (y + width) / page.height;\n\t\t\t} else {\n\t\t\t\tregion.u2 = (x + width) / page.width;\n\t\t\t\tregion.v2 = (y + height) / page.height;\n\t\t\t}\n\t\t\tregion.x = x;\n\t\t\tregion.y = y;\n\t\t\tregion.width = Math.abs(width);\n\t\t\tregion.height = Math.abs(height);\n\n\t\t\tif (reader.readTuple(tuple) == 4) { // split is optional\n\t\t\t\tregion.splits = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])];\n\n\t\t\t\tif (reader.readTuple(tuple) == 4) { // pad is optional, but only present with splits\n\t\t\t\t\tregion.pads = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])];\n\n\t\t\t\t\treader.readTuple(tuple);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tregion.originalWidth = parseInt(tuple[0]);\n\t\t\tregion.originalHeight = parseInt(tuple[1]);\n\n\t\t\treader.readTuple(tuple);\n\t\t\tregion.offsetX = parseInt(tuple[0]);\n\t\t\tregion.offsetY = parseInt(tuple[1]);\n\n\t\t\tregion.index = parseInt(reader.readValue());\n\n\t\t\tthis.regions.push(region);\n\t\t}\n\t}\n};\nspine.Atlas.prototype = {\n\tfindRegion: function (name) {\n\t\tvar regions = this.regions;\n\t\tfor (var i = 0, n = regions.length; i < n; i++)\n\t\t\tif (regions[i].name == name) return regions[i];\n\t\treturn null;\n\t},\n\tdispose: function () {\n\t\tvar pages = this.pages;\n\t\tfor (var i = 0, n = pages.length; i < n; i++)\n\t\t\tthis.textureLoader.unload(pages[i].rendererObject);\n\t},\n\tupdateUVs: function (page) {\n\t\tvar regions = this.regions;\n\t\tfor (var i = 0, n = regions.length; i < n; i++) {\n\t\t\tvar region = regions[i];\n\t\t\tif (region.page != page) continue;\n\t\t\tregion.u = region.x / page.width;\n\t\t\tregion.v = region.y / page.height;\n\t\t\tif (region.rotate) {\n\t\t\t\tregion.u2 = (region.x + region.height) / page.width;\n\t\t\t\tregion.v2 = (region.y + region.width) / page.height;\n\t\t\t} else {\n\t\t\t\tregion.u2 = (region.x + region.width) / page.width;\n\t\t\t\tregion.v2 = (region.y + region.height) / page.height;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.Atlas.Format = {\n\talpha: 0,\n\tintensity: 1,\n\tluminanceAlpha: 2,\n\trgb565: 3,\n\trgba4444: 4,\n\trgb888: 5,\n\trgba8888: 6\n};\n\nspine.Atlas.TextureFilter = {\n\tnearest: 0,\n\tlinear: 1,\n\tmipMap: 2,\n\tmipMapNearestNearest: 3,\n\tmipMapLinearNearest: 4,\n\tmipMapNearestLinear: 5,\n\tmipMapLinearLinear: 6\n};\n\nspine.Atlas.TextureWrap = {\n\tmirroredRepeat: 0,\n\tclampToEdge: 1,\n\trepeat: 2\n};\n\nspine.AtlasPage = function () {};\nspine.AtlasPage.prototype = {\n\tname: null,\n\tformat: null,\n\tminFilter: null,\n\tmagFilter: null,\n\tuWrap: null,\n\tvWrap: null,\n\trendererObject: null,\n\twidth: 0,\n\theight: 0\n};\n\nspine.AtlasRegion = function () {};\nspine.AtlasRegion.prototype = {\n\tpage: null,\n\tname: null,\n\tx: 0, y: 0,\n\twidth: 0, height: 0,\n\tu: 0, v: 0, u2: 0, v2: 0,\n\toffsetX: 0, offsetY: 0,\n\toriginalWidth: 0, originalHeight: 0,\n\tindex: 0,\n\trotate: false,\n\tsplits: null,\n\tpads: null\n};\n\nspine.AtlasReader = function (text) {\n\tthis.lines = text.split(/\\r\\n|\\r|\\n/);\n};\nspine.AtlasReader.prototype = {\n\tindex: 0,\n\ttrim: function (value) {\n\t\treturn value.replace(/^\\s+|\\s+$/g, \"\");\n\t},\n\treadLine: function () {\n\t\tif (this.index >= this.lines.length) return null;\n\t\treturn this.lines[this.index++];\n\t},\n\treadValue: function () {\n\t\tvar line = this.readLine();\n\t\tvar colon = line.indexOf(\":\");\n\t\tif (colon == -1) throw \"Invalid line: \" + line;\n\t\treturn this.trim(line.substring(colon + 1));\n\t},\n\t/** Returns the number of tuple values read (1, 2 or 4). */\n\treadTuple: function (tuple) {\n\t\tvar line = this.readLine();\n\t\tvar colon = line.indexOf(\":\");\n\t\tif (colon == -1) throw \"Invalid line: \" + line;\n\t\tvar i = 0, lastMatch = colon + 1;\n\t\tfor (; i < 3; i++) {\n\t\t\tvar comma = line.indexOf(\",\", lastMatch);\n\t\t\tif (comma == -1) break;\n\t\t\ttuple[i] = this.trim(line.substr(lastMatch, comma - lastMatch));\n\t\t\tlastMatch = comma + 1;\n\t\t}\n\t\ttuple[i] = this.trim(line.substring(lastMatch));\n\t\treturn i + 1;\n\t}\n};\n\nspine.AtlasAttachmentLoader = function (atlas) {\n\tthis.atlas = atlas;\n};\nspine.AtlasAttachmentLoader.prototype = {\n\tnewRegionAttachment: function (skin, name, path) {\n\t\tvar region = this.atlas.findRegion(path);\n\t\tif (!region) throw \"Region not found in atlas: \" + path + \" (region attachment: \" + name + \")\";\n\t\tvar attachment = new spine.RegionAttachment(name);\n\t\tattachment.rendererObject = region;\n\t\tattachment.setUVs(region.u, region.v, region.u2, region.v2, region.rotate);\n\t\tattachment.regionOffsetX = region.offsetX;\n\t\tattachment.regionOffsetY = region.offsetY;\n\t\tattachment.regionWidth = region.width;\n\t\tattachment.regionHeight = region.height;\n\t\tattachment.regionOriginalWidth = region.originalWidth;\n\t\tattachment.regionOriginalHeight = region.originalHeight;\n\t\treturn attachment;\n\t},\n\tnewMeshAttachment: function (skin, name, path) {\n\t\tvar region = this.atlas.findRegion(path);\n\t\tif (!region) throw \"Region not found in atlas: \" + path + \" (mesh attachment: \" + name + \")\";\n\t\tvar attachment = new spine.MeshAttachment(name);\n\t\tattachment.rendererObject = region;\n\t\tattachment.regionU = region.u;\n\t\tattachment.regionV = region.v;\n\t\tattachment.regionU2 = region.u2;\n\t\tattachment.regionV2 = region.v2;\n\t\tattachment.regionRotate = region.rotate;\n\t\tattachment.regionOffsetX = region.offsetX;\n\t\tattachment.regionOffsetY = region.offsetY;\n\t\tattachment.regionWidth = region.width;\n\t\tattachment.regionHeight = region.height;\n\t\tattachment.regionOriginalWidth = region.originalWidth;\n\t\tattachment.regionOriginalHeight = region.originalHeight;\n\t\treturn attachment;\n\t},\n\tnewSkinnedMeshAttachment: function (skin, name, path) {\n\t\tvar region = this.atlas.findRegion(path);\n\t\tif (!region) throw \"Region not found in atlas: \" + path + \" (skinned mesh attachment: \" + name + \")\";\n\t\tvar attachment = new spine.SkinnedMeshAttachment(name);\n\t\tattachment.rendererObject = region;\n\t\tattachment.regionU = region.u;\n\t\tattachment.regionV = region.v;\n\t\tattachment.regionU2 = region.u2;\n\t\tattachment.regionV2 = region.v2;\n\t\tattachment.regionRotate = region.rotate;\n\t\tattachment.regionOffsetX = region.offsetX;\n\t\tattachment.regionOffsetY = region.offsetY;\n\t\tattachment.regionWidth = region.width;\n\t\tattachment.regionHeight = region.height;\n\t\tattachment.regionOriginalWidth = region.originalWidth;\n\t\tattachment.regionOriginalHeight = region.originalHeight;\n\t\treturn attachment;\n\t},\n\tnewBoundingBoxAttachment: function (skin, name) {\n\t\treturn new spine.BoundingBoxAttachment(name);\n\t}\n};\n\nspine.SkeletonBounds = function () {\n\tthis.polygonPool = [];\n\tthis.polygons = [];\n\tthis.boundingBoxes = [];\n};\nspine.SkeletonBounds.prototype = {\n\tminX: 0, minY: 0, maxX: 0, maxY: 0,\n\tupdate: function (skeleton, updateAabb) {\n\t\tvar slots = skeleton.slots;\n\t\tvar slotCount = slots.length;\n\t\tvar x = skeleton.x, y = skeleton.y;\n\t\tvar boundingBoxes = this.boundingBoxes;\n\t\tvar polygonPool = this.polygonPool;\n\t\tvar polygons = this.polygons;\n\n\t\tboundingBoxes.length = 0;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++)\n\t\t\tpolygonPool.push(polygons[i]);\n\t\tpolygons.length = 0;\n\n\t\tfor (var i = 0; i < slotCount; i++) {\n\t\t\tvar slot = slots[i];\n\t\t\tvar boundingBox = slot.attachment;\n\t\t\tif (boundingBox.type != spine.AttachmentType.boundingbox) continue;\n\t\t\tboundingBoxes.push(boundingBox);\n\n\t\t\tvar poolCount = polygonPool.length, polygon;\n\t\t\tif (poolCount > 0) {\n\t\t\t\tpolygon = polygonPool[poolCount - 1];\n\t\t\t\tpolygonPool.splice(poolCount - 1, 1);\n\t\t\t} else\n\t\t\t\tpolygon = [];\n\t\t\tpolygons.push(polygon);\n\n\t\t\tpolygon.length = boundingBox.vertices.length;\n\t\t\tboundingBox.computeWorldVertices(x, y, slot.bone, polygon);\n\t\t}\n\n\t\tif (updateAabb) this.aabbCompute();\n\t},\n\taabbCompute: function () {\n\t\tvar polygons = this.polygons;\n\t\tvar minX = Number.MAX_VALUE, minY = Number.MAX_VALUE, maxX = Number.MIN_VALUE, maxY = Number.MIN_VALUE;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++) {\n\t\t\tvar vertices = polygons[i];\n\t\t\tfor (var ii = 0, nn = vertices.length; ii < nn; ii += 2) {\n\t\t\t\tvar x = vertices[ii];\n\t\t\t\tvar y = vertices[ii + 1];\n\t\t\t\tminX = Math.min(minX, x);\n\t\t\t\tminY = Math.min(minY, y);\n\t\t\t\tmaxX = Math.max(maxX, x);\n\t\t\t\tmaxY = Math.max(maxY, y);\n\t\t\t}\n\t\t}\n\t\tthis.minX = minX;\n\t\tthis.minY = minY;\n\t\tthis.maxX = maxX;\n\t\tthis.maxY = maxY;\n\t},\n\t/** Returns true if the axis aligned bounding box contains the point. */\n\taabbContainsPoint: function (x, y) {\n\t\treturn x >= this.minX && x <= this.maxX && y >= this.minY && y <= this.maxY;\n\t},\n\t/** Returns true if the axis aligned bounding box intersects the line segment. */\n\taabbIntersectsSegment: function (x1, y1, x2, y2) {\n\t\tvar minX = this.minX, minY = this.minY, maxX = this.maxX, maxY = this.maxY;\n\t\tif ((x1 <= minX && x2 <= minX) || (y1 <= minY && y2 <= minY) || (x1 >= maxX && x2 >= maxX) || (y1 >= maxY && y2 >= maxY))\n\t\t\treturn false;\n\t\tvar m = (y2 - y1) / (x2 - x1);\n\t\tvar y = m * (minX - x1) + y1;\n\t\tif (y > minY && y < maxY) return true;\n\t\ty = m * (maxX - x1) + y1;\n\t\tif (y > minY && y < maxY) return true;\n\t\tvar x = (minY - y1) / m + x1;\n\t\tif (x > minX && x < maxX) return true;\n\t\tx = (maxY - y1) / m + x1;\n\t\tif (x > minX && x < maxX) return true;\n\t\treturn false;\n\t},\n\t/** Returns true if the axis aligned bounding box intersects the axis aligned bounding box of the specified bounds. */\n\taabbIntersectsSkeleton: function (bounds) {\n\t\treturn this.minX < bounds.maxX && this.maxX > bounds.minX && this.minY < bounds.maxY && this.maxY > bounds.minY;\n\t},\n\t/** Returns the first bounding box attachment that contains the point, or null. When doing many checks, it is usually more\n\t * efficient to only call this method if {@link #aabbContainsPoint(float, float)} returns true. */\n\tcontainsPoint: function (x, y) {\n\t\tvar polygons = this.polygons;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++)\n\t\t\tif (this.polygonContainsPoint(polygons[i], x, y)) return this.boundingBoxes[i];\n\t\treturn null;\n\t},\n\t/** Returns the first bounding box attachment that contains the line segment, or null. When doing many checks, it is usually\n\t * more efficient to only call this method if {@link #aabbIntersectsSegment(float, float, float, float)} returns true. */\n\tintersectsSegment: function (x1, y1, x2, y2) {\n\t\tvar polygons = this.polygons;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++)\n\t\t\tif (polygons[i].intersectsSegment(x1, y1, x2, y2)) return this.boundingBoxes[i];\n\t\treturn null;\n\t},\n\t/** Returns true if the polygon contains the point. */\n\tpolygonContainsPoint: function (polygon, x, y) {\n\t\tvar nn = polygon.length;\n\t\tvar prevIndex = nn - 2;\n\t\tvar inside = false;\n\t\tfor (var ii = 0; ii < nn; ii += 2) {\n\t\t\tvar vertexY = polygon[ii + 1];\n\t\t\tvar prevY = polygon[prevIndex + 1];\n\t\t\tif ((vertexY < y && prevY >= y) || (prevY < y && vertexY >= y)) {\n\t\t\t\tvar vertexX = polygon[ii];\n\t\t\t\tif (vertexX + (y - vertexY) / (prevY - vertexY) * (polygon[prevIndex] - vertexX) < x) inside = !inside;\n\t\t\t}\n\t\t\tprevIndex = ii;\n\t\t}\n\t\treturn inside;\n\t},\n\t/** Returns true if the polygon contains the line segment. */\n\tpolygonIntersectsSegment: function (polygon, x1, y1, x2, y2) {\n\t\tvar nn = polygon.length;\n\t\tvar width12 = x1 - x2, height12 = y1 - y2;\n\t\tvar det1 = x1 * y2 - y1 * x2;\n\t\tvar x3 = polygon[nn - 2], y3 = polygon[nn - 1];\n\t\tfor (var ii = 0; ii < nn; ii += 2) {\n\t\t\tvar x4 = polygon[ii], y4 = polygon[ii + 1];\n\t\t\tvar det2 = x3 * y4 - y3 * x4;\n\t\t\tvar width34 = x3 - x4, height34 = y3 - y4;\n\t\t\tvar det3 = width12 * height34 - height12 * width34;\n\t\t\tvar x = (det1 * width34 - width12 * det2) / det3;\n\t\t\tif (((x >= x3 && x <= x4) || (x >= x4 && x <= x3)) && ((x >= x1 && x <= x2) || (x >= x2 && x <= x1))) {\n\t\t\t\tvar y = (det1 * height34 - height12 * det2) / det3;\n\t\t\t\tif (((y >= y3 && y <= y4) || (y >= y4 && y <= y3)) && ((y >= y1 && y <= y2) || (y >= y2 && y <= y1))) return true;\n\t\t\t}\n\t\t\tx3 = x4;\n\t\t\ty3 = y4;\n\t\t}\n\t\treturn false;\n\t},\n\tgetPolygon: function (attachment) {\n\t\tvar index = this.boundingBoxes.indexOf(attachment);\n\t\treturn index == -1 ? null : this.polygons[index];\n\t},\n\tgetWidth: function () {\n\t\treturn this.maxX - this.minX;\n\t},\n\tgetHeight: function () {\n\t\treturn this.maxY - this.minY;\n\t}\n};\n","/**\n * @file        Main export of the PIXI spine library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    Spine:              require('./Spine')\n};\n","var core = require('../core');\n\n/**\n * A BitmapText object will create a line or multiple lines of text using bitmap font. To\n * split a line you can use '\\n', '\\r' or '\\r\\n' in your string. You can generate the fnt files using:\n *\n * http://www.angelcode.com/products/bmfont/ for windows or\n * http://www.bmglyph.com/ for mac.\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param text {string} The copy that you would like the text to display\n * @param style {object} The style parameters\n * @param style.font {string} The size (optional) and bitmap font id (required) eq 'Arial' or '20px Arial' (must have loaded previously)\n * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text\n */\nfunction BitmapText(text, style) {\n    core.DisplayObjectContainer.call(this);\n\n    /**\n     * The width of the overall text, different from fontSize,\n     * which is defined in the style object\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.textWidth = 0;\n\n    /**\n     * The height of the overall text, different from fontSize,\n     * which is defined in the style object\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.textHeight = 0;\n\n    /**\n     * Private tracker for the letter sprite pool.\n     *\n     * @member {Sprite[]}\n     * @private\n     */\n    this._pool = [];\n\n    /**\n     * Private tracker for the current style.\n     *\n     * @member {object}\n     * @private\n     */\n    this._style = {\n        tint: style.tint,\n        align: style.align,\n        fontName: null,\n        fontSize: 0\n    };\n    this.font = style.font; // run font setter\n\n    /**\n     * Private tracker for the current text.\n     *\n     * @member {string}\n     * @private\n     */\n    this._text = text;\n\n    /**\n     * The dirty state of this object.\n     *\n     * @member {boolean}\n     */\n    this.dirty = false;\n\n    this.updateText();\n}\n\n// constructor\nBitmapText.prototype = Object.create(core.DisplayObjectContainer.prototype);\nBitmapText.prototype.constructor = BitmapText;\nmodule.exports = BitmapText;\n\nObject.defineProperties(BitmapText.prototype, {\n    /**\n     * The tint of the BitmapText object\n     *\n     * @member {number}\n     * @memberof BitmapText#\n     */\n    tint: {\n        get: function () {\n            return this._style.tint;\n        },\n        set: function (value) {\n            this._style.tint = (typeof value === 'number' && value >= 0) ? value : 0xFFFFFF;\n\n            this.dirty = true;\n        }\n    },\n\n    /**\n     * The tint of the BitmapText object\n     *\n     * @member {string}\n     * @default 'left'\n     * @memberof BitmapText#\n     */\n    align: {\n        get: function () {\n            return this._style.align;\n        },\n        set: function (value) {\n            this._style.align = value;\n\n            this.dirty = true;\n        }\n    },\n\n    /**\n     * The tint of the BitmapText object\n     *\n     * @member {Font}\n     * @memberof BitmapText#\n     */\n    font: {\n        get: function () {\n            return this._style.font;\n        },\n        set: function (value) {\n            value = value.split(' ');\n\n            // TODO - This should be object-based not string based like it has been.\n            this._style.fontName = value[value.length - 1];\n            this._style.fontSize = value.length >= 2 ? parseInt(value[value.length - 2], 10) : BitmapText.fonts[this.fontName].size;\n\n            this.dirty = true;\n        }\n    },\n\n    /**\n     * The text of the BitmapText object\n     *\n     * @member {string}\n     * @memberof BitmapText#\n     */\n    text: {\n        get: function () {\n            return this._text;\n        },\n        set: function (value) {\n            this._text = value;\n\n            this.dirty = true;\n        }\n    }\n});\n\n/**\n * Renders text and updates it when needed\n *\n * @private\n */\nBitmapText.prototype.updateText = function () {\n    var data = BitmapText.fonts[this.fontName];\n    var pos = new core.math.Point();\n    var prevCharCode = null;\n    var chars = [];\n    var maxLineWidth = 0;\n    var lineWidths = [];\n    var line = 0;\n    var scale = this.fontSize / data.size;\n\n    for (var i = 0; i < this.text.length; i++) {\n        var charCode = this.text.charCodeAt(i);\n\n        if (/(?:\\r\\n|\\r|\\n)/.test(this.text.charAt(i))) {\n            lineWidths.push(pos.x);\n            maxLineWidth = Math.max(maxLineWidth, pos.x);\n            line++;\n\n            pos.x = 0;\n            pos.y += data.lineHeight;\n            prevCharCode = null;\n            continue;\n        }\n\n        var charData = data.chars[charCode];\n\n        if (!charData) {\n            continue;\n        }\n\n        if (prevCharCode && charData.kerning[prevCharCode]) {\n            pos.x += charData.kerning[prevCharCode];\n        }\n\n        chars.push({texture:charData.texture, line: line, charCode: charCode, position: new core.math.Point(pos.x + charData.xOffset, pos.y + charData.yOffset)});\n        pos.x += charData.xAdvance;\n\n        prevCharCode = charCode;\n    }\n\n    lineWidths.push(pos.x);\n    maxLineWidth = Math.max(maxLineWidth, pos.x);\n\n    var lineAlignOffsets = [];\n\n    for (i = 0; i <= line; i++) {\n        var alignOffset = 0;\n\n        if (this.style.align === 'right') {\n            alignOffset = maxLineWidth - lineWidths[i];\n        }\n        else if (this.style.align === 'center') {\n            alignOffset = (maxLineWidth - lineWidths[i]) / 2;\n        }\n\n        lineAlignOffsets.push(alignOffset);\n    }\n\n    var lenChildren = this.children.length;\n    var lenChars = chars.length;\n    var tint = this.tint;\n\n    for (i = 0; i < lenChars; i++) {\n        var c = i < lenChildren ? this.children[i] : this._pool.pop(); // get old child if have. if not - take from pool.\n\n        if (c) {\n            c.setTexture(chars[i].texture); // check if got one before.\n        }\n        else {\n            c = new core.Sprite(chars[i].texture); // if no create new one.\n        }\n\n        c.position.x = (chars[i].position.x + lineAlignOffsets[chars[i].line]) * scale;\n        c.position.y = chars[i].position.y * scale;\n        c.scale.x = c.scale.y = scale;\n        c.tint = tint;\n\n        if (!c.parent) {\n            this.addChild(c);\n        }\n    }\n\n    // remove unnecessary children.\n    // and put their into the pool.\n    while(this.children.length > lenChars) {\n        var child = this.getChildAt(this.children.length - 1);\n        this._pool.push(child);\n        this.removeChild(child);\n    }\n\n    this.textWidth = maxLineWidth * scale;\n    this.textHeight = (pos.y + data.lineHeight) * scale;\n};\n\n/**\n * Updates the transform of this object\n *\n * @private\n */\nBitmapText.prototype.updateTransform = function () {\n    if (this.dirty) {\n        this.updateText();\n        this.dirty = false;\n    }\n\n    this.displayObjectContainerUpdateTransform();\n};\n\nBitmapText.fonts = {};\n","var core = require('../core');\n\n/**\n * A Text Object will create a line or multiple lines of text. To split a line you can use '\\n' in your text string,\n * or add a wordWrap property set to true and and wordWrapWidth property with a value in the style object.\n *\n * @class\n * @extends Sprite\n * @namespace PIXI\n * @param text {string} The copy that you would like the text to display\n * @param [style] {object} The style parameters\n * @param [style.font] {string} default 'bold 20px Arial' The style and size of the font\n * @param [style.fill='black'] {String|Number} A canvas fillstyle that will be used on the text e.g 'red', '#00FF00'\n * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text\n * @param [style.stroke] {String|Number} A canvas fillstyle that will be used on the text stroke e.g 'blue', '#FCFF00'\n * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke)\n * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used\n * @param [style.wordWrapWidth=100] {number} The width at which text will wrap, it needs wordWrap to be set to true\n * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text\n * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00'\n * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow\n * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow\n */\nfunction Text(text, style) {\n    /**\n     * The canvas element that everything is drawn to\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.canvas = document.createElement('canvas');\n\n    /**\n     * The canvas 2d context that everything is drawn with\n     * @member {HTMLCanvasElement}\n     */\n    this.context = this.canvas.getContext('2d');\n\n    /**\n     * The resolution of the canvas.\n     * @member {number}\n     */\n    this.resolution = 1;\n\n    core.Sprite.call(this, core.Texture.fromCanvas(this.canvas));\n\n    this.setText(text);\n    this.setStyle(style);\n}\n\n// constructor\nText.prototype = Object.create(core.Sprite.prototype);\nText.prototype.constructor = Text;\nmodule.exports = Text;\n\nObject.defineProperties(Text.prototype, {\n    /**\n     * The width of the Text, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof Text#\n     */\n    width: {\n        get: function () {\n            if (this.dirty) {\n                this.updateText();\n                this.dirty = false;\n            }\n\n            return this.scale.x * this.texture.frame.width;\n        },\n        set: function (value) {\n            this.scale.x = value / this.texture.frame.width;\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the Text, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof Text#\n     */\n    height: {\n        get: function () {\n            if (this.dirty) {\n                this.updateText();\n                this.dirty = false;\n            }\n\n            return  this.scale.y * this.texture.frame.height;\n        },\n        set: function (value) {\n            this.scale.y = value / this.texture.frame.height;\n            this._height = value;\n        }\n    }\n});\n\n/**\n * Set the style of the text\n *\n * @param [style] {object} The style parameters\n * @param [style.font='bold 20pt Arial'] {string} The style and size of the font\n * @param [style.fill='black'] {object} A canvas fillstyle that will be used on the text eg 'red', '#00FF00'\n * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text\n * @param [style.stroke='black'] {string} A canvas fillstyle that will be used on the text stroke eg 'blue', '#FCFF00'\n * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke)\n * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used\n * @param [style.wordWrapWidth=100] {number} The width at which text will wrap\n * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text\n * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00'\n * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow\n * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow\n */\nText.prototype.setStyle = function (style) {\n    style = style || {};\n    style.font = style.font || 'bold 20pt Arial';\n    style.fill = style.fill || 'black';\n    style.align = style.align || 'left';\n    style.stroke = style.stroke || 'black'; //provide a default, see: https://github.com/GoodBoyDigital/pixi.js/issues/136\n    style.strokeThickness = style.strokeThickness || 0;\n    style.wordWrap = style.wordWrap || false;\n    style.wordWrapWidth = style.wordWrapWidth || 100;\n\n    style.dropShadow = style.dropShadow || false;\n    style.dropShadowAngle = style.dropShadowAngle || Math.PI / 6;\n    style.dropShadowDistance = style.dropShadowDistance || 4;\n    style.dropShadowColor = style.dropShadowColor || 'black';\n\n    this.style = style;\n    this.dirty = true;\n};\n\n/**\n * Set the copy for the text object. To split a line you can use '\\n'.\n *\n * @param text {string} The copy that you would like the text to display\n */\nText.prototype.setText = function (text) {\n    this.text = text.toString() || ' ';\n    this.dirty = true;\n};\n\n/**\n * Renders text and updates it when needed\n *\n * @private\n */\nText.prototype.updateText = function () {\n    this.texture.baseTexture.resolution = this.resolution;\n\n    this.context.font = this.style.font;\n\n    var outputText = this.text;\n\n    // word wrap\n    // preserve original text\n    if (this.style.wordWrap) {\n        outputText = this.wordWrap(this.text);\n    }\n\n    //split text into lines\n    var lines = outputText.split(/(?:\\r\\n|\\r|\\n)/);\n\n    //calculate text width\n    var lineWidths = [];\n    var maxLineWidth = 0;\n    var fontProperties = this.determineFontProperties(this.style.font);\n    for (var i = 0; i < lines.length; i++) {\n        var lineWidth = this.context.measureText(lines[i]).width;\n        lineWidths[i] = lineWidth;\n        maxLineWidth = Math.max(maxLineWidth, lineWidth);\n    }\n\n    var width = maxLineWidth + this.style.strokeThickness;\n    if (this.style.dropShadow) {\n        width += this.style.dropShadowDistance;\n    }\n\n    this.canvas.width = ( width + this.context.lineWidth ) * this.resolution;\n\n    //calculate text height\n    var lineHeight = fontProperties.fontSize + this.style.strokeThickness;\n\n    var height = lineHeight * lines.length;\n    if (this.style.dropShadow) {\n        height += this.style.dropShadowDistance;\n    }\n\n    this.canvas.height = height * this.resolution;\n\n    this.context.scale( this.resolution, this.resolution);\n\n    if (navigator.isCocoonJS) {\n        this.context.clearRect(0, 0, this.canvas.width, this.canvas.height);\n    }\n\n    // used for debugging..\n    //this.context.fillStyle =\"#FF0000\"\n    //this.context.fillRect(0, 0, this.canvas.width,this.canvas.height);\n\n    this.context.font = this.style.font;\n    this.context.strokeStyle = this.style.stroke;\n    this.context.lineWidth = this.style.strokeThickness;\n    this.context.textBaseline = 'alphabetic';\n    //this.context.lineJoin = 'round';\n\n    var linePositionX;\n    var linePositionY;\n\n    if (this.style.dropShadow) {\n        this.context.fillStyle = this.style.dropShadowColor;\n\n        var xShadowOffset = Math.cos(this.style.dropShadowAngle) * this.style.dropShadowDistance;\n        var yShadowOffset = Math.sin(this.style.dropShadowAngle) * this.style.dropShadowDistance;\n\n        for (i = 0; i < lines.length; i++) {\n            linePositionX = this.style.strokeThickness / 2;\n            linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent;\n\n            if (this.style.align === 'right') {\n                linePositionX += maxLineWidth - lineWidths[i];\n            }\n            else if (this.style.align === 'center') {\n                linePositionX += (maxLineWidth - lineWidths[i]) / 2;\n            }\n\n            if (this.style.fill) {\n                this.context.fillText(lines[i], linePositionX + xShadowOffset, linePositionY + yShadowOffset);\n            }\n\n          //  if (dropShadow)\n        }\n    }\n\n    //set canvas text styles\n    this.context.fillStyle = this.style.fill;\n\n    //draw lines line by line\n    for (i = 0; i < lines.length; i++) {\n        linePositionX = this.style.strokeThickness / 2;\n        linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent;\n\n        if (this.style.align === 'right') {\n            linePositionX += maxLineWidth - lineWidths[i];\n        }\n        else if (this.style.align === 'center') {\n            linePositionX += (maxLineWidth - lineWidths[i]) / 2;\n        }\n\n        if (this.style.stroke && this.style.strokeThickness) {\n            this.context.strokeText(lines[i], linePositionX, linePositionY);\n        }\n\n        if (this.style.fill) {\n            this.context.fillText(lines[i], linePositionX, linePositionY);\n        }\n\n      //  if (dropShadow)\n    }\n\n    this.updateTexture();\n};\n\n/**\n * Updates texture size based on canvas size\n *\n * @private\n */\nText.prototype.updateTexture = function () {\n    this.texture.baseTexture.width = this.canvas.width;\n    this.texture.baseTexture.height = this.canvas.height;\n    this.texture.crop.width = this.texture.frame.width = this.canvas.width;\n    this.texture.crop.height = this.texture.frame.height = this.canvas.height;\n\n    this._width = this.canvas.width;\n    this._height = this.canvas.height;\n\n    // update the dirty base textures\n    this.texture.baseTexture.dirty();\n};\n\n/**\n * Renders the object using the WebGL renderer\n*\n * @param renderer {WebGLRenderer}\n */\nText.prototype.renderWebGL = function (renderer) {\n    if (this.dirty) {\n        this.resolution = renderer.resolution;\n\n        this.updateText();\n        this.dirty = false;\n    }\n\n    core.Sprite.prototype.renderWebGL.call(this, renderer);\n};\n\n/**\n * Renders the object using the Canvas renderer\n*\n * @param renderer {CanvasRenderer}\n */\nText.prototype.renderCanvas = function (renderer) {\n    if (this.dirty) {\n        this.resolution = renderer.resolution;\n\n        this.updateText();\n        this.dirty = false;\n    }\n\n    core.Sprite.prototype.renderCanvas.call(this, renderer);\n};\n\n/**\n * Calculates the ascent, descent and fontSize of a given fontStyle\n*\n * @param fontStyle {object}\n * @private\n */\nText.prototype.determineFontProperties = function (fontStyle) {\n    var properties = Text.fontPropertiesCache[fontStyle];\n\n    if (!properties) {\n        properties = {};\n\n        var canvas = Text.fontPropertiesCanvas;\n        var context = Text.fontPropertiesContext;\n\n        context.font = fontStyle;\n\n        var width = Math.ceil(context.measureText('|Mq').width);\n        var baseline = Math.ceil(context.measureText('M').width);\n        var height = 2 * baseline;\n\n        baseline = baseline * 1.4 | 0;\n\n        canvas.width = width;\n        canvas.height = height;\n\n        context.fillStyle = '#f00';\n        context.fillRect(0, 0, width, height);\n\n        context.font = fontStyle;\n\n        context.textBaseline = 'alphabetic';\n        context.fillStyle = '#000';\n        context.fillText('|MÉq', 0, baseline);\n\n        var imagedata = context.getImageData(0, 0, width, height).data;\n        var pixels = imagedata.length;\n        var line = width * 4;\n\n        var i, j;\n\n        var idx = 0;\n        var stop = false;\n\n        // ascent. scan from top to bottom until we find a non red pixel\n        for (i = 0; i < baseline; i++) {\n            for (j = 0; j < line; j += 4) {\n                if (imagedata[idx + j] !== 255) {\n                    stop = true;\n                    break;\n                }\n            }\n            if (!stop) {\n                idx += line;\n            }\n            else {\n                break;\n            }\n        }\n\n        properties.ascent = baseline - i;\n\n        idx = pixels - line;\n        stop = false;\n\n        // descent. scan from bottom to top until we find a non red pixel\n        for (i = height; i > baseline; i--) {\n            for (j = 0; j < line; j += 4) {\n                if (imagedata[idx + j] !== 255) {\n                    stop = true;\n                    break;\n                }\n            }\n            if (!stop) {\n                idx -= line;\n            }\n            else {\n                break;\n            }\n        }\n\n        properties.descent = i - baseline;\n        //TODO might need a tweak. kind of a temp fix!\n        properties.descent += 6;\n        properties.fontSize = properties.ascent + properties.descent;\n\n        Text.fontPropertiesCache[fontStyle] = properties;\n    }\n\n    return properties;\n};\n\n/**\n * Applies newlines to a string to have it optimally fit into the horizontal\n * bounds set by the Text object's wordWrapWidth property.\n *\n * @param text {string}\n * @private\n */\nText.prototype.wordWrap = function (text) {\n    // Greedy wrapping algorithm that will wrap words as the line grows longer\n    // than its horizontal bounds.\n    var result = '';\n    var lines = text.split('\\n');\n    for (var i = 0; i < lines.length; i++) {\n        var spaceLeft = this.style.wordWrapWidth;\n        var words = lines[i].split(' ');\n        for (var j = 0; j < words.length; j++) {\n            var wordWidth = this.context.measureText(words[j]).width;\n            var wordWidthWithSpace = wordWidth + this.context.measureText(' ').width;\n            if (j === 0 || wordWidthWithSpace > spaceLeft) {\n                // Skip printing the newline if it's the first word of the line that is\n                // greater than the word wrap width.\n                if (j > 0) {\n                    result += '\\n';\n                }\n                result += words[j];\n                spaceLeft = this.style.wordWrapWidth - wordWidth;\n            }\n            else {\n                spaceLeft -= wordWidthWithSpace;\n                result += ' ' + words[j];\n            }\n        }\n\n        if (i < lines.length-1) {\n            result += '\\n';\n        }\n    }\n    return result;\n};\n\n/**\n * Returns the bounds of the Text as a rectangle. The bounds calculation takes the worldTransform into account.\n *\n * @param matrix {Matrix} the transformation matrix of the Text\n * @return {Rectangle} the framing rectangle\n */\nText.prototype.getBounds = function (matrix) {\n    if (this.dirty) {\n        this.updateText();\n        this.dirty = false;\n    }\n\n    return core.Sprite.prototype.getBounds.call(this, matrix);\n};\n\n/**\n * Destroys this text object.\n *\n * @param destroyBaseTexture {boolean} whether to destroy the base texture as well\n */\nText.prototype.destroy = function (destroyBaseTexture) {\n    // make sure to reset the the context and canvas.. dont want this hanging around in memory!\n    this.context = null;\n    this.canvas = null;\n\n    this.texture.destroy(destroyBaseTexture === undefined ? true : destroyBaseTexture);\n};\n\nText.fontPropertiesCache = {};\nText.fontPropertiesCanvas = document.createElement('canvas');\nText.fontPropertiesContext = Text.fontPropertiesCanvas.getContext('2d');\n","/**\n * @file        Main export of the PIXI text library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    Text:       require('./Text'),\n    BitmapText: require('./BitmapText')\n};\n"]} diff --git a/dist/pixi.js b/dist/pixi.js deleted file mode 100644 index 2e6b3f4..0000000 --- a/dist/pixi.js +++ /dev/null @@ -1,19896 +0,0 @@ -!function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var f;"undefined"!=typeof window?f=window:"undefined"!=typeof global?f=global:"undefined"!=typeof self&&(f=self),f.PIXI=e()}}(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o", - "Richard Davey " - ], - "main": "./src/index.js", - "homepage": "http://goodboydigital.com/", - "bugs": "https://github.com/GoodBoyDigital/pixi.js/issues", - "license": "MIT", - "repository": { - "type": "git", - "url": "https://github.com/GoodBoyDigital/pixi.js.git" - }, - "scripts": { - "test": "gulp test", - "docs": "./node_modules/.bin/jsdoc -c ./gulp/util/jsdoc.conf.json" - }, - "devDependencies": { - "browserify": "^8.0.2", - "chai": "^1.10.0", - "del": "^1.1.0", - "gulp": "^3.8.10", - "gulp-jshint": "^1.9.0", - "gulp-plumber": "^0.6.6", - "gulp-rename": "^1.2.0", - "gulp-uglify": "^1.0.2", - "gulp-util": "^3.0.1", - "ink-docstrap": "^0.4.12", - "jsdoc": "^3.3.0-alpha13", - "jshint-summary": "^0.4.0", - "karma": "^0.12.28", - "karma-firefox-launcher": "^0.1.0", - "karma-mocha": "^0.1.10", - "karma-spec-reporter": "^0.0.16", - "mocha": "^2.1.0", - "require-dir": "^0.1.0", - "run-sequence": "^1.0.2", - "vinyl-buffer": "^1.0.0", - "vinyl-source-stream": "^1.0.0", - "watchify": "^2.2.1" - }, - "dependencies": { - "webgl-enabled": "^1.0.2" - } -} - -},{}],4:[function(require,module,exports){ -/** - * Constant values used in pixi - * - * @mixin const - */ -module.exports = { - /** - * Constant to identify the WEBGL Renderer Type - * - * @static - * @constant - * @property {number} WEBGL_RENDERER - */ - WEBGL_RENDERER: 1, - - /** - * Constant to identify the CANVAS Renderer Type - * - * @static - * @constant - * @property {number} CANVAS_RENDERER - */ - CANVAS_RENDERER: 2, - - /** - * String of the current PIXI version - * - * @static - * @constant - * @property {string} VERSION - */ - VERSION: require('../../package.json').version, - - /** - * Various blend modes supported by PIXI. IMPORTANT - The WebGL renderer only supports - * the NORMAL, ADD, MULTIPLY and SCREEN blend modes. Anything else will silently act like - * NORMAL. - * - * @static - * @constant - * @property {object} blendModes - * @property {number} blendModes.NORMAL - * @property {number} blendModes.ADD - * @property {number} blendModes.MULTIPLY - * @property {number} blendModes.SCREEN - * @property {number} blendModes.OVERLAY - * @property {number} blendModes.DARKEN - * @property {number} blendModes.LIGHTEN - * @property {number} blendModes.COLOR_DODGE - * @property {number} blendModes.COLOR_BURN - * @property {number} blendModes.HARD_LIGHT - * @property {number} blendModes.SOFT_LIGHT - * @property {number} blendModes.DIFFERENCE - * @property {number} blendModes.EXCLUSION - * @property {number} blendModes.HUE - * @property {number} blendModes.SATURATION - * @property {number} blendModes.COLOR - * @property {number} blendModes.LUMINOSITY - */ - blendModes: { - NORMAL: 0, - ADD: 1, - MULTIPLY: 2, - SCREEN: 3, - OVERLAY: 4, - DARKEN: 5, - LIGHTEN: 6, - COLOR_DODGE: 7, - COLOR_BURN: 8, - HARD_LIGHT: 9, - SOFT_LIGHT: 10, - DIFFERENCE: 11, - EXCLUSION: 12, - HUE: 13, - SATURATION: 14, - COLOR: 15, - LUMINOSITY: 16 - }, - - /** - * The scale modes that are supported by pixi. - * - * The DEFAULT scale mode affects the default scaling mode of future operations. - * It can be re-assigned to either LINEAR or NEAREST, depending upon suitability. - * - * @static - * @constant - * @property {object} scaleModes - * @property {number} scaleModes.DEFAULT=LINEAR - * @property {number} scaleModes.LINEAR Smooth scaling - * @property {number} scaleModes.NEAREST Pixelating scaling - */ - scaleModes: { - DEFAULT: 0, - LINEAR: 0, - NEAREST: 1 - }, - - /** - * The prefix that denotes a URL is for a retina asset - * - * @static - * @constant - * @property {string} RETINA_PREFIX - */ - RETINA_PREFIX: '@2x', - - /** - * The default render options if none are supplied to {@link PIXI.WebGLRenderer} - * or {@link PIXI.CanvasRenderer}. - * - * @static - * @constant - * @property {object} defaultRenderOptions - * @property {HTMLCanvasElement} defaultRenderOptions.view=null - * @property {boolean} defaultRenderOptions.transparent=false - * @property {boolean} defaultRenderOptions.antialias=false - * @property {boolean} defaultRenderOptions.preserveDrawingBuffer=false - * @property {number} defaultRenderOptions.resolution=1 - * @property {number} defaultRenderOptions.backgroundColor=0x000000 - * @property {boolean} defaultRenderOptions.clearBeforeRender=true - * @property {boolean} defaultRenderOptions.autoResize=false - */ - defaultRenderOptions: { - view: null, - resolution: 1, - antialias: false, - autoResize: false, - transparent: false, - backgroundColor: 0x000000, - clearBeforeRender: true, - preserveDrawingBuffer: false - }, - - /** - * Constants that identify shapes, mainly to prevent `instanceof` calls. - * - * @static - * @constant - * @property {object} SHAPES - * @property {object} SHAPES.POLY=0 - * @property {object} SHAPES.RECT=1 - * @property {object} SHAPES.CIRC=2 - * @property {object} SHAPES.ELIP=3 - * @property {object} SHAPES.RREC=4 - */ - SHAPES: { - POLY: 0, - RECT: 1, - CIRC: 2, - ELIP: 3, - RREC: 4 - } -}; - -},{"../../package.json":3}],5:[function(require,module,exports){ -var math = require('../math'); - -/** - * The base class for all objects that are rendered on the screen. - * This is an abstract class and should not be used on its own rather it should be extended. - * - * @class - * @namespace PIXI - */ -function DisplayObject() { - /** - * The coordinate of the object relative to the local coordinates of the parent. - * - * @member {Point} - */ - this.position = new math.Point(); - - /** - * The scale factor of the object. - * - * @member {Point} - */ - this.scale = new math.Point(1, 1); - - /** - * The pivot point of the displayObject that it rotates around - * - * @member {Point} - */ - this.pivot = new math.Point(0, 0); - - /** - * The rotation of the object in radians. - * - * @member {number} - */ - this.rotation = 0; - - /** - * The opacity of the object. - * - * @member {number} - */ - this.alpha = 1; - - /** - * The visibility of the object. If false the object will not be drawn, and - * the updateTransform function will not be called. - * - * @member {boolean} - */ - this.visible = true; - - /** - * Can this object be rendered, if false the object will not be drawn but the updateTransform - * methods will still be called. - * - * @member {boolean} - */ - this.renderable = false; - - /** - * The display object container that contains this display object. - * - * @member {DisplayObjectContainer} - * @readOnly - */ - this.parent = null; - - /** - * The multiplied alpha of the displayObject - * - * @member {number} - * @readOnly - */ - this.worldAlpha = 1; - - /** - * Current transform of the object based on world (parent) factors - * - * @member {Matrix} - * @readOnly - */ - this.worldTransform = new math.Matrix(); - - /** - * The area the filter is applied to. This is used as more of an optimisation - * rather than figuring out the dimensions of the displayObject each frame you can set this rectangle - * - * @member {Rectangle} - */ - this.filterArea = null; - - /** - * cached sin rotation - * - * @member {number} - * @private - */ - this._sr = 0; - - /** - * cached cos rotation - * - * @member {number} - * @private - */ - this._cr = 1; - - /** - * The original, cached bounds of the object - * - * @member {Rectangle} - * @private - */ - this._bounds = new math.Rectangle(0, 0, 1, 1); - - /** - * The most up-to-date bounds of the object - * - * @member {Rectangle} - * @private - */ - this._currentBounds = null; - - /** - * The original, cached mask of the object - * - * @member {Rectangle} - * @private - */ - this._mask = null; - - /** - * Cached internal flag. - * - * @member {boolean} - * @private - */ - this._cacheIsDirty = false; -} - -// constructor -DisplayObject.prototype.constructor = DisplayObject; -module.exports = DisplayObject; - -Object.defineProperties(DisplayObject.prototype, { - /** - * The position of the displayObject on the x axis relative to the local coordinates of the parent. - * - * @member {number} - * @memberof DisplayObject# - */ - x: { - get: function () { - return this.position.x; - }, - set: function (value) { - this.position.x = value; - } - }, - - /** - * The position of the displayObject on the y axis relative to the local coordinates of the parent. - * - * @member {number} - * @memberof DisplayObject# - */ - y: { - get: function () { - return this.position.y; - }, - set: function (value) { - this.position.y = value; - } - }, - - /** - * Indicates if the sprite is globally visible. - * - * @member {boolean} - * @memberof DisplayObject# - * @readonly - */ - worldVisible: { - get: function () { - var item = this; - - do { - if (!item.visible) { - return false; - } - - item = item.parent; - } while(item); - - return true; - } - }, - - /** - * Sets a mask for the displayObject. A mask is an object that limits the visibility of an object to the shape of the mask applied to it. - * In PIXI a regular mask must be a PIXI.Graphics object. This allows for much faster masking in canvas as it utilises shape clipping. - * To remove a mask, set this property to null. - * - * @member {Graphics} - * @memberof DisplayObject# - */ - mask: { - get: function () { - return this._mask; - }, - set: function (value) { - if (this._mask) { - this._mask.isMask = false; - } - - this._mask = value; - - if (this._mask) { - this._mask.isMask = true; - } - } - }, - - /** - * Sets the filters for the displayObject. - * * IMPORTANT: This is a webGL only feature and will be ignored by the canvas renderer. - * To remove filters simply set this property to 'null' - * - * @member {Filter[]} - * @memberof DisplayObject# - */ - filters: { - get: function () { - return this._filters; - }, - set: function (value) { - if (value) { - // now put all the passes in one place.. - var passes = []; - - for (var i = 0; i < value.length; i++) { - var filterPasses = value[i].passes; - - for (var j = 0; j < filterPasses.length; j++) { - passes.push(filterPasses[j]); - } - } - - // TODO change this as it is legacy - this._filterBlock = { target: this, filterPasses: passes }; - } - - this._filters = value; - } - } -}); - -/* - * Updates the object transform for rendering - * - * TODO - Optimization pass! - * - * @private - */ -DisplayObject.prototype.updateTransform = function () { - if (!this.parent) { - return; - } - - // create some matrix refs for easy access - var pt = this.parent.worldTransform; - var wt = this.worldTransform; - - // temporary matrix variables - var a, b, c, d, tx, ty; - - // so if rotation is between 0 then we can simplify the multiplication process.. - if (this.rotation % math.PI_2) { - // check to see if the rotation is the same as the previous render. This means we only need to use sin and cos when rotation actually changes - if (this.rotation !== this.rotationCache) { - this.rotationCache = this.rotation; - this._sr = Math.sin(this.rotation); - this._cr = Math.cos(this.rotation); - } - - // get the matrix values of the displayobject based on its transform properties.. - a = this._cr * this.scale.x; - b = this._sr * this.scale.x; - c = -this._sr * this.scale.y; - d = this._cr * this.scale.y; - tx = this.position.x; - ty = this.position.y; - - // check for pivot.. not often used so geared towards that fact! - if (this.pivot.x || this.pivot.y) { - tx -= this.pivot.x * a + this.pivot.y * c; - ty -= this.pivot.x * b + this.pivot.y * d; - } - - // concat the parent matrix with the objects transform. - wt.a = a * pt.a + b * pt.c; - wt.b = a * pt.b + b * pt.d; - wt.c = c * pt.a + d * pt.c; - wt.d = c * pt.b + d * pt.d; - wt.tx = tx * pt.a + ty * pt.c + pt.tx; - wt.ty = tx * pt.b + ty * pt.d + pt.ty; - } - else { - // lets do the fast version as we know there is no rotation.. - a = this.scale.x; - d = this.scale.y; - - tx = this.position.x - this.pivot.x * a; - ty = this.position.y - this.pivot.y * d; - - wt.a = a * pt.a; - wt.b = a * pt.b; - wt.c = d * pt.c; - wt.d = d * pt.d; - wt.tx = tx * pt.a + ty * pt.c + pt.tx; - wt.ty = tx * pt.b + ty * pt.d + pt.ty; - } - - // multiply the alphas.. - this.worldAlpha = this.alpha * this.parent.worldAlpha; -}; - -// performance increase to avoid using call.. (10x faster) -DisplayObject.prototype.displayObjectUpdateTransform = DisplayObject.prototype.updateTransform; - -/** - * Retrieves the bounds of the displayObject as a rectangle object - * - * @param matrix {Matrix} - * @return {Rectangle} the rectangular bounding area - */ -DisplayObject.prototype.getBounds = function (/* matrix */) { - return math.Rectangle.EMPTY; -}; - -/** - * Retrieves the local bounds of the displayObject as a rectangle object - * - * @return {Rectangle} the rectangular bounding area - */ -DisplayObject.prototype.getLocalBounds = function () { - return this.getBounds(math.Matrix.IDENTITY); -}; - -/** - * Calculates the global position of the display object - * - * @param position {Point} The world origin to calculate from - * @return {Point} A point object representing the position of this object - */ -DisplayObject.prototype.toGlobal = function (position) { - // don't need to u[date the lot - this.displayObjectUpdateTransform(); - return this.worldTransform.apply(position); -}; - -/** - * Calculates the local position of the display object relative to another point - * - * @param position {Point} The world origin to calculate from - * @param [from] {DisplayObject} The DisplayObject to calculate the global position from - * @return {Point} A point object representing the position of this object - */ -DisplayObject.prototype.toLocal = function (position, from) { - if (from) { - position = from.toGlobal(position); - } - - // don't need to update the lot - this.displayObjectUpdateTransform(); - return this.worldTransform.applyInverse(position); -}; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} The renderer - * @private - */ -DisplayObject.prototype.renderWebGL = function (/* renderer */) { - // OVERWRITE; -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} The renderer - * @private - */ -DisplayObject.prototype.renderCanvas = function (/* renderer */) { - // OVERWRITE; -}; - -},{"../math":12}],6:[function(require,module,exports){ -var math = require('../math'), - DisplayObject = require('./DisplayObject'), - RenderTexture = require('../textures/RenderTexture'), - // Sprite = require('./Sprite'), - _tempMatrix = new math.Matrix(); - -/** - * A DisplayObjectContainer represents a collection of display objects. - * It is the base class of all display objects that act as a container for other objects. - * - * @class - * @extends DisplayObject - * @namespace PIXI - */ -function DisplayObjectContainer() { - DisplayObject.call(this); - - /** - * The array of children of this container. - * - * @member {DisplayObject[]} - * @readonly - */ - this.children = []; - - /** - * Cached internal flag. - * - * @member {boolean} - * @private - */ - this._cacheAsBitmap = false; - - this._cachedSprite = null; -} - -// constructor -DisplayObjectContainer.prototype = Object.create(DisplayObject.prototype); -DisplayObjectContainer.prototype.constructor = DisplayObjectContainer; -module.exports = DisplayObjectContainer; - -Object.defineProperties(DisplayObjectContainer.prototype, { - /** - * The width of the displayObjectContainer, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof DisplayObjectContainer# - */ - width: { - get: function () { - return this.scale.x * this.getLocalBounds().width; - }, - set: function (value) { - - var width = this.getLocalBounds().width; - - if(width !== 0) { - this.scale.x = value / width; - } - else { - this.scale.x = 1; - } - - - this._width = value; - } - }, - - /** - * The height of the displayObjectContainer, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof DisplayObjectContainer# - */ - height: { - get: function () { - return this.scale.y * this.getLocalBounds().height; - }, - set: function (value) { - - var height = this.getLocalBounds().height; - - if (height !== 0) { - this.scale.y = value / height ; - } - else { - this.scale.y = 1; - } - - this._height = value; - } - }, - - /** - * Set if this display object is cached as a bitmap. - * This basically takes a snap shot of the display object as it is at that moment. It can provide a performance benefit for complex static displayObjects. - * To remove simply set this property to 'null' - * - * @member {boolean} - * @memberof DisplayObject# - */ - cacheAsBitmap: { - get: function () { - return this._cacheAsBitmap; - }, - set: function (value) { - if (this._cacheAsBitmap === value) { - return; - } - - if (value) { - this._generateCachedSprite(); - } - else { - this._destroyCachedSprite(); - } - - this._cacheAsBitmap = value; - } - } -}); - -/** - * Adds a child to the container. - * - * @param child {DisplayObject} The DisplayObject to add to the container - * @return {DisplayObject} The child that was added. - */ -DisplayObjectContainer.prototype.addChild = function (child) { - return this.addChildAt(child, this.children.length); -}; - -/** - * Adds a child to the container at a specified index. If the index is out of bounds an error will be thrown - * - * @param child {DisplayObject} The child to add - * @param index {Number} The index to place the child in - * @return {DisplayObject} The child that was added. - */ -DisplayObjectContainer.prototype.addChildAt = function (child, index) { - // prevent adding self as child - if (child === this) { - return; - } - - if (index >= 0 && index <= this.children.length) { - if (child.parent) { - child.parent.removeChild(child); - } - - child.parent = this; - - this.children.splice(index, 0, child); - - return child; - } - else { - throw new Error(child + 'addChildAt: The index '+ index +' supplied is out of bounds ' + this.children.length); - } -}; - -/** - * Swaps the position of 2 Display Objects within this container. - * - * @param child {DisplayObject} - * @param child2 {DisplayObject} - */ -DisplayObjectContainer.prototype.swapChildren = function (child, child2) { - if (child === child2) { - return; - } - - var index1 = this.getChildIndex(child); - var index2 = this.getChildIndex(child2); - - if (index1 < 0 || index2 < 0) { - throw new Error('swapChildren: Both the supplied DisplayObjects must be a child of the caller.'); - } - - this.children[index1] = child2; - this.children[index2] = child; -}; - -/** - * Returns the index position of a child DisplayObject instance - * - * @param child {DisplayObject} The DisplayObject instance to identify - * @return {Number} The index position of the child display object to identify - */ -DisplayObjectContainer.prototype.getChildIndex = function (child) { - var index = this.children.indexOf(child); - - if (index === -1) { - throw new Error('The supplied DisplayObject must be a child of the caller'); - } - - return index; -}; - -/** - * Changes the position of an existing child in the display object container - * - * @param child {DisplayObject} The child DisplayObject instance for which you want to change the index number - * @param index {Number} The resulting index number for the child display object - */ -DisplayObjectContainer.prototype.setChildIndex = function (child, index) { - if (index < 0 || index >= this.children.length) { - throw new Error('The supplied index is out of bounds'); - } - - var currentIndex = this.getChildIndex(child); - - this.children.splice(currentIndex, 1); //remove from old position - this.children.splice(index, 0, child); //add at new position -}; - -/** - * Returns the child at the specified index - * - * @param index {Number} The index to get the child from - * @return {DisplayObject} The child at the given index, if any. - */ -DisplayObjectContainer.prototype.getChildAt = function (index) { - if (index < 0 || index >= this.children.length) { - throw new Error('getChildAt: Supplied index ' + index + ' does not exist in the child list, or the supplied DisplayObject must be a child of the caller'); - } - - return this.children[index]; -}; - -/** - * Removes a child from the container. - * - * @param child {DisplayObject} The DisplayObject to remove - * @return {DisplayObject} The child that was removed. - */ -DisplayObjectContainer.prototype.removeChild = function (child) { - var index = this.children.indexOf(child); - - if (index === -1) { - return; - } - - return this.removeChildAt(index); -}; - -/** - * Removes a child from the specified index position. - * - * @param index {Number} The index to get the child from - * @return {DisplayObject} The child that was removed. - */ -DisplayObjectContainer.prototype.removeChildAt = function (index) { - var child = this.getChildAt(index); - - child.parent = null; - this.children.splice(index, 1); - - return child; -}; - -/** - * Removes all children from this container that are within the begin and end indexes. - * - * @param beginIndex {Number} The beginning position. Default value is 0. - * @param endIndex {Number} The ending position. Default value is size of the container. - */ -DisplayObjectContainer.prototype.removeChildren = function (beginIndex, endIndex) { - var begin = beginIndex || 0; - var end = typeof endIndex === 'number' ? endIndex : this.children.length; - var range = end - begin; - - if (range > 0 && range <= end) { - var removed = this.children.splice(begin, range); - - for (var i = 0; i < removed.length; ++i) { - removed[i].parent = null; - } - - return removed; - } - else if (range === 0 && this.children.length === 0) { - return []; - } - else { - throw new RangeError('removeChildren: numeric values are outside the acceptable range.'); - } -}; - -/** - * Generates and updates the cached sprite for this object. - * - */ -DisplayObjectContainer.prototype.updateCachedSprite = function () { - this._generateCachedSprite(); -}; - -/** - * Useful function that returns a texture of the displayObject object that can then be used to create sprites - * This can be quite useful if your displayObject is static / complicated and needs to be reused multiple times. - * - * @param resolution {Number} The resolution of the texture being generated - * @param scaleMode {Number} See {{#crossLink "PIXI/scaleModes:property"}}PIXI.scaleModes{{/crossLink}} for possible values - * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used to generate the texture. - * @return {Texture} a texture of the graphics object - */ -DisplayObjectContainer.prototype.generateTexture = function (resolution, scaleMode, renderer) { - var bounds = this.getLocalBounds(); - - var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0, renderer, scaleMode, resolution); - - _tempMatrix.tx = -bounds.x; - _tempMatrix.ty = -bounds.y; - - renderTexture.render(this, _tempMatrix); - - return renderTexture; -}; - -/* - * Updates the transform on all children of this container for rendering - * - * @private - */ -DisplayObjectContainer.prototype.updateTransform = function () { - if (!this.visible) { - return; - } - - this.displayObjectUpdateTransform(); - - if (this._cacheAsBitmap) { - return; - } - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].updateTransform(); - } -}; - -// performance increase to avoid using call.. (10x faster) -DisplayObjectContainer.prototype.displayObjectContainerUpdateTransform = DisplayObjectContainer.prototype.updateTransform; - -/** - * Retrieves the bounds of the displayObjectContainer as a rectangle. The bounds calculation takes all visible children into consideration. - * - * @return {Rectangle} The rectangular bounding area - */ -DisplayObjectContainer.prototype.getBounds = function () { - if (this.children.length === 0) { - return math.Rectangle.EMPTY; - } - - // TODO the bounds have already been calculated this render session so return what we have - - var minX = Infinity; - var minY = Infinity; - - var maxX = -Infinity; - var maxY = -Infinity; - - var childBounds; - var childMaxX; - var childMaxY; - - var childVisible = false; - - for (var i = 0, j = this.children.length; i < j; ++i) { - var child = this.children[i]; - - if (!child.visible) { - continue; - } - - childVisible = true; - - childBounds = this.children[i].getBounds(); - - minX = minX < childBounds.x ? minX : childBounds.x; - minY = minY < childBounds.y ? minY : childBounds.y; - - childMaxX = childBounds.width + childBounds.x; - childMaxY = childBounds.height + childBounds.y; - - maxX = maxX > childMaxX ? maxX : childMaxX; - maxY = maxY > childMaxY ? maxY : childMaxY; - } - - if (!childVisible) { - return math.Rectangle.EMPTY; - } - - this._bounds.x = minX; - this._bounds.y = minY; - this._bounds.width = maxX - minX; - this._bounds.height = maxY - minY; - - // TODO: store a reference so that if this function gets called again in the render cycle we do not have to recalculate - //this._currentBounds = bounds; - - return this._bounds; -}; - -/** - * Retrieves the non-global local bounds of the displayObjectContainer as a rectangle. - * The calculation takes all visible children into consideration. - * - * @return {Rectangle} The rectangular bounding area - */ -DisplayObjectContainer.prototype.getLocalBounds = function () { - var matrixCache = this.worldTransform; - - this.worldTransform = math.Matrix.IDENTITY; - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].updateTransform(); - } - - this.worldTransform = matrixCache; - - return this.getBounds(); -}; - -/** - * Renders the object using the WebGL renderer - * - * TODO - Optimization pass! - * - * @param renderer {WebGLRenderer} The renderer - */ -DisplayObjectContainer.prototype.renderWebGL = function (renderer) { - // if the object is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0) { - return; - } - - if (this._cacheAsBitmap) { - this._renderCachedSprite(renderer); - return; - } - - var i, j; - - // do a quick check to see if this element has a mask or a filter. - if (this._mask || this._filters) { - // push filter first as we need to ensure the stencil buffer is correct for any masking - if (this._filters) { - renderer.spriteBatch.flush(); - renderer.filterManager.pushFilter(this._filterBlock); - } - - if (this._mask) { - renderer.spriteBatch.stop(); - renderer.maskManager.pushMask(this.mask, renderer); - renderer.spriteBatch.start(); - } - - // add this object to the batch, only rendered if it has a texture. - if (this.texture) { - renderer.spriteBatch.render(this); - } - - // now loop through the children and make sure they get rendered - for (i = 0, j = this.children.length; i < j; i++) { - this.children[i].renderWebGL(renderer); - } - - // time to stop the sprite batch as either a mask element or a filter draw will happen next - renderer.spriteBatch.stop(); - - if (this._mask) { - renderer.maskManager.popMask(this._mask, renderer); - } - - if (this._filters) { - renderer.filterManager.popFilter(); - } - - renderer.spriteBatch.start(); - } - else { - if (this.texture) { - renderer.spriteBatch.render(this); - } - - // simple render children! - for (i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderWebGL(renderer); - } - - } -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} The renderer - */ -DisplayObjectContainer.prototype.renderCanvas = function (renderer) { - if (!this.visible || this.alpha <= 0) { - return; - } - - if (this._cacheAsBitmap) { - this._renderCachedSprite(renderer); - return; - } - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderCanvas(renderer); - } - - if (this._mask) { - renderer.maskManager.popMask(renderer); - } -}; - -/** - * Internal method. - * - * @param renderer {WebGLRenderer|CanvasRenderer} The renderer - * @private - */ -DisplayObjectContainer.prototype._renderCachedSprite = function (renderer) { - this._cachedSprite.worldAlpha = this.worldAlpha; - - if (renderer.gl) { - this._cachedSprite.renderWebGL(renderer); - } - else { - this._cachedSprite.renderCanvas(renderer); - } -}; - -/** - * Internal method. - * - * @private - */ -DisplayObjectContainer.prototype._generateCachedSprite = function () { - var bounds = this.getLocalBounds(); - - if (!this._cachedSprite) { - // TODO - RenderTexture now *requires* a renderer instance, so this is like broken - // because `renderer` isn't actually in scope here :P - var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0); - - this._cachedSprite = new Sprite(renderTexture); - this._cachedSprite.worldTransform = this.worldTransform; - } - else { - this._cachedSprite.texture.resize(bounds.width | 0, bounds.height | 0); - } - - var tempFilters = this._filters; - this._filters = null; - - this._cachedSprite.filters = tempFilters; - - _tempMatrix.tx = -bounds.x; - _tempMatrix.ty = -bounds.y; - - this._cachedSprite.texture.render(this, _tempMatrix, true); - - this._cachedSprite.anchor.x = -(bounds.x / bounds.width); - this._cachedSprite.anchor.y = -(bounds.y / bounds.height); - - this._filters = tempFilters; -}; - -/** - * Destroys the cached sprite. - * - * @private - */ -DisplayObjectContainer.prototype._destroyCachedSprite = function () { - if (!this._cachedSprite) { - return; - } - - // TODO: Pool this sprite - this._cachedSprite.destroy(true, true); - this._cachedSprite = null; -}; - -},{"../math":12,"../textures/RenderTexture":43,"./DisplayObject":5}],7:[function(require,module,exports){ -var math = require('../math'), - Texture = require('../textures/Texture'), - DisplayObjectContainer = require('./DisplayObjectContainer'), - CanvasTinter = require('../renderers/canvas/utils/CanvasTinter'), - utils = require('../utils'), - CONST = require('../const'); - -/** - * The Sprite object is the base for all textured objects that are rendered to the screen - * - * A sprite can be created directly from an image like this: - * - * ```js - * var sprite = new Sprite.fromImage('assets/image.png'); - * ``` - * - * @class Sprite - * @extends DisplayObjectContainer - * @namespace PIXI - * @param texture {Texture} The texture for this sprite - */ -function Sprite(texture) { - DisplayObjectContainer.call(this); - - /** - * The anchor sets the origin point of the texture. - * The default is 0,0 this means the texture's origin is the top left - * Setting than anchor to 0.5,0.5 means the textures origin is centered - * Setting the anchor to 1,1 would mean the textures origin points will be the bottom right corner - * - * @member {Point} - */ - this.anchor = new math.Point(); - - /** - * The texture that the sprite is using - * - * @member {Texture} - * @private - */ - this._texture = null; - - /** - * The width of the sprite (this is initially set by the texture) - * - * @member {number} - * @private - */ - this._width = 0; - - /** - * The height of the sprite (this is initially set by the texture) - * - * @member {number} - * @private - */ - this._height = 0; - - /** - * The tint applied to the sprite. This is a hex value. A value of 0xFFFFFF will remove any tint effect. - * - * @member {number} - * @default 0xFFFFFF - */ - this.tint = 0xFFFFFF; - - /** - * The blend mode to be applied to the sprite. Set to CONST.blendModes.NORMAL to remove any blend mode. - * - * @member {number} - * @default CONST.blendModes.NORMAL; - */ - this.blendMode = CONST.blendModes.NORMAL; - - /** - * The shader that will be used to render the sprite. Set to null to remove a current shader. - * - * @member {AbstractFilter} - */ - this.shader = null; - - this.renderable = true; - - // call texture setter - this.texture = texture || Texture.EMPTY; -} - -Sprite.prototype.destroy = function (destroyTexture, destroyBaseTexture) { - DisplayObjectContainer.prototype.destroy.call(this); - - this.anchor = null; - - if (destroyTexture) { - this._texture.destroy(destroyBaseTexture); - } - - this._texture = null; - this.shader = null; -}; - -// constructor -Sprite.prototype = Object.create(DisplayObjectContainer.prototype); -Sprite.prototype.constructor = Sprite; -module.exports = Sprite; - -Object.defineProperties(Sprite.prototype, { - /** - * The width of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member - * @memberof Sprite# - */ - width: { - get: function () { - return this.scale.x * this.texture.frame.width; - }, - set: function (value) { - this.scale.x = value / this.texture.frame.width; - this._width = value; - } - }, - - /** - * The height of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member - * @memberof Sprite# - */ - height: { - get: function () { - return this.scale.y * this.texture.frame.height; - }, - set: function (value) { - this.scale.y = value / this.texture.frame.height; - this._height = value; - } - }, - - /** - * The height of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member - * @memberof Sprite# - */ - texture: { - get: function () { - return this._texture; - }, - set: function (value) { - if (this._texture === value) { - return; - } - - this._texture = value; - this.cachedTint = 0xFFFFFF; - - if (value) { - // wait for the texture to load - if (value.baseTexture.hasLoaded) { - this._onTextureUpdate(); - } - else { - value.once('update', this._onTextureUpdate.bind(this)); - } - } - } - }, -}); - -/** - * When the texture is updated, this event will fire to update the scale and frame - * - * @private - */ -Sprite.prototype._onTextureUpdate = function () { - // so if _width is 0 then width was not set.. - if (this._width) { - this.scale.x = this._width / this.texture.frame.width; - } - - if (this._height) { - this.scale.y = this._height / this.texture.frame.height; - } -}; - -/** - * Returns the bounds of the Sprite as a rectangle. The bounds calculation takes the worldTransform into account. - * - * @param matrix {Matrix} the transformation matrix of the sprite - * @return {Rectangle} the framing rectangle - */ -Sprite.prototype.getBounds = function (matrix) { - var width = this.texture.frame.width; - var height = this.texture.frame.height; - - var w0 = width * (1-this.anchor.x); - var w1 = width * -this.anchor.x; - - var h0 = height * (1-this.anchor.y); - var h1 = height * -this.anchor.y; - - var worldTransform = matrix || this.worldTransform ; - - var a = worldTransform.a; - var b = worldTransform.b; - var c = worldTransform.c; - var d = worldTransform.d; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var minX, - maxX, - minY, - maxY; - - if(b === 0 && c === 0) - { - // scale may be negative! - if (a < 0) { - a *= -1; - } - - if (d < 0) { - d *= -1; - } - - // this means there is no rotation going on right? RIGHT? - // if thats the case then we can avoid checking the bound values! yay - minX = a * w1 + tx; - maxX = a * w0 + tx; - minY = d * h1 + ty; - maxY = d * h0 + ty; - } - else - { - var x1 = a * w1 + c * h1 + tx; - var y1 = d * h1 + b * w1 + ty; - - var x2 = a * w0 + c * h1 + tx; - var y2 = d * h1 + b * w0 + ty; - - var x3 = a * w0 + c * h0 + tx; - var y3 = d * h0 + b * w0 + ty; - - var x4 = a * w1 + c * h0 + tx; - var y4 = d * h0 + b * w1 + ty; - - minX = x1; - minX = x2 < minX ? x2 : minX; - minX = x3 < minX ? x3 : minX; - minX = x4 < minX ? x4 : minX; - - minY = y1; - minY = y2 < minY ? y2 : minY; - minY = y3 < minY ? y3 : minY; - minY = y4 < minY ? y4 : minY; - - maxX = x1; - maxX = x2 > maxX ? x2 : maxX; - maxX = x3 > maxX ? x3 : maxX; - maxX = x4 > maxX ? x4 : maxX; - - maxY = y1; - maxY = y2 > maxY ? y2 : maxY; - maxY = y3 > maxY ? y3 : maxY; - maxY = y4 > maxY ? y4 : maxY; - } - - var bounds = this._bounds; - - bounds.x = minX; - bounds.width = maxX - minX; - - bounds.y = minY; - bounds.height = maxY - minY; - - // store a reference so that if this function gets called again in the render cycle we do not have to recalculate - this._currentBounds = bounds; - - return bounds; -}; - -/** -* Renders the object using the Canvas renderer -* -* @param renderer {CanvasRenderer} The renderer -*/ -Sprite.prototype.renderCanvas = function (renderer) { - if (!this.visible || this.alpha <= 0 || this.texture.crop.width <= 0 || this.texture.crop.height <= 0) { - return; - } - - if (this.blendMode !== renderer.currentBlendMode) { - renderer.currentBlendMode = this.blendMode; - renderer.context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode]; - } - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - // Ignore null sources - if (this.texture.valid) { - var resolution = this.texture.baseTexture.resolution / renderer.resolution; - - renderer.context.globalAlpha = this.worldAlpha; - - // If smoothingEnabled is supported and we need to change the smoothing property for this texture - if (renderer.smoothProperty && renderer.scaleMode !== this.texture.baseTexture.scaleMode) { - renderer.scaleMode = this.texture.baseTexture.scaleMode; - renderer.context[renderer.smoothProperty] = (renderer.scaleMode === CONST.scaleModes.LINEAR); - } - - // If the texture is trimmed we offset by the trim x/y, otherwise we use the frame dimensions - var dx = (this.texture.trim ? this.texture.trim.x : 0) - (this.anchor.x * this.texture.trim.width); - var dy = (this.texture.trim ? this.texture.trim.y : 0) - (this.anchor.y * this.texture.trim.height); - - // Allow for pixel rounding - if (renderer.roundPixels) { - renderer.context.setTransform( - this.worldTransform.a, - this.worldTransform.b, - this.worldTransform.c, - this.worldTransform.d, - (this.worldTransform.tx * renderer.resolution) | 0, - (this.worldTransform.ty * renderer.resolution) | 0 - ); - - dx = dx | 0; - dy = dy | 0; - } - else { - renderer.context.setTransform( - this.worldTransform.a, - this.worldTransform.b, - this.worldTransform.c, - this.worldTransform.d, - this.worldTransform.tx * renderer.resolution, - this.worldTransform.ty * renderer.resolution - ); - } - - if (this.tint !== 0xFFFFFF) { - if (this.cachedTint !== this.tint) { - this.cachedTint = this.tint; - - // TODO clean up caching - how to clean up the caches? - this.tintedTexture = CanvasTinter.getTintedTexture(this, this.tint); - } - - renderer.context.drawImage( - this.tintedTexture, - 0, - 0, - this.texture.crop.width, - this.texture.crop.height, - dx / resolution, - dy / resolution, - this.texture.crop.width / resolution, - this.texture.crop.height / resolution - ); - } - else { - renderer.context.drawImage( - this.texture.baseTexture.source, - this.texture.crop.x, - this.texture.crop.y, - this.texture.crop.width, - this.texture.crop.height, - dx / resolution, - dy / resolution, - this.texture.crop.width / resolution, - this.texture.crop.height / resolution - ); - } - } - - for (var i = 0, j = this.children.length; i < j; i++) { - this.children[i].renderCanvas(renderer); - } - - if (this._mask) { - renderer.maskManager.popMask(renderer); - } -}; - -// some helper functions.. - -/** - * Helper function that creates a sprite that will contain a texture from the TextureCache based on the frameId - * The frame ids are created when a Texture packer file has been loaded - * - * @static - * @param frameId {String} The frame Id of the texture in the cache - * @return {Sprite} A new Sprite using a texture from the texture cache matching the frameId - */ -Sprite.fromFrame = function (frameId) { - var texture = utils.TextureCache[frameId]; - - if (!texture) { - throw new Error('The frameId "' + frameId + '" does not exist in the texture cache' + this); - } - - return new Sprite(texture); -}; - -/** - * Helper function that creates a sprite that will contain a texture based on an image url - * If the image is not in the texture cache it will be loaded - * - * @static - * @param imageId {String} The image url of the texture - * @return {Sprite} A new Sprite using a texture from the texture cache matching the image id - */ -Sprite.fromImage = function (imageId, crossorigin, scaleMode) { - return new Sprite(Texture.fromImage(imageId, crossorigin, scaleMode)); -}; - -},{"../const":4,"../math":12,"../renderers/canvas/utils/CanvasTinter":24,"../textures/Texture":44,"../utils":50,"./DisplayObjectContainer":6}],8:[function(require,module,exports){ -var DisplayObjectContainer = require('./DisplayObjectContainer'), - WebGLFastSpriteBatch = require('../renderers/webgl/utils/WebGLFastSpriteBatch'); - -/** - * The SpriteBatch class is a really fast version of the DisplayObjectContainer built solely for speed, - * so use when you need a lot of sprites or particles. The tradeoff of the SpriteBatch is that advanced - * functionality will not work. SpriteBatch implements only the basic object transform (position, scale, rotation). - * Any other functionality like tinting, masking, etc will not work on sprites in this batch. - * - * It's extremely easy to use : - * - * ```js - * var container = new SpriteBatch(); - * - * for(var i = 0; i < 100; ++i) { - * var sprite = new PIXI.Sprite.fromImage("myImage.png"); - * container.addChild(sprite); - * } - * ``` - * - * And here you have a hundred sprites that will be renderer at the speed of light. - * - * @class - * @namespace PIXI - */ - -//TODO RENAME to PARTICLE CONTAINER? -function SpriteBatch() { - DisplayObjectContainer.call(this); -} - -SpriteBatch.prototype = Object.create(DisplayObjectContainer.prototype); -SpriteBatch.prototype.constructor = SpriteBatch; -module.exports = SpriteBatch; - -/** - * Updates the object transform for rendering - * - * @private - */ -SpriteBatch.prototype.updateTransform = function () { - // TODO don't need to! - this.displayObjectUpdateTransform(); - // PIXI.DisplayObjectContainer.prototype.updateTransform.call( this ); -}; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} The webgl renderer - * @private - */ -SpriteBatch.prototype.renderWebGL = function (renderer) { - if (!this.visible || this.alpha <= 0 || !this.children.length) { - return; - } - - renderer.spriteBatch.stop(); - - renderer.shaderManager.setShader(renderer.shaderManager.fastShader); - - renderer.fastSpriteBatch.begin(this); - renderer.fastSpriteBatch.render(this); - - renderer.spriteBatch.start(); -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} The canvas renderer - * @private - */ -SpriteBatch.prototype.renderCanvas = function (renderer) { - if (!this.visible || this.alpha <= 0 || !this.children.length) { - return; - } - - var context = renderer.context; - var transform = this.worldTransform; - var isRotated = true; - - context.globalAlpha = this.worldAlpha; - - this.displayObjectUpdateTransform(); - - for (var i = 0; i < this.children.length; ++i) { - var child = this.children[i]; - - if (!child.visible) { - continue; - } - - var frame = child.texture.frame; - - context.globalAlpha = this.worldAlpha * child.alpha; - - if (child.rotation % (Math.PI * 2) === 0) { - // this is the fastest way to optimise! - if rotation is 0 then we can avoid any kind of setTransform call - if (isRotated) { - context.setTransform( - transform.a, - transform.b, - transform.c, - transform.d, - transform.tx, - transform.ty - ); - - isRotated = false; - } - - context.drawImage( - child.texture.baseTexture.source, - frame.x, - frame.y, - frame.width, - frame.height, - ((child.anchor.x) * (-frame.width * child.scale.x) + child.position.x + 0.5) | 0, - ((child.anchor.y) * (-frame.height * child.scale.y) + child.position.y + 0.5) | 0, - frame.width * child.scale.x, - frame.height * child.scale.y - ); - } - else { - if (!isRotated) { - isRotated = true; - } - - child.displayObjectUpdateTransform(); - - var childTransform = child.worldTransform; - - if (renderer.roundPixels) { - context.setTransform( - childTransform.a, - childTransform.b, - childTransform.c, - childTransform.d, - childTransform.tx | 0, - childTransform.ty | 0 - ); - } - else { - context.setTransform( - childTransform.a, - childTransform.b, - childTransform.c, - childTransform.d, - childTransform.tx, - childTransform.ty - ); - } - - context.drawImage( - child.texture.baseTexture.source, - frame.x, - frame.y, - frame.width, - frame.height, - ((child.anchor.x) * (-frame.width) + 0.5) | 0, - ((child.anchor.y) * (-frame.height) + 0.5) | 0, - frame.width, - frame.height - ); - } - } -}; - -},{"../renderers/webgl/utils/WebGLFastSpriteBatch":38,"./DisplayObjectContainer":6}],9:[function(require,module,exports){ -/** - * @file Main export of the PIXI core library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -var core = module.exports = { - CONST: require('./const'), - - // utils - utils: require('./utils'), - math: require('./math'), - - // display - DisplayObject: require('./display/DisplayObject'), - DisplayObjectContainer: require('./display/DisplayObjectContainer'), - Sprite: require('./display/Sprite'), - SpriteBatch: require('./display/SpriteBatch'), - - // primitives - Graphics: require('./primitives/Graphics'), - GraphicsData: require('./primitives/GraphicsData'), - - // textures - Texture: require('./textures/Texture'), - BaseTexture: require('./textures/BaseTexture'), - RenderTexture: require('./textures/RenderTexture'), - VideoBaseTexture: require('./textures/VideoBaseTexture'), - - // renderers - canvas - CanvasRenderer: require('./renderers/canvas/CanvasRenderer'), - CanvasGraphics: require('./renderers/canvas/utils/CanvasGraphics'), - CanvasBuffer: require('./renderers/canvas/utils/CanvasBuffer'), - - // renderers - webgl - WebGLRenderer: require('./renderers/webgl/WebGLRenderer'), - WebGLGraphics: require('./renderers/webgl/utils/WebGLGraphics'), - - /** - * This helper function will automatically detect which renderer you should be using. - * WebGL is the preferred renderer as it is a lot faster. If webGL is not supported by - * the browser then this function will return a canvas renderer - * - * @param width=800 {number} the width of the renderers view - * @param height=600 {number} the height of the renderers view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment) - * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you - * need to call toDataUrl on the webgl context - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - * @param [noWebGL=false] {Boolean} prevents selection of WebGL renderer, even if such is present - * - * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer - */ - autoDetectRenderer: function (width, height, options, noWebGL) { - width = width || 800; - height = height || 600; - - if (!noWebGL && require('webgl-enabled')()) { - return new core.WebGLRenderer(width, height, options); - } - - return new core.CanvasRenderer(width, height, options); - }, - - /** - * This helper function will automatically detect which renderer you should be using. This function is very - * similar to the autoDetectRenderer function except that is will return a canvas renderer for android. - * Even thought both android chrome supports webGL the canvas implementation perform better at the time of writing. - * This function will likely change and update as webGL performance improves on these devices. - * - * @param width=800 {number} the width of the renderers view - * @param height=600 {number} the height of the renderers view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment) - * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you - * need to call toDataUrl on the webgl context - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - * - * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer - */ - autoDetectRecommendedRenderer: function (width, height, options) { - var isAndroid = /Android/i.test(navigator.userAgent); - - return core.autoDetectRenderer(width, height, options, isAndroid); - } -}; - -},{"./const":4,"./display/DisplayObject":5,"./display/DisplayObjectContainer":6,"./display/Sprite":7,"./display/SpriteBatch":8,"./math":12,"./primitives/Graphics":18,"./primitives/GraphicsData":19,"./renderers/canvas/CanvasRenderer":20,"./renderers/canvas/utils/CanvasBuffer":21,"./renderers/canvas/utils/CanvasGraphics":22,"./renderers/webgl/WebGLRenderer":25,"./renderers/webgl/utils/WebGLGraphics":39,"./textures/BaseTexture":42,"./textures/RenderTexture":43,"./textures/Texture":44,"./textures/VideoBaseTexture":46,"./utils":50,"webgl-enabled":2}],10:[function(require,module,exports){ -var Point = require('./Point'); - -/** - * The Matrix class is now an object, which makes it a lot faster, - * here is a representation of it : - * | a | b | tx| - * | c | d | ty| - * | 0 | 0 | 1 | - * - * @class - * @namespace PIXI - */ -function Matrix() { - /** - * @member {number} - * @default 1 - */ - this.a = 1; - - /** - * @member {number} - * @default 0 - */ - this.b = 0; - - /** - * @member {number} - * @default 0 - */ - this.c = 0; - - /** - * @member {number} - * @default 1 - */ - this.d = 1; - - /** - * @member {number} - * @default 0 - */ - this.tx = 0; - - /** - * @member {number} - * @default 0 - */ - this.ty = 0; -} - -Matrix.prototype.constructor = Matrix; -module.exports = Matrix; - -/** - * Creates a Matrix object based on the given array. The Element to Matrix mapping order is as follows: - * - * a = array[0] - * b = array[1] - * c = array[3] - * d = array[4] - * tx = array[2] - * ty = array[5] - * - * @param array {number[]} The array that the matrix will be populated from. - */ -Matrix.prototype.fromArray = function (array) { - this.a = array[0]; - this.b = array[1]; - this.c = array[3]; - this.d = array[4]; - this.tx = array[2]; - this.ty = array[5]; -}; - -/** - * Creates an array from the current Matrix object. - * - * @param transpose {boolean} Whether we need to transpose the matrix or not - * @return {number[]} the newly created array which contains the matrix - */ -Matrix.prototype.toArray = function (transpose) { - if (!this.array) { - this.array = new Float32Array(9); - } - - var array = this.array; - - if (transpose) { - array[0] = this.a; - array[1] = this.b; - array[2] = 0; - array[3] = this.c; - array[4] = this.d; - array[5] = 0; - array[6] = this.tx; - array[7] = this.ty; - array[8] = 1; - } - else { - array[0] = this.a; - array[1] = this.c; - array[2] = this.tx; - array[3] = this.b; - array[4] = this.d; - array[5] = this.ty; - array[6] = 0; - array[7] = 0; - array[8] = 1; - } - - return array; -}; - -/** - * Get a new position with the current transformation applied. - * Can be used to go from a child's coordinate space to the world coordinate space. (e.g. rendering) - * - * @param pos {Point} The origin - * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input) - * @return {Point} The new point, transformed through this matrix - */ -Matrix.prototype.apply = function (pos, newPos) { - newPos = newPos || new Point(); - - newPos.x = this.a * pos.x + this.c * pos.y + this.tx; - newPos.y = this.b * pos.x + this.d * pos.y + this.ty; - - return newPos; -}; - -/** - * Get a new position with the inverse of the current transformation applied. - * Can be used to go from the world coordinate space to a child's coordinate space. (e.g. input) - * - * @param pos {Point} The origin - * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input) - * @return {Point} The new point, inverse-transformed through this matrix - */ -Matrix.prototype.applyInverse = function (pos, newPos) { - newPos = newPos || new Point(); - - var id = 1 / (this.a * this.d + this.c * -this.b); - - newPos.x = this.d * id * pos.x + -this.c * id * pos.y + (this.ty * this.c - this.tx * this.d) * id; - newPos.y = this.a * id * pos.y + -this.b * id * pos.x + (-this.ty * this.a + this.tx * this.b) * id; - - return newPos; -}; - -/** - * Translates the matrix on the x and y. - * - * @param {number} x - * @param {number} y - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.translate = function (x, y) { - this.tx += x; - this.ty += y; - - return this; -}; - -/** - * Applies a scale transformation to the matrix. - * - * @param {number} x The amount to scale horizontally - * @param {number} y The amount to scale vertically - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.scale = function (x, y) { - this.a *= x; - this.d *= y; - this.c *= x; - this.b *= y; - this.tx *= x; - this.ty *= y; - - return this; -}; - - -/** - * Applies a rotation transformation to the matrix. - * - * @param {number} angle - The angle in radians. - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.rotate = function (angle) { - var cos = Math.cos( angle ); - var sin = Math.sin( angle ); - - var a1 = this.a; - var c1 = this.c; - var tx1 = this.tx; - - this.a = a1 * cos-this.b * sin; - this.b = a1 * sin+this.b * cos; - this.c = c1 * cos-this.d * sin; - this.d = c1 * sin+this.d * cos; - this.tx = tx1 * cos - this.ty * sin; - this.ty = tx1 * sin + this.ty * cos; - - return this; -}; - -/** - * Appends the given Matrix to this Matrix. - * - * @param {Matrix} matrix - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.append = function (matrix) { - var a1 = this.a; - var b1 = this.b; - var c1 = this.c; - var d1 = this.d; - - this.a = matrix.a * a1 + matrix.b * c1; - this.b = matrix.a * b1 + matrix.b * d1; - this.c = matrix.c * a1 + matrix.d * c1; - this.d = matrix.c * b1 + matrix.d * d1; - - this.tx = matrix.tx * a1 + matrix.ty * c1 + this.tx; - this.ty = matrix.tx * b1 + matrix.ty * d1 + this.ty; - - return this; -}; - -/** - * Resets this Matix to an identity (default) matrix. - * - * @return {Matrix} This matrix. Good for chaining method calls. - */ -Matrix.prototype.identity = function () { - this.a = 1; - this.b = 0; - this.c = 0; - this.d = 1; - this.tx = 0; - this.ty = 0; - - return this; -}; - -Matrix.IDENTITY = new Matrix(); - -},{"./Point":11}],11:[function(require,module,exports){ -/** - * The Point object represents a location in a two-dimensional coordinate system, where x represents - * the horizontal axis and y represents the vertical axis. - * - * @class - * @namespace PIXI - * @param [x=0] {number} position of the point on the x axis - * @param [y=0] {number} position of the point on the y axis - */ -function Point(x, y) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; -} - -Point.prototype.constructor = Point; -module.exports = Point; - -/** - * Creates a clone of this point - * - * @return {Point} a copy of the point - */ -Point.prototype.clone = function () { - return new Point(this.x, this.y); -}; - -/** - * Sets the point to a new x and y position. - * If y is omitted, both x and y will be set to x. - * - * @param [x=0] {number} position of the point on the x axis - * @param [y=0] {number} position of the point on the y axis - */ -Point.prototype.set = function (x, y) { - this.x = x || 0; - this.y = y || ( (y !== 0) ? this.x : 0 ) ; -}; - -},{}],12:[function(require,module,exports){ -/** - * @namespace PIXI.math - */ -module.exports = { - /** - * @property {number} PI_2 - Math.PI x 2 - * @constant - * @static - */ - PI_2: Math.PI * 2, - - /** - * @property {number} RAD_TO_DEG - Constant conversion factor for converting radians to degrees - * @constant - * @static - */ - RAD_TO_DEG: 180 / Math.PI, - - /** - * @property {Number} DEG_TO_RAD - Constant conversion factor for converting degrees to radians - * @constant - * @static - */ - DEG_TO_RAD: Math.PI / 180, - - Point: require('./Point'), - Matrix: require('./Matrix'), - - Circle: require('./shapes/Circle'), - Ellipse: require('./shapes/Ellipse'), - Polygon: require('./shapes/Polygon'), - Rectangle: require('./shapes/Rectangle'), - RoundedRectangle: require('./shapes/RoundedRectangle') -}; - -},{"./Matrix":10,"./Point":11,"./shapes/Circle":13,"./shapes/Ellipse":14,"./shapes/Polygon":15,"./shapes/Rectangle":16,"./shapes/RoundedRectangle":17}],13:[function(require,module,exports){ -var Rectangle = require('./Rectangle'), - CONST = require('../../const'); - -/** - * The Circle object can be used to specify a hit area for displayObjects - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the center of this circle - * @param y {number} The Y coordinate of the center of this circle - * @param radius {number} The radius of the circle - */ -function Circle(x, y, radius) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.radius = radius || 0; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.CIRC; -} - -Circle.prototype.constructor = Circle; -module.exports = Circle; - -/** - * Creates a clone of this Circle instance - * - * @method clone - * @return {Circle} a copy of the Circle - */ -Circle.prototype.clone = function () { - return new Circle(this.x, this.y, this.radius); -}; - -/** - * Checks whether the x and y coordinates given are contained within this circle - * - * @method contains - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this Circle - */ -Circle.prototype.contains = function (x, y) { - if (this.radius <= 0) { - return false; - } - - var dx = (this.x - x), - dy = (this.y - y), - r2 = this.radius * this.radius; - - dx *= dx; - dy *= dy; - - return (dx + dy <= r2); -}; - -/** -* Returns the framing rectangle of the circle as a Rectangle object -* -* @method getBounds -* @return {Rectangle} the framing rectangle -*/ -Circle.prototype.getBounds = function () { - return new Rectangle(this.x - this.radius, this.y - this.radius, this.radius * 2, this.radius * 2); -}; - -},{"../../const":4,"./Rectangle":16}],14:[function(require,module,exports){ -var Rectangle = require('./Rectangle'), - CONST = require('../../const'); - -/** - * The Ellipse object can be used to specify a hit area for displayObjects - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the center of the ellipse - * @param y {number} The Y coordinate of the center of the ellipse - * @param width {number} The half width of this ellipse - * @param height {number} The half height of this ellipse - */ -function Ellipse(x, y, width, height) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.width = width || 0; - - /** - * @member {number} - * @default 0 - */ - this.height = height || 0; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.ELIP; -} - -Ellipse.prototype.constructor = Ellipse; -module.exports = Ellipse; - -/** - * Creates a clone of this Ellipse instance - * - * @method clone - * @return {Ellipse} a copy of the ellipse - */ -Ellipse.prototype.clone = function () { - return new Ellipse(this.x, this.y, this.width, this.height); -}; - -/** - * Checks whether the x and y coordinates given are contained within this ellipse - * - * @method contains - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coords are within this ellipse - */ -Ellipse.prototype.contains = function (x, y) { - if (this.width <= 0 || this.height <= 0) { - return false; - } - - //normalize the coords to an ellipse with center 0,0 - var normx = ((x - this.x) / this.width), - normy = ((y - this.y) / this.height); - - normx *= normx; - normy *= normy; - - return (normx + normy <= 1); -}; - -/** -* Returns the framing rectangle of the ellipse as a Rectangle object -* -* @method getBounds -* @return {Rectangle} the framing rectangle -*/ -Ellipse.prototype.getBounds = function () { - return new Rectangle(this.x - this.width, this.y - this.height, this.width, this.height); -}; - -},{"../../const":4,"./Rectangle":16}],15:[function(require,module,exports){ -var Point = require('../Point'), - CONST = require('../../const'); - -/** - * @class - * @namespace PIXI - * @param points* {Point[]|number[]|Point...|number...} This can be an array of Points that form the polygon, - * a flat array of numbers that will be interpreted as [x,y, x,y, ...], or the arguments passed can be - * all the points of the polygon e.g. `new Polygon(new Point(), new Point(), ...)`, or the - * arguments passed can be flat x,y values e.g. `new Polygon(x,y, x,y, x,y, ...)` where `x` and `y` are - * Numbers. - */ -function Polygon(points) { - //if points isn't an array, use arguments as the array - if (!(points instanceof Array)) { - points = Array.prototype.slice.call(arguments); - } - - //if this is a flat array of numbers, convert it to points - if (points[0] instanceof Point) { - var p = []; - for (var i = 0, il = points.length; i < il; i++) { - p.push(points[i].x, points[i].y); - } - - points = p; - } - - this.closed = true; - - /** - * An array of the points of this polygon - * - * @member {Point[]} - */ - this.points = points; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.POLY; -} - -Polygon.prototype.constructor = Polygon; -module.exports = Polygon; - -/** - * Creates a clone of this polygon - * - * @return {Polygon} a copy of the polygon - */ -Polygon.prototype.clone = function () { - return new Polygon(this.points.slice()); -}; - -/** - * Checks whether the x and y coordinates passed to this function are contained within this polygon - * - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this polygon - */ -Polygon.prototype.contains = function (x, y) { - var inside = false; - - // use some raycasting to test hits - // https://github.com/substack/point-in-polygon/blob/master/index.js - var length = this.points.length / 2; - - for (var i = 0, j = length - 1; i < length; j = i++) { - var xi = this.points[i * 2], yi = this.points[i * 2 + 1], - xj = this.points[j * 2], yj = this.points[j * 2 + 1], - intersect = ((yi > y) !== (yj > y)) && (x < (xj - xi) * (y - yi) / (yj - yi) + xi); - - if (intersect) { - inside = !inside; - } - } - - return inside; -}; - -},{"../../const":4,"../Point":11}],16:[function(require,module,exports){ -var CONST = require('../../const'); - -/** - * the Rectangle object is an area defined by its position, as indicated by its top-left corner point (x, y) and by its width and its height. - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the upper-left corner of the rectangle - * @param y {number} The Y coordinate of the upper-left corner of the rectangle - * @param width {number} The overall width of this rectangle - * @param height {number} The overall height of this rectangle - */ -function Rectangle(x, y, width, height) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.width = width || 0; - - /** - * @member {number} - * @default 0 - */ - this.height = height || 0; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.RECT; -} - -Rectangle.prototype.constructor = Rectangle; -module.exports = Rectangle; - -/** - * A constant empty rectangle. - * - * @static - * @constant - */ -Rectangle.EMPTY = new Rectangle(0, 0, 0, 0); - - -/** - * Creates a clone of this Rectangle - * - * @return {Rectangle} a copy of the rectangle - */ -Rectangle.prototype.clone = function () { - return new Rectangle(this.x, this.y, this.width, this.height); -}; - -/** - * Checks whether the x and y coordinates given are contained within this Rectangle - * - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this Rectangle - */ -Rectangle.prototype.contains = function (x, y) { - if (this.width <= 0 || this.height <= 0) { - return false; - } - - if (x >= this.x && x <= this.x + this.width) { - if (y >= this.y && y <= this.y + this.height) { - return true; - } - } - - return false; -}; - -},{"../../const":4}],17:[function(require,module,exports){ -var CONST = require('../../const'); - -/** - * The Rounded Rectangle object is an area defined by its position and has nice rounded corners, as indicated by its top-left corner point (x, y) and by its width and its height. - * - * @class - * @namespace PIXI - * @param x {number} The X coordinate of the upper-left corner of the rounded rectangle - * @param y {number} The Y coordinate of the upper-left corner of the rounded rectangle - * @param width {number} The overall width of this rounded rectangle - * @param height {number} The overall height of this rounded rectangle - * @param radius {number} Controls the radius of the rounded corners - */ -function RoundedRectangle(x, y, width, height, radius) { - /** - * @member {number} - * @default 0 - */ - this.x = x || 0; - - /** - * @member {number} - * @default 0 - */ - this.y = y || 0; - - /** - * @member {number} - * @default 0 - */ - this.width = width || 0; - - /** - * @member {number} - * @default 0 - */ - this.height = height || 0; - - /** - * @member {number} - * @default 20 - */ - this.radius = radius || 20; - - /** - * The type of the object, mainly used to avoid `instanceof` checks - * - * @member {number} - */ - this.type = CONST.SHAPES.RREC; -} - -RoundedRectangle.prototype.constructor = RoundedRectangle; -module.exports = RoundedRectangle; - -/** - * Creates a clone of this Rounded Rectangle - * - * @return {RoundedRectangle} a copy of the rounded rectangle - */ -RoundedRectangle.prototype.clone = function () { - return new RoundedRectangle(this.x, this.y, this.width, this.height, this.radius); -}; - -/** - * Checks whether the x and y coordinates given are contained within this Rounded Rectangle - * - * @param x {number} The X coordinate of the point to test - * @param y {number} The Y coordinate of the point to test - * @return {boolean} Whether the x/y coordinates are within this Rounded Rectangle - */ -RoundedRectangle.prototype.contains = function (x, y) { - if (this.width <= 0 || this.height <= 0) { - return false; - } - - if (x >= this.x && x <= this.x + this.width) { - if (y >= this.y && y <= this.y + this.height) { - return true; - } - } - - return false; -}; - -},{"../../const":4}],18:[function(require,module,exports){ -var DisplayObjectContainer = require('../display/DisplayObjectContainer'), - Sprite = require('../display/Sprite'), - Texture = require('../textures/Texture'), - CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'), - CanvasGraphics = require('../renderers/canvas/utils/CanvasGraphics'), - WebGLGraphics = require('../renderers/webgl/utils/WebGLGraphics'), - GraphicsData = require('./GraphicsData'), - math = require('../math'), - CONST = require('../const'); - -/** - * The Graphics class contains methods used to draw primitive shapes such as lines, circles and - * rectangles to the display, and color and fill them. - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - */ -function Graphics() { - DisplayObjectContainer.call(this); - - this.renderable = true; - - /** - * The alpha value used when filling the Graphics object. - * - * @member {number} - * @default 1 - */ - this.fillAlpha = 1; - - /** - * The width (thickness) of any lines drawn. - * - * @member {number} - * @default 0 - */ - this.lineWidth = 0; - - /** - * The color of any lines drawn. - * - * @member {string} - * @default 0 - */ - this.lineColor = 0; - - /** - * Graphics data - * - * @member {GraphicsData[]} - * @private - */ - this.graphicsData = []; - - /** - * The tint applied to the graphic shape. This is a hex value. Apply a value of 0xFFFFFF to reset the tint. - * - * @member {number} - * @default 0xFFFFFF - */ - this.tint = 0xFFFFFF; - - /** - * The blend mode to be applied to the graphic shape. Apply a value of blendModes.NORMAL to reset the blend mode. - * - * @member {number} - * @default CONST.blendModes.NORMAL; - */ - this.blendMode = CONST.blendModes.NORMAL; - - /** - * Current path - * - * @member {GraphicsData} - * @private - */ - this.currentPath = null; - - /** - * Array containing some WebGL-related properties used by the WebGL renderer. - * - * @member {object} - * @private - */ - // TODO - _webgl should use a prototype object, not a random undocumented object... - this._webGL = {}; - - /** - * Whether this shape is being used as a mask. - * - * @member {boolean} - */ - this.isMask = false; - - /** - * The bounds' padding used for bounds calculation. - * - * @member {number} - */ - this.boundsPadding = 0; - - /** - * A cache of the local bounds to prevent recalculation. - * - * @member {Rectangle} - * @private - */ - this._localBounds = new math.Rectangle(0,0,1,1); - - /** - * Used to detect if the graphics object has changed. If this is set to true then the graphics - * object will be recalculated. - * - * @member {boolean} - * @private - */ - this.dirty = true; - - /** - * Used to detect if the WebGL graphics object has changed. If this is set to true then the - * graphics object will be recalculated. - * - * @member {boolean} - * @private - */ - this.glDirty = false; - - /** - * Used to detect if the cached sprite object needs to be updated. - * - * @member {boolean} - * @private - */ - this.cachedSpriteDirty = false; -} - -// constructor -Graphics.prototype = Object.create(DisplayObjectContainer.prototype); -Graphics.prototype.constructor = Graphics; -module.exports = Graphics; - -Object.defineProperties(Graphics.prototype, { - /** - * When cacheAsBitmap is set to true the graphics object will be rendered as if it was a sprite. - * This is useful if your graphics element does not change often, as it will speed up the rendering - * of the object in exchange for taking up texture memory. It is also useful if you need the graphics - * object to be anti-aliased, because it will be rendered using canvas. This is not recommended if - * you are constantly redrawing the graphics element. - * - * @member {boolean} - * @memberof Graphics# - * @default false - * @private - */ - cacheAsBitmap: { - get: function () { - return this._cacheAsBitmap; - }, - set: function (value) { - this._cacheAsBitmap = value; - - if (this._cacheAsBitmap) { - this._generateCachedSprite(); - } - else { - this.destroyCachedSprite(); - this.dirty = true; - } - } - } -}); - -/** - * Creates a new Graphics object with the same values as this one. - * - * @return {Graphics} - */ -GraphicsData.prototype.clone = function () { - var clone = new Graphics(); - - clone.renderable = this.renderable; - clone.fillAlpha = this.fillAlpha; - clone.lineWidth = this.lineWidth; - clone.lineColor = this.lineColor; - clone.tint = this.tint; - clone.blendMode = this.blendMode; - clone.isMask = this.isMask; - clone.boundsPadding = this.boundsPadding; - clone.dirty = this.dirty; - clone.glDirty = this.glDirty; - clone.cachedSpriteDirty = this.cachedSpriteDirty; - - // copy graphics data - for (var i = 0; i < this.graphicsData.length; ++i) { - clone.graphicsData.push(this.graphicsData.clone()); - } - - clone.currentPath = clone.graphicsData[clone.graphicsData.length - 1]; - - clone.updateLocalBounds(); - - return clone; -}; - -/** - * Specifies the line style used for subsequent calls to Graphics methods such as the lineTo() method or the drawCircle() method. - * - * @param lineWidth {number} width of the line to draw, will update the objects stored style - * @param color {number} color of the line to draw, will update the objects stored style - * @param alpha {number} alpha of the line to draw, will update the objects stored style - * @return {Graphics} - */ -Graphics.prototype.lineStyle = function (lineWidth, color, alpha) { - this.lineWidth = lineWidth || 0; - this.lineColor = color || 0; - this.lineAlpha = (arguments.length < 3) ? 1 : alpha; - - if (this.currentPath) { - if (this.currentPath.shape.points.length) { - // halfway through a line? start a new one! - this.drawShape( new math.Polygon( this.currentPath.shape.points.slice(-2) )); - } - else { - // otherwise its empty so lets just set the line properties - this.currentPath.lineWidth = this.lineWidth; - this.currentPath.lineColor = this.lineColor; - this.currentPath.lineAlpha = this.lineAlpha; - } - } - - return this; -}; - -/** - * Moves the current drawing position to x, y. - * - * @param x {number} the X coordinate to move to - * @param y {number} the Y coordinate to move to - * @return {Graphics} - */ -Graphics.prototype.moveTo = function (x, y) { - this.drawShape(new math.Polygon([x,y])); - - return this; -}; - -/** - * Draws a line using the current line style from the current drawing position to (x, y); - * The current drawing position is then set to (x, y). - * - * @param x {number} the X coordinate to draw to - * @param y {number} the Y coordinate to draw to - * @return {Graphics} - */ -Graphics.prototype.lineTo = function (x, y) { - this.currentPath.shape.points.push(x, y); - this.dirty = true; - - return this; -}; - -/** - * Calculate the points for a quadratic bezier curve and then draws it. - * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c - * - * @param cpX {number} Control point x - * @param cpY {number} Control point y - * @param toX {number} Destination point x - * @param toY {number} Destination point y - * @return {Graphics} - */ -Graphics.prototype.quadraticCurveTo = function (cpX, cpY, toX, toY) { - if (this.currentPath) { - if (this.currentPath.shape.points.length === 0) { - this.currentPath.shape.points = [0, 0]; - } - } - else { - this.moveTo(0,0); - } - - var xa, - ya, - n = 20, - points = this.currentPath.shape.points; - - if (points.length === 0) { - this.moveTo(0, 0); - } - - var fromX = points[points.length-2]; - var fromY = points[points.length-1]; - - var j = 0; - for (var i = 1; i <= n; ++i) { - j = i / n; - - xa = fromX + ( (cpX - fromX) * j ); - ya = fromY + ( (cpY - fromY) * j ); - - points.push( xa + ( ((cpX + ( (toX - cpX) * j )) - xa) * j ), - ya + ( ((cpY + ( (toY - cpY) * j )) - ya) * j ) ); - } - - this.dirty = true; - - return this; -}; - -/** - * Calculate the points for a bezier curve and then draws it. - * - * @param cpX {number} Control point x - * @param cpY {number} Control point y - * @param cpX2 {number} Second Control point x - * @param cpY2 {number} Second Control point y - * @param toX {number} Destination point x - * @param toY {number} Destination point y - * @return {Graphics} - */ -Graphics.prototype.bezierCurveTo = function (cpX, cpY, cpX2, cpY2, toX, toY) { - if (this.currentPath) { - if (this.currentPath.shape.points.length === 0) { - this.currentPath.shape.points = [0, 0]; - } - } - else { - this.moveTo(0,0); - } - - var n = 20, - dt, - dt2, - dt3, - t2, - t3, - points = this.currentPath.shape.points; - - var fromX = points[points.length-2]; - var fromY = points[points.length-1]; - - var j = 0; - - for (var i = 1; i <= n; ++i) { - j = i / n; - - dt = (1 - j); - dt2 = dt * dt; - dt3 = dt2 * dt; - - t2 = j * j; - t3 = t2 * j; - - points.push( dt3 * fromX + 3 * dt2 * j * cpX + 3 * dt * t2 * cpX2 + t3 * toX, - dt3 * fromY + 3 * dt2 * j * cpY + 3 * dt * t2 * cpY2 + t3 * toY); - } - - this.dirty = true; - - return this; -}; - -/** - * The arcTo() method creates an arc/curve between two tangents on the canvas. - * - * "borrowed" from https://code.google.com/p/fxcanvas/ - thanks google! - * - * @param x1 {number} The x-coordinate of the beginning of the arc - * @param y1 {number} The y-coordinate of the beginning of the arc - * @param x2 {number} The x-coordinate of the end of the arc - * @param y2 {number} The y-coordinate of the end of the arc - * @param radius {number} The radius of the arc - * @return {Graphics} - */ -Graphics.prototype.arcTo = function (x1, y1, x2, y2, radius) { - if (this.currentPath) { - if (this.currentPath.shape.points.length === 0) { - this.currentPath.shape.points.push(x1, y1); - } - } - else { - this.moveTo(x1, y1); - } - - var points = this.currentPath.shape.points, - fromX = points[points.length-2], - fromY = points[points.length-1], - a1 = fromY - y1, - b1 = fromX - x1, - a2 = y2 - y1, - b2 = x2 - x1, - mm = Math.abs(a1 * b2 - b1 * a2); - - if (mm < 1.0e-8 || radius === 0) { - if (points[points.length-2] !== x1 || points[points.length-1] !== y1) { - points.push(x1, y1); - } - } - else { - var dd = a1 * a1 + b1 * b1, - cc = a2 * a2 + b2 * b2, - tt = a1 * a2 + b1 * b2, - k1 = radius * Math.sqrt(dd) / mm, - k2 = radius * Math.sqrt(cc) / mm, - j1 = k1 * tt / dd, - j2 = k2 * tt / cc, - cx = k1 * b2 + k2 * b1, - cy = k1 * a2 + k2 * a1, - px = b1 * (k2 + j1), - py = a1 * (k2 + j1), - qx = b2 * (k1 + j2), - qy = a2 * (k1 + j2), - startAngle = Math.atan2(py - cy, px - cx), - endAngle = Math.atan2(qy - cy, qx - cx); - - this.arc(cx + x1, cy + y1, radius, startAngle, endAngle, b1 * a2 > b2 * a1); - } - - this.dirty = true; - - return this; -}; - -/** - * The arc method creates an arc/curve (used to create circles, or parts of circles). - * - * @param cx {number} The x-coordinate of the center of the circle - * @param cy {number} The y-coordinate of the center of the circle - * @param radius {number} The radius of the circle - * @param startAngle {number} The starting angle, in radians (0 is at the 3 o'clock position of the arc's circle) - * @param endAngle {number} The ending angle, in radians - * @param anticlockwise {boolean} Optional. Specifies whether the drawing should be counterclockwise or clockwise. False is default, and indicates clockwise, while true indicates counter-clockwise. - * @return {Graphics} - */ -Graphics.prototype.arc = function (cx, cy, radius, startAngle, endAngle, anticlockwise) { - var startX = cx + Math.cos(startAngle) * radius; - var startY = cy + Math.sin(startAngle) * radius; - var points; - - // TODO - This if-else makes no sense. It uses currentPath in the else where it doesn't exist... - if (this.currentPath) { - points = this.currentPath.shape.points; - - if (points.length === 0) { - points.push(startX, startY); - } - else if (points[points.length-2] !== startX || points[points.length-1] !== startY) { - points.push(startX, startY); - } - } - else { - this.moveTo(startX, startY); - points = this.currentPath.shape.points; - } - - if (startAngle === endAngle) { - return this; - } - - if (!anticlockwise && endAngle <= startAngle) { - endAngle += Math.PI * 2; - } - else if (anticlockwise && startAngle <= endAngle) { - startAngle += Math.PI * 2; - } - - var sweep = anticlockwise ? (startAngle - endAngle) *-1 : (endAngle - startAngle); - var segs = (Math.abs(sweep)/ (Math.PI * 2)) * 40; - - if (sweep === 0) { - return this; - } - - var theta = sweep/(segs*2); - var theta2 = theta*2; - - var cTheta = Math.cos(theta); - var sTheta = Math.sin(theta); - - var segMinus = segs - 1; - - var remainder = ( segMinus % 1 ) / segMinus; - - for (var i = 0; i <= segMinus; ++i) { - var real = i + remainder * i; - var angle = ((theta) + startAngle + (theta2 * real)); - - var c = Math.cos(angle); - var s = -Math.sin(angle); - - points.push(( (cTheta * c) + (sTheta * s) ) * radius + cx, - ( (cTheta * -s) + (sTheta * c) ) * radius + cy); - } - - this.dirty = true; - - return this; -}; - -/** - * Specifies a simple one-color fill that subsequent calls to other Graphics methods - * (such as lineTo() or drawCircle()) use when drawing. - * - * @param color {number} the color of the fill - * @param alpha {number} the alpha of the fill - * @return {Graphics} - */ -Graphics.prototype.beginFill = function (color, alpha) { - this.filling = true; - this.fillColor = color || 0; - this.fillAlpha = (alpha === undefined) ? 1 : alpha; - - if (this.currentPath) { - if (this.currentPath.shape.points.length <= 2) { - this.currentPath.fill = this.filling; - this.currentPath.fillColor = this.fillColor; - this.currentPath.fillAlpha = this.fillAlpha; - } - } - return this; -}; - -/** - * Applies a fill to the lines and shapes that were added since the last call to the beginFill() method. - * - * @return {Graphics} - */ -Graphics.prototype.endFill = function () { - this.filling = false; - this.fillColor = null; - this.fillAlpha = 1; - - return this; -}; - -/** - * - * @param x {number} The X coord of the top-left of the rectangle - * @param y {number} The Y coord of the top-left of the rectangle - * @param width {number} The width of the rectangle - * @param height {number} The height of the rectangle - * @return {Graphics} - */ -Graphics.prototype.drawRect = function ( x, y, width, height ) { - this.drawShape(new math.Rectangle(x,y, width, height)); - - return this; -}; - -/** - * - * @param x {number} The X coord of the top-left of the rectangle - * @param y {number} The Y coord of the top-left of the rectangle - * @param width {number} The width of the rectangle - * @param height {number} The height of the rectangle - * @param radius {number} Radius of the rectangle corners - */ -Graphics.prototype.drawRoundedRect = function ( x, y, width, height, radius ) { - this.drawShape(new math.RoundedRectangle(x, y, width, height, radius)); - - return this; -}; - -/** - * Draws a circle. - * - * @param x {number} The X coordinate of the center of the circle - * @param y {number} The Y coordinate of the center of the circle - * @param radius {number} The radius of the circle - * @return {Graphics} - */ -Graphics.prototype.drawCircle = function (x, y, radius) { - this.drawShape(new math.Circle(x,y, radius)); - - return this; -}; - -/** - * Draws an ellipse. - * - * @param x {number} The X coordinate of the center of the ellipse - * @param y {number} The Y coordinate of the center of the ellipse - * @param width {number} The half width of the ellipse - * @param height {number} The half height of the ellipse - * @return {Graphics} - */ -Graphics.prototype.drawEllipse = function (x, y, width, height) { - this.drawShape(new math.Ellipse(x, y, width, height)); - - return this; -}; - -/** - * Draws a polygon using the given path. - * - * @param path {Array} The path data used to construct the polygon. - * @return {Graphics} - */ -Graphics.prototype.drawPolygon = function (path) { - if (!(path instanceof Array)) { - path = Array.prototype.slice.call(arguments); - } - - this.drawShape(new math.Polygon(path)); - - return this; -}; - -/** - * Clears the graphics that were drawn to this Graphics object, and resets fill and line style settings. - * - * @return {Graphics} - */ -Graphics.prototype.clear = function () { - this.lineWidth = 0; - this.filling = false; - - this.dirty = true; - this.clearDirty = true; - this.graphicsData = []; - - return this; -}; - -/** - * Useful function that returns a texture of the graphics object that can then be used to create sprites - * This can be quite useful if your geometry is complicated and needs to be reused multiple times. - * - * @param resolution {number} The resolution of the texture being generated - * @param scaleMode {number} Should be one of the scaleMode consts - * @return {Texture} a texture of the graphics object - */ -Graphics.prototype.generateTexture = function (resolution, scaleMode) { - resolution = resolution || 1; - - var bounds = this.getBounds(); - - var canvasBuffer = new CanvasBuffer(bounds.width * resolution, bounds.height * resolution); - - var texture = Texture.fromCanvas(canvasBuffer.canvas, scaleMode); - texture.baseTexture.resolution = resolution; - - canvasBuffer.context.scale(resolution, resolution); - - canvasBuffer.context.translate(-bounds.x,-bounds.y); - - CanvasGraphics.renderGraphics(this, canvasBuffer.context); - - return texture; -}; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} - */ -Graphics.prototype.renderWebGL = function (renderer) { - // if the sprite is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0 || this.isMask === true) { - return; - } - - if (this._cacheAsBitmap) { - if (this.dirty || this.cachedSpriteDirty) { - this._generateCachedSprite(); - - // we will also need to update the texture on the gpu too! - this.updateCachedSpriteTexture(); - - this.cachedSpriteDirty = false; - this.dirty = false; - } - - this._cachedSprite.worldAlpha = this.worldAlpha; - - Sprite.prototype.renderWebGL.call(this._cachedSprite, renderer); - - return; - } - else { - renderer.spriteBatch.stop(); - renderer.blendModeManager.setBlendMode(this.blendMode); - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - if (this._filters) { - renderer.filterManager.pushFilter(this._filterBlock); - } - - // check blend mode - if (this.blendMode !== renderer.spriteBatch.currentBlendMode) { - renderer.spriteBatch.currentBlendMode = this.blendMode; - - var blendModeWebGL = renderer.blendModes[renderer.spriteBatch.currentBlendMode]; - - renderer.spriteBatch.gl.blendFunc(blendModeWebGL[0], blendModeWebGL[1]); - } - - // check if the webgl graphic needs to be updated - if (this.glDirty) { - this.dirty = true; - this.glDirty = false; - } - - WebGLGraphics.renderGraphics(this, renderer); - - // only render if it has children! - if (this.children.length) { - renderer.spriteBatch.start(); - - // simple render children! - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderWebGL(renderer); - } - - renderer.spriteBatch.stop(); - } - - if (this._filters) { - renderer.filterManager.popFilter(); - } - - if (this._mask) { - renderer.maskManager.popMask(this.mask, renderer); - } - - renderer.drawCount++; - - renderer.spriteBatch.start(); - } -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} - * @private - */ -Graphics.prototype.renderCanvas = function (renderer) { - // if the sprite is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0 || this.isMask === true) { - return; - } - - if (this._cacheAsBitmap) { - if (this.dirty || this.cachedSpriteDirty) { - this._generateCachedSprite(); - - // we will also need to update the texture - this.updateCachedSpriteTexture(); - - this.cachedSpriteDirty = false; - this.dirty = false; - } - - this._cachedSprite.alpha = this.alpha; - - Sprite.prototype.renderCanvas.call(this._cachedSprite, renderer); - - return; - } - else { - var context = renderer.context; - var transform = this.worldTransform; - - if (this.blendMode !== renderer.currentBlendMode) { - renderer.currentBlendMode = this.blendMode; - context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode]; - } - - if (this._mask) { - renderer.maskManager.pushMask(this._mask, renderer); - } - - var resolution = renderer.resolution; - context.setTransform( - transform.a * resolution, - transform.b * resolution, - transform.c * resolution, - transform.d * resolution, - transform.tx * resolution, - transform.ty * resolution - ); - - CanvasGraphics.renderGraphics(this, context); - - for (var i = 0, j = this.children.length; i < j; ++i) { - this.children[i].renderCanvas(renderer); - } - - if (this._mask) { - renderer.maskManager.popMask(renderer); - } - } -}; - -/** - * Retrieves the bounds of the graphic shape as a rectangle object - * - * @return {Rectangle} the rectangular bounding area - */ -Graphics.prototype.getBounds = function (matrix) { - // return an empty object if the item is a mask! - if (this.isMask) { - return math.Rectangle.EMPTY; - } - - if (this.dirty) { - this.updateLocalBounds(); - - this.glDirty = true; - this.cachedSpriteDirty = true; - this.dirty = false; - } - - var bounds = this._localBounds; - - var w0 = bounds.x; - var w1 = bounds.width + bounds.x; - - var h0 = bounds.y; - var h1 = bounds.height + bounds.y; - - var worldTransform = matrix || this.worldTransform; - - var a = worldTransform.a; - var b = worldTransform.b; - var c = worldTransform.c; - var d = worldTransform.d; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var x1 = a * w1 + c * h1 + tx; - var y1 = d * h1 + b * w1 + ty; - - var x2 = a * w0 + c * h1 + tx; - var y2 = d * h1 + b * w0 + ty; - - var x3 = a * w0 + c * h0 + tx; - var y3 = d * h0 + b * w0 + ty; - - var x4 = a * w1 + c * h0 + tx; - var y4 = d * h0 + b * w1 + ty; - - var maxX = x1; - var maxY = y1; - - var minX = x1; - var minY = y1; - - minX = x2 < minX ? x2 : minX; - minX = x3 < minX ? x3 : minX; - minX = x4 < minX ? x4 : minX; - - minY = y2 < minY ? y2 : minY; - minY = y3 < minY ? y3 : minY; - minY = y4 < minY ? y4 : minY; - - maxX = x2 > maxX ? x2 : maxX; - maxX = x3 > maxX ? x3 : maxX; - maxX = x4 > maxX ? x4 : maxX; - - maxY = y2 > maxY ? y2 : maxY; - maxY = y3 > maxY ? y3 : maxY; - maxY = y4 > maxY ? y4 : maxY; - - this._bounds.x = minX; - this._bounds.width = maxX - minX; - - this._bounds.y = minY; - this._bounds.height = maxY - minY; - - return this._bounds; -}; - -/** - * Update the bounds of the object - * - */ -Graphics.prototype.updateLocalBounds = function () { - var minX = Infinity; - var maxX = -Infinity; - - var minY = Infinity; - var maxY = -Infinity; - - if (this.graphicsData.length) { - var shape, points, x, y, w, h; - - for (var i = 0; i < this.graphicsData.length; i++) { - var data = this.graphicsData[i]; - var type = data.type; - var lineWidth = data.lineWidth; - shape = data.shape; - - if (type === CONST.SHAPES.RECT || type === CONST.SHAPES.RREC) { - x = shape.x - lineWidth/2; - y = shape.y - lineWidth/2; - w = shape.width + lineWidth; - h = shape.height + lineWidth; - - minX = x < minX ? x : minX; - maxX = x + w > maxX ? x + w : maxX; - - minY = y < minY ? y : minY; - maxY = y + h > maxY ? y + h : maxY; - } - else if (type === CONST.SHAPES.CIRC) { - x = shape.x; - y = shape.y; - w = shape.radius + lineWidth/2; - h = shape.radius + lineWidth/2; - - minX = x - w < minX ? x - w : minX; - maxX = x + w > maxX ? x + w : maxX; - - minY = y - h < minY ? y - h : minY; - maxY = y + h > maxY ? y + h : maxY; - } - else if (type === CONST.SHAPES.ELIP) { - x = shape.x; - y = shape.y; - w = shape.width + lineWidth/2; - h = shape.height + lineWidth/2; - - minX = x - w < minX ? x - w : minX; - maxX = x + w > maxX ? x + w : maxX; - - minY = y - h < minY ? y - h : minY; - maxY = y + h > maxY ? y + h : maxY; - } - else { - // POLY - points = shape.points; - - for (var j = 0; j < points.length; j += 2) { - x = points[j]; - y = points[j+1]; - - minX = x-lineWidth < minX ? x-lineWidth : minX; - maxX = x+lineWidth > maxX ? x+lineWidth : maxX; - - minY = y-lineWidth < minY ? y-lineWidth : minY; - maxY = y+lineWidth > maxY ? y+lineWidth : maxY; - } - } - } - } - else { - minX = 0; - maxX = 0; - minY = 0; - maxY = 0; - } - - var padding = this.boundsPadding; - - this._localBounds.x = minX - padding; - this._localBounds.width = (maxX - minX) + padding * 2; - - this._localBounds.y = minY - padding; - this._localBounds.height = (maxY - minY) + padding * 2; -}; - -/** - * Generates the cached sprite when the sprite has cacheAsBitmap = true - * - * @private - */ -Graphics.prototype._generateCachedSprite = function () { - var bounds = this.getLocalBounds(); - - if (!this._cachedSprite) { - var canvasBuffer = new CanvasBuffer(bounds.width, bounds.height); - var texture = Texture.fromCanvas(canvasBuffer.canvas); - - this._cachedSprite = new Sprite(texture); - this._cachedSprite.buffer = canvasBuffer; - - this._cachedSprite.worldTransform = this.worldTransform; - } - else { - this._cachedSprite.buffer.resize(bounds.width, bounds.height); - } - - // leverage the anchor to account for the offset of the element - this._cachedSprite.anchor.x = -( bounds.x / bounds.width ); - this._cachedSprite.anchor.y = -( bounds.y / bounds.height ); - - // this._cachedSprite.buffer.context.save(); - this._cachedSprite.buffer.context.translate(-bounds.x,-bounds.y); - - // make sure we set the alpha of the graphics to 1 for the render.. - this.worldAlpha = 1; - - // now render the graphic.. - CanvasGraphics.renderGraphics(this, this._cachedSprite.buffer.context); - - this._cachedSprite.alpha = this.alpha; -}; - -/** - * Updates texture size based on canvas size - * - * @private - */ -Graphics.prototype.updateCachedSpriteTexture = function () { - var cachedSprite = this._cachedSprite; - var texture = cachedSprite.texture; - var canvas = cachedSprite.buffer.canvas; - - texture.baseTexture.width = canvas.width; - texture.baseTexture.height = canvas.height; - texture.crop.width = texture.frame.width = canvas.width; - texture.crop.height = texture.frame.height = canvas.height; - - cachedSprite._width = canvas.width; - cachedSprite._height = canvas.height; - - // update the dirty base textures - texture.baseTexture.dirty(); -}; - -/** - * Destroys a previous cached sprite. - * - */ -Graphics.prototype.destroyCachedSprite = function () { - this._cachedSprite.texture.destroy(true); - - // let the gc collect the unused sprite - // TODO could be object pooled! - this._cachedSprite = null; -}; - -/** - * Draws the given shape to this Graphics object. Can be any of Circle, Rectangle, Ellipse, Line or Polygon. - * - * @param {Circle|Rectangle|Ellipse|Line|Polygon} shape The Shape object to draw. - * @return {GraphicsData} The generated GraphicsData object. - */ -Graphics.prototype.drawShape = function (shape) { - if (this.currentPath) { - // check current path! - if (this.currentPath.shape.points.length <= 2) { - this.graphicsData.pop(); - } - } - - this.currentPath = null; - - var data = new GraphicsData(this.lineWidth, this.lineColor, this.lineAlpha, this.fillColor, this.fillAlpha, this.filling, shape); - - this.graphicsData.push(data); - - if (data.type === CONST.SHAPES.POLY) { - data.shape.closed = this.filling; - this.currentPath = data; - } - - this.dirty = true; - - return data; -}; - -},{"../const":4,"../display/DisplayObjectContainer":6,"../display/Sprite":7,"../math":12,"../renderers/canvas/utils/CanvasBuffer":21,"../renderers/canvas/utils/CanvasGraphics":22,"../renderers/webgl/utils/WebGLGraphics":39,"../textures/Texture":44,"./GraphicsData":19}],19:[function(require,module,exports){ -/** - * A GraphicsData object. - * - * @class - * @namespace PIXI - */ -function GraphicsData(lineWidth, lineColor, lineAlpha, fillColor, fillAlpha, fill, shape) { - this.lineWidth = lineWidth; - this.lineColor = lineColor; - this.lineAlpha = lineAlpha; - this._lineTint = lineColor; - - this.fillColor = fillColor; - this.fillAlpha = fillAlpha; - this._fillTint = fillColor; - this.fill = fill; - - this.shape = shape; - this.type = shape.type; -} - -GraphicsData.prototype.constructor = GraphicsData; -module.exports = GraphicsData; - -/** - * Creates a new GraphicsData object with the same values as this one. - * - * @return {GraphicsData} - */ -GraphicsData.prototype.clone = function () { - return new GraphicsData( - this.lineWidth, - this.lineColor, - this.lineAlpha, - this.fillColor, - this.fillAlpha, - this.fill, - this.shape - ); -}; - -},{}],20:[function(require,module,exports){ -var CanvasMaskManager = require('./utils/CanvasMaskManager'), - utils = require('../../utils'), - CONST = require('../../const'); - -/** - * The CanvasRenderer draws the scene and all its content onto a 2d canvas. This renderer should be used for browsers that do not support webGL. - * Don't forget to add the CanvasRenderer.view to your DOM or you will not see anything :) - * - * @class - * @namespace PIXI - * @param [width=800] {number} the width of the canvas view - * @param [height=600] {number} the height of the canvas view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - * @param [options.clearBeforeRender=true] {boolean} This sets if the CanvasRenderer will clear the canvas or not before the new render pass. - */ -function CanvasRenderer(width, height, options) { - utils.sayHello('Canvas'); - - if (options) { - for (var i in CONST.defaultRenderOptions) { - if (typeof options[i] === 'undefined') { - options[i] = CONST.defaultRenderOptions[i]; - } - } - } - else { - options = CONST.defaultRenderOptions; - } - - /** - * The renderer type. - * - * @member {number} - */ - this.type = CONST.CANVAS_RENDERER; - - /** - * The resolution of the canvas. - * - * @member {number} - */ - this.resolution = options.resolution; - - /** - * This sets if the CanvasRenderer will clear the canvas or not before the new render pass. - * If the scene is NOT transparent Pixi will use a canvas sized fillRect operation every frame to set the canvas background color. - * If the scene is transparent Pixi will use clearRect to clear the canvas every frame. - * Disable this by setting this to false. For example if your game has a canvas filling background image you often don't need this set. - * - * @member {boolean} - * @default - */ - this.clearBeforeRender = options.clearBeforeRender; - - /** - * The background color as a number. - * - * @member {number} - * @private - */ - this._backgroundColor = 0x000000; - - /** - * The background color as a string. - * - * @member {string} - * @private - */ - this._backgroundColorString = '#000000'; - - this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter - - /** - * Whether the render view is transparent - * - * @member {boolean} - */ - this.transparent = options.transparent; - - /** - * Whether the render view should be resized automatically - * - * @member {boolean} - */ - this.autoResize = options.autoResize || false; - - - /** - * The width of the canvas view - * - * @member {number} - * @default 800 - */ - this.width = width || 800; - - /** - * The height of the canvas view - * - * @member {number} - * @default 600 - */ - this.height = height || 600; - - this.width *= this.resolution; - this.height *= this.resolution; - - /** - * The canvas element that everything is drawn to. - * - * @member {HTMLCanvasElement} - */ - this.view = options.view || document.createElement('canvas'); - - /** - * The canvas 2d context that everything is drawn with - * @member {CanvasRenderingContext2D} - */ - this.context = this.view.getContext('2d', { alpha: this.transparent }); - - /** - * Boolean flag controlling canvas refresh. - * - * @member {boolean} - */ - this.refresh = true; - - this.view.width = this.width * this.resolution; - this.view.height = this.height * this.resolution; - - /** - * Internal var. - * - * @member {number} - */ - this.count = 0; - - /** - * Instance of a CanvasMaskManager, handles masking when using the canvas renderer - * @member {CanvasMaskManager} - */ - this.maskManager = new CanvasMaskManager(); - - /** - * If true Pixi will Math.floor() x/y values when rendering, stopping pixel interpolation. - * Handy for crisp pixel art and speed on legacy devices. - * - * @member {boolean} - */ - this.roundPixels = false; - - this.scaleMode = null; - - this.smoothProperty = null; - - if (this.context.imageSmoothingEnabled) { - this.smoothProperty = 'imageSmoothingEnabled'; - } - else if (this.context.webkitImageSmoothingEnabled) { - this.smoothProperty = 'webkitImageSmoothingEnabled'; - } - else if (this.context.mozImageSmoothingEnabled) { - this.smoothProperty = 'mozImageSmoothingEnabled'; - } - else if (this.context.oImageSmoothingEnabled) { - this.smoothProperty = 'oImageSmoothingEnabled'; - } - else if (this.context.msImageSmoothingEnabled) { - this.smoothProperty = 'msImageSmoothingEnabled'; - } - - this.currentBlendMode = CONST.blendModes.NORMAL; - - this.blendModes = null; - - this._mapBlendModes(); - - this.resize(width, height); -} - -// constructor -CanvasRenderer.prototype.constructor = CanvasRenderer; -module.exports = CanvasRenderer; - -Object.defineProperties(CanvasRenderer.prototype, { - /** - * The background color to fill if not transparent - * - * @member {number} - * @memberof CanvasRenderer# - */ - backgroundColor: { - get: function () { - return this._backgroundColor; - }, - set: function (val) { - this._backgroundColor = val; - this._backgroundColorString = utils.hex2string(val); - } - } -}); - -/** - * Renders the object to this canvas view - * - * @param object {DisplayObject} the object to be rendered - */ -CanvasRenderer.prototype.render = function (object) { - object.updateTransform(); - - this.context.setTransform(1,0,0,1,0,0); - - this.context.globalAlpha = 1; - - this.currentBlendMode = CONST.blendModes.NORMAL; - this.context.globalCompositeOperation = this.blendModes[CONST.blendModes.NORMAL]; - - if (navigator.isCocoonJS && this.view.screencanvas) { - this.context.fillStyle = 'black'; - this.context.clear(); - } - - if (this.clearBeforeRender) { - if (this.transparent) { - this.context.clearRect(0, 0, this.width, this.height); - } - else { - this.context.fillStyle = this._backgroundColorString; - this.context.fillRect(0, 0, this.width , this.height); - } - } - - this.renderDisplayObject(object); -}; - -/** - * Removes everything from the renderer and optionally removes the Canvas DOM element. - * - * @param [removeView=false] {boolean} Removes the Canvas element from the DOM. - */ -CanvasRenderer.prototype.destroy = function (removeView) { - if (removeView && this.view.parent) { - this.view.parent.removeChild(this.view); - } - - this.view = null; - this.context = null; - this.maskManager = null; -}; - -/** - * Resizes the canvas view to the specified width and height - * - * @param width {number} the new width of the canvas view - * @param height {number} the new height of the canvas view - */ -CanvasRenderer.prototype.resize = function (width, height) { - this.width = width * this.resolution; - this.height = height * this.resolution; - - this.view.width = this.width; - this.view.height = this.height; - - if (this.autoResize) { - this.view.style.width = this.width / this.resolution + 'px'; - this.view.style.height = this.height / this.resolution + 'px'; - } -}; - -/** - * Renders a display object - * - * @param displayObject {DisplayObject} The displayObject to render - * @private - */ -CanvasRenderer.prototype.renderDisplayObject = function (displayObject) { - displayObject.renderCanvas(this); -}; - -/** - * Maps Pixi blend modes to canvas blend modes. - * - * @private - */ -CanvasRenderer.prototype._mapBlendModes = function () { - if (!this.blendModes) { - this.blendModes = {}; - - if (utils.canUseNewCanvasBlendModes()) { - this.blendModes[CONST.blendModes.NORMAL] = 'source-over'; - this.blendModes[CONST.blendModes.ADD] = 'lighter'; //IS THIS OK??? - this.blendModes[CONST.blendModes.MULTIPLY] = 'multiply'; - this.blendModes[CONST.blendModes.SCREEN] = 'screen'; - this.blendModes[CONST.blendModes.OVERLAY] = 'overlay'; - this.blendModes[CONST.blendModes.DARKEN] = 'darken'; - this.blendModes[CONST.blendModes.LIGHTEN] = 'lighten'; - this.blendModes[CONST.blendModes.COLOR_DODGE] = 'color-dodge'; - this.blendModes[CONST.blendModes.COLOR_BURN] = 'color-burn'; - this.blendModes[CONST.blendModes.HARD_LIGHT] = 'hard-light'; - this.blendModes[CONST.blendModes.SOFT_LIGHT] = 'soft-light'; - this.blendModes[CONST.blendModes.DIFFERENCE] = 'difference'; - this.blendModes[CONST.blendModes.EXCLUSION] = 'exclusion'; - this.blendModes[CONST.blendModes.HUE] = 'hue'; - this.blendModes[CONST.blendModes.SATURATION] = 'saturation'; - this.blendModes[CONST.blendModes.COLOR] = 'color'; - this.blendModes[CONST.blendModes.LUMINOSITY] = 'luminosity'; - } - else { - // this means that the browser does not support the cool new blend modes in canvas 'cough' ie 'cough' - this.blendModes[CONST.blendModes.NORMAL] = 'source-over'; - this.blendModes[CONST.blendModes.ADD] = 'lighter'; //IS THIS OK??? - this.blendModes[CONST.blendModes.MULTIPLY] = 'source-over'; - this.blendModes[CONST.blendModes.SCREEN] = 'source-over'; - this.blendModes[CONST.blendModes.OVERLAY] = 'source-over'; - this.blendModes[CONST.blendModes.DARKEN] = 'source-over'; - this.blendModes[CONST.blendModes.LIGHTEN] = 'source-over'; - this.blendModes[CONST.blendModes.COLOR_DODGE] = 'source-over'; - this.blendModes[CONST.blendModes.COLOR_BURN] = 'source-over'; - this.blendModes[CONST.blendModes.HARD_LIGHT] = 'source-over'; - this.blendModes[CONST.blendModes.SOFT_LIGHT] = 'source-over'; - this.blendModes[CONST.blendModes.DIFFERENCE] = 'source-over'; - this.blendModes[CONST.blendModes.EXCLUSION] = 'source-over'; - this.blendModes[CONST.blendModes.HUE] = 'source-over'; - this.blendModes[CONST.blendModes.SATURATION] = 'source-over'; - this.blendModes[CONST.blendModes.COLOR] = 'source-over'; - this.blendModes[CONST.blendModes.LUMINOSITY] = 'source-over'; - } - } -}; - -},{"../../const":4,"../../utils":50,"./utils/CanvasMaskManager":23}],21:[function(require,module,exports){ -/** - * Creates a Canvas element of the given size. - * - * @class - * @namespace PIXI - * @param width {number} the width for the newly created canvas - * @param height {number} the height for the newly created canvas - */ -function CanvasBuffer(width, height) { - /** - * The Canvas object that belongs to this CanvasBuffer. - * - * @member {HTMLCanvasElement} - */ - this.canvas = document.createElement('canvas'); - - /** - * A CanvasRenderingContext2D object representing a two-dimensional rendering context. - * - * @member {CanvasRenderingContext2D} - */ - this.context = this.canvas.getContext('2d'); - - this.canvas.width = width; - this.canvas.height = height; -} - -CanvasBuffer.prototype.constructor = CanvasBuffer; -module.exports = CanvasBuffer; - -Object.defineProperties(CanvasBuffer.prototype, { - /** - * The width of the canvas buffer in pixels. - * - * @member {number} - * @memberof CanvasBuffer# - */ - width: { - get: function () { - return this.canvas.width; - }, - set: function (val) { - this.canvas.width = val; - } - }, - /** - * The height of the canvas buffer in pixels. - * - * @member {number} - * @memberof CanvasBuffer# - */ - height: { - get: function () { - return this.canvas.height; - }, - set: function (val) { - this.canvas.height = val; - } - } -}); - -/** - * Clears the canvas that was created by the CanvasBuffer class. - * - * @private - */ -CanvasBuffer.prototype.clear = function () { - this.context.setTransform(1, 0, 0, 1, 0, 0); - this.context.clearRect(0,0, this.canvas.width, this.canvas.height); -}; - -/** - * Resizes the canvas to the specified width and height. - * - * @param width {number} the new width of the canvas - * @param height {number} the new height of the canvas - */ -CanvasBuffer.prototype.resize = function (width, height) { - this.canvas.width = width; - this.canvas.height = height; -}; - -},{}],22:[function(require,module,exports){ -var CONST = require('../../../const'); - -/** - * A set of functions used by the canvas renderer to draw the primitive graphics data. - * - * @namespace PIXI - */ -var CanvasGraphics = module.exports = {}; - -/* - * Renders a Graphics object to a canvas. - * - * @param graphics {Graphics} the actual graphics object to render - * @param context {CanvasRenderingContext2D} the 2d drawing method of the canvas - */ -CanvasGraphics.renderGraphics = function (graphics, context) { - var worldAlpha = graphics.worldAlpha; - - if (graphics.dirty) { - this.updateGraphicsTint(graphics); - graphics.dirty = false; - } - - for (var i = 0; i < graphics.graphicsData.length; i++) { - var data = graphics.graphicsData[i]; - var shape = data.shape; - - var fillColor = data._fillTint; - var lineColor = data._lineTint; - - context.lineWidth = data.lineWidth; - - if (data.type === CONST.SHAPES.POLY) { - context.beginPath(); - - var points = shape.points; - - context.moveTo(points[0], points[1]); - - for (var j=1; j < points.length/2; j++) { - context.lineTo(points[j * 2], points[j * 2 + 1]); - } - - if (shape.closed) { - context.lineTo(points[0], points[1]); - } - - // if the first and last point are the same close the path - much neater :) - if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) { - context.closePath(); - } - - if (data.fill) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - else if (data.type === CONST.SHAPES.RECT) { - - if (data.fillColor || data.fillColor === 0) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fillRect(shape.x, shape.y, shape.width, shape.height); - - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.strokeRect(shape.x, shape.y, shape.width, shape.height); - } - } - else if (data.type === CONST.SHAPES.CIRC) { - // TODO - need to be Undefined! - context.beginPath(); - context.arc(shape.x, shape.y, shape.radius,0,2*Math.PI); - context.closePath(); - - if (data.fill) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - else if (data.type === CONST.SHAPES.ELIP) { - // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas - - var w = shape.width * 2; - var h = shape.height * 2; - - var x = shape.x - w/2; - var y = shape.y - h/2; - - context.beginPath(); - - var kappa = 0.5522848, - ox = (w / 2) * kappa, // control point offset horizontal - oy = (h / 2) * kappa, // control point offset vertical - xe = x + w, // x-end - ye = y + h, // y-end - xm = x + w / 2, // x-middle - ym = y + h / 2; // y-middle - - context.moveTo(x, ym); - context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y); - context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym); - context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye); - context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym); - - context.closePath(); - - if (data.fill) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - else if (data.type === CONST.SHAPES.RREC) { - var rx = shape.x; - var ry = shape.y; - var width = shape.width; - var height = shape.height; - var radius = shape.radius; - - var maxRadius = Math.min(width, height) / 2 | 0; - radius = radius > maxRadius ? maxRadius : radius; - - context.beginPath(); - context.moveTo(rx, ry + radius); - context.lineTo(rx, ry + height - radius); - context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height); - context.lineTo(rx + width - radius, ry + height); - context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius); - context.lineTo(rx + width, ry + radius); - context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry); - context.lineTo(rx + radius, ry); - context.quadraticCurveTo(rx, ry, rx, ry + radius); - context.closePath(); - - if (data.fillColor || data.fillColor === 0) { - context.globalAlpha = data.fillAlpha * worldAlpha; - context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6); - context.fill(); - - } - if (data.lineWidth) { - context.globalAlpha = data.lineAlpha * worldAlpha; - context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6); - context.stroke(); - } - } - } -}; - -/* - * Renders a graphics mask - * - * @private - * @param graphics {Graphics} the graphics which will be used as a mask - * @param context {CanvasRenderingContext2D} the context 2d method of the canvas - */ -CanvasGraphics.renderGraphicsMask = function (graphics, context) { - var len = graphics.graphicsData.length; - - if (len === 0) { - return; - } - - context.beginPath(); - - for (var i = 0; i < len; i++) { - var data = graphics.graphicsData[i]; - var shape = data.shape; - - if (data.type === CONST.SHAPES.POLY) { - - var points = shape.points; - - context.moveTo(points[0], points[1]); - - for (var j=1; j < points.length/2; j++) { - context.lineTo(points[j * 2], points[j * 2 + 1]); - } - - // if the first and last point are the same close the path - much neater :) - if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) { - context.closePath(); - } - - } - else if (data.type === CONST.SHAPES.RECT) { - context.rect(shape.x, shape.y, shape.width, shape.height); - context.closePath(); - } - else if (data.type === CONST.SHAPES.CIRC) { - // TODO - need to be Undefined! - context.arc(shape.x, shape.y, shape.radius, 0, 2 * Math.PI); - context.closePath(); - } - else if (data.type === CONST.SHAPES.ELIP) { - - // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas - - var w = shape.width * 2; - var h = shape.height * 2; - - var x = shape.x - w/2; - var y = shape.y - h/2; - - var kappa = 0.5522848, - ox = (w / 2) * kappa, // control point offset horizontal - oy = (h / 2) * kappa, // control point offset vertical - xe = x + w, // x-end - ye = y + h, // y-end - xm = x + w / 2, // x-middle - ym = y + h / 2; // y-middle - - context.moveTo(x, ym); - context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y); - context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym); - context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye); - context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym); - context.closePath(); - } - else if (data.type === CONST.SHAPES.RREC) { - - var rx = shape.x; - var ry = shape.y; - var width = shape.width; - var height = shape.height; - var radius = shape.radius; - - var maxRadius = Math.min(width, height) / 2 | 0; - radius = radius > maxRadius ? maxRadius : radius; - - context.moveTo(rx, ry + radius); - context.lineTo(rx, ry + height - radius); - context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height); - context.lineTo(rx + width - radius, ry + height); - context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius); - context.lineTo(rx + width, ry + radius); - context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry); - context.lineTo(rx + radius, ry); - context.quadraticCurveTo(rx, ry, rx, ry + radius); - context.closePath(); - } - } -}; - -CanvasGraphics.updateGraphicsTint = function (graphics) { - if (graphics.tint === 0xFFFFFF) { - return; - } - - var tintR = (graphics.tint >> 16 & 0xFF) / 255; - var tintG = (graphics.tint >> 8 & 0xFF) / 255; - var tintB = (graphics.tint & 0xFF)/ 255; - - for (var i = 0; i < graphics.graphicsData.length; i++) { - var data = graphics.graphicsData[i]; - - var fillColor = data.fillColor | 0; - var lineColor = data.lineColor | 0; - - /* - var colorR = (fillColor >> 16 & 0xFF) / 255; - var colorG = (fillColor >> 8 & 0xFF) / 255; - var colorB = (fillColor & 0xFF) / 255; - - colorR *= tintR; - colorG *= tintG; - colorB *= tintB; - - fillColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255); - - colorR = (lineColor >> 16 & 0xFF) / 255; - colorG = (lineColor >> 8 & 0xFF) / 255; - colorB = (lineColor & 0xFF) / 255; - - colorR *= tintR; - colorG *= tintG; - colorB *= tintB; - - lineColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255); - */ - - // super inline cos im an optimization NAZI :) - data._fillTint = (((fillColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((fillColor >> 8 & 0xFF) / 255 * tintG*255 << 8) + (fillColor & 0xFF) / 255 * tintB*255); - data._lineTint = (((lineColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((lineColor >> 8 & 0xFF) / 255 * tintG*255 << 8) + (lineColor & 0xFF) / 255 * tintB*255); - - } -}; - - -},{"../../../const":4}],23:[function(require,module,exports){ -var CanvasGraphics = require('./CanvasGraphics'); - -/** - * A set of functions used to handle masking. - * - * @class - * @namespace PIXI - */ -function CanvasMaskManager() {} - -CanvasMaskManager.prototype.constructor = CanvasMaskManager; -module.exports = CanvasMaskManager; - -/** - * This method adds it to the current stack of masks. - * - * @param maskData {object} the maskData that will be pushed - * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use. - */ -CanvasMaskManager.prototype.pushMask = function (maskData, renderer) { - renderer.context.save(); - - var cacheAlpha = maskData.alpha; - var transform = maskData.worldTransform; - var resolution = renderer.resolution; - - renderer.context.setTransform( - transform.a * resolution, - transform.b * resolution, - transform.c * resolution, - transform.d * resolution, - transform.tx * resolution, - transform.ty * resolution - ); - - CanvasGraphics.renderGraphicsMask(maskData, renderer.context); - - renderer.context.clip(); - - maskData.worldAlpha = cacheAlpha; -}; - -/** - * Restores the current drawing context to the state it was before the mask was applied. - * - * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use. - */ -CanvasMaskManager.prototype.popMask = function (renderer) { - renderer.context.restore(); -}; - -},{"./CanvasGraphics":22}],24:[function(require,module,exports){ -var utils = require('../../../utils'); - -/** - * Utility methods for Sprite/Texture tinting. - * - * @namespace PIXI - */ -var CanvasTinter = module.exports = {}; - -/** - * Basically this method just needs a sprite and a color and tints the sprite with the given color. - * - * @param sprite {Sprite} the sprite to tint - * @param color {number} the color to use to tint the sprite with - * @return {HTMLCanvasElement} The tinted canvas - */ -CanvasTinter.getTintedTexture = function (sprite, color) { - var texture = sprite.texture; - - color = CanvasTinter.roundColor(color); - - var stringColor = '#' + ('00000' + ( color | 0).toString(16)).substr(-6); - - texture.tintCache = texture.tintCache || {}; - - if (texture.tintCache[stringColor]) { - return texture.tintCache[stringColor]; - } - - // clone texture.. - var canvas = CanvasTinter.canvas || document.createElement('canvas'); - - //CanvasTinter.tintWithPerPixel(texture, stringColor, canvas); - CanvasTinter.tintMethod(texture, color, canvas); - - if (CanvasTinter.convertTintToImage) { - // is this better? - var tintImage = new Image(); - tintImage.src = canvas.toDataURL(); - - texture.tintCache[stringColor] = tintImage; - } - else { - texture.tintCache[stringColor] = canvas; - // if we are not converting the texture to an image then we need to lose the reference to the canvas - CanvasTinter.canvas = null; - } - - return canvas; -}; - -/** - * Tint a texture using the 'multiply' operation. - * - * @param texture {Texture} the texture to tint - * @param color {number} the color to use to tint the sprite with - * @param canvas {HTMLCanvasElement} the current canvas - */ -CanvasTinter.tintWithMultiply = function (texture, color, canvas) { - var context = canvas.getContext( '2d' ); - - var crop = texture.crop; - - canvas.width = crop.width; - canvas.height = crop.height; - - context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6); - - context.fillRect(0, 0, crop.width, crop.height); - - context.globalCompositeOperation = 'multiply'; - - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); - - context.globalCompositeOperation = 'destination-atop'; - - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); -}; - -/** - * Tint a texture using the 'overlay' operation. - * - * @param texture {Texture} the texture to tint - * @param color {number} the color to use to tint the sprite with - * @param canvas {HTMLCanvasElement} the current canvas - */ -CanvasTinter.tintWithOverlay = function (texture, color, canvas) { - var context = canvas.getContext( '2d' ); - - var crop = texture.crop; - - canvas.width = crop.width; - canvas.height = crop.height; - - context.globalCompositeOperation = 'copy'; - context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6); - context.fillRect(0, 0, crop.width, crop.height); - - context.globalCompositeOperation = 'destination-atop'; - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); - - // context.globalCompositeOperation = 'copy'; -}; - -/** - * Tint a texture pixel per pixel. - * - * @param texture {Texture} the texture to tint - * @param color {number} the color to use to tint the sprite with - * @param canvas {HTMLCanvasElement} the current canvas - */ -CanvasTinter.tintWithPerPixel = function (texture, color, canvas) { - var context = canvas.getContext( '2d' ); - - var crop = texture.crop; - - canvas.width = crop.width; - canvas.height = crop.height; - - context.globalCompositeOperation = 'copy'; - context.drawImage( - texture.baseTexture.source, - crop.x, - crop.y, - crop.width, - crop.height, - 0, - 0, - crop.width, - crop.height - ); - - var rgbValues = utils.hex2rgb(color); - var r = rgbValues[0], g = rgbValues[1], b = rgbValues[2]; - - var pixelData = context.getImageData(0, 0, crop.width, crop.height); - - var pixels = pixelData.data; - - for (var i = 0; i < pixels.length; i += 4) { - pixels[i+0] *= r; - pixels[i+1] *= g; - pixels[i+2] *= b; - } - - context.putImageData(pixelData, 0, 0); -}; - -/** - * Rounds the specified color according to the CanvasTinter.cacheStepsPerColorChannel. - * - * @param color {number} the color to round, should be a hex color - */ -CanvasTinter.roundColor = function (color) { - var step = CanvasTinter.cacheStepsPerColorChannel; - - var rgbValues = utils.hex2rgb(color); - - rgbValues[0] = Math.min(255, (rgbValues[0] / step) * step); - rgbValues[1] = Math.min(255, (rgbValues[1] / step) * step); - rgbValues[2] = Math.min(255, (rgbValues[2] / step) * step); - - return utils.rgb2hex(rgbValues); -}; - -/** - * Number of steps which will be used as a cap when rounding colors. - * - * @member - */ -CanvasTinter.cacheStepsPerColorChannel = 8; - -/** - * Tint cache boolean flag. - * - * @member - */ -CanvasTinter.convertTintToImage = false; - -/** - * Whether or not the Canvas BlendModes are supported, consequently the ability to tint using the multiply method. - * - * @member - */ -CanvasTinter.canUseMultiply = utils.canUseNewCanvasBlendModes(); - -/** - * The tinting method that will be used. - * - */ -CanvasTinter.tintMethod = CanvasTinter.canUseMultiply ? CanvasTinter.tintWithMultiply : CanvasTinter.tintWithPerPixel; - -},{"../../../utils":50}],25:[function(require,module,exports){ -var WebGLSpriteBatch = require('./utils/WebGLSpriteBatch'), - WebGLFastSpriteBatch = require('./utils/WebGLFastSpriteBatch'), - WebGLShaderManager = require('./managers/WebGLShaderManager'), - WebGLMaskManager = require('./managers/WebGLMaskManager'), - WebGLFilterManager = require('./managers/WebGLFilterManager'), - WebGLStencilManager = require('./managers/WebGLStencilManager'), - WebGLBlendModeManager = require('./managers/WebGLBlendModeManager'), - math = require('../../math'), - utils = require('../../utils'), - CONST = require('../../const'); - -/** - * The WebGLRenderer draws the scene and all its content onto a webGL enabled canvas. This renderer - * should be used for browsers that support webGL. This Render works by automatically managing webGLBatchs. - * So no need for Sprite Batches or Sprite Clouds. - * Don't forget to add the view to your DOM or you will not see anything :) - * - * @class - * @namespace PIXI - * @param [width=0] {number} the width of the canvas view - * @param [height=0] {number} the height of the canvas view - * @param [options] {object} The optional renderer parameters - * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional - * @param [options.transparent=false] {boolean} If the render view is transparent, default false - * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false - * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment) - * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you need to call toDataUrl on the webgl context - * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2 - */ -function WebGLRenderer(width, height, options) { - utils.sayHello('webGL'); - - if (options) { - for (var i in CONST.defaultRenderOptions) { - if (typeof options[i] === 'undefined') { - options[i] = CONST.defaultRenderOptions[i]; - } - } - } - else { - options = CONST.defaultRenderOptions; - } - - this.uuid = utils.uuid(); - - /** - * @member {number} - */ - this.type = CONST.WEBGL_RENDERER; - - /** - * The resolution of the renderer - * - * @member {number} - * @default 1 - */ - this.resolution = options.resolution; - - // do a catch.. only 1 webGL renderer.. - - /** - * Whether the render view is transparent - * - * @member {boolean} - */ - this.transparent = options.transparent; - - /** - * The background color as a number. - * - * @member {number} - * @private - */ - this._backgroundColor = 0x000000; - - /** - * The background color as an [R, G, B] array. - * - * @member {number[]} - * @private - */ - this._backgroundColorRgb = [0, 0, 0]; - - this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter - - /** - * Whether the render view should be resized automatically - * - * @member {boolean} - */ - this.autoResize = options.autoResize || false; - - /** - * The value of the preserveDrawingBuffer flag affects whether or not the contents of the stencil buffer is retained after rendering. - * - * @member {boolean} - */ - this.preserveDrawingBuffer = options.preserveDrawingBuffer; - - /** - * This sets if the WebGLRenderer will clear the context texture or not before the new render pass. If true: - * If the renderer is NOT transparent, Pixi will clear to alpha (0, 0, 0, 0). - * If the renderer is transparent, Pixi will clear to the target Stage's background color. - * Disable this by setting this to false. For example: if your game has a canvas filling background image, you often don't need this set. - * - * @member {boolean} - * @default - */ - this.clearBeforeRender = options.clearBeforeRender; - - /** - * The width of the canvas view - * - * @member {number} - * @default 800 - */ - this.width = width || 800; - - /** - * The height of the canvas view - * - * @member {number} - * @default 600 - */ - this.height = height || 600; - - /** - * The canvas element that everything is drawn to - * - * @member {HTMLCanvasElement} - */ - this.view = options.view || document.createElement( 'canvas' ); - - // deal with losing context.. - - /** - * @member {Function} - */ - this.contextLostBound = this.handleContextLost.bind(this); - - /** - * @member {Function} - */ - this.contextRestoredBound = this.handleContextRestored.bind(this); - - this.view.addEventListener('webglcontextlost', this.contextLostBound, false); - this.view.addEventListener('webglcontextrestored', this.contextRestoredBound, false); - - /** - * @member {object} - * @private - */ - this._contextOptions = { - alpha: this.transparent, - antialias: options.antialias, // SPEED UP?? - premultipliedAlpha:this.transparent && this.transparent !== 'notMultiplied', - stencil:true, - preserveDrawingBuffer: options.preserveDrawingBuffer - }; - - /** - * @member {Point} - */ - this.projection = new math.Point(); - - /** - * @member {Point} - */ - this.offset = new math.Point(0, 0); - - /** - * Counter for the number of draws made each frame - * - * @member {number} - */ - this.drawCount = 0; - - // time to create the render managers! each one focuses on managing a state in webGL - - /** - * Deals with managing the shader programs and their attribs - * @member {WebGLShaderManager} - */ - this.shaderManager = new WebGLShaderManager(this); - - /** - * Manages the rendering of sprites - * @member {WebGLSpriteBatch} - */ - this.spriteBatch = new WebGLSpriteBatch(this); - - /** - * Manages the rendering of sprites - * @member {WebGLFastSpriteBatch} - */ - this.fastSpriteBatch = new WebGLFastSpriteBatch(this); - - /** - * Manages the masks using the stencil buffer - * @member {WebGLMaskManager} - */ - this.maskManager = new WebGLMaskManager(this); - - /** - * Manages the filters - * @member {WebGLFilterManager} - */ - this.filterManager = new WebGLFilterManager(this); - - /** - * Manages the stencil buffer - * @member {WebGLStencilManager} - */ - this.stencilManager = new WebGLStencilManager(this); - - /** - * Manages the blendModes - * @member {WebGLBlendModeManager} - */ - this.blendModeManager = new WebGLBlendModeManager(this); - - this.blendModes = null; - - this._boundUpdateTexture = this.updateTexture.bind(this); - this._boundDestroyTexture = this.destroyTexture.bind(this); - - // time init the context.. - this._initContext(); - - // map some webGL blend modes.. - this._mapBlendModes(); -} - -// constructor -WebGLRenderer.prototype.constructor = WebGLRenderer; -module.exports = WebGLRenderer; - -utils.eventTarget.mixin(WebGLRenderer.prototype); - -Object.defineProperties(WebGLRenderer.prototype, { - /** - * The background color to fill if not transparent - * - * @member {number} - * @memberof WebGLRenderer# - */ - backgroundColor: { - get: function () { - return this._backgroundColor; - }, - set: function (val) { - this._backgroundColor = val; - utils.hex2rgb(val, this._backgroundColorRgb); - } - } -}); - -/** - * - * @private - */ -WebGLRenderer.prototype._initContext = function () { - var gl = this.view.getContext('webgl', this._contextOptions) || this.view.getContext('experimental-webgl', this._contextOptions); - this.gl = gl; - - if (!gl) { - // fail, not able to get a context - throw new Error('This browser does not support webGL. Try using the canvas renderer'); - } - - this.glContextId = WebGLRenderer.glContextId++; - gl.id = this.glContextId; - gl.renderer = this; - - // set up the default pixi settings.. - gl.disable(gl.DEPTH_TEST); - gl.disable(gl.CULL_FACE); - gl.enable(gl.BLEND); - - this.emit('context', gl); - - // now resize and we are good to go! - this.resize(this.width, this.height); -}; - -/** - * Renders the object to its webGL view - * - * @param object {DisplayObject} the object to be rendered - */ -WebGLRenderer.prototype.render = function (object) { - // no point rendering if our context has been blown up! - if (this.gl.isContextLost()) { - return; - } - - // update the scene graph - object.updateTransform(); - - var gl = this.gl; - - // -- Does this need to be set every frame? -- // - gl.viewport(0, 0, this.width, this.height); - - // make sure we are bound to the main frame buffer - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - - if (this.clearBeforeRender) { - if (this.transparent) { - gl.clearColor(0, 0, 0, 0); - } - else { - gl.clearColor(this._backgroundColorRgb[0], this._backgroundColorRgb[1], this._backgroundColorRgb[2], 1); - } - - gl.clear(gl.COLOR_BUFFER_BIT); - } - - this.renderDisplayObject(object, this.projection); -}; - -/** - * Renders a Display Object. - * - * @param displayObject {DisplayObject} The DisplayObject to render - * @param projection {Point} The projection - * @param buffer {Array} a standard WebGL buffer - */ -WebGLRenderer.prototype.renderDisplayObject = function (displayObject, projection, buffer) { - this.blendModeManager.setBlendMode(CONST.blendModes.NORMAL); - - // reset the render session data.. - this.drawCount = 0; - - // make sure to flip the Y if using a render texture.. - this.flipY = buffer ? -1 : 1; - - // set the default projection - this.projection = projection; - - //set the default offset - this.offset = this.offset; - - // start the sprite batch - this.spriteBatch.begin(); - - // start the filter manager - this.filterManager.begin(buffer); - - // render the scene! - displayObject.renderWebGL(this); - - // finish the sprite batch - this.spriteBatch.end(); -}; - -/** - * Resizes the webGL view to the specified width and height. - * - * @param width {number} the new width of the webGL view - * @param height {number} the new height of the webGL view - */ -WebGLRenderer.prototype.resize = function (width, height) { - this.width = width * this.resolution; - this.height = height * this.resolution; - - this.view.width = this.width; - this.view.height = this.height; - - if (this.autoResize) { - this.view.style.width = this.width / this.resolution + 'px'; - this.view.style.height = this.height / this.resolution + 'px'; - } - - this.gl.viewport(0, 0, this.width, this.height); - - this.projection.x = this.width / 2 / this.resolution; - this.projection.y = -this.height / 2 / this.resolution; -}; - -/** - * Updates and/or Creates a WebGL texture for the renderer's context. - * - * @param texture {BaseTexture|Texture} the texture to update - */ -WebGLRenderer.prototype.updateTexture = function (texture) { - texture = texture.baseTexture || texture; - - if (!texture.hasLoaded) { - return; - } - - var gl = this.gl; - - if (!texture._glTextures[gl.id]) { - texture._glTextures[gl.id] = gl.createTexture(); - texture.on('update', this._boundUpdateTexture); - texture.on('dispose', this._boundDestroyTexture); - } - - gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]); - - gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, texture.premultipliedAlpha); - gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, texture.source); - - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - - - if (texture.mipmap && utils.isPowerOfTwo(texture.width, texture.height)) { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR_MIPMAP_LINEAR : gl.NEAREST_MIPMAP_NEAREST); - gl.generateMipmap(gl.TEXTURE_2D); - } - else { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - } - - if (!texture._powerOf2) { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); - } - else { - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT); - } - - return texture._glTextures[gl.id]; -}; - -WebGLRenderer.prototype.destroyTexture = function (texture) { - texture = texture.baseTexture || texture; - - if (!texture.hasLoaded) { - return; - } - - if (texture._glTextures[this.gl.id]) { - this.gl.deleteTexture(texture._glTextures[this.gl.id]); - } -}; - -/** - * Handles a lost webgl context - * - * @param event {Event} - * @private - */ -WebGLRenderer.prototype.handleContextLost = function (event) { - event.preventDefault(); -}; - -/** - * Handles a restored webgl context - * - * @param event {Event} - * @private - */ -WebGLRenderer.prototype.handleContextRestored = function () { - this._initContext(); - - // empty all the ol gl textures as they are useless now - for (var key in utils.TextureCache) { - var texture = utils.TextureCache[key].baseTexture; - texture._glTextures = []; - } -}; - -/** - * Removes everything from the renderer (event listeners, spritebatch, etc...) - * - * @param [removeView=false] {boolean} Removes the Canvas element from the DOM. - */ -WebGLRenderer.prototype.destroy = function (removeView) { - if (removeView && this.view.parent) { - this.view.parent.removeChild(this.view); - } - - // remove listeners - this.view.removeEventListener('webglcontextlost', this.contextLostBound); - this.view.removeEventListener('webglcontextrestored', this.contextRestoredBound); - - // time to create the render managers! each one focuses on managine a state in webGL - this.shaderManager.destroy(); - this.spriteBatch.destroy(); - this.maskManager.destroy(); - this.filterManager.destroy(); - - - // this.uuid = utils.uuid(); - // this.type = CONST.WEBGL_RENDERER; - - // this.resolution = options.resolution; - // this.transparent = options.transparent; - - this._backgroundColor = 0x000000; - this._backgroundColorRgb = null; - - // this.backgroundColor = null; - // this.autoResize = options.autoResize || false; - // this.preserveDrawingBuffer = options.preserveDrawingBuffer; - // this.clearBeforeRender = options.clearBeforeRender; - // this.width = width || 800; - // this.height = height || 600; - - this.view = null; - - this.contextLostBound = null; - this.contextRestoredBound = null; - - this._contextOptions = null; - - this.projection = null; - this.offset = null; - this.drawCount = 0; - - this.shaderManager = null; - this.spriteBatch = null; - this.maskManager = null; - this.filterManager = null; - this.stencilManager = null; - this.blendModeManager = null; - - this.blendModes = null; - - this.gl = null; - this.blendModes = null; -}; - -/** - * Maps Pixi blend modes to WebGL blend modes. - * - * @private - */ -WebGLRenderer.prototype._mapBlendModes = function () { - var gl = this.gl; - - if (!this.blendModes) { - this.blendModes = {}; - - this.blendModes[CONST.blendModes.NORMAL] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.ADD] = [gl.SRC_ALPHA, gl.DST_ALPHA]; - this.blendModes[CONST.blendModes.MULTIPLY] = [gl.DST_COLOR, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.SCREEN] = [gl.SRC_ALPHA, gl.ONE]; - this.blendModes[CONST.blendModes.OVERLAY] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.DARKEN] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.LIGHTEN] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.COLOR_DODGE] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.COLOR_BURN] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.HARD_LIGHT] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.SOFT_LIGHT] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.DIFFERENCE] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.EXCLUSION] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.HUE] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.SATURATION] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.COLOR] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - this.blendModes[CONST.blendModes.LUMINOSITY] = [gl.ONE, gl.ONE_MINUS_SRC_ALPHA]; - } -}; - -WebGLRenderer.glContextId = 0; - -},{"../../const":4,"../../math":12,"../../utils":50,"./managers/WebGLBlendModeManager":26,"./managers/WebGLFilterManager":27,"./managers/WebGLMaskManager":29,"./managers/WebGLShaderManager":30,"./managers/WebGLStencilManager":31,"./utils/WebGLFastSpriteBatch":38,"./utils/WebGLSpriteBatch":41}],26:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLBlendModeManager(renderer) { - WebGLManager.call(this, renderer); - - /** - * @member {number} - */ - this.currentBlendMode = 99999; -} - -WebGLBlendModeManager.prototype = Object.create(WebGLManager.prototype); -WebGLBlendModeManager.prototype.constructor = WebGLBlendModeManager; -module.exports = WebGLBlendModeManager; - -/** - * Sets-up the given blendMode from WebGL's point of view. - * - * @param blendMode {number} the blendMode, should be a Pixi const, such as BlendModes.ADD - */ -WebGLBlendModeManager.prototype.setBlendMode = function (blendMode) { - if (this.currentBlendMode === blendMode) { - return false; - } - - this.currentBlendMode = blendMode; - - var mode = this.renderer.blendModes[this.currentBlendMode]; - this.renderer.gl.blendFunc(mode[0], mode[1]); - - return true; -}; - -},{"./WebGLManager":28}],27:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - FilterTexture = require('../utils/FilterTexture'), - Shader = require('../shaders/Shader'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLFilterManager(renderer) { - WebGLManager.call(this, renderer); - - /** - * @member {any[]} - */ - this.filterStack = []; - - /** - * @member {any[]]} - */ - this.texturePool = []; - - /** - * @member {number} - */ - this.offsetX = 0; - - /** - * @member {number} - */ - this.offsetY = 0; - - // listen for context and update necessary buffers - var self = this; - this.renderer.on('context', function () { - self.texturePool.length = 0; - self.initShaderBuffers(); - }); -} - -WebGLFilterManager.prototype = Object.create(WebGLManager.prototype); -WebGLFilterManager.prototype.constructor = WebGLFilterManager; -module.exports = WebGLFilterManager; - -/** - * @param renderer {WebGLRenderer} - * @param buffer {ArrayBuffer} - */ -WebGLFilterManager.prototype.begin = function (buffer) { - this.defaultShader = this.renderer.shaderManager.defaultShader; - - this.width = this.renderer.projection.x * 2; - this.height = -this.renderer.projection.y * 2; - - this.buffer = buffer; -}; - -/** - * Applies the filter and adds it to the current filter stack. - * - * @param filterBlock {object} the filter that will be pushed to the current filter stack - */ -WebGLFilterManager.prototype.pushFilter = function (filterBlock) { - var gl = this.renderer.gl; - - var projection = this.renderer.projection; - var offset = this.renderer.offset; - - filterBlock._filterArea = filterBlock.target.filterArea || filterBlock.target.getBounds(); - - // filter program - // OPTIMISATION - the first filter is free if its a simple color change? - this.filterStack.push(filterBlock); - - var filter = filterBlock.filterPasses[0]; - - this.offsetX += filterBlock._filterArea.x; - this.offsetY += filterBlock._filterArea.y; - - var texture = this.texturePool.pop(); - if (!texture) { - texture = new FilterTexture(this.renderer.gl, this.width, this.height); - } - else { - texture.resize(this.width, this.height); - } - - gl.bindTexture(gl.TEXTURE_2D, texture.texture); - - var filterArea = filterBlock._filterArea;// filterBlock.target.getBounds();///filterBlock.target.filterArea; - - var padding = filter.padding; - filterArea.x -= padding; - filterArea.y -= padding; - filterArea.width += padding * 2; - filterArea.height += padding * 2; - - var localX = filterArea.x, - localY = filterArea.y; - - if (filterArea.x < 0) { - filterArea.width += filterArea.x; - filterArea.x = 0; - } - - if (filterArea.y < 0) { - filterArea.height += filterArea.y; - filterArea.y = 0; - } - - if (localX + filterArea.width > this.width) { - filterArea.width = this.width - localX; - } - - if (localY + filterArea.height > this.height) { - filterArea.height = this.height - localY; - } - - if (filterArea.width < 0) { - filterArea.width = 0; - } - - if (filterArea.height < 0) { - filterArea.height = 0; - } - - //gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, filterArea.width, filterArea.height, 0, gl.RGBA, gl.UNSIGNED_BYTE, null); - gl.bindFramebuffer(gl.FRAMEBUFFER, texture.frameBuffer); - - // set view port - gl.viewport(0, 0, filterArea.width, filterArea.height); - - projection.x = filterArea.width/2; - projection.y = -filterArea.height/2; - - offset.x = -filterArea.x; - offset.y = -filterArea.y; - - // update projection - // now restore the regular shader.. - // this.renderer.shaderManager.setShader(this.defaultShader); - //gl.uniform2f(this.defaultShader.projectionVector, filterArea.width/2, -filterArea.height/2); - //gl.uniform2f(this.defaultShader.offsetVector, -filterArea.x, -filterArea.y); - - gl.colorMask(true, true, true, true); - gl.clearColor(0,0,0, 0); - gl.clear(gl.COLOR_BUFFER_BIT); - - filterBlock._glFilterTexture = texture; - -}; - -/** - * Removes the last filter from the filter stack and doesn't return it. - * - */ -WebGLFilterManager.prototype.popFilter = function () { - var gl = this.renderer.gl; - - var filterBlock = this.filterStack.pop(); - var filterArea = filterBlock._filterArea; - var texture = filterBlock._glFilterTexture; - var projection = this.renderer.projection; - var offset = this.renderer.offset; - - if (filterBlock.filterPasses.length > 1) { - gl.viewport(0, 0, filterArea.width, filterArea.height); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - - this.vertexArray[0] = 0; - this.vertexArray[1] = filterArea.height; - - this.vertexArray[2] = filterArea.width; - this.vertexArray[3] = filterArea.height; - - this.vertexArray[4] = 0; - this.vertexArray[5] = 0; - - this.vertexArray[6] = filterArea.width; - this.vertexArray[7] = 0; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - // now set the uvs.. - this.uvArray[2] = filterArea.width/this.width; - this.uvArray[5] = filterArea.height/this.height; - this.uvArray[6] = filterArea.width/this.width; - this.uvArray[7] = filterArea.height/this.height; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray); - - var inputTexture = texture; - var outputTexture = this.texturePool.pop(); - if (!outputTexture) { - outputTexture = new FilterTexture(this.renderer.gl, this.width, this.height); - } - outputTexture.resize(this.width, this.height); - - // need to clear this FBO as it may have some left over elements from a previous filter. - gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer ); - gl.clear(gl.COLOR_BUFFER_BIT); - - gl.disable(gl.BLEND); - - for (var i = 0; i < filterBlock.filterPasses.length-1; i++) { - var filterPass = filterBlock.filterPasses[i]; - - gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer ); - - // set texture - gl.activeTexture(gl.TEXTURE0); - gl.bindTexture(gl.TEXTURE_2D, inputTexture.texture); - - // draw texture.. - //filterPass.applyFilterPass(filterArea.width, filterArea.height); - this.applyFilterPass(filterPass, filterArea, filterArea.width, filterArea.height); - - // swap the textures.. - var temp = inputTexture; - inputTexture = outputTexture; - outputTexture = temp; - } - - gl.enable(gl.BLEND); - - texture = inputTexture; - this.texturePool.push(outputTexture); - } - - var filter = filterBlock.filterPasses[filterBlock.filterPasses.length-1]; - - this.offsetX -= filterArea.x; - this.offsetY -= filterArea.y; - - var sizeX = this.width; - var sizeY = this.height; - - var offsetX = 0; - var offsetY = 0; - - var buffer = this.buffer; - - // time to render the filters texture to the previous scene - if (this.filterStack.length === 0) { - gl.colorMask(true, true, true, true);//this.transparent); - } - else { - var currentFilter = this.filterStack[this.filterStack.length-1]; - filterArea = currentFilter._filterArea; - - sizeX = filterArea.width; - sizeY = filterArea.height; - - offsetX = filterArea.x; - offsetY = filterArea.y; - - buffer = currentFilter._glFilterTexture.frameBuffer; - } - - // TODO need to remove these global elements.. - projection.x = sizeX/2; - projection.y = -sizeY/2; - - offset.x = offsetX; - offset.y = offsetY; - - filterArea = filterBlock._filterArea; - - var x = filterArea.x-offsetX; - var y = filterArea.y-offsetY; - - // update the buffers.. - // make sure to flip the y! - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - - this.vertexArray[0] = x; - this.vertexArray[1] = y + filterArea.height; - - this.vertexArray[2] = x + filterArea.width; - this.vertexArray[3] = y + filterArea.height; - - this.vertexArray[4] = x; - this.vertexArray[5] = y; - - this.vertexArray[6] = x + filterArea.width; - this.vertexArray[7] = y; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - - this.uvArray[2] = filterArea.width/this.width; - this.uvArray[5] = filterArea.height/this.height; - this.uvArray[6] = filterArea.width/this.width; - this.uvArray[7] = filterArea.height/this.height; - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray); - - gl.viewport(0, 0, sizeX, sizeY); - - // bind the buffer - gl.bindFramebuffer(gl.FRAMEBUFFER, buffer ); - - // set the blend mode! - //gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA) - - // set texture - gl.activeTexture(gl.TEXTURE0); - gl.bindTexture(gl.TEXTURE_2D, texture.texture); - - // apply! - this.applyFilterPass(filter, filterArea, sizeX, sizeY); - - // now restore the regular shader.. should happen automatically now.. - // this.renderer.shaderManager.setShader(this.defaultShader); - // gl.uniform2f(this.defaultShader.projectionVector, sizeX/2, -sizeY/2); - // gl.uniform2f(this.defaultShader.offsetVector, -offsetX, -offsetY); - - // return the texture to the pool - this.texturePool.push(texture); - filterBlock._glFilterTexture = null; -}; - - -/** - * Applies the filter to the specified area. - * - * @param filter {AbstractFilter} the filter that needs to be applied - * @param filterArea {Texture} TODO - might need an update - * @param width {number} the horizontal range of the filter - * @param height {number} the vertical range of the filter - */ -WebGLFilterManager.prototype.applyFilterPass = function (filter, filterArea, width, height) { - // use program - var gl = this.renderer.gl; - - var shader = filter.shaders[gl.id]; - - if (!shader) { - shader = new Shader(gl); - - shader.fragmentSrc = filter.fragmentSrc; - shader.uniforms = filter.uniforms; - shader.init(); - - filter.shaders[gl.id] = shader; - } - - // set the shader - this.renderer.shaderManager.setShader(shader); - -// gl.useProgram(shader.program); - - gl.uniform2f(shader.projectionVector, width/2, -height/2); - gl.uniform2f(shader.offsetVector, 0,0); - - if (filter.uniforms.dimensions) { - filter.uniforms.dimensions.value[0] = this.width;//width; - filter.uniforms.dimensions.value[1] = this.height;//height; - filter.uniforms.dimensions.value[2] = this.vertexArray[0]; - filter.uniforms.dimensions.value[3] = this.vertexArray[5];//filterArea.height; - } - - shader.syncUniforms(); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer); - gl.vertexAttribPointer(shader.aColor, 2, gl.FLOAT, false, 0, 0); - - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - - // draw the filter... - gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0 ); - - this.renderer.drawCount++; -}; - -/** - * Initialises the shader buffers. - * - */ -WebGLFilterManager.prototype.initShaderBuffers = function () { - var gl = this.renderer.gl; - - // create some buffers - this.vertexBuffer = gl.createBuffer(); - this.uvBuffer = gl.createBuffer(); - this.colorBuffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - - // bind and upload the vertexs.. - // keep a reference to the vertexFloatData.. - this.vertexArray = new Float32Array([0.0, 0.0, - 1.0, 0.0, - 0.0, 1.0, - 1.0, 1.0]); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertexArray, gl.STATIC_DRAW); - - // bind and upload the uv buffer - this.uvArray = new Float32Array([0.0, 0.0, - 1.0, 0.0, - 0.0, 1.0, - 1.0, 1.0]); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.uvArray, gl.STATIC_DRAW); - - this.colorArray = new Float32Array([1.0, 0xFFFFFF, - 1.0, 0xFFFFFF, - 1.0, 0xFFFFFF, - 1.0, 0xFFFFFF]); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.colorArray, gl.STATIC_DRAW); - - // bind and upload the index - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array([0, 1, 2, 1, 3, 2]), gl.STATIC_DRAW); - -}; - -/** - * Destroys the filter and removes it from the filter stack. - * - */ -WebGLFilterManager.prototype.destroy = function () { - var gl = this.renderer.gl; - - this.filterStack = null; - - this.offsetX = 0; - this.offsetY = 0; - - // destroy textures - for (var i = 0; i < this.texturePool.length; i++) { - this.texturePool[i].destroy(); - } - - this.texturePool = null; - - //destroy buffers.. - gl.deleteBuffer(this.vertexBuffer); - gl.deleteBuffer(this.uvBuffer); - gl.deleteBuffer(this.colorBuffer); - gl.deleteBuffer(this.indexBuffer); - - this.renderer = null; -}; - -},{"../shaders/Shader":35,"../utils/FilterTexture":37,"./WebGLManager":28}],28:[function(require,module,exports){ -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLManager(renderer) { - /** - * The renderer this manager works for. - * - * @member {WebGLRenderer} - */ - this.renderer = renderer; -} - -WebGLManager.prototype.constructor = WebGLManager; -module.exports = WebGLManager; - -WebGLManager.prototype.destroy = function () { - this.renderer = null; -}; - -},{}],29:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - WebGLGraphics = require('../utils/WebGLGraphics'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLMaskManager(renderer) { - WebGLManager.call(this, renderer); -} - -WebGLMaskManager.prototype = Object.create(WebGLManager.prototype); -WebGLMaskManager.prototype.constructor = WebGLMaskManager; -module.exports = WebGLMaskManager; - -/** - * Applies the Mask and adds it to the current filter stack. - * - * @param maskData {any[]} - */ -WebGLMaskManager.prototype.pushMask = function (maskData) { - if (maskData.dirty) { - WebGLGraphics.updateGraphics(maskData, this.renderer.gl); - } - - if (!maskData._webGL[this.renderer.gl.id].data.length) { - return; - } - - this.renderer.stencilManager.pushStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer); -}; - -/** - * Removes the last filter from the filter stack and doesn't return it. - * - * @param maskData {any[]} - */ -WebGLMaskManager.prototype.popMask = function (maskData) { - this.renderer.stencilManager.popStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer); -}; - -},{"../utils/WebGLGraphics":39,"./WebGLManager":28}],30:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - PrimitiveShader = require('../shaders/PrimitiveShader'), - ComplexPrimitiveShader = require('../shaders/ComplexPrimitiveShader'), - Shader = require('../shaders/Shader'), - FastShader = require('../shaders/FastShader'), - StripShader = require('../shaders/StripShader'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLShaderManager(renderer) { - WebGLManager.call(this, renderer); - - /** - * @member {number} - */ - this.maxAttibs = 10; - - /** - * @member {any[]} - */ - this.attribState = []; - - /** - * @member {any[]} - */ - this.tempAttribState = []; - - for (var i = 0; i < this.maxAttibs; i++) { - this.attribState[i] = false; - } - - /** - * @member {any[]} - */ - this.stack = []; - - /** - * @member {number} - * @private - */ - this._currentId = -1; - - /** - * @member {Shader} - * @private - */ - this.currentShader = null; - - // this shader is used for rendering primitives - this.primitiveShader = null; - - // this shader is used for rendering triangle strips - this.complexPrimitiveShader = null; - - // this shader is used for the default sprite rendering - this.defaultShader = null; - - // this shader is used for the fast sprite rendering - this.fastShader = null; - - // the next one is used for rendering triangle strips - this.stripShader = null; - - // listen for context and update necessary shaders - var self = this; - this.renderer.on('context', function (event) { - var gl = event.data; - - // this shader is used for rendering primitives - self.primitiveShader = new PrimitiveShader(gl); - - // this shader is used for rendering triangle strips - self.complexPrimitiveShader = new ComplexPrimitiveShader(gl); - - // this shader is used for the default sprite rendering - self.defaultShader = new Shader(gl); - - // this shader is used for the fast sprite rendering - self.fastShader = new FastShader(gl); - - // the next one is used for rendering triangle strips - self.stripShader = new StripShader(gl); - - self.setShader(self.defaultShader); - }); -} - -WebGLShaderManager.prototype = Object.create(WebGLManager.prototype); -WebGLShaderManager.prototype.constructor = WebGLShaderManager; -module.exports = WebGLShaderManager; - -/** - * Takes the attributes given in parameters. - * - * @param attribs {Array} attribs - */ -WebGLShaderManager.prototype.setAttribs = function (attribs) { - // reset temp state - var i; - - for (i = 0; i < this.tempAttribState.length; i++) { - this.tempAttribState[i] = false; - } - - // set the new attribs - for (var a in attribs) { - this.tempAttribState[attribs[a]] = true; - } - - var gl = this.renderer.gl; - - for (i = 0; i < this.attribState.length; i++) { - if (this.attribState[i] !== this.tempAttribState[i]) { - this.attribState[i] = this.tempAttribState[i]; - - if (this.attribState[i]) { - gl.enableVertexAttribArray(i); - } - else { - gl.disableVertexAttribArray(i); - } - } - } -}; - -/** - * Sets the current shader. - * - * @param shader {Any} - */ -WebGLShaderManager.prototype.setShader = function (shader) { - if (this._currentId === shader.uuid) { - return false; - } - - this._currentId = shader.uuid; - - this.currentShader = shader; - - this.renderer.gl.useProgram(shader.program); - this.setAttribs(shader.attributes); - - return true; -}; - -/** - * Destroys this object. - * - */ -WebGLShaderManager.prototype.destroy = function () { - this.attribState = null; - - this.tempAttribState = null; - - this.primitiveShader.destroy(); - this.primitiveShader = null; - - this.complexPrimitiveShader.destroy(); - this.complexPrimitiveShader = null; - - this.defaultShader.destroy(); - this.defaultShader = null; - - this.fastShader.destroy(); - this.fastShader = null; - - this.stripShader.destroy(); - this.stripShader = null; - - this.renderer = null; -}; - -},{"../shaders/ComplexPrimitiveShader":32,"../shaders/FastShader":33,"../shaders/PrimitiveShader":34,"../shaders/Shader":35,"../shaders/StripShader":36,"./WebGLManager":28}],31:[function(require,module,exports){ -var WebGLManager = require('./WebGLManager'), - utils = require('../../../utils'); - -/** - * @class - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this manager works for. - */ -function WebGLStencilManager(renderer) { - WebGLManager.call(this, renderer); - - this.stencilStack = []; - this.reverse = true; - this.count = 0; -} - -WebGLStencilManager.prototype = Object.create(WebGLManager.prototype); -WebGLStencilManager.prototype.constructor = WebGLStencilManager; -module.exports = WebGLStencilManager; - -/** - * Applies the Mask and adds it to the current filter stack. - * - * @param graphics {Graphics} - * @param webGLData {any[]} - */ -WebGLStencilManager.prototype.pushStencil = function (graphics, webGLData) { - var gl = this.renderer.gl; - - this.bindGraphics(graphics, webGLData, this.renderer); - - if (this.stencilStack.length === 0) { - gl.enable(gl.STENCIL_TEST); - gl.clear(gl.STENCIL_BUFFER_BIT); - this.reverse = true; - this.count = 0; - } - - this.stencilStack.push(webGLData); - - var level = this.count; - - gl.colorMask(false, false, false, false); - - gl.stencilFunc(gl.ALWAYS,0,0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT); - - // draw the triangle strip! - - if (webGLData.mode === 1) { - gl.drawElements(gl.TRIANGLE_FAN, webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 ); - - if (this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - - // draw a quad to increment.. - gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 ); - - if (this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - } - - this.reverse = !this.reverse; - } - else { - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - - gl.drawElements(gl.TRIANGLE_STRIP, webGLData.indices.length, gl.UNSIGNED_SHORT, 0 ); - - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - } - } - - gl.colorMask(true, true, true, true); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP); - - this.count++; -}; - -/** - * TODO this does not belong here! - * - * @param graphics {Graphics} - * @param webGLData {Array} - */ -WebGLStencilManager.prototype.bindGraphics = function (graphics, webGLData) { - //if (this._currentGraphics === graphics)return; - this._currentGraphics = graphics; - - var gl = this.renderer.gl; - - // bind the graphics object.. - var projection = this.renderer.projection, - offset = this.renderer.offset, - shader;// = this.renderer.shaderManager.primitiveShader; - - if (webGLData.mode === 1) { - shader = this.renderer.shaderManager.complexPrimitiveShader; - - this.renderer.shaderManager.setShader(shader); - - gl.uniform1f(shader.flipY, this.renderer.flipY); - - gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true)); - - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - - gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint)); - gl.uniform3fv(shader.color, webGLData.color); - - gl.uniform1f(shader.alpha, graphics.worldAlpha * webGLData.alpha); - - gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer); - - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 2, 0); - - - // now do the rest.. - // set the index buffer! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer); - } - else { - //this.renderer.shaderManager.activatePrimitiveShader(); - shader = this.renderer.shaderManager.primitiveShader; - this.renderer.shaderManager.setShader( shader ); - - gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true)); - - gl.uniform1f(shader.flipY, this.renderer.flipY); - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - - gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint)); - - gl.uniform1f(shader.alpha, graphics.worldAlpha); - - gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer); - - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0); - gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4); - - // set the index buffer! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer); - } -}; - -/** - * @param graphics {Graphics} - * @param webGLData {Array} - */ -WebGLStencilManager.prototype.popStencil = function (graphics, webGLData) { - var gl = this.renderer.gl; - - this.stencilStack.pop(); - - this.count--; - - if (this.stencilStack.length === 0) { - // the stack is empty! - gl.disable(gl.STENCIL_TEST); - - } - else { - - var level = this.count; - - this.bindGraphics(graphics, webGLData, this.renderer); - - gl.colorMask(false, false, false, false); - - if (webGLData.mode === 1) { - this.reverse = !this.reverse; - - if (this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - - // draw a quad to increment.. - gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 ); - - gl.stencilFunc(gl.ALWAYS,0,0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT); - - // draw the triangle strip! - gl.drawElements(gl.TRIANGLE_FAN, webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 ); - - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - } - - } - else { - // console.log("<<>>") - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR); - } - else { - gl.stencilFunc(gl.EQUAL,level+1, 0xFF); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR); - } - - gl.drawElements(gl.TRIANGLE_STRIP, webGLData.indices.length, gl.UNSIGNED_SHORT, 0 ); - - if (!this.reverse) { - gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF); - } - else { - gl.stencilFunc(gl.EQUAL,level, 0xFF); - } - } - - gl.colorMask(true, true, true, true); - gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP); - - - } -}; - -/** - * Destroys the mask stack. - * - */ -WebGLStencilManager.prototype.destroy = function () { - this.renderer = null; - this.stencilStack = null; -}; - -},{"../../../utils":50,"./WebGLManager":28}],32:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function ComplexPrimitiveShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - // 'attribute vec2 aTextureCoord;', - // 'attribute vec4 aColor;', - - 'uniform mat3 translationMatrix;', - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - - 'uniform vec3 tint;', - 'uniform float alpha;', - 'uniform vec3 color;', - 'uniform float flipY;', - 'varying vec4 vColor;', - - 'void main(void) {', - ' vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);', - ' v -= offsetVector.xyx;', - ' gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);', - ' vColor = vec4(color * alpha * tint, alpha);',//" * vec4(tint * alpha, alpha);', - '}' - ].join('\n'), - // fragment shader - [ - 'precision mediump float;', - - 'varying vec4 vColor;', - - 'void main(void) {', - ' gl_FragColor = vColor;', - '}' - ].join('\n'), - // custom uniforms - { - tint: { type: '3f', value: [0, 0, 0] }, - flipY: { type: '1f', value: 0 }, - alpha: { type: '1f', value: 0 }, - translationMatrix: { type: 'mat3', value: new Float32Array(9) } - } - ); -} - -ComplexPrimitiveShader.prototype = Object.create(Shader.prototype); -ComplexPrimitiveShader.prototype.constructor = ComplexPrimitiveShader; -module.exports = ComplexPrimitiveShader; - -},{"./Shader":35}],33:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @extends Shader - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function FastShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - 'attribute vec2 aTextureCoord;', - 'attribute vec4 aColor;', - - 'attribute vec2 aPositionCoord;', - 'attribute vec2 aScale;', - 'attribute float aRotation;', - - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - 'uniform mat3 uMatrix;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'const vec2 center = vec2(-1.0, 1.0);', - - 'void main(void) {', - ' vec2 v;', - ' vec2 sv = aVertexPosition * aScale;', - ' v.x = (sv.x) * cos(aRotation) - (sv.y) * sin(aRotation);', - ' v.y = (sv.x) * sin(aRotation) + (sv.y) * cos(aRotation);', - ' v = ( uMatrix * vec3(v + aPositionCoord , 1.0) ).xy ;', - ' gl_Position = vec4( ( v / projectionVector) + center , 0.0, 1.0);', - ' vTextureCoord = aTextureCoord;', - // ' vec3 color = mod(vec3(aColor.y/65536.0, aColor.y/256.0, aColor.y), 256.0) / 256.0;', - ' vColor = aColor;', - '}' - ].join('\n'), - // fragment shader, use default - null, - // custom uniforms - { - uMatrix: { type: 'mat3', value: new Float32Array(9) } - }, - // custom attributes - { - aPositionCoord: 0, - aRotation: 0, - aScale: 0 - } - ); -} - -FastShader.prototype = Object.create(Shader.prototype); -FastShader.prototype.constructor = FastShader; -module.exports = FastShader; - -},{"./Shader":35}],34:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function PrimitiveShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - // 'attribute vec2 aTextureCoord;', - 'attribute vec4 aColor;', - - 'uniform mat3 translationMatrix;', - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - 'uniform float alpha;', - 'uniform float flipY;', - 'uniform vec3 tint;', - - 'varying vec4 vColor;', - - 'void main(void) {', - ' vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);', - ' v -= offsetVector.xyx;', - ' gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);', - ' vColor = aColor * vec4(tint * alpha, alpha);', - '}' - ].join('\n'), - // fragment shader - [ - 'precision mediump float;', - - 'varying vec4 vColor;', - - 'void main(void) {', - ' gl_FragColor = vColor;', - '}' - ].join('\n'), - // custom uniforms - { - tint: { type: '3f', value: [0, 0, 0] }, - flipY: { type: '1f', value: 0 }, - alpha: { type: '1f', value: 0 }, - translationMatrix: { type: 'mat3', value: new Float32Array(9) } - } - ); -} - -PrimitiveShader.prototype = Object.create(Shader.prototype); -PrimitiveShader.prototype.constructor = PrimitiveShader; -module.exports = PrimitiveShader; - -},{"./Shader":35}],35:[function(require,module,exports){ -var utils = require('../../../utils'); - -/** - * @class - * @namespace PIXI - * @param [fragmentSrc] {string} The source of the fragment shader. - * @param [vertexSrc] {string} The source of the vertex shader. - */ -function Shader(gl, vertexSrc, fragmentSrc, customUniforms, customAttributes) { - /** - * @member {number} - * @readonly - */ - this.uuid = utils.uuid(); - - /** - * @member {WebGLContext} - * @readonly - */ - this.gl = gl; - - /** - * The WebGL program. - * @member {WebGLProgram} - * @readonly - */ - this.program = null; - - this.uniforms = { - uSampler: { type: 'sampler2D', value: 0 }, - projectionVector: { type: '2f', value: { x: 0, y: 0 } }, - offsetVector: { type: '2f', value: { x: 0, y: 0 } }, - dimensions: { type: '4f', value: new Float32Array(4) } - }; - - for (var u in customUniforms) { - this.uniforms[u] = customUniforms[u]; - } - - this.attributes = { - aVertexPosition: 0, - aTextureCoord: 0, - aColor: 0 - }; - - for (var a in customAttributes) { - this.attributes[a] = customAttributes[a]; - } - - this.textureCount = 0; - - /** - * The vertex shader. - * @member {Array} - */ - this.vertexSrc = vertexSrc || [ - 'attribute vec2 aVertexPosition;', - 'attribute vec2 aTextureCoord;', - 'attribute vec4 aColor;', - - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'const vec2 center = vec2(-1.0, 1.0);', - - 'void main(void) {', - ' gl_Position = vec4( ((aVertexPosition + offsetVector) / projectionVector) + center , 0.0, 1.0);', - ' vTextureCoord = aTextureCoord;', - ' vColor = vec4(aColor.rgb * aColor.a, aColor.a);', - '}' - ].join('\n'); - - /** - * The fragment shader. - * @member {Array} - */ - this.fragmentSrc = fragmentSrc || [ - 'precision lowp float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord) * vColor ;', - '}' - ].join('\n'); - - this.init(); -} - -Shader.prototype.constructor = Shader; -module.exports = Shader; - -Shader.prototype.init = function () { - this.compile(); - - this.gl.useProgram(this.program); - - this.cacheUniformLocations(Object.keys(this.uniforms)); - this.cacheAttributeLocations(Object.keys(this.attributes)); -}; - -Shader.prototype.cacheUniformLocations = function (keys) { - for (var i = 0; i < keys.length; ++i) { - this.uniforms[keys[i]]._location = this.gl.getUniformLocation(this.program, keys[i]); - } -}; - -Shader.prototype.cacheAttributeLocations = function (keys) { - for (var i = 0; i < keys.length; ++i) { - this.attributes[keys[i]] = this.gl.getAttribLocation(this.program, keys[i]); - } - - // TODO: Check if this is needed anymore... - - // Begin worst hack eva // - - // WHY??? ONLY on my chrome pixel the line above returns -1 when using filters? - // maybe its something to do with the current state of the gl context. - // I'm convinced this is a bug in the chrome browser as there is NO reason why this should be returning -1 especially as it only manifests on my chrome pixel - // If theres any webGL people that know why could happen please help :) - // if (this.attributes.aColor === -1) { - // this.attributes.aColor = 2; - // } - - // End worst hack eva // -}; - -Shader.prototype.compile = function () { - var gl = this.gl; - - var glVertShader = this._glCompile(gl.VERTEX_SHADER, this.vertexSrc); - var glFragShader = this._glCompile(gl.FRAGMENT_SHADER, this.fragmentSrc); - - var program = gl.createProgram(); - - gl.attachShader(program, glVertShader); - gl.attachShader(program, glFragShader); - gl.linkProgram(program); - - // if linking fails, then log and cleanup - if (!gl.getProgramParameter(program, gl.LINK_STATUS)) { - window.console.error('Pixi.js Error: Could not initialize shader.'); - window.console.error('gl.VALIDATE_STATUS', gl.getProgramParameter(program, gl.VALIDATE_STATUS)); - window.console.error('gl.getError()', gl.getError()); - - // if there is a program info log, log it - if (gl.getProgramInfoLog(program) !== '') { - window.console.warn('Pixi.js Warning: gl.getProgramInfoLog()', gl.getProgramInfoLog(program)); - } - - gl.deleteProgram(program); - program = null; - } - - // clean up some shaders - gl.deleteShader(glVertShader); - gl.deleteShader(glFragShader); - - return (this.program = program); -}; - -Shader.prototype.syncUniforms = function () { - var gl = this.gl; - - this.textureCount = 1; - - for (var key in this.uniforms) { - var uniform = this.uniforms[key], - location = uniform._location, - value = uniform.value, - i, il; - - switch (uniform.type) { - case 'i': - case '1i': - gl.uniform1i(location, value); - break; - - case 'f': - case '1f': - gl.uniform1f(location, value); - break; - - case '2f': - gl.uniform2f(location, value[0], value[1]); - break; - - case '3f': - gl.uniform3f(location, value[0], value[1], value[2]); - break; - - case '4f': - gl.uniform4f(location, value[0], value[1], value[2], value[3]); - break; - - // a 2D Point object - case 'v2': - gl.uniform2f(location, value.x, value.y); - break; - - // a 3D Point object - case 'v3': - gl.uniform3f(location, value.x, value.y, value.z); - break; - - // a 4D Point object - case 'v4': - gl.uniform4f(location, value.x, value.y, value.z, value.w); - break; - - case '1iv': - gl.uniform1iv(location, value); - break; - - case '3iv': - gl.uniform3iv(location, value); - break; - - case '1fv': - gl.uniform1fv(location, value); - break; - - case '2fv': - gl.uniform2fv(location, value); - break; - - case '3fv': - gl.uniform3fv(location, value); - break; - - case '4fv': - gl.uniform4fv(location, value); - break; - - case 'm2': - case 'mat2': - case 'Matrix2fv': - gl.uniformMatrix2fv(location, uniform.transpose, value); - break; - - case 'm3': - case 'mat3': - case 'Matrix3fv': - gl.uniformMatrix3fv(location, uniform.transpose, value); - break; - - case 'm4': - case 'mat4': - case 'Matrix4fv': - gl.uniformMatrix4fv(location, uniform.transpose, value); - break; - - // a Color Value - case 'c': - if (typeof value === 'number') { - value = utils.hex2rgb(value); - } - - gl.uniform3f(location, value[0], value[1], value[2]); - break; - - // flat array of integers (JS or typed array) - case 'iv1': - gl.uniform1iv(location, value); - break; - - // flat array of integers with 3 x N size (JS or typed array) - case 'iv': - gl.uniform3iv(location, value); - break; - - // flat array of floats (JS or typed array) - case 'fv1': - gl.uniform1fv(location, value); - break; - - // flat array of floats with 3 x N size (JS or typed array) - case 'fv': - gl.uniform3fv(location, value); - break; - - // array of 2D Point objects - case 'v2v': - if (!uniform._array) { - uniform._array = new Float32Array(2 * value.length); - } - - for (i = 0, il = value.length; i < il; ++i) { - uniform._array[i * 2] = value[i].x; - uniform._array[i * 2 + 1] = value[i].y; - } - - gl.uniform2fv(location, uniform._array); - break; - - // array of 3D Point objects - case 'v3v': - if (!uniform._array) { - uniform._array = new Float32Array(3 * value.length); - } - - for (i = 0, il = value.length; i < il; ++i) { - uniform._array[i * 3] = value[i].x; - uniform._array[i * 3 + 1] = value[i].y; - uniform._array[i * 3 + 2] = value[i].z; - - } - - gl.uniform3fv(location, uniform._array); - break; - - // array of 4D Point objects - case 'v4v': - if (!uniform._array) { - uniform._array = new Float32Array(4 * value.length); - } - - for (i = 0, il = value.length; i < il; ++i) { - uniform._array[i * 4] = value[i].x; - uniform._array[i * 4 + 1] = value[i].y; - uniform._array[i * 4 + 2] = value[i].z; - uniform._array[i * 4 + 3] = value[i].w; - - } - - gl.uniform4fv(location, uniform._array); - break; - - case 't': - case 'sampler2D': - if (!uniform.value || !uniform.value.baseTexture || !uniform.value.baseTexture.hasLoaded) { - break; - } - - // activate this texture - gl.activeTexture(gl['TEXTURE' + this.textureCount]); - - // bind the texture - gl.bindTexture(gl.TEXTURE_2D, uniform.value.baseTexture._glTextures[gl.id]); - - // set uniform to texture index - gl.uniform1i(uniform._location, this.textureCount); - - // increment next texture id - this.textureCount++; - - // initialize the texture if we haven't yet - if (!uniform._init) { - this.initSampler2D(uniform); - - uniform._init = true; - } - break; - - default: - window.console.warn('Pixi.js Shader Warning: Unknown uniform type: ' + uniform.type); - } - } -}; - - -/** - * Initialises a Sampler2D uniform (which may only be available later on after initUniforms once the texture has loaded) - * - */ -Shader.prototype.initSampler2D = function (uniform) { - var gl = this.gl; - - // Extended texture data - if (uniform.textureData) { - var data = uniform.textureData; - - // GLTexture = mag linear, min linear_mipmap_linear, wrap repeat + gl.generateMipmap(gl.TEXTURE_2D); - // GLTextureLinear = mag/min linear, wrap clamp - // GLTextureNearestRepeat = mag/min NEAREST, wrap repeat - // GLTextureNearest = mag/min nearest, wrap clamp - // AudioTexture = whatever + luminance + width 512, height 2, border 0 - // KeyTexture = whatever + luminance + width 256, height 2, border 0 - - // magFilter can be: gl.LINEAR, gl.LINEAR_MIPMAP_LINEAR or gl.NEAREST - // wrapS/T can be: gl.CLAMP_TO_EDGE or gl.REPEAT - - var magFilter = (data.magFilter) ? data.magFilter : gl.LINEAR; - var minFilter = (data.minFilter) ? data.minFilter : gl.LINEAR; - var wrapS = (data.wrapS) ? data.wrapS : gl.CLAMP_TO_EDGE; - var wrapT = (data.wrapT) ? data.wrapT : gl.CLAMP_TO_EDGE; - var format = (data.luminance) ? gl.LUMINANCE : gl.RGBA; - - if (data.repeat) { - wrapS = gl.REPEAT; - wrapT = gl.REPEAT; - } - - gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, !!data.flipY); - - if (data.width) { - var width = (data.width) ? data.width : 512; - var height = (data.height) ? data.height : 2; - var border = (data.border) ? data.border : 0; - - // void texImage2D(GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, ArrayBufferView? pixels); - gl.texImage2D(gl.TEXTURE_2D, 0, format, width, height, border, format, gl.UNSIGNED_BYTE, null); - } - else { - // void texImage2D(GLenum target, GLint level, GLenum internalformat, GLenum format, GLenum type, ImageData? pixels); - gl.texImage2D(gl.TEXTURE_2D, 0, format, gl.RGBA, gl.UNSIGNED_BYTE, uniform.value.baseTexture.source); - } - - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, magFilter); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrapS); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrapT); - } -}; - -/** - * Destroys the shader. - * - */ -Shader.prototype.destroy = function () { - this.gl.deleteProgram(this.program); - - this.gl = null; - this.uniforms = null; - this.attributes = null; - - this.vertexSrc = null; - this.fragmentSrc = null; -}; - -Shader.prototype._glCompile = function (type, src) { - var shader = this.gl.createShader(type); - - if (Array.isArray(src)) { - debugger; - } - - this.gl.shaderSource(shader, src); - this.gl.compileShader(shader); - - if (!this.gl.getShaderParameter(shader, this.gl.COMPILE_STATUS)) { - window.console.log(this.gl.getShaderInfoLog(shader)); - return null; - } - - return shader; -}; - -},{"../../../utils":50}],36:[function(require,module,exports){ -var Shader = require('./Shader'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - */ -function StripShader(gl) { - Shader.call(this, - gl, - // vertex shader - [ - 'attribute vec2 aVertexPosition;', - 'attribute vec2 aTextureCoord;', - // 'attribute vec4 aColor;', - - 'uniform mat3 translationMatrix;', - 'uniform vec2 projectionVector;', - 'uniform vec2 offsetVector;', - - 'varying vec2 vTextureCoord;', - - 'void main(void) {', - ' vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);', - ' v -= offsetVector.xyx;', - ' gl_Position = vec4( v.x / projectionVector.x -1.0, v.y / -projectionVector.y + 1.0 , 0.0, 1.0);', - ' vTextureCoord = aTextureCoord;', - '}' - ].join('\n'), - // fragment shader - [ - 'precision mediump float;', - - 'uniform float alpha;', - 'uniform sampler2D uSampler;', - - 'varying vec2 vTextureCoord;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * alpha;', - '}' - ].join('\n'), - // custom uniforms - { - alpha: { type: '1f', value: 0 }, - translationMatrix: { type: 'mat3', value: new Float32Array(9) } - } - ); -} - -StripShader.prototype = Object.create(Shader.prototype); -StripShader.prototype.constructor = StripShader; -module.exports = StripShader; - -},{"./Shader":35}],37:[function(require,module,exports){ -var CONST = require('../../../const'); - -/** - * @class - * @namespace PIXI - * @param gl {WebGLContext} the current WebGL drawing context - * @param width {number} the horizontal range of the filter - * @param height {number} the vertical range of the filter - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - */ -function FilterTexture(gl, width, height, scaleMode) { - /** - * @member {WebGLContext} - */ - this.gl = gl; - - // next time to create a frame buffer and texture - - /** - * @member {Any} - */ - this.frameBuffer = gl.createFramebuffer(); - - /** - * @member {Any} - */ - this.texture = gl.createTexture(); - - /** - * @member {number} - */ - scaleMode = scaleMode || CONST.scaleModes.DEFAULT; - - gl.bindTexture(gl.TEXTURE_2D, this.texture); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE); - gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE); - gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer ); - - gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer ); - gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0); - - // required for masking a mask?? - this.renderBuffer = gl.createRenderbuffer(); - gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer); - gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, this.renderBuffer); - - // reset render buffer - gl.bindRenderbuffer(gl.RENDERBUFFER, null); - - this.resize(width, height); -} - -FilterTexture.prototype.constructor = FilterTexture; -module.exports = FilterTexture; - -/** - * Clears the filter texture. - * - */ -FilterTexture.prototype.clear = function () { - var gl = this.gl; - - gl.clearColor(0,0,0, 0); - gl.clear(gl.COLOR_BUFFER_BIT); -}; - -/** - * Resizes the texture to the specified width and height - * - * @param width {number} the new width of the texture - * @param height {number} the new height of the texture - */ -FilterTexture.prototype.resize = function (width, height) { - if (this.width === width && this.height === height) { - return; - } - - this.width = width; - this.height = height; - - var gl = this.gl; - - gl.bindTexture(gl.TEXTURE_2D, this.texture); - gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, width , height , 0, gl.RGBA, gl.UNSIGNED_BYTE, null); - // update the stencil buffer width and height - gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer); - gl.renderbufferStorage(gl.RENDERBUFFER, gl.DEPTH_STENCIL, width , height); - - // reset render buffer - gl.bindRenderbuffer(gl.RENDERBUFFER, null); -}; - -/** - * Destroys the filter texture. - * - */ -FilterTexture.prototype.destroy = function () { - var gl = this.gl; - gl.deleteFramebuffer( this.frameBuffer ); - gl.deleteTexture( this.texture ); - - this.frameBuffer = null; - this.texture = null; -}; - -},{"../../../const":4}],38:[function(require,module,exports){ -/** - * @author Mat Groves - * - * Big thanks to the very clever Matt DesLauriers https://github.com/mattdesl/ - * for creating the original pixi version! - * - * Heavily inspired by LibGDX's WebGLSpriteBatch: - * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java - */ - -/** - * @class - * @private - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this sprite batch works for. - */ -function WebGLFastSpriteBatch(renderer) { - /** - * The renderer instance this sprite batch operates on. - * - * @member {WebGLRenderer} - */ - this.renderer = renderer; - - /** - * - * - * @member {number} - */ - this.vertSize = 10; - - /** - * - * - * @member {number} - */ - this.maxSize = 6000;//Math.pow(2, 16) / this.vertSize; - - /** - * - * - * @member {number} - */ - this.size = this.maxSize; - - //the total number of floats in our batch - var numVerts = this.size * 4 * this.vertSize; - - //the total number of indices in our batch - var numIndices = this.maxSize * 6; - - /** - * Vertex data - * - * @member {Float32Array} - */ - this.vertices = new Float32Array(numVerts); - - /** - * Index data - * - * @member {Uint16Array} - */ - this.indices = new Uint16Array(numIndices); - - /** - * - * - * @member {object} - */ - this.vertexBuffer = null; - - /** - * - * - * @member {object} - */ - this.indexBuffer = null; - - /** - * - * - * @member {number} - */ - this.lastIndexCount = 0; - - for (var i=0, j=0; i < numIndices; i += 6, j += 4) { - this.indices[i + 0] = j + 0; - this.indices[i + 1] = j + 1; - this.indices[i + 2] = j + 2; - this.indices[i + 3] = j + 0; - this.indices[i + 4] = j + 2; - this.indices[i + 5] = j + 3; - } - - /** - * - * - * @member {boolean} - */ - this.drawing = false; - - /** - * - * - * @member {number} - */ - this.currentBatchSize = 0; - - /** - * - * - * @member {BaseTexture} - */ - this.currentBaseTexture = null; - - /** - * - * - * @member {number} - */ - this.currentBlendMode = 0; - - /** - * - * - * @member {object} - */ - this.shader = null; - - /** - * - * - * @member {Matrix} - */ - this.matrix = null; - - // listen for context and update necessary buffers - var self = this; - this.renderer.on('context', function () { - self.setupContext(); - }); -} - -WebGLFastSpriteBatch.prototype.constructor = WebGLFastSpriteBatch; -module.exports = WebGLFastSpriteBatch; - -/** - * Sets the WebGL Context. - * - * @param gl {WebGLContext} the current WebGL drawing context - */ -WebGLFastSpriteBatch.prototype.setupContext = function () { - var gl = this.renderer.gl; - - // create a couple of buffers - this.vertexBuffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - - // 65535 is max index, so 65535 / 6 = 10922. - - //upload the index data - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW); -}; - -/** - * @param spriteBatch {SpriteBatch} The SpriteBatch container to prepare for. - */ -WebGLFastSpriteBatch.prototype.begin = function (spriteBatch) { - this.shader = this.renderer.shaderManager.fastShader; - - this.matrix = spriteBatch.worldTransform.toArray(true); - - this.start(); -}; - -/** - */ -WebGLFastSpriteBatch.prototype.end = function () { - this.flush(); -}; - -/** - * @param spriteBatch {SpriteBatch} The SpriteBatch container to render. - */ -WebGLFastSpriteBatch.prototype.render = function (spriteBatch) { - var children = spriteBatch.children; - var sprite = children[0]; - - // if the uvs have not updated then no point rendering just yet! - - // check texture. - if (!sprite.texture._uvs) { - return; - } - - this.currentBaseTexture = sprite.texture.baseTexture; - - // check blend mode - if (sprite.blendMode !== this.renderer.blendModeManager.currentBlendMode) { - this.flush(); - this.renderer.blendModeManager.setBlendMode(sprite.blendMode); - } - - for (var i=0,j= children.length; i= this.size) { - this.flush(); - } -}; - -/** - * - */ -WebGLFastSpriteBatch.prototype.flush = function () { - // If the batch is length 0 then return as there is nothing to draw - if (this.currentBatchSize === 0) { - return; - } - - var gl = this.renderer.gl; - - // bind the current texture - if (!this.currentBaseTexture._glTextures[gl.id]) { - this.renderer.updateTexture(this.currentBaseTexture, gl); - } - //TODO-SHOUD THIS BE ELSE??!?!?! - else { - gl.bindTexture(gl.TEXTURE_2D, this.currentBaseTexture._glTextures[gl.id]); - } - - // upload the verts to the buffer - - if (this.currentBatchSize > ( this.size * 0.5 ) ) { - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices); - } - else { - var view = this.vertices.subarray(0, this.currentBatchSize * 4 * this.vertSize); - - gl.bufferSubData(gl.ARRAY_BUFFER, 0, view); - } - - // now draw those suckas! - gl.drawElements(gl.TRIANGLES, this.currentBatchSize * 6, gl.UNSIGNED_SHORT, 0); - - // then reset the batch! - this.currentBatchSize = 0; - - // increment the draw count - this.renderer.drawCount++; -}; - - -/** - * Ends the batch and flushes - * - */ -WebGLFastSpriteBatch.prototype.stop = function () { - this.flush(); -}; - -/** - * - */ -WebGLFastSpriteBatch.prototype.start = function () { - var gl = this.renderer.gl; - - // bind the main texture - gl.activeTexture(gl.TEXTURE0); - - // bind the buffers - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - - // set the projection - var projection = this.renderer.projection; - gl.uniform2f(this.shader.projectionVector, projection.x, projection.y); - - // set the matrix - gl.uniformMatrix3fv(this.shader.uMatrix, false, this.matrix); - - // set the pointers - var stride = this.vertSize * 4; - - gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0); - gl.vertexAttribPointer(this.shader.attributes.aPositionCoord, 2, gl.FLOAT, false, stride, 2 * 4); - gl.vertexAttribPointer(this.shader.attributes.aScale, 2, gl.FLOAT, false, stride, 4 * 4); - gl.vertexAttribPointer(this.shader.attributes.aRotation, 1, gl.FLOAT, false, stride, 6 * 4); - gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 7 * 4); - gl.vertexAttribPointer(this.shader.attributes.aColor, 1, gl.FLOAT, false, stride, 9 * 4); -}; - -},{}],39:[function(require,module,exports){ -var utils = require('../../../utils'), - math = require('../../../math'), - CONST = require('../../../const'), - WebGLGraphicsData = require('./WebGLGraphicsData'); - -/** - * A set of functions used by the webGL renderer to draw the primitive graphics data - * - * @namespace PIXI - * @private - */ -var WebGLGraphics = module.exports = {}; - -/** - * Renders the graphics object - * - * @static - * @private - * @param graphics {Graphics} - * @param renderer {WebGLRenderer} - */ -WebGLGraphics.renderGraphics = function (graphics, renderer) {//projection, offset) { - var gl = renderer.gl; - - var projection = renderer.projection, - offset = renderer.offset, - shader = renderer.shaderManager.primitiveShader, - webGLData; - - if (graphics.dirty) { - WebGLGraphics.updateGraphics(graphics, gl); - } - - var webGL = graphics._webGL[gl.id]; - - // This could be speeded up for sure! - - for (var i = 0; i < webGL.data.length; i++) { - if (webGL.data[i].mode === 1) { - webGLData = webGL.data[i]; - - renderer.stencilManager.pushStencil(graphics, webGLData, renderer); - - // render quad.. - gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 ); - - renderer.stencilManager.popStencil(graphics, webGLData, renderer); - } - else { - webGLData = webGL.data[i]; - - - renderer.shaderManager.setShader( shader );//activatePrimitiveShader(); - shader = renderer.shaderManager.primitiveShader; - gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true)); - - gl.uniform1f(shader.flipY, 1); - - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - - gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint)); - - gl.uniform1f(shader.alpha, graphics.worldAlpha); - - - gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer); - - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0); - gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4); - - // set the index buffer! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer); - gl.drawElements(gl.TRIANGLE_STRIP, webGLData.indices.length, gl.UNSIGNED_SHORT, 0 ); - } - } -}; - -/** - * Updates the graphics object - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object to update - * @param gl {WebGLContext} the current WebGL drawing context - */ -WebGLGraphics.updateGraphics = function (graphics, gl) { - // get the contexts graphics object - var webGL = graphics._webGL[gl.id]; - - // if the graphics object does not exist in the webGL context time to create it! - if (!webGL) { - webGL = graphics._webGL[gl.id] = {lastIndex:0, data:[], gl:gl}; - } - - // flag the graphics as not dirty as we are about to update it... - graphics.dirty = false; - - var i; - - // if the user cleared the graphics object we will need to clear every object - if (graphics.clearDirty) { - graphics.clearDirty = false; - - // lop through and return all the webGLDatas to the object pool so than can be reused later on - for (i = 0; i < webGL.data.length; i++) { - var graphicsData = webGL.data[i]; - graphicsData.reset(); - WebGLGraphics.graphicsDataPool.push( graphicsData ); - } - - // clear the array and reset the index.. - webGL.data = []; - webGL.lastIndex = 0; - } - - var webGLData; - - // loop through the graphics datas and construct each one.. - // if the object is a complex fill then the new stencil buffer technique will be used - // other wise graphics objects will be pushed into a batch.. - for (i = webGL.lastIndex; i < graphics.graphicsData.length; i++) { - var data = graphics.graphicsData[i]; - - if (data.type === CONST.SHAPES.POLY) { - // need to add the points the the graphics object.. - data.points = data.shape.points.slice(); - if (data.shape.closed) { - // close the poly if the value is true! - if (data.points[0] !== data.points[data.points.length-2] || data.points[1] !== data.points[data.points.length-1]) { - data.points.push(data.points[0], data.points[1]); - } - } - - // MAKE SURE WE HAVE THE CORRECT TYPE.. - if (data.fill) { - if (data.points.length >= 6) { - if (data.points.length < 6 * 2) { - webGLData = WebGLGraphics.switchMode(webGL, 0); - - var canDrawUsingSimple = WebGLGraphics.buildPoly(data, webGLData); - // console.log(canDrawUsingSimple); - - if (!canDrawUsingSimple) { - // console.log("<>>>") - webGLData = WebGLGraphics.switchMode(webGL, 1); - WebGLGraphics.buildComplexPoly(data, webGLData); - } - - } - else { - webGLData = WebGLGraphics.switchMode(webGL, 1); - WebGLGraphics.buildComplexPoly(data, webGLData); - } - } - } - - if (data.lineWidth > 0) { - webGLData = WebGLGraphics.switchMode(webGL, 0); - WebGLGraphics.buildLine(data, webGLData); - - } - } - else { - webGLData = WebGLGraphics.switchMode(webGL, 0); - - if (data.type === CONST.SHAPES.RECT) { - WebGLGraphics.buildRectangle(data, webGLData); - } - else if (data.type === CONST.SHAPES.CIRC || data.type === CONST.SHAPES.ELIP) { - WebGLGraphics.buildCircle(data, webGLData); - } - else if (data.type === CONST.SHAPES.RREC) { - WebGLGraphics.buildRoundedRectangle(data, webGLData); - } - } - - webGL.lastIndex++; - } - - // upload all the dirty data... - for (i = 0; i < webGL.data.length; i++) { - webGLData = webGL.data[i]; - - if (webGLData.dirty) { - webGLData.upload(); - } - } -}; - -/** - * @static - * @private - * @param webGL {WebGLContext} - * @param type {number} - */ -WebGLGraphics.switchMode = function (webGL, type) { - var webGLData; - - if (!webGL.data.length) { - webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl); - webGLData.mode = type; - webGL.data.push(webGLData); - } - else { - webGLData = webGL.data[webGL.data.length-1]; - - if (webGLData.mode !== type || type === 1) { - webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl); - webGLData.mode = type; - webGL.data.push(webGLData); - } - } - - webGLData.dirty = true; - - return webGLData; -}; - -/** - * Builds a rectangle to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildRectangle = function (graphicsData, webGLData) { - // --- // - // need to convert points to a nice regular data - // - var rectData = graphicsData.shape; - var x = rectData.x; - var y = rectData.y; - var width = rectData.width; - var height = rectData.height; - - if (graphicsData.fill) { - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var verts = webGLData.points; - var indices = webGLData.indices; - - var vertPos = verts.length/6; - - // start - verts.push(x, y); - verts.push(r, g, b, alpha); - - verts.push(x + width, y); - verts.push(r, g, b, alpha); - - verts.push(x , y + height); - verts.push(r, g, b, alpha); - - verts.push(x + width, y + height); - verts.push(r, g, b, alpha); - - // insert 2 dead triangles.. - indices.push(vertPos, vertPos, vertPos+1, vertPos+2, vertPos+3, vertPos+3); - } - - if (graphicsData.lineWidth) { - var tempPoints = graphicsData.points; - - graphicsData.points = [x, y, - x + width, y, - x + width, y + height, - x, y + height, - x, y]; - - - WebGLGraphics.buildLine(graphicsData, webGLData); - - graphicsData.points = tempPoints; - } -}; - -/** - * Builds a rounded rectangle to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildRoundedRectangle = function (graphicsData, webGLData) { - var rrectData = graphicsData.shape; - var x = rrectData.x; - var y = rrectData.y; - var width = rrectData.width; - var height = rrectData.height; - - var radius = rrectData.radius; - - var recPoints = []; - recPoints.push(x, y + radius); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x, y + height - radius, x, y + height, x + radius, y + height)); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width - radius, y + height, x + width, y + height, x + width, y + height - radius)); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width, y + radius, x + width, y, x + width - radius, y)); - recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + radius, y, x, y, x, y + radius)); - - if (graphicsData.fill) { - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var verts = webGLData.points; - var indices = webGLData.indices; - - var vecPos = verts.length/6; - - var triangles = utils.PolyK.Triangulate(recPoints); - - // - - var i = 0; - for (i = 0; i < triangles.length; i+=3) { - indices.push(triangles[i] + vecPos); - indices.push(triangles[i] + vecPos); - indices.push(triangles[i+1] + vecPos); - indices.push(triangles[i+2] + vecPos); - indices.push(triangles[i+2] + vecPos); - } - - - for (i = 0; i < recPoints.length; i++) { - verts.push(recPoints[i], recPoints[++i], r, g, b, alpha); - } - } - - if (graphicsData.lineWidth) { - var tempPoints = graphicsData.points; - - graphicsData.points = recPoints; - - WebGLGraphics.buildLine(graphicsData, webGLData); - - graphicsData.points = tempPoints; - } -}; - -/** - * Calculate the points for a quadratic bezier curve. (helper function..) - * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c - * - * @static - * @private - * @param fromX {number} Origin point x - * @param fromY {number} Origin point x - * @param cpX {number} Control point x - * @param cpY {number} Control point y - * @param toX {number} Destination point x - * @param toY {number} Destination point y - * @return {number[]} - */ -WebGLGraphics.quadraticBezierCurve = function (fromX, fromY, cpX, cpY, toX, toY) { - - var xa, - ya, - xb, - yb, - x, - y, - n = 20, - points = []; - - function getPt(n1 , n2, perc) { - var diff = n2 - n1; - - return n1 + ( diff * perc ); - } - - var j = 0; - for (var i = 0; i <= n; i++ ) { - j = i / n; - - // The Green Line - xa = getPt( fromX , cpX , j ); - ya = getPt( fromY , cpY , j ); - xb = getPt( cpX , toX , j ); - yb = getPt( cpY , toY , j ); - - // The Black Dot - x = getPt( xa , xb , j ); - y = getPt( ya , yb , j ); - - points.push(x, y); - } - return points; -}; - -/** - * Builds a circle to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object to draw - * @param webGLData {object} - */ -WebGLGraphics.buildCircle = function (graphicsData, webGLData) { - // need to convert points to a nice regular data - var circleData = graphicsData.shape; - var x = circleData.x; - var y = circleData.y; - var width; - var height; - - // TODO - bit hacky?? - if (graphicsData.type === CONST.SHAPES.CIRC) { - width = circleData.radius; - height = circleData.radius; - } - else { - width = circleData.width; - height = circleData.height; - } - - var totalSegs = 40; - var seg = (Math.PI * 2) / totalSegs ; - - var i = 0; - - if (graphicsData.fill) { - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var verts = webGLData.points; - var indices = webGLData.indices; - - var vecPos = verts.length/6; - - indices.push(vecPos); - - for (i = 0; i < totalSegs + 1 ; i++) { - verts.push(x,y, r, g, b, alpha); - - verts.push(x + Math.sin(seg * i) * width, - y + Math.cos(seg * i) * height, - r, g, b, alpha); - - indices.push(vecPos++, vecPos++); - } - - indices.push(vecPos-1); - } - - if (graphicsData.lineWidth) { - var tempPoints = graphicsData.points; - - graphicsData.points = []; - - for (i = 0; i < totalSegs + 1; i++) { - graphicsData.points.push(x + Math.sin(seg * i) * width, - y + Math.cos(seg * i) * height); - } - - WebGLGraphics.buildLine(graphicsData, webGLData); - - graphicsData.points = tempPoints; - } -}; - -/** - * Builds a line to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildLine = function (graphicsData, webGLData) { - // TODO OPTIMISE! - var i = 0; - var points = graphicsData.points; - - if (points.length === 0) { - return; - } - - // if the line width is an odd number add 0.5 to align to a whole pixel - if (graphicsData.lineWidth%2) { - for (i = 0; i < points.length; i++) { - points[i] += 0.5; - } - } - - // get first and last point.. figure out the middle! - var firstPoint = new math.Point(points[0], points[1]); - var lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]); - - // if the first point is the last point - gonna have issues :) - if (firstPoint.x === lastPoint.x && firstPoint.y === lastPoint.y) { - // need to clone as we are going to slightly modify the shape.. - points = points.slice(); - - points.pop(); - points.pop(); - - lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]); - - var midPointX = lastPoint.x + (firstPoint.x - lastPoint.x) *0.5; - var midPointY = lastPoint.y + (firstPoint.y - lastPoint.y) *0.5; - - points.unshift(midPointX, midPointY); - points.push(midPointX, midPointY); - } - - var verts = webGLData.points; - var indices = webGLData.indices; - var length = points.length / 2; - var indexCount = points.length; - var indexStart = verts.length/6; - - // DRAW the Line - var width = graphicsData.lineWidth / 2; - - // sort color - var color = utils.hex2rgb(graphicsData.lineColor); - var alpha = graphicsData.lineAlpha; - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var px, py, p1x, p1y, p2x, p2y, p3x, p3y; - var perpx, perpy, perp2x, perp2y, perp3x, perp3y; - var a1, b1, c1, a2, b2, c2; - var denom, pdist, dist; - - p1x = points[0]; - p1y = points[1]; - - p2x = points[2]; - p2y = points[3]; - - perpx = -(p1y - p2y); - perpy = p1x - p2x; - - dist = Math.sqrt(perpx*perpx + perpy*perpy); - - perpx /= dist; - perpy /= dist; - perpx *= width; - perpy *= width; - - // start - verts.push(p1x - perpx , p1y - perpy, - r, g, b, alpha); - - verts.push(p1x + perpx , p1y + perpy, - r, g, b, alpha); - - for (i = 1; i < length-1; i++) { - p1x = points[(i-1)*2]; - p1y = points[(i-1)*2 + 1]; - - p2x = points[(i)*2]; - p2y = points[(i)*2 + 1]; - - p3x = points[(i+1)*2]; - p3y = points[(i+1)*2 + 1]; - - perpx = -(p1y - p2y); - perpy = p1x - p2x; - - dist = Math.sqrt(perpx*perpx + perpy*perpy); - perpx /= dist; - perpy /= dist; - perpx *= width; - perpy *= width; - - perp2x = -(p2y - p3y); - perp2y = p2x - p3x; - - dist = Math.sqrt(perp2x*perp2x + perp2y*perp2y); - perp2x /= dist; - perp2y /= dist; - perp2x *= width; - perp2y *= width; - - a1 = (-perpy + p1y) - (-perpy + p2y); - b1 = (-perpx + p2x) - (-perpx + p1x); - c1 = (-perpx + p1x) * (-perpy + p2y) - (-perpx + p2x) * (-perpy + p1y); - a2 = (-perp2y + p3y) - (-perp2y + p2y); - b2 = (-perp2x + p2x) - (-perp2x + p3x); - c2 = (-perp2x + p3x) * (-perp2y + p2y) - (-perp2x + p2x) * (-perp2y + p3y); - - denom = a1*b2 - a2*b1; - - if (Math.abs(denom) < 0.1 ) { - - denom+=10.1; - verts.push(p2x - perpx , p2y - perpy, - r, g, b, alpha); - - verts.push(p2x + perpx , p2y + perpy, - r, g, b, alpha); - - continue; - } - - px = (b1*c2 - b2*c1)/denom; - py = (a2*c1 - a1*c2)/denom; - - - pdist = (px -p2x) * (px -p2x) + (py -p2y) + (py -p2y); - - - if (pdist > 140 * 140) { - perp3x = perpx - perp2x; - perp3y = perpy - perp2y; - - dist = Math.sqrt(perp3x*perp3x + perp3y*perp3y); - perp3x /= dist; - perp3y /= dist; - perp3x *= width; - perp3y *= width; - - verts.push(p2x - perp3x, p2y -perp3y); - verts.push(r, g, b, alpha); - - verts.push(p2x + perp3x, p2y +perp3y); - verts.push(r, g, b, alpha); - - verts.push(p2x - perp3x, p2y -perp3y); - verts.push(r, g, b, alpha); - - indexCount++; - } - else { - - verts.push(px , py); - verts.push(r, g, b, alpha); - - verts.push(p2x - (px-p2x), p2y - (py - p2y)); - verts.push(r, g, b, alpha); - } - } - - p1x = points[(length-2)*2]; - p1y = points[(length-2)*2 + 1]; - - p2x = points[(length-1)*2]; - p2y = points[(length-1)*2 + 1]; - - perpx = -(p1y - p2y); - perpy = p1x - p2x; - - dist = Math.sqrt(perpx*perpx + perpy*perpy); - perpx /= dist; - perpy /= dist; - perpx *= width; - perpy *= width; - - verts.push(p2x - perpx , p2y - perpy); - verts.push(r, g, b, alpha); - - verts.push(p2x + perpx , p2y + perpy); - verts.push(r, g, b, alpha); - - indices.push(indexStart); - - for (i = 0; i < indexCount; i++) { - indices.push(indexStart++); - } - - indices.push(indexStart-1); -}; - -/** - * Builds a complex polygon to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildComplexPoly = function (graphicsData, webGLData) { - //TODO - no need to copy this as it gets turned into a FLoat32Array anyways.. - var points = graphicsData.points.slice(); - - if (points.length < 6) { - return; - } - - // get first and last point.. figure out the middle! - var indices = webGLData.indices; - webGLData.points = points; - webGLData.alpha = graphicsData.fillAlpha; - webGLData.color = utils.hex2rgb(graphicsData.fillColor); - - // calclate the bounds.. - var minX = Infinity; - var maxX = -Infinity; - - var minY = Infinity; - var maxY = -Infinity; - - var x,y; - - // get size.. - for (var i = 0; i < points.length; i+=2) { - x = points[i]; - y = points[i+1]; - - minX = x < minX ? x : minX; - maxX = x > maxX ? x : maxX; - - minY = y < minY ? y : minY; - maxY = y > maxY ? y : maxY; - } - - // add a quad to the end cos there is no point making another buffer! - points.push(minX, minY, - maxX, minY, - maxX, maxY, - minX, maxY); - - // push a quad onto the end.. - - //TODO - this aint needed! - var length = points.length / 2; - for (i = 0; i < length; i++) { - indices.push( i ); - } - -}; - -/** - * Builds a polygon to draw - * - * @static - * @private - * @param graphicsData {Graphics} The graphics object containing all the necessary properties - * @param webGLData {object} - */ -WebGLGraphics.buildPoly = function (graphicsData, webGLData) { - var points = graphicsData.points; - - if (points.length < 6) { - return; - } - // get first and last point.. figure out the middle! - var verts = webGLData.points; - var indices = webGLData.indices; - - var length = points.length / 2; - - // sort color - var color = utils.hex2rgb(graphicsData.fillColor); - var alpha = graphicsData.fillAlpha; - var r = color[0] * alpha; - var g = color[1] * alpha; - var b = color[2] * alpha; - - var triangles = utils.PolyK.Triangulate(points); - - if (!triangles) { - return false; - } - - var vertPos = verts.length / 6; - - var i = 0; - - for (i = 0; i < triangles.length; i+=3) { - indices.push(triangles[i] + vertPos); - indices.push(triangles[i] + vertPos); - indices.push(triangles[i+1] + vertPos); - indices.push(triangles[i+2] +vertPos); - indices.push(triangles[i+2] + vertPos); - } - - for (i = 0; i < length; i++) { - verts.push(points[i * 2], points[i * 2 + 1], - r, g, b, alpha); - } - - return true; -}; - -WebGLGraphics.graphicsDataPool = []; - -},{"../../../const":4,"../../../math":12,"../../../utils":50,"./WebGLGraphicsData":40}],40:[function(require,module,exports){ -/** - * @class - * @private - */ -function WebGLGraphicsData(gl) { - this.gl = gl; - - //TODO does this need to be split before uploding?? - this.color = [0, 0, 0]; // color split! - this.points = []; - this.indices = []; - this.buffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - this.mode = 1; - this.alpha = 1; - this.dirty = true; -} - -WebGLGraphicsData.prototype.constructor = WebGLGraphicsData; -module.exports = WebGLGraphicsData; - -/** - * - */ -WebGLGraphicsData.prototype.reset = function () { - this.points.length = 0; - this.indices.length = 0; -}; - -/** - * - */ -WebGLGraphicsData.prototype.upload = function () { - var gl = this.gl; - -// this.lastIndex = graphics.graphicsData.length; - this.glPoints = new Float32Array(this.points); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.buffer); - gl.bufferData(gl.ARRAY_BUFFER, this.glPoints, gl.STATIC_DRAW); - - this.glIndicies = new Uint16Array(this.indices); - - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.glIndicies, gl.STATIC_DRAW); - - this.dirty = false; -}; - -},{}],41:[function(require,module,exports){ -var TextureUvs = require('../../../textures/TextureUvs'), - Shader = require('../shaders/Shader'); - -/** - * @author Mat Groves - * - * Big thanks to the very clever Matt DesLauriers https://github.com/mattdesl/ - * for creating the original pixi version! - * Also a thanks to https://github.com/bchevalier for tweaking the tint and alpha so that they now share 4 bytes on the vertex buffer - * - * Heavily inspired by LibGDX's WebGLSpriteBatch: - * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java - */ - -/** - * - * @class - * @private - * @namespace PIXI - * @param renderer {WebGLRenderer} The renderer this sprite batch works for. - */ -function WebGLSpriteBatch(renderer) { - /** - * - * - * @member {WebGLRenderer} - */ - this.renderer = renderer; - - /** - * - * - * @member {number} - */ - this.vertSize = 5; - - /** - * The number of images in the SpriteBatch before it flushes. - * - * @member {number} - */ - this.size = 2000;//Math.pow(2, 16) / this.vertSize; - - // the total number of bytes in our batch - var numVerts = this.size * 4 * 4 * this.vertSize; - // the total number of indices in our batch - var numIndices = this.size * 6; - - /** - * Holds the vertices - * - * @member {ArrayBuffer} - */ - this.vertices = new ArrayBuffer(numVerts); - - /** - * View on the vertices as a Float32Array - * - * @member {Float32Array} - */ - this.positions = new Float32Array(this.vertices); - - /** - * View on the vertices as a Uint32Array - * - * @member {Uint32Array} - */ - this.colors = new Uint32Array(this.vertices); - - /** - * Holds the indices - * - * @member {Uint16Array} - */ - this.indices = new Uint16Array(numIndices); - - /** - * - * - * @member {number} - */ - this.lastIndexCount = 0; - - for (var i=0, j=0; i < numIndices; i += 6, j += 4) { - this.indices[i + 0] = j + 0; - this.indices[i + 1] = j + 1; - this.indices[i + 2] = j + 2; - this.indices[i + 3] = j + 0; - this.indices[i + 4] = j + 2; - this.indices[i + 5] = j + 3; - } - - /** - * - * - * @member {boolean} - */ - this.drawing = false; - - /** - * - * - * @member {number} - */ - this.currentBatchSize = 0; - - /** - * - * - * @member {BaseTexture} - */ - this.currentBaseTexture = null; - - /** - * - * - * @member {boolean} - */ - this.dirty = true; - - /** - * - * - * @member {Array} - */ - this.textures = []; - - /** - * - * - * @member {Array} - */ - this.blendModes = []; - - /** - * - * - * @member {Array} - */ - this.shaders = []; - - /** - * - * - * @member {Array} - */ - this.sprites = []; - - /** - * The default shader that is used if a sprite doesn't have a more specific one. - * - * @member {Shader} - */ - this.shader = null; - - // listen for context and update necessary buffers - var self = this; - this.renderer.on('context', function () { - self.setupContext(); - }); -} - -WebGLSpriteBatch.prototype.constructor = WebGLSpriteBatch; -module.exports = WebGLSpriteBatch; - -/** - * @param gl {WebGLContext} the current WebGL drawing context - */ -WebGLSpriteBatch.prototype.setupContext = function () { - var gl = this.renderer.gl; - - // setup default shader - this.shader = new Shader(gl); - - // create a couple of buffers - this.vertexBuffer = gl.createBuffer(); - this.indexBuffer = gl.createBuffer(); - - // 65535 is max index, so 65535 / 6 = 10922. - - //upload the index data - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW); - - this.currentBlendMode = 99999; -}; - -/** - * - */ -WebGLSpriteBatch.prototype.begin = function () { - // this.shader = this.renderer.shaderManager.defaultShader; - - this.start(); -}; - -/** - */ -WebGLSpriteBatch.prototype.end = function () { - this.flush(); -}; - -/** - * @param sprite {Sprite} the sprite to render when using this spritebatch - */ -WebGLSpriteBatch.prototype.render = function (sprite) { - var texture = sprite.texture; - - //TODO set blend modes.. - // check texture.. - if (this.currentBatchSize >= this.size) { - this.flush(); - this.currentBaseTexture = texture.baseTexture; - } - - // get the uvs for the texture - var uvs = texture._uvs; - - // if the uvs have not updated then no point rendering just yet! - if (!uvs) { - return; - } - - // TODO trim?? - var aX = sprite.anchor.x; - var aY = sprite.anchor.y; - - var w0, w1, h0, h1; - - if (texture.trim) { - // if the sprite is trimmed then we need to add the extra space before transforming the sprite coords.. - var trim = texture.trim; - - w1 = trim.x - aX * trim.width; - w0 = w1 + texture.crop.width; - - h1 = trim.y - aY * trim.height; - h0 = h1 + texture.crop.height; - - } - else { - w0 = (texture.frame.width ) * (1-aX); - w1 = (texture.frame.width ) * -aX; - - h0 = texture.frame.height * (1-aY); - h1 = texture.frame.height * -aY; - } - - var index = this.currentBatchSize * 4 * this.vertSize; - - var resolution = texture.baseTexture.resolution; - - var worldTransform = sprite.worldTransform; - - var a = worldTransform.a / resolution; - var b = worldTransform.b / resolution; - var c = worldTransform.c / resolution; - var d = worldTransform.d / resolution; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var colors = this.colors; - var positions = this.positions; - - if (this.renderer.roundPixels) { - // xy - positions[index] = a * w1 + c * h1 + tx | 0; - positions[index+1] = d * h1 + b * w1 + ty | 0; - - // xy - positions[index+5] = a * w0 + c * h1 + tx | 0; - positions[index+6] = d * h1 + b * w0 + ty | 0; - - // xy - positions[index+10] = a * w0 + c * h0 + tx | 0; - positions[index+11] = d * h0 + b * w0 + ty | 0; - - // xy - positions[index+15] = a * w1 + c * h0 + tx | 0; - positions[index+16] = d * h0 + b * w1 + ty | 0; - } - else { - // xy - positions[index] = a * w1 + c * h1 + tx; - positions[index+1] = d * h1 + b * w1 + ty; - - // xy - positions[index+5] = a * w0 + c * h1 + tx; - positions[index+6] = d * h1 + b * w0 + ty; - - // xy - positions[index+10] = a * w0 + c * h0 + tx; - positions[index+11] = d * h0 + b * w0 + ty; - - // xy - positions[index+15] = a * w1 + c * h0 + tx; - positions[index+16] = d * h0 + b * w1 + ty; - } - - // uv - positions[index+2] = uvs.x0; - positions[index+3] = uvs.y0; - - // uv - positions[index+7] = uvs.x1; - positions[index+8] = uvs.y1; - - // uv - positions[index+12] = uvs.x2; - positions[index+13] = uvs.y2; - - // uv - positions[index+17] = uvs.x3; - positions[index+18] = uvs.y3; - - // color and alpha - var tint = sprite.tint; - colors[index+4] = colors[index+9] = colors[index+14] = colors[index+19] = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (sprite.worldAlpha * 255 << 24); - - // increment the batchsize - this.sprites[this.currentBatchSize++] = sprite; - - -}; - -/** - * Renders a TilingSprite using the spriteBatch. - * - * @param sprite {TilingSprite} the tilingSprite to render - */ -WebGLSpriteBatch.prototype.renderTilingSprite = function (tilingSprite) { - var texture = tilingSprite.tilingTexture; - - // check texture.. - if (this.currentBatchSize >= this.size) { - //return; - this.flush(); - this.currentBaseTexture = texture.baseTexture; - } - - // set the textures uvs temporarily - // TODO create a separate texture so that we can tile part of a texture - - if (!tilingSprite._uvs) { - tilingSprite._uvs = new TextureUvs(); - } - - var uvs = tilingSprite._uvs; - - tilingSprite.tilePosition.x %= texture.baseTexture.width * tilingSprite.tileScaleOffset.x; - tilingSprite.tilePosition.y %= texture.baseTexture.height * tilingSprite.tileScaleOffset.y; - - var offsetX = tilingSprite.tilePosition.x/(texture.baseTexture.width*tilingSprite.tileScaleOffset.x); - var offsetY = tilingSprite.tilePosition.y/(texture.baseTexture.height*tilingSprite.tileScaleOffset.y); - - var scaleX = (tilingSprite.width / texture.baseTexture.width) / (tilingSprite.tileScale.x * tilingSprite.tileScaleOffset.x); - var scaleY = (tilingSprite.height / texture.baseTexture.height) / (tilingSprite.tileScale.y * tilingSprite.tileScaleOffset.y); - - uvs.x0 = 0 - offsetX; - uvs.y0 = 0 - offsetY; - - uvs.x1 = (1 * scaleX) - offsetX; - uvs.y1 = 0 - offsetY; - - uvs.x2 = (1 * scaleX) - offsetX; - uvs.y2 = (1 * scaleY) - offsetY; - - uvs.x3 = 0 - offsetX; - uvs.y3 = (1 * scaleY) - offsetY; - - // get the tilingSprites current alpha and tint and combining them into a single color - var tint = tilingSprite.tint; - var color = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (tilingSprite.alpha * 255 << 24); - - var positions = this.positions; - var colors = this.colors; - - var width = tilingSprite.width; - var height = tilingSprite.height; - - // TODO trim?? - var aX = tilingSprite.anchor.x; - var aY = tilingSprite.anchor.y; - var w0 = width * (1-aX); - var w1 = width * -aX; - - var h0 = height * (1-aY); - var h1 = height * -aY; - - var index = this.currentBatchSize * 4 * this.vertSize; - - var resolution = texture.baseTexture.resolution; - - var worldTransform = tilingSprite.worldTransform; - - var a = worldTransform.a / resolution;//[0]; - var b = worldTransform.b / resolution;//[3]; - var c = worldTransform.c / resolution;//[1]; - var d = worldTransform.d / resolution;//[4]; - var tx = worldTransform.tx;//[2]; - var ty = worldTransform.ty;//[5]; - - // xy - positions[index++] = a * w1 + c * h1 + tx; - positions[index++] = d * h1 + b * w1 + ty; - // uv - positions[index++] = uvs.x0; - positions[index++] = uvs.y0; - // color - colors[index++] = color; - - // xy - positions[index++] = (a * w0 + c * h1 + tx); - positions[index++] = d * h1 + b * w0 + ty; - // uv - positions[index++] = uvs.x1; - positions[index++] = uvs.y1; - // color - colors[index++] = color; - - // xy - positions[index++] = a * w0 + c * h0 + tx; - positions[index++] = d * h0 + b * w0 + ty; - // uv - positions[index++] = uvs.x2; - positions[index++] = uvs.y2; - // color - colors[index++] = color; - - // xy - positions[index++] = a * w1 + c * h0 + tx; - positions[index++] = d * h0 + b * w1 + ty; - // uv - positions[index++] = uvs.x3; - positions[index++] = uvs.y3; - // color - colors[index++] = color; - - // increment the batchsize - this.sprites[this.currentBatchSize++] = tilingSprite; -}; - -/** - * Renders the content and empties the current batch. - * - */ -WebGLSpriteBatch.prototype.flush = function () { - // If the batch is length 0 then return as there is nothing to draw - if (this.currentBatchSize === 0) { - return; - } - - var gl = this.renderer.gl; - var shader; - - if (this.dirty) { - this.dirty = false; - // bind the main texture - gl.activeTexture(gl.TEXTURE0); - - // bind the buffers - gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer); - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer); - - // this is the same for each shader? - var stride = this.vertSize * 4; - gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0); - gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 2 * 4); - - // color attributes will be interpreted as unsigned bytes and normalized - gl.vertexAttribPointer(this.shader.attributes.aColor, 4, gl.UNSIGNED_BYTE, true, stride, 4 * 4); - } - - // upload the verts to the buffer - if (this.currentBatchSize > ( this.size * 0.5 ) ) { - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices); - } - else { - var view = this.positions.subarray(0, this.currentBatchSize * 4 * this.vertSize); - gl.bufferSubData(gl.ARRAY_BUFFER, 0, view); - } - - var nextTexture, nextBlendMode, nextShader; - var batchSize = 0; - var start = 0; - - var currentBaseTexture = null; - var currentBlendMode = this.renderer.blendModeManager.currentBlendMode; - var currentShader = null; - - var blendSwap = false; - var shaderSwap = false; - var sprite; - - for (var i = 0, j = this.currentBatchSize; i < j; i++) { - - sprite = this.sprites[i]; - - nextTexture = sprite.texture.baseTexture; - nextBlendMode = sprite.blendMode; - nextShader = sprite.shader || this.shader; - - blendSwap = currentBlendMode !== nextBlendMode; - shaderSwap = currentShader !== nextShader; // should I use uuidS??? - - if (currentBaseTexture !== nextTexture || blendSwap || shaderSwap) { - this.renderBatch(currentBaseTexture, batchSize, start); - - start = i; - batchSize = 0; - currentBaseTexture = nextTexture; - - if (blendSwap) { - currentBlendMode = nextBlendMode; - this.renderer.blendModeManager.setBlendMode( currentBlendMode ); - } - - if (shaderSwap) { - currentShader = nextShader; - - shader = currentShader.shaders ? currentShader.shaders[gl.id] : currentShader; - - if (!shader) { - shader = new Shader(gl, null, currentShader.fragmentSrc, currentShader.uniforms); - currentShader.shaders[gl.id] = shader; - } - - // set shader function??? - this.renderer.shaderManager.setShader(shader); - - if (shader.dirty) { - shader.syncUniforms(); - } - - // both thease only need to be set if they are changing.. - // set the projection - var projection = this.renderer.projection; - gl.uniform2f(shader.uniforms.projectionVector._location, projection.x, projection.y); - - // TODO - this is temprorary! - var offsetVector = this.renderer.offset; - gl.uniform2f(shader.uniforms.offsetVector._location, offsetVector.x, offsetVector.y); - - // set the pointers - } - } - - batchSize++; - } - - this.renderBatch(currentBaseTexture, batchSize, start); - - // then reset the batch! - this.currentBatchSize = 0; -}; - -/** - * @param texture {Texture} - * @param size {number} - * @param startIndex {number} - */ -WebGLSpriteBatch.prototype.renderBatch = function (texture, size, startIndex) { - if (size === 0) { - return; - } - - var gl = this.renderer.gl; - - if (!texture._glTextures[gl.id]) { - this.renderer.updateTexture(texture); - } - else { - // bind the current texture - gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]); - } - - // now draw those suckas! - gl.drawElements(gl.TRIANGLES, size * 6, gl.UNSIGNED_SHORT, startIndex * 6 * 2); - - // increment the draw count - this.renderer.drawCount++; -}; - -/** - * - */ -WebGLSpriteBatch.prototype.stop = function () { - this.flush(); - this.dirty = true; -}; - -/** - * - */ -WebGLSpriteBatch.prototype.start = function () { - this.dirty = true; -}; - -/** - * Destroys the SpriteBatch. - * - */ -WebGLSpriteBatch.prototype.destroy = function () { - this.renderer.gl.deleteBuffer(this.vertexBuffer); - this.renderer.gl.deleteBuffer(this.indexBuffer); - - this.vertices = null; - this.indices = null; - - this.vertexBuffer = null; - this.indexBuffer = null; - - this.currentBaseTexture = null; - - this.renderer = null; -}; - -},{"../../../textures/TextureUvs":45,"../shaders/Shader":35}],42:[function(require,module,exports){ -var utils = require('../utils'), - CONST = require('../const'); - -/** - * A texture stores the information that represents an image. All textures have a base texture. - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param source {Image|Canvas} the source object of the texture. - * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values - */ -function BaseTexture(source, scaleMode) { - this.uuid = utils.uuid(); - - /** - * The Resolution of the texture. - * - * @member {number} - */ - this.resolution = 1; - - /** - * The width of the base texture set when the image has loaded - * - * @member {number} - * @readOnly - */ - this.width = 100; - - /** - * The height of the base texture set when the image has loaded - * - * @member {number} - * @readOnly - */ - this.height = 100; - - /** - * The scale mode to apply when scaling this texture - * - * @member {{number}} - * @default scaleModes.LINEAR - */ - this.scaleMode = scaleMode || CONST.scaleModes.DEFAULT; - - /** - * Set to true once the base texture has successfully loaded. - * - * This is never true if the underlying source fails to load or has no texture data. - * - * @member {boolean} - * @readOnly - */ - this.hasLoaded = false; - - /** - * Set to true if the source is currently loading. - * - * If an Image source is loading the 'loaded' or 'error' event will be - * dispatched when the operation ends. An underyling source that is - * immediately-available bypasses loading entirely. - * - * @member {boolean} - * @readonly - */ - this.isLoading = false; - - /** - * The image source that is used to create the texture. - * - * TODO: Make this a setter that calls loadSource(); - * - * @member {Image|Canvas} - * @readonly - */ - this.source = null; // set in loadSource, if at all - - /** - * Controls if RGB channels should be pre-multiplied by Alpha (WebGL only) - * - * @member {boolean} - * @default true - */ - this.premultipliedAlpha = true; - - /** - * @member {string} - */ - this.imageUrl = null; - - /** - * @member {boolean} - * @private - */ - this._powerOf2 = false; - - // used for webGL - - /** - * - * Set this to true if a mipmap of this texture needs to be generated. This value needs to be set before the texture is used - * Also the texture must be a power of two size to work - * - * @member {boolean} - */ - this.mipmap = false; - - /** - * A map of renderer IDs to webgl textures - * - * @member {object} - * @private - */ - this._glTextures = {}; - - /** - * Does the texture on the GPU need to be updated? - * - * @member {boolean} - * @private - */ - this._needsUpdate = false; - - // if no source passed don't try to load - if (source) { - this.loadSource(source); - } - - /** - * Fired when a not-immediately-available source finishes loading. - * - * @event loaded - * @protected - */ - - /** - * Fired when a not-immediately-available source fails to load. - * - * @event error - * @protected - */ -} - -BaseTexture.prototype.constructor = BaseTexture; -module.exports = BaseTexture; - -utils.eventTarget.mixin(BaseTexture.prototype); - -Object.defineProperties(BaseTexture.prototype, { - needsUpdate: { - get: function () { - return this._needsUpdate; - }, - set: function (val) { - this._needsUpdate = val; - - if (val) { - this.emit('update', this); - } - } - } -}); - -/** - * Load a source. - * - * If the source is not-immediately-available, such as an image that needs to be - * downloaded, then the 'loaded' or 'error' event will be dispatched in the future - * and `hasLoaded` will remain false after this call. - * - * The logic state after calling `loadSource` directly or indirectly (eg. `fromImage`, `new BaseTexture`) is: - * - * if (texture.hasLoaded) { - * // texture ready for use - * } else if (texture.isLoading) { - * // listen to 'loaded' and/or 'error' events on texture - * } else { - * // not loading, not going to load UNLESS the source is reloaded - * // (it may still make sense to listen to the events) - * } - * - * @protected - * @param source {Image|Canvas} the source object of the texture. - */ -BaseTexture.prototype.loadSource = function (source) { - var wasLoading = this.isLoading; - this.hasLoaded = false; - this.isLoading = false; - - if (wasLoading && this.source) { - this.source.onload = null; - this.source.onerror = null; - } - - this.source = source; - - // Apply source if loaded. Otherwise setup appropriate loading monitors. - if ((this.source.complete || this.source.getContext) && this.source.width && this.source.height) { - this._sourceLoaded(); - } - else if(!source.getContext) { - // Image fail / not ready - this.isLoading = true; - - var scope = this; - - source.onload = function () { - source.onload = null; - source.onerror = null; - - if(!scope.isLoading) { - return; - } - - scope.isLoading = false; - scope._sourceLoaded(); - - scope.emit('loaded', scope); - }; - - source.onerror = function () { - source.onload = null; - source.onerror = null; - - if(!scope.isLoading) { - return; - } - - scope.isLoading = false; - scope.emit('error', scope); - }; - - // Per http://www.w3.org/TR/html5/embedded-content-0.html#the-img-element - // "The value of `complete` can thus change while a script is executing." - // So complete needs to be re-checked after the callbacks have been added.. - if (source.complete) { - this.isLoading = false; - - // ..and if we're complete now, no need for callbacks - source.onload = null; - source.onerror = null; - - if (source.width && source.height) { - this._sourceLoaded(); - - // If any previous subscribers possible - if (wasLoading) { - this.emit('loaded', this); - } - } - else { - // If any previous subscribers possible - if (wasLoading) { - this.emit('error', this); - } - } - } - } -}; - -/** - * Used internally to update the width, height, and some other tracking vars once - * a source has successfully loaded. - * - * @private - */ -BaseTexture.prototype._sourceLoaded = function () { - this.hasLoaded = true; - - this.width = this.source.naturalWidth || this.source.width; - this.height = this.source.naturalHeight || this.source.height; - - this.needsUpdate = true; -}; - -/** - * Destroys this base texture - * - */ -BaseTexture.prototype.destroy = function () { - if (this.imageUrl) { - delete utils.BaseTextureCache[this.imageUrl]; - delete utils.TextureCache[this.imageUrl]; - this.imageUrl = null; - if (!navigator.isCocoonJS) { - this.source.src = ''; - } - } - else if (this.source && this.source._pixiId) { - delete utils.BaseTextureCache[this.source._pixiId]; - } - this.source = null; - - this.dispose(); -}; - -/** - * Frees the texture from WebGL memory without destroying this texture object. - * This means you can still use the texture later which will upload it to GPU - * memory again. - * - */ -BaseTexture.prototype.dispose = function () { - this.emit('dispose', this); -}; - -/** - * Changes the source image of the texture. - * The original source must be an Image element. - * - * @param newSrc {string} the path of the image - */ -BaseTexture.prototype.updateSourceImage = function (newSrc) { - this.source.src = newSrc; - this.loadSource(this.source); -}; - -/** - * Helper function that creates a base texture from the given image url. - * If the image is not in the base texture cache it will be created and loaded. - * - * @static - * @param imageUrl {string} The image url of the texture - * @param [crossorigin=(auto)] {boolean} Should use anonymouse CORS? Defaults to true if the URL is not a data-URI. - * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values - * @return BaseTexture - */ -BaseTexture.fromImage = function (imageUrl, crossorigin, scaleMode) { - var baseTexture = utils.BaseTextureCache[imageUrl]; - - if (crossorigin === undefined && imageUrl.indexOf('data:') !== 0) { - crossorigin = true; - } - - if (!baseTexture) { - // new Image() breaks tex loading in some versions of Chrome. - // See https://code.google.com/p/chromium/issues/detail?id=238071 - var image = new Image();//document.createElement('img'); - if (crossorigin) { - image.crossOrigin = ''; - } - - image.src = imageUrl; - baseTexture = new BaseTexture(image, scaleMode); - baseTexture.imageUrl = imageUrl; - utils.BaseTextureCache[imageUrl] = baseTexture; - - // if there is an @2x at the end of the url we are going to assume its a highres image - if ( imageUrl.indexOf(CONST.RETINA_PREFIX + '.') !== -1) { - baseTexture.resolution = 2; - } - } - - return baseTexture; -}; - -/** - * Helper function that creates a base texture from the given canvas element. - * - * @static - * @param canvas {Canvas} The canvas element source of the texture - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return BaseTexture - */ -BaseTexture.fromCanvas = function (canvas, scaleMode) { - if (!canvas._pixiId) { - canvas._pixiId = 'canvas_' + utils.uuid(); - } - - var baseTexture = utils.BaseTextureCache[canvas._pixiId]; - - if (!baseTexture) { - baseTexture = new BaseTexture(canvas, scaleMode); - utils.BaseTextureCache[canvas._pixiId] = baseTexture; - } - - return baseTexture; -}; - -},{"../const":4,"../utils":50}],43:[function(require,module,exports){ -var BaseTexture = require('./BaseTexture'), - Texture = require('./Texture'), - FilterTexture = require('../renderers/webgl/utils/FilterTexture'), - CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'), - math = require('../math'), - CONST = require('../const'); - -/** - * A RenderTexture is a special texture that allows any Pixi display object to be rendered to it. - * - * __Hint__: All DisplayObjects (i.e. Sprites) that render to a RenderTexture should be preloaded - * otherwise black rectangles will be drawn instead. - * - * A RenderTexture takes a snapshot of any Display Object given to its render method. The position - * and rotation of the given Display Objects is ignored. For example: - * - * ```js - * var renderTexture = new PIXI.RenderTexture(800, 600); - * var sprite = PIXI.Sprite.fromImage("spinObj_01.png"); - * - * sprite.position.x = 800/2; - * sprite.position.y = 600/2; - * sprite.anchor.x = 0.5; - * sprite.anchor.y = 0.5; - * - * renderTexture.render(sprite); - * ``` - * - * The Sprite in this case will be rendered to a position of 0,0. To render this sprite at its actual - * position a DisplayObjectContainer should be used: - * - * ```js - * var doc = new DisplayObjectContainer(); - * - * doc.addChild(sprite); - * - * renderTexture.render(doc); // Renders to center of renderTexture - * ``` - * - * @class - * @extends Texture - * @namespace PIXI - * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used for this RenderTexture - * @param [width=100] {number} The width of the render texture - * @param [height=100] {number} The height of the render texture - * @param [scaleMode] {number} See {@link scaleModes} for possible values - * @param [resolution=1] {number} The resolution of the texture being generated - */ -function RenderTexture(renderer, width, height, scaleMode, resolution) { - if (!renderer) { - throw new Error('Unable to create RenderTexture, you must pass a renderer into the constructor.'); - } - - /** - * The with of the render texture - * - * @member {number} - */ - this.width = width || 100; - - /** - * The height of the render texture - * - * @member {number} - */ - this.height = height || 100; - - /** - * The Resolution of the texture. - * - * @member {number} - */ - this.resolution = resolution || 1; - - /** - * The framing rectangle of the render texture - * - * @member {Rectangle} - */ - this.frame = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution); - - /** - * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering, - * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases) - * - * @member {Rectangle} - */ - this.crop = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution); - - /** - * Draw/render the given DisplayObject onto the texture. - * - * The displayObject and descendents are transformed during this operation. - * If `restoreWorldTransform` is true then the transformations will be restored before the - * method returns. Otherwise it is up to the calling code to correctly use or reset - * the transformed display objects. - * - * The display object is always rendered with a worldAlpha value of 1. - * - * @method - * @param displayObject {DisplayObject} The display object to render this texture on - * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering. - * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn - * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children - * transformations will be restored. Not restoring this information will be a little faster. - */ - this.render = null; - - /** - * The base texture object that this texture uses - * - * @member {BaseTexture} - */ - this.baseTexture = new BaseTexture(); - this.baseTexture.width = this.width * this.resolution; - this.baseTexture.height = this.height * this.resolution; - this.baseTexture._glTextures = []; - this.baseTexture.resolution = this.resolution; - - this.baseTexture.scaleMode = scaleMode || CONST.scaleModes.DEFAULT; - - this.baseTexture.hasLoaded = true; - - Texture.call(this, - this.baseTexture, - new math.Rectangle(0, 0, this.width, this.height) - ); - - /** - * The renderer this RenderTexture uses. A RenderTexture can only belong to one renderer at the moment if its webGL. - * - * @member {CanvasRenderer|WebGLRenderer} - */ - this.renderer = renderer; - - if (this.renderer.type === CONST.WEBGL_RENDERER) { - var gl = this.renderer.gl; - - this.textureBuffer = new FilterTexture(gl, this.width * this.resolution, this.height * this.resolution, this.baseTexture.scaleMode); - this.baseTexture._glTextures[gl.id] = this.textureBuffer.texture; - - this.render = this.renderWebGL; - this.projection = new math.Point(this.width*0.5, -this.height*0.5); - } - else { - this.render = this.renderCanvas; - this.textureBuffer = new CanvasBuffer(this.width* this.resolution, this.height* this.resolution); - this.baseTexture.source = this.textureBuffer.canvas; - } - - /** - * @member {boolean} - */ - this.valid = true; - - this._updateUvs(); -} - -RenderTexture.prototype = Object.create(Texture.prototype); -RenderTexture.prototype.constructor = RenderTexture; -module.exports = RenderTexture; - -/** - * Resizes the RenderTexture. - * - * @param width {number} The width to resize to. - * @param height {number} The height to resize to. - * @param updateBase {boolean} Should the baseTexture.width and height values be resized as well? - */ -RenderTexture.prototype.resize = function (width, height, updateBase) { - if (width === this.width && height === this.height) { - return; - } - - this.valid = (width > 0 && height > 0); - - this.width = this.frame.width = this.crop.width = width; - this.height = this.frame.height = this.crop.height = height; - - if (updateBase) { - this.baseTexture.width = this.width; - this.baseTexture.height = this.height; - } - - if (this.renderer.type === CONST.WEBGL_RENDERER) { - this.projection.x = this.width / 2; - this.projection.y = -this.height / 2; - } - - if (!this.valid) { - return; - } - - this.textureBuffer.resize(this.width * this.resolution, this.height * this.resolution); -}; - -/** - * Clears the RenderTexture. - * - */ -RenderTexture.prototype.clear = function () { - if (!this.valid) { - return; - } - - if (this.renderer.type === CONST.WEBGL_RENDERER) { - this.renderer.gl.bindFramebuffer(this.renderer.gl.FRAMEBUFFER, this.textureBuffer.frameBuffer); - } - - this.textureBuffer.clear(); -}; - -/** - * Internal method assigned to the `render` property if using a CanvasRenderer. - * - * @private - * @param displayObject {DisplayObject} The display object to render this texture on - * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering. - * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn - * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children - * transformations will be restored. Not restoring this information will be a little faster. - */ -RenderTexture.prototype.renderWebGL = function (displayObject, matrix, clear, restoreWorldTransform) { - if (!this.valid) { - return; - } - - if (typeof restoreWorldTransform === 'undefined') { - restoreWorldTransform = true; - } - - var tempAlpha, - tempTransform; - - if (restoreWorldTransform) { - tempAlpha = displayObject.worldAlpha; - tempTransform = displayObject.worldTransform.toArray(); - } - - //TOOD replace position with matrix.. - - //Lets create a nice matrix to apply to our display object. Frame buffers come in upside down so we need to flip the matrix - var wt = displayObject.worldTransform; - - wt.identity(); - wt.translate(0, this.projection.y * 2); - - if (matrix) { - wt.append(matrix); - } - - wt.scale(1,-1); - - // setWorld Alpha to ensure that the object is renderer at full opacity - displayObject.worldAlpha = 1; - - // Time to update all the children of the displayObject with the new matrix.. - var children = displayObject.children; - var i, j; - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - - // time for the webGL fun stuff! - var gl = this.renderer.gl; - - gl.viewport(0, 0, this.width * this.resolution, this.height * this.resolution); - - gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer ); - - if (clear) { - this.textureBuffer.clear(); - } - - this.renderer.spriteBatch.dirty = true; - - this.renderer.renderDisplayObject(displayObject, this.projection, this.textureBuffer.frameBuffer); - - this.renderer.spriteBatch.dirty = true; - - if (restoreWorldTransform) { - displayObject.worldAlpha = tempAlpha; - displayObject.worldTransform.fromArray(tempTransform); - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - } -}; - - -/** - * Internal method assigned to the `render` property if using a CanvasRenderer. - * - * @private - * @param displayObject {DisplayObject} The display object to render this texture on - * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering. - * @param [clear] {boolean} If true the texture will be cleared before the displayObject is drawn - * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children - * transformations will be restored. Not restoring this information will be a little faster. - */ -RenderTexture.prototype.renderCanvas = function (displayObject, matrix, clear, restoreWorldTransform) { - if (!this.valid) { - return; - } - - if (typeof restoreWorldTransform === 'undefined') { - restoreWorldTransform = true; - } - - var tempAlpha, - tempTransform; - - if (restoreWorldTransform) { - tempAlpha = displayObject.worldAlpha; - tempTransform = displayObject.worldTransform.toArray(); - } - - var wt = displayObject.worldTransform; - wt.identity(); - if (matrix) { - wt.append(matrix); - } - - // setWorld Alpha to ensure that the object is renderer at full opacity - displayObject.worldAlpha = 1; - - // Time to update all the children of the displayObject with the new matrix.. - var children = displayObject.children; - var i, j; - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - - if (clear) { - this.textureBuffer.clear(); - } - - var context = this.textureBuffer.context; - - var realResolution = this.renderer.resolution; - - this.renderer.resolution = this.resolution; - - this.renderer.renderDisplayObject(displayObject, context); - - this.renderer.resolution = realResolution; - - if (restoreWorldTransform) { - displayObject.worldAlpha = tempAlpha; - displayObject.worldTransform.fromArray(tempTransform); - - for (i = 0, j = children.length; i < j; ++i) { - children[i].updateTransform(); - } - } -}; - -/** - * Will return a HTML Image of the texture - * - * @return {Image} - */ -RenderTexture.prototype.getImage = function () { - var image = new Image(); - image.src = this.getBase64(); - return image; -}; - -/** - * Will return a a base64 encoded string of this texture. It works by calling RenderTexture.getCanvas and then running toDataURL on that. - * - * @return {string} A base64 encoded string of the texture. - */ -RenderTexture.prototype.getBase64 = function () { - return this.getCanvas().toDataURL(); -}; - -/** - * Creates a Canvas element, renders this RenderTexture to it and then returns it. - * - * @return {HTMLCanvasElement} A Canvas element with the texture rendered on. - */ -RenderTexture.prototype.getCanvas = function () { - if (this.renderer.type === CONST.WEBGL_RENDERER) { - var gl = this.renderer.gl; - var width = this.textureBuffer.width; - var height = this.textureBuffer.height; - - var webGLPixels = new Uint8Array(4 * width * height); - - gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer); - gl.readPixels(0, 0, width, height, gl.RGBA, gl.UNSIGNED_BYTE, webGLPixels); - gl.bindFramebuffer(gl.FRAMEBUFFER, null); - - var tempCanvas = new CanvasBuffer(width, height); - var canvasData = tempCanvas.context.getImageData(0, 0, width, height); - canvasData.data.set(webGLPixels); - - tempCanvas.context.putImageData(canvasData, 0, 0); - - return tempCanvas.canvas; - } - else { - return this.textureBuffer.canvas; - } -}; - -},{"../const":4,"../math":12,"../renderers/canvas/utils/CanvasBuffer":21,"../renderers/webgl/utils/FilterTexture":37,"./BaseTexture":42,"./Texture":44}],44:[function(require,module,exports){ -var BaseTexture = require('./BaseTexture'), - VideoBaseTexture = require('./VideoBaseTexture'), - TextureUvs = require('./TextureUvs'), - eventTarget = require('../utils/eventTarget'), - math = require('../math'), - utils = require('../utils'); - -/** - * A texture stores the information that represents an image or part of an image. It cannot be added - * to the display list directly. Instead use it as the texture for a Sprite. If no frame is provided then the whole image is used. - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param baseTexture {BaseTexture} The base texture source to create the texture from - * @param [frame] {Rectangle} The rectangle frame of the texture to show - * @param [crop] {Rectangle} The area of original texture - * @param [trim] {Rectangle} Trimmed texture rectangle - */ -function Texture(baseTexture, frame, crop, trim) { - /** - * Does this Texture have any frame data assigned to it? - * - * @member {boolean} - */ - this.noFrame = false; - - if (!frame) { - this.noFrame = true; - frame = new math.Rectangle(0, 0, 1, 1); - } - - if (baseTexture instanceof Texture) { - baseTexture = baseTexture.baseTexture; - } - - /** - * The base texture that this texture uses. - * - * @member {BaseTexture} - */ - this.baseTexture = baseTexture; - - /** - * The frame specifies the region of the base texture that this texture uses - * - * @member {Rectangle} - * @private - */ - this._frame = frame; - - /** - * The texture trim data. - * - * @member {Rectangle} - */ - this.trim = trim; - - /** - * This will let the renderer know if the texture is valid. If it's not then it cannot be rendered. - * - * @member {boolean} - */ - this.valid = false; - - /** - * This will let a renderer know that a texture has been updated (used mainly for webGL uv updates) - * - * @member {boolean} - */ - this.requiresUpdate = false; - - /** - * The WebGL UV data cache. - * - * @member {object} - * @private - */ - this._uvs = null; - - /** - * The width of the Texture in pixels. - * - * @member {number} - */ - this.width = 0; - - /** - * The height of the Texture in pixels. - * - * @member {number} - */ - this.height = 0; - - /** - * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering, - * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases) - * - * @member {Rectangle} - */ - this.crop = crop || new math.Rectangle(0, 0, 1, 1); - - if (baseTexture.hasLoaded) { - if (this.noFrame) { - frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height); - } - this.frame = frame; - } - else { - baseTexture.addEventListener('loaded', this.onBaseTextureLoaded.bind(this)); - } -} - -Texture.prototype.constructor = Texture; -module.exports = Texture; - -eventTarget.mixin(Texture.prototype); - -Object.defineProperties(Texture.prototype, { - needsUpdate: { - get: function () { - return this.baseTexture.needsUpdate; - }, - set: function (val) { - this.baseTexture.needsUpdate = val; - } - }, - - frame: { - get: function () { - return this._frame; - }, - set: function (frame) { - this._frame = frame; - - this.noFrame = false; - - this.width = frame.width; - this.height = frame.height; - - this.crop.x = frame.x; - this.crop.y = frame.y; - this.crop.width = frame.width; - this.crop.height = frame.height; - - if (!this.trim && (frame.x + frame.width > this.baseTexture.width || frame.y + frame.height > this.baseTexture.height)) { - throw new Error('Texture Error: frame does not fit inside the base Texture dimensions ' + this); - } - - this.valid = frame && frame.width && frame.height && this.baseTexture.source && this.baseTexture.hasLoaded; - - if (this.trim) { - this.width = this.trim.width; - this.height = this.trim.height; - this._frame.width = this.trim.width; - this._frame.height = this.trim.height; - } - - if (this.valid) { - this._updateUvs(); - } - } - } -}); - -/** - * Called when the base texture is loaded - * - * @private - */ -Texture.prototype.onBaseTextureLoaded = function () { - var baseTexture = this.baseTexture; - baseTexture.removeEventListener('loaded', this.onLoaded); - - if (this.noFrame) { - this.frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height); - } - - this.dispatchEvent( { type: 'update', content: this } ); -}; - -/** - * Destroys this texture - * - * @param destroyBase {boolean} Whether to destroy the base texture as well - */ -Texture.prototype.destroy = function (destroyBase) { - if (destroyBase) { - this.baseTexture.destroy(); - } - - this.valid = false; -}; - -/** - * Updates the internal WebGL UV cache. - * - * @private - */ -Texture.prototype._updateUvs = function () { - if (!this._uvs) { - this._uvs = new TextureUvs(); - } - - var frame = this.crop; - var tw = this.baseTexture.width; - var th = this.baseTexture.height; - - this._uvs.x0 = frame.x / tw; - this._uvs.y0 = frame.y / th; - - this._uvs.x1 = (frame.x + frame.width) / tw; - this._uvs.y1 = frame.y / th; - - this._uvs.x2 = (frame.x + frame.width) / tw; - this._uvs.y2 = (frame.y + frame.height) / th; - - this._uvs.x3 = frame.x / tw; - this._uvs.y3 = (frame.y + frame.height) / th; -}; - -/** - * Helper function that creates a Texture object from the given image url. - * If the image is not in the texture cache it will be created and loaded. - * - * @static - * @param imageUrl {string} The image url of the texture - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return Texture - */ -Texture.fromImage = function (imageUrl, crossorigin, scaleMode) { - var texture = utils.TextureCache[imageUrl]; - - if (!texture) { - texture = new Texture(BaseTexture.fromImage(imageUrl, crossorigin, scaleMode)); - utils.TextureCache[imageUrl] = texture; - } - - return texture; -}; - -/** - * Helper function that returns a Texture objected based on the given frame id. - * If the frame id is not in the texture cache an error will be thrown. - * - * @static - * @param frameId {string} The frame id of the texture - * @return Texture - */ -Texture.fromFrame = function (frameId) { - var texture = utils.TextureCache[frameId]; - if (!texture) { - throw new Error('The frameId "' + frameId + '" does not exist in the texture cache '); - } - return texture; -}; - -/** - * Helper function that creates a new Texture based on the given canvas element. - * - * @static - * @param canvas {Canvas} The canvas element source of the texture - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return {Texture} - */ -Texture.fromCanvas = function (canvas, scaleMode) { - return new Texture(BaseTexture.fromCanvas(canvas, scaleMode)); -}; - -/** - * Helper function that creates a new Texture based on the given video element. - * - * @static - * @param video {HTMLVideoElement} - * @param scaleMode {number} See {{#crossLink "PIXI/scaleModes:property"}}scaleModes{{/crossLink}} for possible values - * @return {Texture} A Texture - */ -Texture.fromVideo = function (video, scaleMode) { - return new Texture(VideoBaseTexture.baseTextureFromVideo(video, scaleMode)); -}; - -/** - * Adds a texture to the global utils.TextureCache. This cache is shared across the whole PIXI object. - * - * @static - * @param texture {Texture} The Texture to add to the cache. - * @param id {string} The id that the texture will be stored against. - */ -Texture.addTextureToCache = function (texture, id) { - utils.TextureCache[id] = texture; -}; - -/** - * Remove a texture from the global utils.TextureCache. - * - * @static - * @param id {string} The id of the texture to be removed - * @return {Texture} The texture that was removed - */ -Texture.removeTextureFromCache = function (id) { - var texture = utils.TextureCache[id]; - - delete utils.TextureCache[id]; - delete utils.BaseTextureCache[id]; - - return texture; -}; - -Texture.emptyTexture = new Texture(new BaseTexture()); - -},{"../math":12,"../utils":50,"../utils/eventTarget":49,"./BaseTexture":42,"./TextureUvs":45,"./VideoBaseTexture":46}],45:[function(require,module,exports){ -function TextureUvs() { - this.x0 = 0; - this.y0 = 0; - - this.x1 = 0; - this.y1 = 0; - - this.x2 = 0; - this.y2 = 0; - - this.x3 = 0; - this.y3 = 0; -} - -module.exports = TextureUvs; - -},{}],46:[function(require,module,exports){ -var BaseTexture = require('./BaseTexture'), - utils = require('../utils'); - -/** - * A texture of a [playing] Video. - * - * See the ["deus" demo](http://www.goodboydigital.com/pixijs/examples/deus/). - * - * @class - * @extends BaseTexture - * @namespace PIXI - * @param source {HTMLVideoElement} - * @param [scaleMode] {number} See {@link scaleModes} for possible values - */ -function VideoBaseTexture(source, scaleMode) { - if (!source){ - throw new Error('No video source element specified.'); - } - - // hook in here to check if video is already available. - // BaseTexture looks for a source.complete boolean, plus width & height. - - if ((source.readyState === source.HAVE_ENOUGH_DATA || source.readyState === source.HAVE_FUTURE_DATA) && source.width && source.height) { - source.complete = true; - } - - BaseTexture.call(this, source, scaleMode); - - this.autoUpdate = false; - - this._boundOnUpdate = this._onUpdate.bind(this); - this._boundOnCanPlay = this._onCanPlay.bind(this); - - if (!source.complete) { - source.addEventListener('canplay', this._boundOnCanPlay); - source.addEventListener('canplaythrough', this._boundOnCanPlay); - - // started playing.. - source.addEventListener('play', this._onPlayStart.bind(this)); - source.addEventListener('pause', this._onPlayStop.bind(this)); - } - - this.__loaded = false; -} - -VideoBaseTexture.prototype = Object.create(BaseTexture.prototype); -VideoBaseTexture.prototype.constructor = VideoBaseTexture; -module.exports = VideoBaseTexture; - -VideoBaseTexture.prototype._onUpdate = function () { - if (this.autoUpdate) { - window.requestAnimationFrame(this._boundOnUpdate); - this.needsUpdate = true; - } -}; - -VideoBaseTexture.prototype._onPlayStart = function () { - if (!this.autoUpdate) { - window.requestAnimationFrame(this._boundOnUpdate); - this.autoUpdate = true; - } -}; - -VideoBaseTexture.prototype._onPlayStop = function () { - this.autoUpdate = false; -}; - -VideoBaseTexture.prototype._onCanPlay = function () { - this.hasLoaded = true; - - if (this.source) { - this.source.removeEventListener('canplay', this._boundOnCanPlay); - this.source.removeEventListener('canplaythrough', this._boundOnCanPlay); - - this.width = this.source.videoWidth; - this.height = this.source.videoHeight; - - this.source.play(); - - // prevent multiple loaded dispatches.. - if(!this.__loaded){ - this.__loaded = true; - this.emit('loaded', this); - } - } -}; - -VideoBaseTexture.prototype.destroy = function () { - if (this.source && this.source._pixiId) { - utils.BaseTextureCache[ this.source._pixiId ] = null; - delete utils.BaseTextureCache[ this.source._pixiId ]; - - this.source._pixiId = null; - delete this.source._pixiId; - } - - BaseTexture.prototype.destroy.call(this); -}; - -/** - * Mimic Pixi BaseTexture.from.... method. - * - * @static - * @param video {HTMLVideoElement} - * @param scaleMode {number} See {@link scaleModes} for possible values - * @return {VideoBaseTexture} - */ -VideoBaseTexture.fromVideo = function (video, scaleMode) { - if (!video._pixiId) { - video._pixiId = 'video_' + utils.uuid(); - } - - var baseTexture = utils.BaseTextureCache[video._pixiId]; - - if (!baseTexture) { - baseTexture = new VideoBaseTexture(video, scaleMode); - utils.BaseTextureCache[ video._pixiId ] = baseTexture; - } - - return baseTexture; -}; - -/** - * Mimic Pixi BaseTexture.from.... method. - * - * This can be used in a couple ways, such as: - * - * ```js - * var texture = PIXI.VideoBaseTexture.fromUrl('http://mydomain.com/video.mp4'); - * - * var texture = PIXI.VideoBaseTexture.fromUrl({ src: 'http://mydomain.com/video.mp4', mime: 'video/mp4' }); - * - * var texture = PIXI.VideoBaseTexture.fromUrls(['/video.webm', '/video.mp4']); - * - * var texture = PIXI.VideoBaseTexture.fromUrls([ - * { src: '/video.webm', mime: 'video/webm' }, - * { src: '/video.mp4', mime: 'video/mp4' } - * ]); - * ``` - * - * @alias fromUrls - * @static - * @param videoSrc {string|object|string[]|object[]} The URL(s) for the video. - * @param [videoSrc.src] {string} One of the source urls for the video - * @param [videoSrc.mime] {string} The mimetype of the video (e.g. 'video/mp4'). If not specified - * the url's extension will be used as the second part of the mime type. - * @param scaleMode {number} See {@link scaleModes} for possible values - * @return {VideoBaseTexture} - */ -VideoBaseTexture.fromUrl = function (videoSrc, scaleMode) { - var video = document.createElement('video'); - - // array of objects or strings - if (Array.isArray(videoSrc)) { - for (var i = 0; i < videoSrc.length; ++i) { - video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime)); - } - } - // single object or string - else { - video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime)); - } - - video.load(); - video.play(); - - return VideoBaseTexture.textureFromVideo(video, scaleMode); -}; - -VideoBaseTexture.fromUrls = VideoBaseTexture.fromUrl; - -function createSource(path, type) { - if (!type) { - type = 'video/' + path.substr(path.lastIndexOf('.') + 1); - } - - var source = document.createElement('source'); - - source.src = path; - source.type = type; - - return source; -} - -},{"../utils":50,"./BaseTexture":42}],47:[function(require,module,exports){ -/** - * Creates an homogenous object for tracking events so users can know what to expect. - * - * @class - * @namespace PIXI - * @param target {object} The target object that the event is called on - * @param name {string} The string name of the event that was triggered - * @param data {object} Arbitrary event data to pass along - */ -function EventData(target, name, data) { - // for duck typing in the ".on()" function - this.__isEventObject = true; - - /** - * Tracks the state of bubbling propagation. Do not - * set this directly, instead use `event.stopPropagation()` - * - * @member {boolean} - * @private - * @readonly - */ - this.stopped = false; - - /** - * Tracks the state of sibling listener propagation. Do not - * set this directly, instead use `event.stopImmediatePropagation()` - * - * @member {boolean} - * @private - * @readonly - */ - this.stoppedImmediate = false; - - /** - * The original target the event triggered on. - * - * @member {object} - * @readonly - */ - this.target = target; - - /** - * The string name of the event that this represents. - * - * @member {string} - * @readonly - */ - this.type = name; - - /** - * The data that was passed in with this event. - * - * @member {object} - * @readonly - */ - this.data = data; - - /** - * The timestamp when the event occurred. - * - * @member {number} - * @readonly - */ - this.timeStamp = Date.now(); -} - -EventData.prototype.constructor = EventData; -module.exports = EventData; - -/** - * Stops the propagation of events up the scene graph (prevents bubbling). - * - */ -EventData.prototype.stopPropagation = function stopPropagation() { - this.stopped = true; -}; - -/** - * Stops the propagation of events to sibling listeners (no longer calls any listeners). - * - */ -EventData.prototype.stopImmediatePropagation = function stopImmediatePropagation() { - this.stoppedImmediate = true; -}; - -},{}],48:[function(require,module,exports){ -//TODO: Have Graphics use https://github.com/mattdesl/shape2d -// and https://github.com/mattdesl/shape2d-triangulate instead of custom code. - -/* - PolyK library - url: http://polyk.ivank.net - Released under MIT licence. - - Copyright (c) 2012 Ivan Kuckir - - Permission is hereby granted, free of charge, to any person - obtaining a copy of this software and associated documentation - files (the "Software"), to deal in the Software without - restriction, including without limitation the rights to use, - copy, modify, merge, publish, distribute, sublicense, and/or sell - copies of the Software, and to permit persons to whom the - Software is furnished to do so, subject to the following - conditions: - - The above copyright notice and this permission notice shall be - included in all copies or substantial portions of the Software. - - THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, - EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES - OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND - NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT - HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, - WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING - FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR - OTHER DEALINGS IN THE SOFTWARE. - - This is an amazing lib! - - Slightly modified by Mat Groves (matgroves.com); -*/ - -/** - * Based on the Polyk library http://polyk.ivank.net released under MIT licence. - * This is an amazing lib! - * Slightly modified by Mat Groves (matgroves.com); - * - * @namespace PIXI - */ -var PolyK = module.exports = {}; - -/** - * Triangulates shapes for webGL graphic fills. - * - */ -PolyK.Triangulate = function (p) { - var sign = true; - - var n = p.length >> 1; - if (n < 3) return []; - - var tgs = []; - var avl = []; - for (var i = 0; i < n; i++) avl.push(i); - - i = 0; - var al = n; - while(al > 3) { - var i0 = avl[(i+0)%al]; - var i1 = avl[(i+1)%al]; - var i2 = avl[(i+2)%al]; - - var ax = p[2*i0], ay = p[2*i0+1]; - var bx = p[2*i1], by = p[2*i1+1]; - var cx = p[2*i2], cy = p[2*i2+1]; - - var earFound = false; - if (PolyK._convex(ax, ay, bx, by, cx, cy, sign)) { - earFound = true; - for (var j = 0; j < al; j++) { - var vi = avl[j]; - if (vi === i0 || vi === i1 || vi === i2) continue; - - if (PolyK._PointInTriangle(p[2*vi], p[2*vi+1], ax, ay, bx, by, cx, cy)) { - earFound = false; - break; - } - } - } - - if (earFound) { - tgs.push(i0, i1, i2); - avl.splice((i+1)%al, 1); - al--; - i = 0; - } - else if (i++ > 3*al) { - // need to flip flip reverse it! - // reset! - if (sign) { - tgs = []; - avl = []; - for (i = 0; i < n; i++) avl.push(i); - - i = 0; - al = n; - - sign = false; - } - else { - // window.console.log("PIXI Warning: shape too complex to fill"); - return null; - } - } - } - - tgs.push(avl[0], avl[1], avl[2]); - return tgs; -}; - -/** - * Checks whether a point is within a triangle - * - * @param px {number} x coordinate of the point to test - * @param py {number} y coordinate of the point to test - * @param ax {number} x coordinate of the a point of the triangle - * @param ay {number} y coordinate of the a point of the triangle - * @param bx {number} x coordinate of the b point of the triangle - * @param by {number} y coordinate of the b point of the triangle - * @param cx {number} x coordinate of the c point of the triangle - * @param cy {number} y coordinate of the c point of the triangle - * @private - * @return {boolean} - */ -PolyK._PointInTriangle = function (px, py, ax, ay, bx, by, cx, cy) { - var v0x = cx-ax; - var v0y = cy-ay; - var v1x = bx-ax; - var v1y = by-ay; - var v2x = px-ax; - var v2y = py-ay; - - var dot00 = v0x*v0x+v0y*v0y; - var dot01 = v0x*v1x+v0y*v1y; - var dot02 = v0x*v2x+v0y*v2y; - var dot11 = v1x*v1x+v1y*v1y; - var dot12 = v1x*v2x+v1y*v2y; - - var invDenom = 1 / (dot00 * dot11 - dot01 * dot01); - var u = (dot11 * dot02 - dot01 * dot12) * invDenom; - var v = (dot00 * dot12 - dot01 * dot02) * invDenom; - - // Check if point is in triangle - return (u >= 0) && (v >= 0) && (u + v < 1); -}; - -/** - * Checks whether a shape is convex - * - * @private - * @return {boolean} - */ -PolyK._convex = function (ax, ay, bx, by, cx, cy, sign) { - return ((ay-by)*(cx-bx) + (bx-ax)*(cy-by) >= 0) === sign; -}; - -},{}],49:[function(require,module,exports){ -var EventData = require('./EventData'); - -/** - * Mixins event emitter functionality to an object. - * - * @mixin - * @namespace PIXI - * @example - * function MyEmitter() {} - * - * eventTarget.mixin(MyEmitter.prototype); - * - * var em = new MyEmitter(); - * em.emit('eventName', 'some data', 'some more data', {}, null, ...); - */ -function eventTarget(obj) { - /** - * Return a list of assigned event listeners. - * - * @param eventName {string} The events that should be listed. - * @return {Array} An array of listener functions - */ - obj.listeners = function listeners(eventName) { - this._listeners = this._listeners || {}; - - return this._listeners[eventName] ? this._listeners[eventName].slice() : []; - }; - - /** - * Emit an event to all registered event listeners. - * - * @alias dispatchEvent - * @param eventName {string} The name of the event. - * @return {boolean} Indication if we've emitted an event. - */ - obj.emit = obj.dispatchEvent = function emit(eventName, data) { - this._listeners = this._listeners || {}; - - // fast return when there are no listeners - if (!this._listeners[eventName]) { - return; - } - - //backwards compat with old method ".emit({ type: 'something' })" - if (typeof eventName === 'object') { - data = eventName; - eventName = eventName.type; - } - - //ensure we are using a real pixi event - if (!data || data.__isEventObject !== true) { - data = new EventData(this, eventName, data); - } - - //iterate the listeners - var listeners = this._listeners[eventName].slice(0), - length = listeners.length, - fn = listeners[0], - i; - - for (i = 0; i < length; fn = listeners[++i]) { - //call the event listener - fn.call(this, data); - - //if "stopImmediatePropagation" is called, stop calling sibling events - if (data.stoppedImmediate) { - return this; - } - } - - //if "stopPropagation" is called then don't bubble the event - if (data.stopped) { - return this; - } - - //bubble this event up the scene graph - if (this.parent && this.parent.emit) { - this.parent.emit.call(this.parent, eventName, data); - } - - return this; - }; - - /** - * Register a new EventListener for the given event. - * - * @alias addEventListener - * @param eventName {string} Name of the event. - * @param callback {Functon} fn Callback function. - */ - obj.on = obj.addEventListener = function on(eventName, fn) { - this._listeners = this._listeners || {}; - - (this._listeners[eventName] = this._listeners[eventName] || []) - .push(fn); - - return this; - }; - - /** - * Add an EventListener that's only called once. - * - * @param eventName {string} Name of the event. - * @param callback {Function} Callback function. - */ - obj.once = function once(eventName, fn) { - this._listeners = this._listeners || {}; - - var self = this; - function onceHandlerWrapper() { - fn.apply(self.off(eventName, onceHandlerWrapper), arguments); - } - onceHandlerWrapper._originalHandler = fn; - - return this.on(eventName, onceHandlerWrapper); - }; - - /** - * Remove event listeners. - * - * @alias removeEventListener - * @param eventName {string} The event we want to remove. - * @param callback {Function} The listener that we need to find. - */ - obj.off = obj.removeEventListener = function off(eventName, fn) { - this._listeners = this._listeners || {}; - - if (!this._listeners[eventName]) { - return this; - } - - var list = this._listeners[eventName], - i = fn ? list.length : 0; - - while(i-- > 0) { - if (list[i] === fn || list[i]._originalHandler === fn) { - list.splice(i, 1); - } - } - - if (list.length === 0) { - delete this._listeners[eventName]; - } - - return this; - }; - - /** - * Remove all listeners or only the listeners for the specified event. - * - * @param eventName {string} The event you want to remove all listeners for. - */ - obj.removeAllListeners = function removeAllListeners(eventName) { - this._listeners = this._listeners || {}; - - if (!this._listeners[eventName]) { - return this; - } - - delete this._listeners[eventName]; - - return this; - }; -} - -module.exports = { - /** - * Mixes in the properties of the eventTarget prototype onto another object - * - * @param object {object} The obj to mix into - */ - mixin: function mixin(obj) { - eventTarget(obj); - } -}; - -},{"./EventData":47}],50:[function(require,module,exports){ -var CONST = require('../const'); - -/** - * @namespace PIXI - */ -var utils = module.exports = { - _uid: 0, - _saidHello: false, - - PolyK: require('./PolyK'), - EventData: require('./EventData'), - eventTarget: require('./eventTarget'), - - /** - * Gets the next uuid - * - * @return {number} The next uuid to use. - */ - uuid: function () { - return ++utils._uid; - }, - - /** - * Converts a hex color number to an [R, G, B] array - * - * @param hex {number} - * @return {number[]} An array representing the [R, G, B] of the color. - */ - hex2rgb: function (hex, out) { - out = out || []; - - out[0] = (hex >> 16 & 0xFF) / 255; - out[1] = (hex >> 8 & 0xFF) / 255; - out[2] = (hex & 0xFF) / 255; - - return out; - }, - - /** - * Converts a hex color number to a string. - * - * @param hex {number} - * @return {string} The string color. - */ - hex2string: function (hex) { - hex = hex.toString(16); - hex = '000000'.substr(0, 6 - hex.length) + hex; - - return '#' + hex; - }, - - /** - * Converts a color as an [R, G, B] array to a hex number - * - * @param rgb {number[]} - * @return {number} The color number - */ - rgb2hex: function (rgb) { - return ((rgb[0]*255 << 16) + (rgb[1]*255 << 8) + rgb[2]*255); - }, - - /** - * Checks whether the Canvas BlendModes are supported by the current browser - * - * @return {boolean} whether they are supported - */ - canUseNewCanvasBlendModes: function () { - if (typeof document === 'undefined') { - return false; - } - - var canvas = document.createElement('canvas'), - context = canvas.getContext('2d'); - - canvas.width = 1; - canvas.height = 1; - - context.fillStyle = '#000'; - context.fillRect(0, 0, 1, 1); - - context.globalCompositeOperation = 'multiply'; - - context.fillStyle = '#fff'; - context.fillRect(0, 0, 1, 1); - - return context.getImageData(0,0,1,1).data[0] === 0; - }, - - /** - * Given a number, this function returns the closest number that is a power of two - * this function is taken from Starling Framework as its pretty neat ;) - * - * @param number {number} - * @return {number} the closest number that is a power of two - */ - getNextPowerOfTwo: function (number) { - // see: http://en.wikipedia.org/wiki/Power_of_two#Fast_algorithm_to_check_if_a_positive_number_is_a_power_of_two - if (number > 0 && (number & (number - 1)) === 0) { - return number; - } - else { - var result = 1; - - while (result < number) { - result <<= 1; - } - - return result; - } - }, - - /** - * checks if the given width and height make a power of two rectangle - * - * @param width {number} - * @param height {number} - * @return {boolean} - */ - isPowerOfTwo: function (width, height) { - return (width > 0 && (width & (width - 1)) === 0 && height > 0 && (height & (height - 1)) === 0); - }, - - /** - * Logs out the version and renderer information for this running instance of PIXI. - * If you don't want to see this message you can set `PIXI.utils._saidHello = true;` - * so the library thinks it already said it. Keep in mind that doing that will forever - * makes you a jerk face. - * - * @param {string} type - The string renderer type to log. - * @constant - * @static - */ - sayHello: function (type) { - if (utils._saidHello) { - return; - } - - if (navigator.userAgent.toLowerCase().indexOf('chrome') > -1) { - var args = [ - '%c %c %c Pixi.js ' + CONST.VERSION + ' - ' + type + ' %c ' + ' %c ' + ' http://www.pixijs.com/ %c %c ♥%c♥%c♥ ', - 'background: #ff66a5', - 'background: #ff66a5', - 'color: #ff66a5; background: #030307;', - 'background: #ff66a5', - 'background: #ffc3dc', - 'background: #ff66a5', - 'color: #ff2424; background: #fff', - 'color: #ff2424; background: #fff', - 'color: #ff2424; background: #fff' - ]; - - console.log.apply(console, args); //jshint ignore:line - } - else if (window.console) { - console.log('Pixi.js ' + CONST.VERSION + ' - ' + type + ' - http://www.pixijs.com/'); //jshint ignore:line - } - - utils._saidHello = true; - }, - - /** - * A wrapper for ajax requests to be handled cross browser - * - * TODO: Replace this wil superagent - * - * @class - * @namespace PIXI - */ - AjaxRequest: function () { - var activexmodes = ['Msxml2.XMLHTTP.6.0', 'Msxml2.XMLHTTP.3.0', 'Microsoft.XMLHTTP']; //activeX versions to check for in IE - - if (window.ActiveXObject) { //Test for support for ActiveXObject in IE first (as XMLHttpRequest in IE7 is broken) - for (var i=0; i= this.textures.length) { - this.gotoAndStop(this.textures.length - 1); - - if (this.onComplete) { - this.onComplete(); - } - } -}; - -/** - * A short hand way of creating a movieclip from an array of frame ids - * - * @static - * @param frames {string[]} the array of frames ids the movieclip will use as its texture frames - */ -MovieClip.fromFrames = function (frames) { - var textures = []; - - for (var i = 0; i < frames.length; ++i) { - textures.push(new core.Texture.fromFrame(frames[i])); - } - - return new MovieClip(textures); -}; - -/** - * A short hand way of creating a movieclip from an array of image ids - * - * @static - * @param images {string[]} the array of image urls the movieclip will use as its texture frames - */ -MovieClip.fromImages = function (images) { - var textures = []; - - for (var i = 0; i < images.length; ++i) { - textures.push(new core.Texture.fromImage(images[i])); - } - - return new MovieClip(textures); -}; - -},{"../core":9}],52:[function(require,module,exports){ -var Strip = require('./Strip'); - -/** - * - * @class - * @namespace PIXI - * @extends Strip - * @param {Texture} texture - The texture to use on the rope. - * @param {Array} points - An array of {Point}. - * - */ -function Rope(texture, points) { - Strip.call(this, texture); - this.points = points; - - this.vertices = new Float32Array(points.length * 4); - this.uvs = new Float32Array(points.length * 4); - this.colors = new Float32Array(points.length * 2); - this.indices = new Uint16Array(points.length * 2); - - this.refresh(); -} - - -// constructor -Rope.prototype = Object.create(Strip.prototype); -Rope.prototype.constructor = Rope; -module.exports = Rope; - -/** - * Refreshes - * - */ -Rope.prototype.refresh = function () { - var points = this.points; - - if (points.length < 1) { - return; - } - - var uvs = this.uvs; - - var indices = this.indices; - var colors = this.colors; - - // this.count -= 0.2; - - uvs[0] = 0; - uvs[1] = 0; - uvs[2] = 0; - uvs[3] = 1; - - colors[0] = 1; - colors[1] = 1; - - indices[0] = 0; - indices[1] = 1; - - var total = points.length, - point, index, amount; - - for (var i = 1; i < total; i++) { - point = points[i]; - index = i * 4; - // time to do some smart drawing! - amount = i / (total-1); - - if (i%2) { - uvs[index] = amount; - uvs[index+1] = 0; - - uvs[index+2] = amount; - uvs[index+3] = 1; - } - else { - uvs[index] = amount; - uvs[index+1] = 0; - - uvs[index+2] = amount; - uvs[index+3] = 1; - } - - index = i * 2; - colors[index] = 1; - colors[index+1] = 1; - - index = i * 2; - indices[index] = index; - indices[index + 1] = index + 1; - } -}; - -/* - * Updates the object transform for rendering - * - * @private - */ -Rope.prototype.updateTransform = function () { - var points = this.points; - - if (points.length < 1) { - return; - } - - var lastPoint = points[0]; - var nextPoint; - var perpX = 0; - var perpY = 0; - - // this.count -= 0.2; - - var vertices = this.vertices; - var total = points.length, - point, index, ratio, perpLength, num; - - for (var i = 0; i < total; i++) { - point = points[i]; - index = i * 4; - - if (i < points.length-1) { - nextPoint = points[i+1]; - } - else { - nextPoint = point; - } - - perpY = -(nextPoint.x - lastPoint.x); - perpX = nextPoint.y - lastPoint.y; - - ratio = (1 - (i / (total-1))) * 10; - - if (ratio > 1) { - ratio = 1; - } - - perpLength = Math.sqrt(perpX * perpX + perpY * perpY); - num = this.texture.height / 2; //(20 + Math.abs(Math.sin((i + this.count) * 0.3) * 50) )* ratio; - perpX /= perpLength; - perpY /= perpLength; - - perpX *= num; - perpY *= num; - - vertices[index] = point.x + perpX; - vertices[index+1] = point.y + perpY; - vertices[index+2] = point.x - perpX; - vertices[index+3] = point.y - perpY; - - lastPoint = point; - } - - this.displayObjectContainerUpdateTransform(); -}; - -/** - * Sets the texture that the Rope will use - * - * @param texture {Texture} the texture that will be used - */ -Rope.prototype.setTexture = function (texture) { - // stop current texture - this.texture = texture; - //this.updateFrame = true; -}; - -},{"./Strip":53}],53:[function(require,module,exports){ -var core = require('../core'); - -/** - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - * @param texture {Texture} The texture to use - * @param width {number} the width - * @param height {number} the height - * - */ -function Strip(texture) { - core.DisplayObjectContainer.call(this); - - /** - * The texture of the strip - * - * @member {Texture} - */ - this.texture = texture; - - // set up the main bits.. - this.uvs = new Float32Array([0, 1, - 1, 1, - 1, 0, - 0, 1]); - - this.vertices = new Float32Array([0, 0, - 100, 0, - 100, 100, - 0, 100]); - - this.colors = new Float32Array([1, 1, 1, 1]); - - this.indices = new Uint16Array([0, 1, 2, 3]); - - /** - * Whether the strip is dirty or not - * - * @member {boolean} - */ - this.dirty = true; - - /** - * The blend mode to be applied to the sprite. Set to blendModes.NORMAL to remove any blend mode. - * - * @member {number} - * @default CONST.blendModes.NORMAL; - */ - this.blendMode = core.CONST.blendModes.NORMAL; - - /** - * Triangles in canvas mode are automatically antialiased, use this value to force triangles to overlap a bit with each other. - * - * @member {number} - */ - this.canvasPadding = 0; - - this.drawMode = Strip.DrawModes.TRIANGLE_STRIP; -} - -// constructor -Strip.prototype = Object.create(core.DisplayObjectContainer.prototype); -Strip.prototype.constructor = Strip; -module.exports = Strip; - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} - */ -Strip.prototype.renderWebGL = function (renderer) { - // if the sprite is not visible or the alpha is 0 then no need to render this element - if (!this.visible || this.alpha <= 0) { - return; - } - - // render triangle strip.. - - renderer.spriteBatch.stop(); - - // init! init! - if (!this._vertexBuffer) { - this._initWebGL(renderer); - } - - renderer.shaderManager.setShader(renderer.shaderManager.stripShader); - - this._renderStrip(renderer); - - ///renderer.shaderManager.activateDefaultShader(); - - renderer.spriteBatch.start(); - - //TODO check culling -}; - -Strip.prototype._initWebGL = function (renderer) { - // build the strip! - var gl = renderer.gl; - - this._vertexBuffer = gl.createBuffer(); - this._indexBuffer = gl.createBuffer(); - this._uvBuffer = gl.createBuffer(); - this._colorBuffer = gl.createBuffer(); - - gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.uvs, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ARRAY_BUFFER, this._colorBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.colors, gl.STATIC_DRAW); - - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); -}; - -Strip.prototype._renderStrip = function (renderer) { - var gl = renderer.gl; - var projection = renderer.projection, - offset = renderer.offset, - shader = renderer.shaderManager.stripShader; - - var drawMode = this.drawMode === Strip.DrawModes.TRIANGLE_STRIP ? gl.TRIANGLE_STRIP : gl.TRIANGLES; - - // gl.uniformMatrix4fv(shaderProgram.mvMatrixUniform, false, mat4Real); - - renderer.blendModeManager.setBlendMode(this.blendMode); - - - // set uniforms - gl.uniformMatrix3fv(shader.translationMatrix, false, this.worldTransform.toArray(true)); - gl.uniform2f(shader.projectionVector, projection.x, -projection.y); - gl.uniform2f(shader.offsetVector, -offset.x, -offset.y); - gl.uniform1f(shader.alpha, this.worldAlpha); - - if (!this.dirty) { - - gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer); - gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices); - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0); - - // update the uvs - gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer); - gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0); - - gl.activeTexture(gl.TEXTURE0); - - // check if a texture is dirty.. - if (this.texture.baseTexture._dirty[gl.id]) { - renderer.updateTexture(this.texture.baseTexture); - } - else { - // bind the current texture - gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]); - } - - // dont need to upload! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer); - - - } - else { - - this.dirty = false; - gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.STATIC_DRAW); - gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0); - - // update the uvs - gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer); - gl.bufferData(gl.ARRAY_BUFFER, this.uvs, gl.STATIC_DRAW); - gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0); - - gl.activeTexture(gl.TEXTURE0); - - // check if a texture is dirty.. - if (this.texture.baseTexture._dirty[gl.id]) { - renderer.updateTexture(this.texture.baseTexture); - } - else { - gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]); - } - - // dont need to upload! - gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer); - gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW); - - } - //console.log(gl.TRIANGLE_STRIP) - // - // - gl.drawElements(drawMode, this.indices.length, gl.UNSIGNED_SHORT, 0); - - -}; - -/** - * Renders the object using the Canvas renderer - * - * @param renderer {CanvasRenderer} - */ -Strip.prototype.renderCanvas = function (renderer) { - var context = renderer.context; - - var transform = this.worldTransform; - - if (renderer.roundPixels) { - context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx | 0, transform.ty | 0); - } - else { - context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx, transform.ty); - } - - if (this.drawMode === Strip.DrawModes.TRIANGLE_STRIP) { - this._renderCanvasTriangleStrip(context); - } - else { - this._renderCanvasTriangles(context); - } -}; - -Strip.prototype._renderCanvasTriangleStrip = function (context) { - // draw triangles!! - var vertices = this.vertices; - var uvs = this.uvs; - - var length = vertices.length / 2; - // this.count++; - - for (var i = 0; i < length - 2; i++) { - // draw some triangles! - var index = i * 2; - this._renderCanvasDrawTriangle(context, vertices, uvs, index, (index + 2), (index + 4)); - } -}; - -Strip.prototype._renderCanvasTriangles = function (context) { - // draw triangles!! - var vertices = this.vertices; - var uvs = this.uvs; - var indices = this.indices; - - var length = indices.length; - // this.count++; - - for (var i = 0; i < length; i += 3) { - // draw some triangles! - var index0 = indices[i] * 2, index1 = indices[i + 1] * 2, index2 = indices[i + 2] * 2; - this._renderCanvasDrawTriangle(context, vertices, uvs, index0, index1, index2); - } -}; - -Strip.prototype._renderCanvasDrawTriangle = function (context, vertices, uvs, index0, index1, index2) { - var textureSource = this.texture.baseTexture.source; - var textureWidth = this.texture.width; - var textureHeight = this.texture.height; - - var x0 = vertices[index0], x1 = vertices[index1], x2 = vertices[index2]; - var y0 = vertices[index0 + 1], y1 = vertices[index1 + 1], y2 = vertices[index2 + 1]; - - var u0 = uvs[index0] * textureWidth, u1 = uvs[index1] * textureWidth, u2 = uvs[index2] * textureWidth; - var v0 = uvs[index0 + 1] * textureHeight, v1 = uvs[index1 + 1] * textureHeight, v2 = uvs[index2 + 1] * textureHeight; - - if (this.canvasPadding > 0) { - var paddingX = this.canvasPadding / this.worldTransform.a; - var paddingY = this.canvasPadding / this.worldTransform.d; - var centerX = (x0 + x1 + x2) / 3; - var centerY = (y0 + y1 + y2) / 3; - - var normX = x0 - centerX; - var normY = y0 - centerY; - - var dist = Math.sqrt(normX * normX + normY * normY); - x0 = centerX + (normX / dist) * (dist + paddingX); - y0 = centerY + (normY / dist) * (dist + paddingY); - - // - - normX = x1 - centerX; - normY = y1 - centerY; - - dist = Math.sqrt(normX * normX + normY * normY); - x1 = centerX + (normX / dist) * (dist + paddingX); - y1 = centerY + (normY / dist) * (dist + paddingY); - - normX = x2 - centerX; - normY = y2 - centerY; - - dist = Math.sqrt(normX * normX + normY * normY); - x2 = centerX + (normX / dist) * (dist + paddingX); - y2 = centerY + (normY / dist) * (dist + paddingY); - } - - context.save(); - context.beginPath(); - - - context.moveTo(x0, y0); - context.lineTo(x1, y1); - context.lineTo(x2, y2); - - context.closePath(); - - context.clip(); - - // Compute matrix transform - var delta = (u0 * v1) + (v0 * u2) + (u1 * v2) - (v1 * u2) - (v0 * u1) - (u0 * v2); - var deltaA = (x0 * v1) + (v0 * x2) + (x1 * v2) - (v1 * x2) - (v0 * x1) - (x0 * v2); - var deltaB = (u0 * x1) + (x0 * u2) + (u1 * x2) - (x1 * u2) - (x0 * u1) - (u0 * x2); - var deltaC = (u0 * v1 * x2) + (v0 * x1 * u2) + (x0 * u1 * v2) - (x0 * v1 * u2) - (v0 * u1 * x2) - (u0 * x1 * v2); - var deltaD = (y0 * v1) + (v0 * y2) + (y1 * v2) - (v1 * y2) - (v0 * y1) - (y0 * v2); - var deltaE = (u0 * y1) + (y0 * u2) + (u1 * y2) - (y1 * u2) - (y0 * u1) - (u0 * y2); - var deltaF = (u0 * v1 * y2) + (v0 * y1 * u2) + (y0 * u1 * v2) - (y0 * v1 * u2) - (v0 * u1 * y2) - (u0 * y1 * v2); - - context.transform(deltaA / delta, deltaD / delta, - deltaB / delta, deltaE / delta, - deltaC / delta, deltaF / delta); - - context.drawImage(textureSource, 0, 0); - context.restore(); -}; - - - -/** - * Renders a flat strip - * - * @param strip {Strip} The Strip to render - * @private - */ -Strip.prototype.renderStripFlat = function (strip) { - var context = this.context; - var vertices = strip.vertices; - - var length = vertices.length/2; - // this.count++; - - context.beginPath(); - for (var i=1; i < length-2; i++) { - // draw some triangles! - var index = i*2; - - var x0 = vertices[index], x1 = vertices[index+2], x2 = vertices[index+4]; - var y0 = vertices[index+1], y1 = vertices[index+3], y2 = vertices[index+5]; - - context.moveTo(x0, y0); - context.lineTo(x1, y1); - context.lineTo(x2, y2); - } - - context.fillStyle = '#FF0000'; - context.fill(); - context.closePath(); -}; - -/* -Strip.prototype.setTexture = function (texture) { - //TODO SET THE TEXTURES - //TODO VISIBILITY - - // stop current texture - this.texture = texture; - this.width = texture.frame.width; - this.height = texture.frame.height; - this.updateFrame = true; -}; - */ - -/** - * When the texture is updated, this event will fire to update the scale and frame - * - * @param event - * @private - */ - -Strip.prototype.onTextureUpdate = function () { - this.updateFrame = true; -}; - -/** - * Returns the bounds of the mesh as a rectangle. The bounds calculation takes the worldTransform into account. - * - * @param matrix {Matrix} the transformation matrix of the sprite - * @return {Rectangle} the framing rectangle - */ -Strip.prototype.getBounds = function (matrix) { - var worldTransform = matrix || this.worldTransform; - - var a = worldTransform.a; - var b = worldTransform.b; - var c = worldTransform.c; - var d = worldTransform.d; - var tx = worldTransform.tx; - var ty = worldTransform.ty; - - var maxX = -Infinity; - var maxY = -Infinity; - - var minX = Infinity; - var minY = Infinity; - - var vertices = this.vertices; - for (var i = 0, n = vertices.length; i < n; i += 2) { - var rawX = vertices[i], rawY = vertices[i + 1]; - var x = (a * rawX) + (c * rawY) + tx; - var y = (d * rawY) + (b * rawX) + ty; - - minX = x < minX ? x : minX; - minY = y < minY ? y : minY; - - maxX = x > maxX ? x : maxX; - maxY = y > maxY ? y : maxY; - } - - if (minX === -Infinity || maxY === Infinity) { - return core.math.Rectangle.EMPTY; - } - - var bounds = this._bounds; - - bounds.x = minX; - bounds.width = maxX - minX; - - bounds.y = minY; - bounds.height = maxY - minY; - - // store a reference so that if this function gets called again in the render cycle we do not have to recalculate - this._currentBounds = bounds; - - return bounds; -}; - -/** - * Different drawing buffer modes supported - * - * @static - * @constant - * @property {object} DrawModes - * @property {number} DrawModes.TRIANGLE_STRIP - * @property {number} DrawModes.TRIANGLES - */ -Strip.DrawModes = { - TRIANGLE_STRIP: 0, - TRIANGLES: 1 -}; - -},{"../core":9}],54:[function(require,module,exports){ -var core = require('../core'); - -/** - * A tiling sprite is a fast way of rendering a tiling image - * - * @class - * @extends Sprite - * @namespace PIXI - * @param texture {Texture} the texture of the tiling sprite - * @param width {number} the width of the tiling sprite - * @param height {number} the height of the tiling sprite - */ -function TilingSprite(texture, width, height) { - core.Sprite.call( this, texture); - - /** - * The with of the tiling sprite - * - * @member {number} - */ - this._width = width || 100; - - /** - * The height of the tiling sprite - * - * @member {number} - */ - this._height = height || 100; - - /** - * The scaling of the image that is being tiled - * - * @member {Point} - */ - this.tileScale = new core.math.Point(1,1); - - /** - * A point that represents the scale of the texture object - * - * @member {Point} - */ - this.tileScaleOffset = new core.math.Point(1,1); - - /** - * The offset position of the image that is being tiled - * - * @member {Point} - */ - this.tilePosition = new core.math.Point(0,0); - - /** - * Whether this sprite is renderable or not - * - * @member {boolean} - * @default true - */ - this.renderable = true; - - /** - * The tint applied to the sprite. This is a hex value - * - * @member {number} - * @default 0xFFFFFF - */ - this.tint = 0xFFFFFF; - - /** - * The blend mode to be applied to the sprite - * - * @member {number} - * @default blendModes.NORMAL; - */ - this.blendMode = core.CONST.blendModes.NORMAL; -} - -TilingSprite.prototype = Object.create(core.Sprite.prototype); -TilingSprite.prototype.constructor = TilingSprite; -module.exports = TilingSprite; - - -Object.defineProperties(TilingSprite.prototype, { - /** - * The width of the sprite, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof TilingSprite# - */ - width: { - get: function () { - return this._width; - }, - set: function (value) { - this._width = value; - } - }, - - /** - * The height of the TilingSprite, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof TilingSprite# - */ - height: { - get: function () { - return this._height; - }, - set: function (value) { - this._height = value; - } - }, - - texture: { - get: function () { - return this._texture; - }, - set: function (value) { - if (this._texture === value) { - return; - } - - this._texture = value; - this.refreshTexture = true; - this.cachedTint = 0xFFFFFF; - } - } -}); - -/** - * Renders the object using the WebGL renderer - * - * @param renderer {WebGLRenderer} - */ -TilingSprite.prototype.renderWebGL = function (renderer) { - if (!this.visible || this.alpha <= 0) { - return; - } - - var i, j; - - if (this._mask) { - renderer.spriteBatch.stop(); - renderer.maskManager.pushMask(this.mask, renderer); - renderer.spriteBatch.start(); - } - - if (this._filters) { - renderer.spriteBatch.flush(); - renderer.filterManager.pushFilter(this._filterBlock); - } - - - - if (!this.tilingTexture || this.refreshTexture) { - this.generateTilingTexture(true); - - if (this.tilingTexture && this.tilingTexture.needsUpdate) { - //TODO - tweaking - renderer.updateTexture(this.tilingTexture.baseTexture); - this.tilingTexture.needsUpdate = false; - // this.tilingTexture._uvs = null; - } - } - else { - renderer.spriteBatch.renderTilingSprite(this); - } - // simple render children! - for (i=0,j=this.children.length; i maxX ? x2 : maxX; - maxX = x3 > maxX ? x3 : maxX; - maxX = x4 > maxX ? x4 : maxX; - - maxY = y1; - maxY = y2 > maxY ? y2 : maxY; - maxY = y3 > maxY ? y3 : maxY; - maxY = y4 > maxY ? y4 : maxY; - - var bounds = this._bounds; - - bounds.x = minX; - bounds.width = maxX - minX; - - bounds.y = minY; - bounds.height = maxY - minY; - - // store a reference so that if this function gets called again in the render cycle we do not have to recalculate - this._currentBounds = bounds; - - return bounds; -}; - -/** - * When the texture is updated, this event will fire to update the scale and frame - * - * @param event - * @private - */ -TilingSprite.prototype.onTextureUpdate = function () { - // overriding the sprite version of this! -}; - -/** - * - * @param forcePowerOfTwo {boolean} Whether we want to force the texture to be a power of two - */ -TilingSprite.prototype.generateTilingTexture = function (forcePowerOfTwo) { - if (!this.texture.baseTexture.hasLoaded) { - return; - } - - var texture = this.originalTexture || this.texture; - var frame = texture.frame; - var targetWidth, targetHeight; - - // Check that the frame is the same size as the base texture. - var isFrame = frame.width !== texture.baseTexture.width || frame.height !== texture.baseTexture.height; - - var newTextureRequired = false; - - if (!forcePowerOfTwo) { - if (isFrame) { - targetWidth = frame.width; - targetHeight = frame.height; - - newTextureRequired = true; - } - } - else { - targetWidth = core.utils.getNextPowerOfTwo(frame.width); - targetHeight = core.utils.getNextPowerOfTwo(frame.height); - - // If the BaseTexture dimensions don't match the texture frame then we need a new texture anyway because it's part of a texture atlas - if (frame.width !== targetWidth || frame.height !== targetHeight || texture.baseTexture.width !== targetWidth || texture.baseTexture.height || targetHeight) { - newTextureRequired = true; - } - } - - if (newTextureRequired) { - var canvasBuffer; - - if (this.tilingTexture && this.tilingTexture.isTiling) { - canvasBuffer = this.tilingTexture.canvasBuffer; - canvasBuffer.resize(targetWidth, targetHeight); - this.tilingTexture.baseTexture.width = targetWidth; - this.tilingTexture.baseTexture.height = targetHeight; - this.tilingTexture.needsUpdate = true; - } - else { - canvasBuffer = new core.CanvasBuffer(targetWidth, targetHeight); - - this.tilingTexture = core.Texture.fromCanvas(canvasBuffer.canvas); - this.tilingTexture.canvasBuffer = canvasBuffer; - this.tilingTexture.isTiling = true; - } - - canvasBuffer.context.drawImage(texture.baseTexture.source, - texture.crop.x, - texture.crop.y, - texture.crop.width, - texture.crop.height, - 0, - 0, - targetWidth, - targetHeight); - - this.tileScaleOffset.x = frame.width / targetWidth; - this.tileScaleOffset.y = frame.height / targetHeight; - } - else { - // TODO - switching? - if (this.tilingTexture && this.tilingTexture.isTiling) { - // destroy the tiling texture! - // TODO could store this somewhere? - this.tilingTexture.destroy(true); - } - - this.tileScaleOffset.x = 1; - this.tileScaleOffset.y = 1; - this.tilingTexture = texture; - } - - this.refreshTexture = false; - - this.originalTexture = this.texture; - this.texture = this.tilingTexture; - - this.tilingTexture.baseTexture._powerOf2 = true; -}; - -},{"../core":9}],55:[function(require,module,exports){ -/** - * @file Main export of the PIXI extras library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - MovieClip: require('./MovieClip'), - Rope: require('./Rope'), - Strip: require('./Strip'), - TilingSprite: require('./TilingSprite') -}; - -},{"./MovieClip":51,"./Rope":52,"./Strip":53,"./TilingSprite":54}],56:[function(require,module,exports){ -/** - * This is the base class for creating a PIXI filter. Currently only WebGL supports filters. - * If you want to make a custom filter this should be your base class. - * - * @class - * @namespace PIXI - * @param fragmentSrc {string|string[]} The fragment source in an array of strings. - * @param uniforms {object} An object containing the uniforms for this filter. - */ -function AbstractFilter(fragmentSrc, uniforms) { - /** - * An array of passes - some filters contain a few steps this array simply stores the steps in a liniear fashion. - * For example the blur filter has two passes blurX and blurY. - * - * @member {AbstractFilter[]} - * @private - */ - this.passes = [this]; - - /** - * @member {Shader[]} - * @private - */ - this.shaders = []; - - /** - * @member {number} - */ - this.padding = 0; - - /** - * @member {object} - * @private - */ - this.uniforms = uniforms || {}; - - /** - * @member {string[]} - * @private - */ - this.fragmentSrc = typeof fragmentSrc === 'string' ? fragmentSrc.split('') : (fragmentSrc || []); -} - -AbstractFilter.prototype.constructor = AbstractFilter; -module.exports = AbstractFilter; - -/** - * Syncs the uniforms between the class object and the shaders. - * - */ -AbstractFilter.prototype.syncUniforms = function () { - for (var i = 0, j = this.shaders.length; i < j; ++i) { - this.shaders[i].dirty = true; - } -}; - -/* -AbstractFilter.prototype.apply = function (frameBuffer) { - // TODO :) -}; -*/ - -},{}],57:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The AlphaMaskFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object. - * You can use this filter to apply all manor of crazy warping effects - * Currently the r property of the texture is used to offset the x and the g property of the texture is used to offset the y. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment - */ -function AlphaMaskFilter(texture) { - AbstractFilter.call(this); - - texture.baseTexture._powerOf2 = true; - - // set the uniforms - this.uniforms = { - mask: { type: 'sampler2D', value: texture }, - mapDimensions: { type: '2f', value: { x: 1, y: 5112 } }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] }, - offset: { type: '2f', value: { x: 0, y: 0 } } - }; - - if (texture.baseTexture.hasLoaded) { - this.uniforms.mask.value.x = texture.width; - this.uniforms.mask.value.y = texture.height; - } - else { - this.boundLoadedFunction = this.onTextureLoaded.bind(this); - - texture.baseTexture.on('loaded', this.boundLoadedFunction); - } - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D uSampler;', - 'uniform sampler2D mask;', - 'uniform vec2 mapDimensions;', - 'uniform vec4 dimensions;', - 'uniform vec2 offset;', - - 'void main() {', - ' vec2 mapCords = vTextureCoord.xy;', - ' mapCords += (dimensions.zw + offset)/ dimensions.xy ;', - ' mapCords.y *= -1.0;', - ' mapCords.y += 1.0;', - ' mapCords *= dimensions.xy / mapDimensions;', - - ' vec4 original = texture2D(uSampler, vTextureCoord);', - ' float maskAlpha = texture2D(mask, mapCords).r;', - ' original *= maskAlpha;', - //' original.rgb *= maskAlpha;', - ' gl_FragColor = original;', - //' gl_FragColor = gl_FragColor;', - '}' - ]; -} - -AlphaMaskFilter.prototype = Object.create(AbstractFilter.prototype); -AlphaMaskFilter.prototype.constructor = AlphaMaskFilter; -module.exports = AlphaMaskFilter; - -/** - * Sets the map dimensions uniforms when the texture becomes available. - * - */ -AlphaMaskFilter.prototype.onTextureLoaded = function () { - this.uniforms.mapDimensions.value.x = this.uniforms.mask.value.width; - this.uniforms.mapDimensions.value.y = this.uniforms.mask.value.height; - - this.uniforms.mask.value.baseTexture.off('loaded', this.boundLoadedFunction); -}; - -Object.defineProperties(AlphaMaskFilter.prototype, { - /** - * The texture used for the displacement map. Must be power of 2 sized texture. - * - * @member {Texture} - * @memberof AlphaMaskFilter# - */ - map: { - get: function () { - return this.uniforms.mask.value; - }, - set: function (value) { - this.uniforms.mask.value = value; - } - }, - - /** - * The offset used to move the displacement map. - * - * @member {Point} - * @memberof AlphaMaskFilter# - */ - offset: { - get: function() { - return this.uniforms.offset.value; - }, - set: function(value) { - this.uniforms.offset.value = value; - } - } -}); - -},{"./AbstractFilter":56}],58:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original shader : https://www.shadertoy.com/view/lssGDj by @movAX13h - */ - -/** - * An ASCII filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function AsciiFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - dimensions: { type: '4fv', value: new Float32Array([10000, 100, 10, 10]) }, - pixelSize: { type: '1f', value: 8} - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'uniform vec4 dimensions;', - 'uniform float pixelSize;', - 'uniform sampler2D uSampler;', - - 'float character(float n, vec2 p)', - '{', - ' p = floor(p*vec2(4.0, -4.0) + 2.5);', - ' if (clamp(p.x, 0.0, 4.0) == p.x && clamp(p.y, 0.0, 4.0) == p.y)', - ' {', - ' if (int(mod(n/exp2(p.x + 5.0*p.y), 2.0)) == 1) return 1.0;', - ' }', - ' return 0.0;', - '}', - - 'void main()', - '{', - ' vec2 uv = gl_FragCoord.xy;', - ' vec3 col = texture2D(uSampler, floor( uv / pixelSize ) * pixelSize / dimensions.xy).rgb;', - - ' #ifdef HAS_GREENSCREEN', - ' float gray = (col.r + col.b)/2.0;', - ' #else', - ' float gray = (col.r + col.g + col.b)/3.0;', - ' #endif', - - ' float n = 65536.0; // .', - ' if (gray > 0.2) n = 65600.0; // :', - ' if (gray > 0.3) n = 332772.0; // *', - ' if (gray > 0.4) n = 15255086.0; // o', - ' if (gray > 0.5) n = 23385164.0; // &', - ' if (gray > 0.6) n = 15252014.0; // 8', - ' if (gray > 0.7) n = 13199452.0; // @', - ' if (gray > 0.8) n = 11512810.0; // #', - - ' vec2 p = mod( uv / ( pixelSize * 0.5 ), 2.0) - vec2(1.0);', - ' col = col * character(n, p);', - - ' gl_FragColor = vec4(col, 1.0);', - '}' - ]; -} - -AsciiFilter.prototype = Object.create(AbstractFilter.prototype); -AsciiFilter.prototype.constructor = AsciiFilter; -module.exports = AsciiFilter; - -Object.defineProperties(AsciiFilter.prototype, { - /** - * The pixel size used by the filter. - * - * @member {number} - * @memberof AsciiFilter# - */ - size: { - get: function () { - return this.uniforms.pixelSize.value; - }, - set: function (value) { - this.uniforms.pixelSize.value = value; - } - } -}); - -},{"./AbstractFilter":56}],59:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - BlurXFilter = require('./BlurXFilter'), - BlurYFilter = require('./BlurYFilter'); - -/** - * The BlurFilter applies a Gaussian blur to an object. - * The strength of the blur can be set for x- and y-axis separately. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function BlurFilter() { - AbstractFilter.call(this); - - this.blurXFilter = new BlurXFilter(); - this.blurYFilter = new BlurYFilter(); - - this.passes = [this.blurXFilter, this.blurYFilter]; -} - -BlurFilter.prototype = Object.create(AbstractFilter.prototype); -BlurFilter.prototype.constructor = BlurFilter; -module.exports = BlurFilter; - -Object.defineProperties(BlurFilter.prototype, { - /** - * Sets the strength of both the blurX and blurY properties simultaneously - * - * @member {number} - * @memberOf BlurFilter# - * @default 2 - */ - blur: { - get: function () { - return this.blurXFilter.blur; - }, - set: function (value) { - this.blurXFilter.blur = this.blurYFilter.blur = value; - } - }, - - /** - * Sets the strength of the blurX property - * - * @member {number} - * @memberOf BlurFilter# - * @default 2 - */ - blurX: { - get: function () { - return this.blurXFilter.blur; - }, - set: function (value) { - this.blurXFilter.blur = value; - } - }, - - /** - * Sets the strength of the blurY property - * - * @member {number} - * @memberOf BlurFilter# - * @default 2 - */ - blurY: { - get: function () { - return this.blurYFilter.blur; - }, - set: function (value) { - this.blurYFilter.blur = value; - } - } -}); - -},{"./AbstractFilter":56,"./BlurXFilter":60,"./BlurYFilter":61}],60:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - blurFactor = 1 / 7000; - -/** - * The BlurXFilter applies a horizontal Gaussian blur to an object. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function BlurXFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 1 / 512 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float blur;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec4 sum = vec4(0.0);', - - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - 4.0*blur, vTextureCoord.y)) * 0.05;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - 3.0*blur, vTextureCoord.y)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - 2.0*blur, vTextureCoord.y)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x - blur, vTextureCoord.y)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + blur, vTextureCoord.y)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + 2.0*blur, vTextureCoord.y)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + 3.0*blur, vTextureCoord.y)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x + 4.0*blur, vTextureCoord.y)) * 0.05;', - - ' gl_FragColor = sum;', - '}' - ]; -} - -BlurXFilter.prototype = Object.create(AbstractFilter.prototype); -BlurXFilter.prototype.constructor = BlurXFilter; -module.exports = BlurXFilter; - -Object.defineProperties(BlurXFilter.prototype, { - /** - * Sets the strength of both the blur. - * - * @member {number} - * @memberof BlurXFilter# - * @default 2 - */ - blur: { - get: function () { - return this.uniforms.blur.value / blurFactor; - }, - set: function (value) { - this.uniforms.blur.value = blurFactor * value; - } - } -}); - -},{"./AbstractFilter":56}],61:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - blurFactor = 1 / 7000; - -/** - * The BlurYFilter applies a vertical Gaussian blur to an object. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function BlurYFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 1 / 512 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float blur;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec4 sum = vec4(0.0);', - - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 4.0*blur)) * 0.05;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 3.0*blur)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 2.0*blur)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - blur)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + blur)) * 0.15;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 2.0*blur)) * 0.12;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 3.0*blur)) * 0.09;', - ' sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 4.0*blur)) * 0.05;', - - ' gl_FragColor = sum;', - '}' - ]; -} - -BlurYFilter.prototype = Object.create(AbstractFilter.prototype); -BlurYFilter.prototype.constructor = BlurYFilter; -module.exports = BlurYFilter; - -Object.defineProperties(BlurYFilter.prototype, { - /** - * Sets the strength of both the blur. - * - * @member {number} - * @memberof BlurYFilter - * @default 2 - */ - blur: { - get: function () { - return this.uniforms.blur.value / blurFactor; - }, - set: function (value) { - //this.padding = value; - this.uniforms.blur.value = blurFactor * value; - } - } -}); - -},{"./AbstractFilter":56}],62:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The ColorMatrixFilter class lets you apply a 4x4 matrix transformation on the RGBA - * color and alpha values of every pixel on your displayObject to produce a result - * with a new set of RGBA color and alpha values. It's pretty powerful! - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function ColorMatrixFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - matrix: { type: 'mat4', value: [1, 0, 0, 0, - 0, 1, 0, 0, - 0, 0, 1, 0, - 0, 0, 0, 1] } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float invert;', - 'uniform mat4 matrix;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord) * matrix;', - '}' - ]; -} - -ColorMatrixFilter.prototype = Object.create(AbstractFilter.prototype); -ColorMatrixFilter.prototype.constructor = ColorMatrixFilter; -module.exports = ColorMatrixFilter; - -Object.defineProperties(ColorMatrixFilter.prototype, { - /** - * Sets the matrix of the color matrix filter - * - * @member {number[]} - * @memberof ColorMatrixFilter# - * @default [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1] - */ - matrix: { - get: function () { - return this.uniforms.matrix.value; - }, - set: function (value) { - this.uniforms.matrix.value = value; - } - } -}); - -},{"./AbstractFilter":56}],63:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This lowers the color depth of your image by the given amount, producing an image with a smaller palette. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function ColorStepFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - step: { type: '1f', value: 5 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - 'uniform sampler2D uSampler;', - 'uniform float step;', - - 'void main(void) {', - ' vec4 color = texture2D(uSampler, vTextureCoord);', - ' color = floor(color * step) / step;', - ' gl_FragColor = color;', - '}' - ]; -} - -ColorStepFilter.prototype = Object.create(AbstractFilter.prototype); -ColorStepFilter.prototype.constructor = ColorStepFilter; -module.exports = ColorStepFilter; - -Object.defineProperties(ColorStepFilter.prototype, { - /** - * The number of steps to reduce the palette by. - * - * @member {number} - * @memberof ColorStepFilter# - */ - step: { - get: function () { - return this.uniforms.step.value; - }, - set: function (value) { - this.uniforms.step.value = value; - } - } -}); - -},{"./AbstractFilter":56}],64:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The ConvolutionFilter class applies a matrix convolution filter effect. - * A convolution combines pixels in the input image with neighboring pixels to produce a new image. - * A wide variety of image effects can be achieved through convolutions, including blurring, edge - * detection, sharpening, embossing, and beveling. The matrix should be specified as a 9 point Array. - * See http://docs.gimp.org/en/plug-in-convmatrix.html for more info. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param matrix {number[]} An array of values used for matrix transformation. Specified as a 9 point Array. - * @param width {number} Width of the object you are transforming - * @param height {number} Height of the object you are transforming - */ -function ConvolutionFilter(matrix, width, height) { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - matrix: { type: '1fv', value: new Float32Array(matrix) }, - texelSizeX: { type: '1f', value: 1 / width }, - texelSizeY: { type: '1f', value: 1 / height } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying mediump vec2 vTextureCoord;', - - 'uniform sampler2D texture;', - 'uniform float texelSizeX;', - 'uniform float texelSizeY;', - 'uniform float matrix[9];', - - 'vec2 px = vec2(texelSizeX, texelSizeY);', - - 'void main(void) {', - ' vec4 c11 = texture2D(texture, vTextureCoord - px);', // top left - ' vec4 c12 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y - px.y));', // top center - ' vec4 c13 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y - px.y));', // top right - - ' vec4 c21 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y) );', // mid left - ' vec4 c22 = texture2D(texture, vTextureCoord);', // mid center - ' vec4 c23 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y) );', // mid right - - ' vec4 c31 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y + px.y) );', // bottom left - ' vec4 c32 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y + px.y) );', // bottom center - ' vec4 c33 = texture2D(texture, vTextureCoord + px );', // bottom right - - ' gl_FragColor = ', - ' c11 * matrix[0] + c12 * matrix[1] + c13 * matrix[2] +', - ' c21 * matrix[3] + c22 * matrix[4] + c23 * matrix[5] +', - ' c31 * matrix[6] + c32 * matrix[7] + c33 * matrix[8];', - ' gl_FragColor.a = c22.a;', - '}' - ]; -} - -ConvolutionFilter.prototype = Object.create(AbstractFilter.prototype); -ConvolutionFilter.prototype.constructor = ConvolutionFilter; -module.exports = ConvolutionFilter; - -Object.defineProperties(ConvolutionFilter.prototype, { - /** - * An array of values used for matrix transformation. Specified as a 9 point Array. - * - * @member {number[]} - * @memberof ConvolutionFilter# - */ - matrix: { - get: function () { - return this.uniforms.matrix.value; - }, - set: function (value) { - this.uniforms.matrix.value = new Float32Array(value); - } - }, - - /** - * Width of the object you are transforming - * - * @member {number} - * @memberof ConvolutionFilter# - */ - width: { - get: function () { - return this.uniforms.texelSizeX.value; - }, - set: function (value) { - this.uniforms.texelSizeX.value = 1/value; - } - }, - - /** - * Height of the object you are transforming - * - * @member {number} - * @memberof ConvolutionFilter# - */ - height: { - get: function () { - return this.uniforms.texelSizeY.value; - }, - set: function (value) { - this.uniforms.texelSizeY.value = 1/value; - } - } -}); - -},{"./AbstractFilter":56}],65:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * A Cross Hatch effect filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function CrossHatchFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 1 / 512 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float blur;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' float lum = length(texture2D(uSampler, vTextureCoord.xy).rgb);', - - ' gl_FragColor = vec4(1.0, 1.0, 1.0, 1.0);', - - ' if (lum < 1.00) {', - ' if (mod(gl_FragCoord.x + gl_FragCoord.y, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - - ' if (lum < 0.75) {', - ' if (mod(gl_FragCoord.x - gl_FragCoord.y, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - - ' if (lum < 0.50) {', - ' if (mod(gl_FragCoord.x + gl_FragCoord.y - 5.0, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - - ' if (lum < 0.3) {', - ' if (mod(gl_FragCoord.x - gl_FragCoord.y - 5.0, 10.0) == 0.0) {', - ' gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);', - ' }', - ' }', - '}' - ]; -} - -CrossHatchFilter.prototype = Object.create(AbstractFilter.prototype); -CrossHatchFilter.prototype.constructor = CrossHatchFilter; -module.exports = CrossHatchFilter; - -Object.defineProperties(CrossHatchFilter.prototype, { - /** - * Sets the strength of both the blur. - * - * @member {number} - * @memberof CrossHatchFilter# - * @default 2 - */ - blur: { - get: function () { - return this.uniforms.blur.value / (1/7000); - }, - set: function (value) { - //this.padding = value; - this.uniforms.blur.value = (1/7000) * value; - } - } -}); - -},{"./AbstractFilter":56}],66:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The DisplacementFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object. - * You can use this filter to apply all manor of crazy warping effects - * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment - */ -function DisplacementFilter(texture) { - AbstractFilter.call(this); - - texture.baseTexture._powerOf2 = true; - - // set the uniforms - this.uniforms = { - displacementMap: { type: 'sampler2D', value: texture }, - scale: { type: '2f', value: { x: 30, y: 30 } }, - offset: { type: '2f', value: { x: 0, y: 0 } }, - mapDimensions: { type: '2f', value: { x: 1, y: 5112 } }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] } - }; - - if (texture.baseTexture.hasLoaded) { - this.onTextureLoaded(); - } - else { - this.boundLoadedFunction = this.onTextureLoaded.bind(this); - - texture.baseTexture.on('loaded', this.boundLoadedFunction); - } - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D displacementMap;', - 'uniform sampler2D uSampler;', - 'uniform vec2 scale;', - 'uniform vec2 offset;', - 'uniform vec4 dimensions;', - 'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);', - // 'const vec2 textureDimensions = vec2(750.0, 750.0);', - - 'void main(void) {', - ' vec2 mapCords = vTextureCoord.xy;', - ' mapCords += (dimensions.zw + offset)/ dimensions.xy ;', - ' mapCords.y *= -1.0;', - ' mapCords.y += 1.0;', - - ' vec2 matSample = texture2D(displacementMap, mapCords).xy;', - ' matSample -= 0.5;', - ' matSample *= scale;', - ' matSample /= mapDimensions;', - - ' gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x + matSample.x, vTextureCoord.y + matSample.y));', - - //TODO: Is this needed? - ' gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb, 1.0);', - '}' - ]; -} - -DisplacementFilter.prototype = Object.create(AbstractFilter.prototype); -DisplacementFilter.prototype.constructor = DisplacementFilter; -module.exports = DisplacementFilter; - -/** - * Sets the map dimensions uniforms when the texture becomes available. - * - * @private - */ -DisplacementFilter.prototype.onTextureLoaded = function () { - this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width; - this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height; - - this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction); -}; - -Object.defineProperties(DisplacementFilter.prototype, { - /** - * The texture used for the displacement map. Must be power of 2 texture. - * - * @member {Texture} - * @memberof DisplacementFilter# - */ - map: { - get: function () { - return this.uniforms.displacementMap.value; - }, - set: function (value) { - this.uniforms.displacementMap.value = value; - } - }, - - /** - * The multiplier used to scale the displacement result from the map calculation. - * - * @member {Point} - * @memberof DisplacementFilter# - */ - scale: { - get: function () { - return this.uniforms.scale.value; - }, - set: function (value) { - this.uniforms.scale.value = value; - } - }, - - /** - * The offset used to move the displacement map. - * - * @member {Point} - * @memberof DisplacementFilter# - */ - offset: { - get: function () { - return this.uniforms.offset.value; - }, - set: function (value) { - this.uniforms.offset.value = value; - } - } -}); - -},{"./AbstractFilter":56}],67:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Mat Groves http://matgroves.com/ @Doormat23 - * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/fun/dotscreen.js - */ - -/** - * This filter applies a dotscreen effect making display objects appear to be made out of - * black and white halftone dots like an old printer. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function DotScreenFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - scale: { type: '1f', value: 1 }, - angle: { type: '1f', value: 5 }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform vec4 dimensions;', - 'uniform sampler2D uSampler;', - - 'uniform float angle;', - 'uniform float scale;', - - 'float pattern() {', - ' float s = sin(angle), c = cos(angle);', - ' vec2 tex = vTextureCoord * dimensions.xy;', - ' vec2 point = vec2(', - ' c * tex.x - s * tex.y,', - ' s * tex.x + c * tex.y', - ' ) * scale;', - ' return (sin(point.x) * sin(point.y)) * 4.0;', - '}', - - 'void main() {', - ' vec4 color = texture2D(uSampler, vTextureCoord);', - ' float average = (color.r + color.g + color.b) / 3.0;', - ' gl_FragColor = vec4(vec3(average * 10.0 - 5.0 + pattern()), color.a);', - '}' - ]; -} - -DotScreenFilter.prototype = Object.create(AbstractFilter.prototype); -DotScreenFilter.prototype.constructor = DotScreenFilter; -module.exports = DotScreenFilter; - -Object.defineProperties(DotScreenFilter.prototype, { - /** - * The scale of the effect. - * @member {number} - * @memberof DotScreenFilter# - */ - scale: { - get: function () { - return this.uniforms.scale.value; - }, - set: function (value) { - this.uniforms.scale.value = value; - } - }, - - /** - * The radius of the effect. - * @member {number} - * @memberof DotScreenFilter# - */ - angle: { - get: function () { - return this.uniforms.angle.value; - }, - set: function (value) { - this.uniforms.angle.value = value; - } - } -}); - -},{"./AbstractFilter":56}],68:[function(require,module,exports){ -/** - * A target and pass info object for filters. - * - * @class - * @namespace PIXI - */ -function FilterBlock() { - /** - * The visible state of this FilterBlock. - * - * @member {boolean} - */ - this.visible = true; - - /** - * The renderable state of this FilterBlock. - * - * @member {boolean} - */ - this.renderable = true; -} - -FilterBlock.prototype.constructor = FilterBlock; -module.exports = FilterBlock; - -},{}],69:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This greyscales the palette of your Display Objects. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function GrayFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - gray: { type: '1f', value: 1 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform sampler2D uSampler;', - 'uniform float gray;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord);', - ' gl_FragColor.rgb = mix(gl_FragColor.rgb, vec3(0.2126*gl_FragColor.r + 0.7152*gl_FragColor.g + 0.0722*gl_FragColor.b), gray);', - // ' gl_FragColor = gl_FragColor;', - '}' - ]; -} - -GrayFilter.prototype = Object.create(AbstractFilter.prototype); -GrayFilter.prototype.constructor = GrayFilter; -module.exports = GrayFilter; - -Object.defineProperties(GrayFilter.prototype, { - /** - * The strength of the gray. 1 will make the object black and white, 0 will make the object its normal color. - * - * @member {number} - * @memberof GrayFilter# - */ - gray: { - get: function () { - return this.uniforms.gray.value; - }, - set: function (value) { - this.uniforms.gray.value = value; - } - } -}); - -},{"./AbstractFilter":56}],70:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This inverts your Display Objects colors. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function InvertFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - invert: { type: '1f', value: 1 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float invert;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord);', - ' gl_FragColor.rgb = mix( (vec3(1)-gl_FragColor.rgb) * gl_FragColor.a, gl_FragColor.rgb, 1.0 - invert);', - //' gl_FragColor.rgb = gl_FragColor.rgb * gl_FragColor.a;', - // ' gl_FragColor = gl_FragColor * vColor;', - '}' - ]; -} - -InvertFilter.prototype = Object.create(AbstractFilter.prototype); -InvertFilter.prototype.constructor = InvertFilter; -module.exports = InvertFilter; - -Object.defineProperties(InvertFilter.prototype, { - /** - * The strength of the invert. 1 will fully invert the colors, 0 will make the object its normal color - * - * @member {number} - * @memberof InvertFilter# - */ - invert: { - get: function () { - return this.uniforms.invert.value; - }, - set: function (value) { - this.uniforms.invert.value = value; - } - } -}); - -},{"./AbstractFilter":56}],71:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/adjust/noise.js - */ - -/** - * A Noise effect filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function NoiseFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - noise: { type: '1f', value: 0.5 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float noise;', - 'uniform sampler2D uSampler;', - - 'float rand(vec2 co) {', - ' return fract(sin(dot(co.xy ,vec2(12.9898,78.233))) * 43758.5453);', - '}', - - 'void main() {', - ' vec4 color = texture2D(uSampler, vTextureCoord);', - - ' float diff = (rand(vTextureCoord) - 0.5) * noise;', - ' color.r += diff;', - ' color.g += diff;', - ' color.b += diff;', - - ' gl_FragColor = color;', - '}' - ]; -} - -NoiseFilter.prototype = Object.create(AbstractFilter.prototype); -NoiseFilter.prototype.constructor = NoiseFilter; -module.exports = NoiseFilter; - -Object.defineProperties(NoiseFilter.prototype, { - /** - * The amount of noise to apply. - * - * @member {number} - * @memberof NoiseFilter# - * @default 0.5 - */ - noise: { - get: function () { - return this.uniforms.noise.value; - }, - set: function (value) { - this.dirty = true; - this.uniforms.noise.value = value; - } - } -}); - -},{"./AbstractFilter":56}],72:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * The NormalMapFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object. - * You can use this filter to apply all manor of crazy warping effects - * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - * @param texture {Texture} The texture used for the normal map, must be power of 2 texture at the moment - */ -function NormalMapFilter(texture) { - AbstractFilter.call(this); - - texture.baseTexture._powerOf2 = true; - - // set the uniforms - this.uniforms = { - displacementMap: { type: 'sampler2D', value: texture }, - scale: { type: '2f', value: { x: 15, y: 15 } }, - offset: { type: '2f', value: { x: 0, y: 0 } }, - mapDimensions: { type: '2f', value: { x: 1, y: 1 } }, - dimensions: { type: '4f', value: [0, 0, 0, 0] }, - // LightDir: { type: 'f3', value: [0, 1, 0] }, - LightPos: { type: '3f', value: [0, 1, 0] } - }; - - if (texture.baseTexture.hasLoaded) { - this.onTextureLoaded(); - } - else { - this.boundLoadedFunction = this.onTextureLoaded.bind(this); - - texture.baseTexture.on('loaded', this.boundLoadedFunction); - } - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying float vColor;', - - 'uniform sampler2D displacementMap;', - 'uniform sampler2D uSampler;', - - 'uniform vec4 dimensions;', - - 'const vec2 Resolution = vec2(1.0,1.0);', //resolution of screen - 'uniform vec3 LightPos;', //light position, normalized - 'const vec4 LightColor = vec4(1.0, 1.0, 1.0, 1.0);', //light RGBA -- alpha is intensity - 'const vec4 AmbientColor = vec4(1.0, 1.0, 1.0, 0.5);', //ambient RGBA -- alpha is intensity - 'const vec3 Falloff = vec3(0.0, 1.0, 0.2);', //attenuation coefficients - - 'uniform vec3 LightDir;',//' = vec3(1.0, 0.0, 1.0);', - - - 'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);', - - - 'void main(void) {', - ' vec2 mapCords = vTextureCoord.xy;', - - ' vec4 color = texture2D(uSampler, vTextureCoord.st);', - ' vec3 nColor = texture2D(displacementMap, vTextureCoord.st).rgb;', - - - ' mapCords *= vec2(dimensions.x/512.0, dimensions.y/512.0);', - - ' mapCords.y *= -1.0;', - ' mapCords.y += 1.0;', - - //RGBA of our diffuse color - ' vec4 DiffuseColor = texture2D(uSampler, vTextureCoord);', - - //RGB of our normal map - ' vec3 NormalMap = texture2D(displacementMap, mapCords).rgb;', - - //The delta position of light - //'vec3 LightDir = vec3(LightPos.xy - (gl_FragCoord.xy / Resolution.xy), LightPos.z);', - ' vec3 LightDir = vec3(LightPos.xy - (mapCords.xy), LightPos.z);', - //Correct for aspect ratio - // ' LightDir.x *= Resolution.x / Resolution.y;', - - //Determine distance (used for attenuation) BEFORE we normalize our LightDir - ' float D = length(LightDir);', - - //normalize our vectors - ' vec3 N = normalize(NormalMap * 2.0 - 1.0);', - ' vec3 L = normalize(LightDir);', - - //Pre-multiply light color with intensity - //Then perform 'N dot L' to determine our diffuse term - ' vec3 Diffuse = (LightColor.rgb * LightColor.a) * max(dot(N, L), 0.0);', - - //pre-multiply ambient color with intensity - ' vec3 Ambient = AmbientColor.rgb * AmbientColor.a;', - - //calculate attenuation - ' float Attenuation = 1.0 / ( Falloff.x + (Falloff.y*D) + (Falloff.z*D*D) );', - - //the calculation which brings it all together - ' vec3 Intensity = Ambient + Diffuse * Attenuation;', - ' vec3 FinalColor = DiffuseColor.rgb * Intensity;', - ' gl_FragColor = vColor * vec4(FinalColor, DiffuseColor.a);', - // ' gl_FragColor = vec4(1.0, 0.0, 0.0, Attenuation);',//vColor * vec4(FinalColor, DiffuseColor.a);', - - /*// normalise color - ' vec3 normal = normalize(nColor * 2.0 - 1.0);', - - ' vec3 deltaPos = vec3( (light.xy - gl_FragCoord.xy) / resolution.xy, light.z );', - - ' float lambert = clamp(dot(normal, lightDir), 0.0, 1.0);', - - ' float d = sqrt(dot(deltaPos, deltaPos));', - ' float att = 1.0 / ( attenuation.x + (attenuation.y*d) + (attenuation.z*d*d) );', - - ' vec3 result = (ambientColor * ambientIntensity) + (lightColor.rgb * lambert) * att;', - ' result *= color.rgb;', - - ' gl_FragColor = vec4(result, 1.0);',*/ - '}' - ]; -} - -NormalMapFilter.prototype = Object.create(AbstractFilter.prototype); -NormalMapFilter.prototype.constructor = NormalMapFilter; -module.exports = NormalMapFilter; - -/** - * Sets the map dimensions uniforms when the texture becomes available. - * - */ -NormalMapFilter.prototype.onTextureLoaded = function () { - this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width; - this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height; - - this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction); -}; - -Object.defineProperties(NormalMapFilter.prototype, { - /** - * The texture used for the displacement map. Must be power of 2 texture. - * - * @member {Texture} - * @memberof NormalMapFilter# - */ - map: { - get: function () { - return this.uniforms.displacementMap.value; - }, - set: function (value) { - this.uniforms.displacementMap.value = value; - } - }, - - /** - * The multiplier used to scale the displacement result from the map calculation. - * - * @member {Point} - * @memberof NormalMapFilter# - */ - scale: { - get: function () { - return this.uniforms.scale.value; - }, - set: function (value) { - this.uniforms.scale.value = value; - } - }, - - /** - * The offset used to move the displacement map. - * - * @member {Point} - * @memberof NormalMapFilter# - */ - offset: { - get: function () { - return this.uniforms.offset.value; - }, - set: function (value) { - this.uniforms.offset.value = value; - } - } -}); - -},{"./AbstractFilter":56}],73:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This filter applies a pixelate effect making display objects appear 'blocky'. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function PixelateFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - invert: { type: '1f', value: 0 }, - dimensions: { type: '4fv', value: new Float32Array([10000, 100, 10, 10]) }, - pixelSize: { type: '2f', value: { x: 10, y: 10 } } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform vec2 testDim;', - 'uniform vec4 dimensions;', - 'uniform vec2 pixelSize;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec2 coord = vTextureCoord;', - - ' vec2 size = dimensions.xy/pixelSize;', - - ' vec2 color = floor( ( vTextureCoord * size ) ) / size + pixelSize/dimensions.xy * 0.5;', - ' gl_FragColor = texture2D(uSampler, color);', - '}' - ]; -} - -PixelateFilter.prototype = Object.create(AbstractFilter.prototype); -PixelateFilter.prototype.constructor = PixelateFilter; -module.exports = PixelateFilter; - -Object.defineProperties(PixelateFilter.prototype, { - /** - * This a point that describes the size of the blocks. x is the width of the block and y is the height. - * - * @member {Point} - * @memberof PixelateFilter# - */ - size: { - get: function () { - return this.uniforms.pixelSize.value; - }, - set: function (value) { - this.uniforms.pixelSize.value = value; - } - } -}); - -},{"./AbstractFilter":56}],74:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * An RGB Split Filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function RGBSplitFilter() { - AbstractFilter.call(this); - - this.passes = [this]; - - // set the uniforms - this.uniforms = { - red: { type: '2f', value: { x: 20, y: 20 } }, - green: { type: '2f', value: { x: -20, y: 20 } }, - blue: { type: '2f', value: { x: 20, y: -20 } }, - dimensions: { type: '4fv', value: [0, 0, 0, 0] } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform vec2 red;', - 'uniform vec2 green;', - 'uniform vec2 blue;', - 'uniform vec4 dimensions;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' gl_FragColor.r = texture2D(uSampler, vTextureCoord + red/dimensions.xy).r;', - ' gl_FragColor.g = texture2D(uSampler, vTextureCoord + green/dimensions.xy).g;', - ' gl_FragColor.b = texture2D(uSampler, vTextureCoord + blue/dimensions.xy).b;', - ' gl_FragColor.a = texture2D(uSampler, vTextureCoord).a;', - '}' - ]; -} - -RGBSplitFilter.prototype = Object.create(AbstractFilter.prototype); -RGBSplitFilter.prototype.constructor = RGBSplitFilter; -module.exports = RGBSplitFilter; - -Object.defineProperties(RGBSplitFilter.prototype, { - /** - * Red channel offset. - * - * @member {Point} - * @memberof RGBSplitFilter# - */ - red: { - get: function () { - return this.uniforms.red.value; - }, - set: function (value) { - this.uniforms.red.value = value; - } - }, - - /** - * Green channel offset. - * - * @member {Point} - * @memberof RGBSplitFilter# - */ - green: { - get: function () { - return this.uniforms.green.value; - }, - set: function (value) { - this.uniforms.green.value = value; - } - }, - - /** - * Blue offset. - * - * @member {Point} - * @memberof RGBSplitFilter# - */ - blue: { - get: function () { - return this.uniforms.blue.value; - }, - set: function (value) { - this.uniforms.blue.value = value; - } - } -}); - -},{"./AbstractFilter":56}],75:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This applies a sepia effect to your Display Objects. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function SepiaFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - sepia: { type: '1f', value: 1 } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float sepia;', - 'uniform sampler2D uSampler;', - - 'const mat3 sepiaMatrix = mat3(0.3588, 0.7044, 0.1368, 0.2990, 0.5870, 0.1140, 0.2392, 0.4696, 0.0912);', - - 'void main(void) {', - ' gl_FragColor = texture2D(uSampler, vTextureCoord);', - ' gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb * sepiaMatrix, sepia);', - '}' - ]; -} - -SepiaFilter.prototype = Object.create(AbstractFilter.prototype); -SepiaFilter.prototype.constructor = SepiaFilter; -module.exports = SepiaFilter; - -Object.defineProperties(SepiaFilter.prototype, { - /** - * The strength of the sepia. 1 will apply the full sepia effect, 0 will make the object its normal color. - * - * @member {number} - * @memberof SepiaFilter# - */ - sepia: { - get: function () { - return this.uniforms.sepia.value; - }, - set: function (value) { - this.uniforms.sepia.value = value; - } - } -}); - -},{"./AbstractFilter":56}],76:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * A Smart Blur Filter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function SmartBlurFilter() { - AbstractFilter.call(this); - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - - 'uniform sampler2D uSampler;', - 'const vec2 delta = vec2(1.0/10.0, 0.0);', - - 'float random(vec3 scale, float seed) {', - ' return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);', - '}', - - - 'void main(void) {', - ' vec4 color = vec4(0.0);', - ' float total = 0.0;', - - ' float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);', - - ' for (float t = -30.0; t <= 30.0; t++) {', - ' float percent = (t + offset - 0.5) / 30.0;', - ' float weight = 1.0 - abs(percent);', - ' vec4 sample = texture2D(uSampler, vTextureCoord + delta * percent);', - ' sample.rgb *= sample.a;', - ' color += sample * weight;', - ' total += weight;', - ' }', - - ' gl_FragColor = color / total;', - ' gl_FragColor.rgb /= gl_FragColor.a + 0.00001;', - '}' - ]; -} - -SmartBlurFilter.prototype = Object.create(AbstractFilter.prototype); -SmartBlurFilter.prototype.constructor = SmartBlurFilter; -module.exports = SmartBlurFilter; - -},{"./AbstractFilter":56}],77:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'), - TiltShiftXFilter = require('./TiltShiftXFilter'), - TiltShiftYFilter = require('./TiltShiftYFilter'); - -/** - * @author Vico @vicocotea - * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/ - */ - -/** - * A TiltShift Filter. Manages the pass of both a TiltShiftXFilter and TiltShiftYFilter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TiltShiftFilter() { - AbstractFilter.call(this); - - this.tiltShiftXFilter = new TiltShiftXFilter(); - this.tiltShiftYFilter = new TiltShiftYFilter(); - - this.tiltShiftXFilter.updateDelta(); - this.tiltShiftXFilter.updateDelta(); - - this.passes = [this.tiltShiftXFilter, this.tiltShiftYFilter]; -} - -TiltShiftFilter.prototype = Object.create(AbstractFilter.prototype); -TiltShiftFilter.prototype.constructor = TiltShiftFilter; -module.exports = TiltShiftFilter; - -Object.defineProperties(TiltShiftFilter.prototype, { - /** - * The strength of the blur. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - blur: { - get: function () { - return this.tiltShiftXFilter.blur; - }, - set: function (value) { - this.tiltShiftXFilter.blur = this.tiltShiftYFilter.blur = value; - } - }, - - /** - * The strength of the gradient blur. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - gradientBlur: { - get: function () { - return this.tiltShiftXFilter.gradientBlur; - }, - set: function (value) { - this.tiltShiftXFilter.gradientBlur = this.tiltShiftYFilter.gradientBlur = value; - } - }, - - /** - * The Y value to start the effect at. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - start: { - get: function () { - return this.tiltShiftXFilter.start; - }, - set: function (value) { - this.tiltShiftXFilter.start = this.tiltShiftYFilter.start = value; - } - }, - - /** - * The Y value to end the effect at. - * - * @member {number} - * @memberof TiltShiftFilter# - */ - end: { - get: function () { - return this.tiltShiftXFilter.end; - }, - set: function (value) { - this.tiltShiftXFilter.end = this.tiltShiftYFilter.end = value; - } - }, -}); - -},{"./AbstractFilter":56,"./TiltShiftXFilter":78,"./TiltShiftYFilter":79}],78:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/ - */ - -/** - * A TiltShiftXFilter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TiltShiftXFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 100 }, - gradientBlur: { type: '1f', value: 600 }, - start: { type: '2f', value: { x: 0, y: window.screenHeight / 2 } }, - end: { type: '2f', value: { x: 600, y: window.screenHeight / 2 } }, - delta: { type: '2f', value: { x: 30, y: 30 } }, - texSize: { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } } - }; - - this.updateDelta(); - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - - 'uniform sampler2D uSampler;', - 'uniform float blur;', - 'uniform float gradientBlur;', - 'uniform vec2 start;', - 'uniform vec2 end;', - 'uniform vec2 delta;', - 'uniform vec2 texSize;', - - 'float random(vec3 scale, float seed) {', - ' return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);', - '}', - - 'void main(void) {', - ' vec4 color = vec4(0.0);', - ' float total = 0.0;', - - ' float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);', - ' vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));', - ' float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;', - - ' for (float t = -30.0; t <= 30.0; t++) {', - ' float percent = (t + offset - 0.5) / 30.0;', - ' float weight = 1.0 - abs(percent);', - ' vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);', - ' sample.rgb *= sample.a;', - ' color += sample * weight;', - ' total += weight;', - ' }', - - ' gl_FragColor = color / total;', - ' gl_FragColor.rgb /= gl_FragColor.a + 0.00001;', - '}' - ]; -} - -TiltShiftXFilter.prototype = Object.create(AbstractFilter.prototype); -TiltShiftXFilter.prototype.constructor = TiltShiftXFilter; -module.exports = TiltShiftXFilter; - -/** - * Updates the filter delta values. - * - */ -TiltShiftXFilter.prototype.updateDelta = function () { - var dx = this.uniforms.end.value.x - this.uniforms.start.value.x; - var dy = this.uniforms.end.value.y - this.uniforms.start.value.y; - var d = Math.sqrt(dx * dx + dy * dy); - - this.uniforms.delta.value.x = dx / d; - this.uniforms.delta.value.y = dy / d; -}; - - -Object.defineProperties(TiltShiftXFilter.prototype, { - /** - * The strength of the blur. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - blur: { - get: function () { - return this.uniforms.blur.value; - }, - set: function (value) { - this.uniforms.blur.value = value; - } - }, - - /** - * The strength of the gradient blur. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - gradientBlur: { - get: function () { - return this.uniforms.gradientBlur.value; - }, - set: function (value) { - this.uniforms.gradientBlur.value = value; - } - }, - - /** - * The X value to start the effect at. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - start: { - get: function () { - return this.uniforms.start.value; - }, - set: function (value) { - this.uniforms.start.value = value; - this.updateDelta(); - } - }, - - /** - * The X value to end the effect at. - * - * @member {number} - * @memberof TilttShiftXFilter# - */ - end: { - get: function () { - return this.uniforms.end.value; - }, - set: function (value) { - this.uniforms.end.value = value; - this.updateDelta(); - } - } -}); - -},{"./AbstractFilter":56}],79:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * @author Vico @vicocotea - * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/ - */ - -/** - * A TiltShiftYFilter. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TiltShiftYFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - blur: { type: '1f', value: 100 }, - gradientBlur: { type: '1f', value: 600 }, - start: { type: '2f', value: { x: 0, y: window.screenHeight / 2 } }, - end: { type: '2f', value: { x: 600, y: window.screenHeight / 2 } }, - delta: { type: '2f', value: { x: 30, y: 30 } }, - texSize: { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } } - }; - - this.updateDelta(); - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - - 'uniform sampler2D uSampler;', - 'uniform float blur;', - 'uniform float gradientBlur;', - 'uniform vec2 start;', - 'uniform vec2 end;', - 'uniform vec2 delta;', - 'uniform vec2 texSize;', - - 'float random(vec3 scale, float seed) {', - ' return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);', - '}', - - 'void main(void) {', - ' vec4 color = vec4(0.0);', - ' float total = 0.0;', - - ' float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);', - ' vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));', - ' float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;', - - ' for (float t = -30.0; t <= 30.0; t++) {', - ' float percent = (t + offset - 0.5) / 30.0;', - ' float weight = 1.0 - abs(percent);', - ' vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);', - ' sample.rgb *= sample.a;', - ' color += sample * weight;', - ' total += weight;', - ' }', - - ' gl_FragColor = color / total;', - ' gl_FragColor.rgb /= gl_FragColor.a + 0.00001;', - '}' - ]; -} - -TiltShiftYFilter.prototype = Object.create(AbstractFilter.prototype); -TiltShiftYFilter.prototype.constructor = TiltShiftYFilter; -module.exports = TiltShiftYFilter; - -/** - * Updates the filter delta values. - * - */ -TiltShiftYFilter.prototype.updateDelta = function (){ - var dx = this.uniforms.end.value.x - this.uniforms.start.value.x; - var dy = this.uniforms.end.value.y - this.uniforms.start.value.y; - var d = Math.sqrt(dx * dx + dy * dy); - this.uniforms.delta.value.x = -dy / d; - this.uniforms.delta.value.y = dx / d; -}; - -Object.defineProperties(TiltShiftYFilter.prototype, { - /** - * The strength of the blur. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - blur: { - get: function () { - return this.uniforms.blur.value; - }, - set: function (value) { - this.uniforms.blur.value = value; - } - }, - - /** - * The strength of the gradient blur. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - gradientBlur: { - get: function () { - return this.uniforms.gradientBlur.value; - }, - set: function (value) { - this.uniforms.gradientBlur.value = value; - } - }, - - /** - * The Y value to start the effect at. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - start: { - get: function () { - return this.uniforms.start.value; - }, - set: function (value) { - this.uniforms.start.value = value; - this.updateDelta(); - } - }, - - /** - * The Y value to end the effect at. - * - * @member {number} - * @memberof TiltShiftYFilter# - */ - end: { - get: function () { - return this.uniforms.end.value; - }, - set: function (value) { - this.uniforms.end.value = value; - this.updateDelta(); - } - } -}); - -},{"./AbstractFilter":56}],80:[function(require,module,exports){ -var AbstractFilter = require('./AbstractFilter'); - -/** - * This filter applies a twist effect making display objects appear twisted in the given direction. - * - * @class - * @extends AbstractFilter - * @namespace PIXI - */ -function TwistFilter() { - AbstractFilter.call(this); - - // set the uniforms - this.uniforms = { - radius: { type: '1f', value: 0.5}, - angle: { type: '1f', value: 5}, - offset: { type: '2f', value: { x: 0.5, y: 0.5 } } - }; - - this.fragmentSrc = [ - 'precision mediump float;', - - 'varying vec2 vTextureCoord;', - 'varying vec4 vColor;', - - 'uniform float radius;', - 'uniform float angle;', - 'uniform vec2 offset;', - 'uniform sampler2D uSampler;', - - 'void main(void) {', - ' vec2 coord = vTextureCoord - offset;', - ' float distance = length(coord);', - - ' if (distance < radius) {', - ' float ratio = (radius - distance) / radius;', - ' float angleMod = ratio * ratio * angle;', - ' float s = sin(angleMod);', - ' float c = cos(angleMod);', - ' coord = vec2(coord.x * c - coord.y * s, coord.x * s + coord.y * c);', - ' }', - - ' gl_FragColor = texture2D(uSampler, coord+offset);', - '}' - ]; -} - -TwistFilter.prototype = Object.create(AbstractFilter.prototype); -TwistFilter.prototype.constructor = TwistFilter; -module.exports = TwistFilter; - -Object.defineProperties(TwistFilter.prototype, { - /** - * This point describes the the offset of the twist. - * - * @member {Point} - * @memberof TwistFilter# - */ - offset: { - get: function () { - return this.uniforms.offset.value; - }, - set: function (value) { - this.uniforms.offset.value = value; - } - }, - - /** - * This radius of the twist. - * - * @member {number} - * @memberof TwistFilter# - */ - radius: { - get: function () { - return this.uniforms.radius.value; - }, - set: function (value) { - this.uniforms.radius.value = value; - } - }, - - /** - * This angle of the twist. - * - * @member {number} - * @memberof TwistFilter# - */ - angle: { - get: function () { - return this.uniforms.angle.value; - }, - set: function (value) { - this.uniforms.angle.value = value; - } - } -}); - -},{"./AbstractFilter":56}],81:[function(require,module,exports){ -/** - * @file Main export of the PIXI filters library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - AbstractFilter: require('./AbstractFilter'), - AlphaMaskFilter: require('./AlphaMaskFilter'), - AsciiFilter: require('./AsciiFilter'), - BlurFilter: require('./BlurFilter'), - BlurXFilter: require('./BlurXFilter'), - BlurYFilter: require('./BlurYFilter'), - ColorMatrixFilter: require('./ColorMatrixFilter'), - ColorStepFilter: require('./ColorStepFilter'), - ConvolutionFilter: require('./ConvolutionFilter'), - CrossHatchFilter: require('./CrossHatchFilter'), - DisplacementFilter: require('./DisplacementFilter'), - DotScreenFilter: require('./DotScreenFilter'), - FilterBlock: require('./FilterBlock'), - GrayFilter: require('./GrayFilter'), - InvertFilter: require('./InvertFilter'), - NoiseFilter: require('./NoiseFilter'), - NormalMapFilter: require('./NormalMapFilter'), - PixelateFilter: require('./PixelateFilter'), - RGBSplitFilter: require('./RGBSplitFilter'), - SepiaFilter: require('./SepiaFilter'), - SmartBlurFilter: require('./SmartBlurFilter'), - TiltShiftFilter: require('./TiltShiftFilter'), - TiltShiftXFilter: require('./TiltShiftXFilter'), - TiltShiftYFilter: require('./TiltShiftYFilter'), - TwistFilter: require('./TwistFilter') -}; - -},{"./AbstractFilter":56,"./AlphaMaskFilter":57,"./AsciiFilter":58,"./BlurFilter":59,"./BlurXFilter":60,"./BlurYFilter":61,"./ColorMatrixFilter":62,"./ColorStepFilter":63,"./ConvolutionFilter":64,"./CrossHatchFilter":65,"./DisplacementFilter":66,"./DotScreenFilter":67,"./FilterBlock":68,"./GrayFilter":69,"./InvertFilter":70,"./NoiseFilter":71,"./NormalMapFilter":72,"./PixelateFilter":73,"./RGBSplitFilter":74,"./SepiaFilter":75,"./SmartBlurFilter":76,"./TiltShiftFilter":77,"./TiltShiftXFilter":78,"./TiltShiftYFilter":79,"./TwistFilter":80}],82:[function(require,module,exports){ -var core = require('../core'); - -/** - * Holds all information related to an Interaction event - * - * @class - * @namespace PIXI - */ -function InteractionData() { - /** - * This point stores the global coords of where the touch/mouse event happened - * - * @member {Point} - */ - this.global = new core.math.Point(); - - /** - * The target Sprite that was interacted with - * - * @member {Sprite} - */ - this.target = null; - - /** - * When passed to an event handler, this will be the original DOM Event that was captured - * - * @member {Event} - */ - this.originalEvent = null; -} - -InteractionData.prototype.constructor = InteractionData; -module.exports = InteractionData; - -/** - * This will return the local coordinates of the specified displayObject for this InteractionData - * - * @param displayObject {DisplayObject} The DisplayObject that you would like the local coords off - * @param [point] {Point} A Point object in which to store the value, optional (otherwise will create a new point) - * @return {Point} A point containing the coordinates of the InteractionData position relative to the DisplayObject - */ -InteractionData.prototype.getLocalPosition = function (displayObject, point) { - var worldTransform = displayObject.worldTransform; - var global = this.global; - - // do a cheeky transform to get the mouse coords; - var a00 = worldTransform.a, a01 = worldTransform.c, a02 = worldTransform.tx, - a10 = worldTransform.b, a11 = worldTransform.d, a12 = worldTransform.ty, - id = 1 / (a00 * a11 + a01 * -a10); - - point = point || new core.math.Point(); - - point.x = a11 * id * global.x + -a01 * id * global.y + (a12 * a01 - a02 * a11) * id; - point.y = a00 * id * global.y + -a10 * id * global.x + (-a12 * a00 + a02 * a10) * id; - - // set the mouse coords... - return point; -}; - -},{"../core":9}],83:[function(require,module,exports){ -var core = require('../core'), - InteractionData = require('./InteractionData'); - -// TODO: Obviously rewrite this... -var INTERACTION_FREQUENCY = 30; -var AUTO_PREVENT_DEFAULT = true; - -/** - * The interaction manager deals with mouse and touch events. Any DisplayObject can be interactive - * if its interactive parameter is set to true - * This manager also supports multitouch. - * - * @class - * @namespace PIXI - * @param stage {Stage} The stage to handle interactions - */ -function InteractionManager(stage) { - /** - * A reference to the stage - * - * @member {Stage} - */ - this.stage = stage; - - /** - * The mouse data - * - * @member {InteractionData} - */ - this.mouse = new InteractionData(); - - /** - * An object that stores current touches (InteractionData) by id reference - * - * @member {object} - */ - this.touches = {}; - - /** - * @member {Point} - * @private - */ - this.tempPoint = new core.math.Point(); - - /** - * @member {boolean} - * @default - */ - this.mouseoverEnabled = true; - - /** - * Tiny little interactiveData pool ! - * - * @member {Array} - */ - this.pool = []; - - /** - * An array containing all the iterative items from the our interactive tree - * - * @member {Array} - * @private - */ - this.interactiveItems = []; - - /** - * The DOM element to bind to. - * - * @member {HTMLElement} - * @private - */ - this.interactionDOMElement = null; - - /** - * Have events been attached to the dom element? - * - * @member {boolean} - * @private - */ - this.eventsAdded = false; - - //this will make it so that you don't have to call bind all the time - - /** - * @member {Function} - */ - this.onMouseMove = this.onMouseMove.bind( this ); - - /** - * @member {Function} - */ - this.onMouseDown = this.onMouseDown.bind(this); - - /** - * @member {Function} - */ - this.onMouseOut = this.onMouseOut.bind(this); - - /** - * @member {Function} - */ - this.onMouseUp = this.onMouseUp.bind(this); - - /** - * @member {Function} - */ - this.onTouchStart = this.onTouchStart.bind(this); - - /** - * @member {Function} - */ - this.onTouchEnd = this.onTouchEnd.bind(this); - - /** - * @member {Function} - */ - this.onTouchMove = this.onTouchMove.bind(this); - - /** - * @member {number} - */ - this.last = 0; - - /** - * The css style of the cursor that is being used - * @member {string} - */ - this.currentCursorStyle = 'inherit'; - - /** - * Is set to true when the mouse is moved out of the canvas - * @member {boolean} - */ - this.mouseOut = false; - - /** - * @member {number} - */ - this.resolution = 1; - - // used for hit testing - this._tempPoint = new core.math.Point(); -} - -InteractionManager.prototype.constructor = InteractionManager; -module.exports = InteractionManager; - -/** - * Collects an interactive sprite recursively to have their interactions managed - * - * @param displayObject {DisplayObject} the displayObject to collect - * @param iParent {DisplayObject} the display object's parent - * @private - */ -InteractionManager.prototype.collectInteractiveSprite = function (displayObject, iParent) { - var children = displayObject.children; - var length = children.length; - - // make an interaction tree... {item.__interactiveParent} - for (var i = length - 1; i >= 0; i--) { - var child = children[i]; - - // push all interactive bits - if (child._interactive) { - iParent.interactiveChildren = true; - //child.__iParent = iParent; - this.interactiveItems.push(child); - - if (child.children.length > 0) { - this.collectInteractiveSprite(child, child); - } - } - else { - child.__iParent = null; - if (child.children.length > 0) { - this.collectInteractiveSprite(child, iParent); - } - } - - } -}; - -/** - * Sets the DOM element which will receive mouse/touch events. This is useful for when you have - * other DOM elements on top of the renderers Canvas element. With this you'll be bale to deletegate - * another DOM element to receive those events. - * - * @param element {HTMLElement} the DOM element which will receive mouse and touch events. - * @param [resolution=1] {number} THe resolution of the new element (relative to the canvas). - * @private - */ -InteractionManager.prototype.setTargetElement = function (element, resolution) { - this.removeEvents(); - - this.interactionDOMElement = element; - - this.resolution = resolution || 1; - - this.addEvents(); -}; - -/** - * - * @private - */ -InteractionManager.prototype.addEvents = function () { - if (!this.interactionDOMElement) { - return; - } - - if (window.navigator.msPointerEnabled) { - this.interactionDOMElement.style['-ms-content-zooming'] = 'none'; - this.interactionDOMElement.style['-ms-touch-action'] = 'none'; - } - - this.interactionDOMElement.addEventListener('mousemove', this.onMouseMove, true); - this.interactionDOMElement.addEventListener('mousedown', this.onMouseDown, true); - this.interactionDOMElement.addEventListener('mouseout', this.onMouseOut, true); - - this.interactionDOMElement.addEventListener('touchstart', this.onTouchStart, true); - this.interactionDOMElement.addEventListener('touchend', this.onTouchEnd, true); - this.interactionDOMElement.addEventListener('touchmove', this.onTouchMove, true); - - window.addEventListener('mouseup', this.onMouseUp, true); - - this.eventsAdded = true; -}; - -/** - * - * @private - */ -InteractionManager.prototype.removeEvents = function () { - if (!this.interactionDOMElement) { - return; - } - - if (window.navigator.msPointerEnabled) { - this.interactionDOMElement.style['-ms-content-zooming'] = ''; - this.interactionDOMElement.style['-ms-touch-action'] = ''; - } - - this.interactionDOMElement.removeEventListener('mousemove', this.onMouseMove, true); - this.interactionDOMElement.removeEventListener('mousedown', this.onMouseDown, true); - this.interactionDOMElement.removeEventListener('mouseout', this.onMouseOut, true); - - this.interactionDOMElement.removeEventListener('touchstart', this.onTouchStart, true); - this.interactionDOMElement.removeEventListener('touchend', this.onTouchEnd, true); - this.interactionDOMElement.removeEventListener('touchmove', this.onTouchMove, true); - - this.interactionDOMElement = null; - - window.removeEventListener('mouseup', this.onMouseUp, true); - - this.eventsAdded = false; -}; - -/** - * updates the state of interactive objects - * - * @private - */ -InteractionManager.prototype.update = function () { - if (!this.interactionDOMElement) { - return; - } - - // frequency of 30fps?? - var now = Date.now(); - var diff = now - this.last; - diff = (diff * INTERACTION_FREQUENCY ) / 1000; - if (diff < 1) { - return; - } - - this.last = now; - - var i = 0; - - // ok.. so mouse events?? - // yes for now :) - // OPTIMISE - how often to check?? - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - // loop through interactive objects! - var length = this.interactiveItems.length; - var cursor = 'inherit'; - var over = false; - - for (i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - // OPTIMISATION - only calculate every time if the mousemove function exists.. - // OK so.. does the object have any other interactive functions? - // hit-test the clip! - // if (item.mouseover || item.mouseout || item.buttonMode) - // { - // ok so there are some functions so lets hit test it.. - item.__hit = this.hitTest(item, this.mouse); - this.mouse.target = item; - // ok so deal with interactions.. - // looks like there was a hit! - if (item.__hit && !over) { - if (item.buttonMode) { - cursor = item.defaultCursor; - } - - if (!item.interactiveChildren) { - over = true; - } - - if (!item.__isOver) { - if (item.mouseover) { - item.mouseover (this.mouse); - } - item.__isOver = true; - } - } - else { - if (item.__isOver) { - // roll out! - if (item.mouseout) { - item.mouseout (this.mouse); - } - item.__isOver = false; - } - } - } - - if (this.currentCursorStyle !== cursor) { - this.currentCursorStyle = cursor; - this.interactionDOMElement.style.cursor = cursor; - } -}; - -/** - * @private - */ -InteractionManager.prototype.rebuildInteractiveGraph = function () { - this.dirty = false; - - var len = this.interactiveItems.length; - - for (var i = 0; i < len; i++) { - this.interactiveItems[i].interactiveChildren = false; - } - - this.interactiveItems.length = 0; - - if (this.stage.interactive) { - this.interactiveItems.push(this.stage); - } - - // Go through and collect all the objects that are interactive.. - this.collectInteractiveSprite(this.stage, this.stage); -}; - -/** - * Is called when the mouse moves across the renderer element - * - * @param event {Event} The DOM event of the mouse moving - * @private - */ -InteractionManager.prototype.onMouseMove = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - // TODO optimize by not check EVERY TIME! maybe half as often? // - var rect = this.interactionDOMElement.getBoundingClientRect(); - - this.mouse.global.x = (event.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) / this.resolution; - this.mouse.global.y = (event.clientY - rect.top) * ( this.interactionDOMElement.height / rect.height) / this.resolution; - - var length = this.interactiveItems.length; - - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - // Call the function! - if (item.mousemove) { - item.mousemove(this.mouse); - } - } -}; - -/** - * Is called when the mouse button is pressed down on the renderer element - * - * @param event {Event} The DOM event of a mouse button being pressed down - * @private - */ -InteractionManager.prototype.onMouseDown = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - if (AUTO_PREVENT_DEFAULT) { - this.mouse.originalEvent.preventDefault(); - } - - // loop through interaction tree... - // hit test each item! -> - // get interactive items under point?? - //stage.__i - var length = this.interactiveItems.length; - - var e = this.mouse.originalEvent; - var isRightButton = e.button === 2 || e.which === 3; - var downFunction = isRightButton ? 'rightdown' : 'mousedown'; - var clickFunction = isRightButton ? 'rightclick' : 'click'; - var buttonIsDown = isRightButton ? '__rightIsDown' : '__mouseIsDown'; - var isDown = isRightButton ? '__isRightDown' : '__isDown'; - - // while - // hit test - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - if (item[downFunction] || item[clickFunction]) { - item[buttonIsDown] = true; - item.__hit = this.hitTest(item, this.mouse); - - if (item.__hit) { - //call the function! - if (item[downFunction]) { - item[downFunction](this.mouse); - } - item[isDown] = true; - - // just the one! - if (!item.interactiveChildren) { - break; - } - } - } - } -}; - -/** - * Is called when the mouse is moved out of the renderer element - * - * @param event {Event} The DOM event of a mouse being moved out - * @private - */ -InteractionManager.prototype.onMouseOut = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - var length = this.interactiveItems.length; - - this.interactionDOMElement.style.cursor = 'inherit'; - - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - if (item.__isOver) { - this.mouse.target = item; - if (item.mouseout) { - item.mouseout(this.mouse); - } - item.__isOver = false; - } - } - - this.mouseOut = true; - - // move the mouse to an impossible position - this.mouse.global.x = -10000; - this.mouse.global.y = -10000; -}; - -/** - * Is called when the mouse button is released on the renderer element - * - * @param event {Event} The DOM event of a mouse button being released - * @private - */ -InteractionManager.prototype.onMouseUp = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - this.mouse.originalEvent = event; - - var length = this.interactiveItems.length; - var up = false; - - var e = this.mouse.originalEvent; - var isRightButton = e.button === 2 || e.which === 3; - - var upFunction = isRightButton ? 'rightup' : 'mouseup'; - var clickFunction = isRightButton ? 'rightclick' : 'click'; - var upOutsideFunction = isRightButton ? 'rightupoutside' : 'mouseupoutside'; - var isDown = isRightButton ? '__isRightDown' : '__isDown'; - - for (var i = 0; i < length; i++) { - var item = this.interactiveItems[i]; - - if (item[clickFunction] || item[upFunction] || item[upOutsideFunction]) { - item.__hit = this.hitTest(item, this.mouse); - - if (item.__hit && !up) { - //call the function! - if (item[upFunction]) { - item[upFunction](this.mouse); - } - if (item[isDown]) { - if (item[clickFunction]) { - item[clickFunction](this.mouse); - } - } - - if (!item.interactiveChildren) { - up = true; - } - } - else { - if (item[isDown]) { - if (item[upOutsideFunction]) { - item[upOutsideFunction](this.mouse); - } - } - } - - item[isDown] = false; - } - } -}; - -/** - * Tests if the current mouse coordinates hit a sprite - * - * @param item {DisplayObject} The displayObject to test for a hit - * @param interactionData {InteractionData} The interactionData object to update in the case there is a hit - * @private - */ -InteractionManager.prototype.hitTest = function (item, interactionData) { - var global = interactionData.global; - - if (!item.worldVisible) { - return false; - } - - // map the global point to local space. - item.worldTransform.applyInverse(global, this._tempPoint); - - var x = this._tempPoint.x, - y = this._tempPoint.y, - i; - - interactionData.target = item; - - //a sprite or display object with a hit area defined - if (item.hitArea && item.hitArea.contains) { - return item.hitArea.contains(x, y); - } - // a sprite with no hitarea defined - else if (item instanceof core.Sprite) { - var width = item.texture.frame.width; - var height = item.texture.frame.height; - var x1 = -width * item.anchor.x; - var y1; - - if (x > x1 && x < x1 + width) { - y1 = -height * item.anchor.y; - - if (y > y1 && y < y1 + height) { - // set the target property if a hit is true! - return true; - } - } - } - else if (item instanceof core.Graphics) { - var graphicsData = item.graphicsData; - for (i = 0; i < graphicsData.length; i++) { - var data = graphicsData[i]; - - if (!data.fill) { - continue; - } - - // only deal with fills.. - if (data.shape) { - if (data.shape.contains(x, y)) { - //interactionData.target = item; - return true; - } - } - } - } - - var length = item.children.length; - - for (i = 0; i < length; i++) { - var tempItem = item.children[i]; - var hit = this.hitTest(tempItem, interactionData); - if (hit) { - // hmm.. TODO SET CORRECT TARGET? - interactionData.target = item; - return true; - } - } - return false; -}; - -/** - * Is called when a touch is moved across the renderer element - * - * @param event {Event} The DOM event of a touch moving across the renderer view - * @private - */ -InteractionManager.prototype.onTouchMove = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - var rect = this.interactionDOMElement.getBoundingClientRect(); - var changedTouches = event.changedTouches; - var touchData; - var i = 0; - - for (i = 0; i < changedTouches.length; i++) { - var touchEvent = changedTouches[i]; - touchData = this.touches[touchEvent.identifier]; - touchData.originalEvent = event; - - // update the touch position - touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution; - touchData.global.y = ( (touchEvent.clientY - rect.top) * (this.interactionDOMElement.height / rect.height) ) / this.resolution; - if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) { - //Support for CocoonJS fullscreen scale modes - touchData.global.x = touchEvent.clientX; - touchData.global.y = touchEvent.clientY; - } - - for (var j = 0; j < this.interactiveItems.length; j++) { - var item = this.interactiveItems[j]; - if (item.touchmove && item.__touchData && item.__touchData[touchEvent.identifier]) { - item.touchmove(touchData); - } - } - } -}; - -/** - * Is called when a touch is started on the renderer element - * - * @param event {Event} The DOM event of a touch starting on the renderer view - * @private - */ -InteractionManager.prototype.onTouchStart = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - var rect = this.interactionDOMElement.getBoundingClientRect(); - - if (AUTO_PREVENT_DEFAULT) { - event.preventDefault(); - } - - var changedTouches = event.changedTouches; - for (var i=0; i < changedTouches.length; i++) { - var touchEvent = changedTouches[i]; - - var touchData = this.pool.pop(); - if (!touchData) { - touchData = new InteractionData(); - } - - touchData.originalEvent = event; - - this.touches[touchEvent.identifier] = touchData; - touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution; - touchData.global.y = ( (touchEvent.clientY - rect.top) * (this.interactionDOMElement.height / rect.height) ) / this.resolution; - if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) { - //Support for CocoonJS fullscreen scale modes - touchData.global.x = touchEvent.clientX; - touchData.global.y = touchEvent.clientY; - } - - var length = this.interactiveItems.length; - - for (var j = 0; j < length; j++) { - var item = this.interactiveItems[j]; - - if (item.touchstart || item.tap) { - item.__hit = this.hitTest(item, touchData); - - if (item.__hit) { - //call the function! - if (item.touchstart) { - item.touchstart(touchData); - } - - item.__isDown = true; - item.__touchData = item.__touchData || {}; - item.__touchData[touchEvent.identifier] = touchData; - - if (!item.interactiveChildren) { - break; - } - } - } - } - } -}; - -/** - * Is called when a touch is ended on the renderer element - * - * @param event {Event} The DOM event of a touch ending on the renderer view - * @private - */ -InteractionManager.prototype.onTouchEnd = function (event) { - if (this.dirty) { - this.rebuildInteractiveGraph(); - } - - var rect = this.interactionDOMElement.getBoundingClientRect(); - var changedTouches = event.changedTouches; - - for (var i=0; i < changedTouches.length; i++) { - var touchEvent = changedTouches[i]; - var touchData = this.touches[touchEvent.identifier]; - var up = false; - touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution; - touchData.global.y = ( (touchEvent.clientY - rect.top) * (this.interactionDOMElement.height / rect.height) ) / this.resolution; - if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) { - //Support for CocoonJS fullscreen scale modes - touchData.global.x = touchEvent.clientX; - touchData.global.y = touchEvent.clientY; - } - - var length = this.interactiveItems.length; - for (var j = 0; j < length; j++) { - var item = this.interactiveItems[j]; - - if (item.__touchData && item.__touchData[touchEvent.identifier]) { - - item.__hit = this.hitTest(item, item.__touchData[touchEvent.identifier]); - - // so this one WAS down... - touchData.originalEvent = event; - // hitTest?? - - if (item.touchend || item.tap) { - if (item.__hit && !up) { - if (item.touchend) { - item.touchend(touchData); - } - if (item.__isDown && item.tap) { - item.tap(touchData); - } - if (!item.interactiveChildren) { - up = true; - } - } - else { - if (item.__isDown && item.touchendoutside) { - item.touchendoutside(touchData); - } - } - - item.__isDown = false; - } - - item.__touchData[touchEvent.identifier] = null; - } - } - // remove the touch.. - this.pool.push(touchData); - this.touches[touchEvent.identifier] = null; - } -}; - -},{"../core":9,"./InteractionData":82}],84:[function(require,module,exports){ -/** - * @file Main export of the PIXI interactions library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - InteractionData: require('./InteractionData'), - InteractionManager: require('./InteractionManager') -}; - -},{"./InteractionData":82,"./InteractionManager":83}],85:[function(require,module,exports){ -var core = require('../core'), - ImageLoader = require('./ImageLoader'); - -/** - * The atlas file loader is used to load in Texture Atlas data and parse it. When loaded this class will dispatch a 'loaded' event. If loading fails this class will dispatch an 'error' event. - * - * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format. - * - * It is highly recommended to use texture atlases (also know as 'sprite sheets') as it allowed sprites to be batched and drawn together for highly increased rendering speed. - * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId() - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function AtlasLoader(url, crossorigin) { - this.url = url; - this.baseUrl = url.replace(/[^\/]*$/, ''); - this.crossorigin = crossorigin; - this.loaded = false; -} - -AtlasLoader.prototype.constructor = AtlasLoader; -module.exports = AtlasLoader; - -core.utils.eventTarget.mixin(AtlasLoader.prototype); - - /** - * Starts loading the JSON file - * - */ -AtlasLoader.prototype.load = function () { - this.ajaxRequest = new core.utils.AjaxRequest(); - this.ajaxRequest.onreadystatechange = this.onAtlasLoaded.bind(this); - - this.ajaxRequest.open('GET', this.url, true); - - if (this.ajaxRequest.overrideMimeType) { - this.ajaxRequest.overrideMimeType('application/json'); - } - - this.ajaxRequest.send(null); -}; - -/** - * Invoked when the Atlas has fully loaded. Parses the JSON and builds the texture frames. - * - * @private - */ -AtlasLoader.prototype.onAtlasLoaded = function () { - if (this.ajaxRequest.readyState === 4) { - if (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1) { - this.atlas = { - meta : { - image : [] - }, - frames : [] - }; - var result = this.ajaxRequest.responseText.split(/\r?\n/); - var lineCount = -3; - - var currentImageId = 0; - var currentFrame = null; - var nameInNextLine = false; - - var i = 0, - j = 0, - selfOnLoaded = this.onLoaded.bind(this); - - // parser without rotation support yet! - for (i = 0; i < result.length; i++) { - result[i] = result[i].replace(/^\s+|\s+$/g, ''); - - if (result[i] === '') { - nameInNextLine = i+1; - } - - if (result[i].length > 0) { - if (nameInNextLine === i) { - this.atlas.meta.image.push(result[i]); - currentImageId = this.atlas.meta.image.length - 1; - this.atlas.frames.push({}); - lineCount = -3; - } else if (lineCount > 0) { - if (lineCount % 7 === 1) { // frame name - if (currentFrame != null) { //jshint ignore:line - this.atlas.frames[currentImageId][currentFrame.name] = currentFrame; - } - currentFrame = { name: result[i], frame : {} }; - } else { - var text = result[i].split(' '); - if (lineCount % 7 === 3) { // position - currentFrame.frame.x = Number(text[1].replace(',', '')); - currentFrame.frame.y = Number(text[2]); - } else if (lineCount % 7 === 4) { // size - currentFrame.frame.w = Number(text[1].replace(',', '')); - currentFrame.frame.h = Number(text[2]); - } else if (lineCount % 7 === 5) { // real size - var realSize = { - x : 0, - y : 0, - w : Number(text[1].replace(',', '')), - h : Number(text[2]) - }; - - if (realSize.w > currentFrame.frame.w || realSize.h > currentFrame.frame.h) { - currentFrame.trimmed = true; - currentFrame.realSize = realSize; - } else { - currentFrame.trimmed = false; - } - } - } - } - lineCount++; - } - } - - if (currentFrame != null) { //jshint ignore:line - this.atlas.frames[currentImageId][currentFrame.name] = currentFrame; - } - - if (this.atlas.meta.image.length > 0) { - this.images = []; - for (j = 0; j < this.atlas.meta.image.length; j++) { - // sprite sheet - var textureUrl = this.baseUrl + this.atlas.meta.image[j]; - var frameData = this.atlas.frames[j]; - this.images.push(new ImageLoader(textureUrl, this.crossorigin)); - - for (i in frameData) { - var rect = frameData[i].frame; - if (rect) { - core.utils.TextureCache[i] = new core.Texture(this.images[j].texture.baseTexture, { - x: rect.x, - y: rect.y, - width: rect.w, - height: rect.h - }); - if (frameData[i].trimmed) { - core.utils.TextureCache[i].realSize = frameData[i].realSize; - // trim in pixi not supported yet, todo update trim properties if it is done ... - core.utils.TextureCache[i].trim.x = 0; - core.utils.TextureCache[i].trim.y = 0; - } - } - } - } - - this.currentImageId = 0; - for (j = 0; j < this.images.length; j++) { - this.images[j].on('loaded', selfOnLoaded); - } - this.images[this.currentImageId].load(); - - } else { - this.onLoaded(); - } - - } else { - this.onError(); - } - } -}; - -/** - * Invoked when json file has loaded. - * - * @private - */ -AtlasLoader.prototype.onLoaded = function () { - if (this.images.length - 1 > this.currentImageId) { - this.currentImageId++; - this.images[this.currentImageId].load(); - } else { - this.loaded = true; - this.emit('loaded', { content: this }); - } -}; - -/** - * Invoked when an error occurs. - * - * @private - */ -AtlasLoader.prototype.onError = function () { - this.emit('error', { content: this }); -}; - -},{"../core":9,"./ImageLoader":87}],86:[function(require,module,exports){ -var core = require('../core'), - ImageLoader = require('./ImageLoader'); - -/** - * The xml loader is used to load in XML bitmap font data ('xml' or 'fnt') - * To generate the data you can use http://www.angelcode.com/products/bmfont/ - * This loader will also load the image file as the data. - * When loaded this class will dispatch a 'loaded' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the sprite sheet JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function BitmapFontLoader(url, crossorigin) { - /** - * The url of the bitmap font data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * The base url of the bitmap font data - * - * @member {String} - * @readOnly - */ - this.baseUrl = url.replace(/[^\/]*$/, ''); - - /** - * The texture of the bitmap font - * - * @member {Texture} - */ - this.texture = null; -} - -// constructor -BitmapFontLoader.prototype.constructor = BitmapFontLoader; -module.exports = BitmapFontLoader; - -core.utils.eventTarget.mixin(BitmapFontLoader.prototype); - -/** - * Loads the XML font data - * - */ -BitmapFontLoader.prototype.load = function () { - this.ajaxRequest = new core.utils.AjaxRequest(); - this.ajaxRequest.onreadystatechange = this.onXMLLoaded.bind(this); - - this.ajaxRequest.open('GET', this.url, true); - - if (this.ajaxRequest.overrideMimeType) { - this.ajaxRequest.overrideMimeType('application/xml'); - } - - this.ajaxRequest.send(null); -}; - -/** - * Invoked when the XML file is loaded, parses the data. - * - * @private - */ -BitmapFontLoader.prototype.onXMLLoaded = function () { - if (this.ajaxRequest.readyState === 4) { - if (this.ajaxRequest.status === 200 || window.location.protocol.indexOf('http') === -1) { - var responseXML = this.ajaxRequest.responseXML; - if (!responseXML || /MSIE 9/i.test(navigator.userAgent) || navigator.isCocoonJS) { - if (typeof(window.DOMParser) === 'function') { - var domparser = new DOMParser(); - responseXML = domparser.parseFromString(this.ajaxRequest.responseText, 'text/xml'); - } else { - var div = document.createElement('div'); - div.innerHTML = this.ajaxRequest.responseText; - responseXML = div; - } - } - - var textureUrl = this.baseUrl + responseXML.getElementsByTagName('page')[0].getAttribute('file'); - var image = new ImageLoader(textureUrl, this.crossorigin); - this.texture = image.texture.baseTexture; - - var data = {}; - var info = responseXML.getElementsByTagName('info')[0]; - var common = responseXML.getElementsByTagName('common')[0]; - data.font = info.getAttribute('face'); - data.size = parseInt(info.getAttribute('size'), 10); - data.lineHeight = parseInt(common.getAttribute('lineHeight'), 10); - data.chars = {}; - - //parse letters - var letters = responseXML.getElementsByTagName('char'); - - for (var i = 0; i < letters.length; i++) { - var charCode = parseInt(letters[i].getAttribute('id'), 10); - - var textureRect = new core.math.Rectangle( - parseInt(letters[i].getAttribute('x'), 10), - parseInt(letters[i].getAttribute('y'), 10), - parseInt(letters[i].getAttribute('width'), 10), - parseInt(letters[i].getAttribute('height'), 10) - ); - - data.chars[charCode] = { - xOffset: parseInt(letters[i].getAttribute('xoffset'), 10), - yOffset: parseInt(letters[i].getAttribute('yoffset'), 10), - xAdvance: parseInt(letters[i].getAttribute('xadvance'), 10), - kerning: {}, - texture: core.utils.TextureCache[charCode] = new core.Texture(this.texture, textureRect) - - }; - } - - //parse kernings - var kernings = responseXML.getElementsByTagName('kerning'); - for (i = 0; i < kernings.length; i++) { - var first = parseInt(kernings[i].getAttribute('first'), 10); - var second = parseInt(kernings[i].getAttribute('second'), 10); - var amount = parseInt(kernings[i].getAttribute('amount'), 10); - - data.chars[second].kerning[first] = amount; - - } - - core.BitmapText.fonts[data.font] = data; - - image.addEventListener('loaded', this.onLoaded.bind(this)); - image.load(); - } - } -}; - -/** - * Invoked when all files are loaded (xml/fnt and texture) - * - * @private - */ -BitmapFontLoader.prototype.onLoaded = function () { - this.emit('loaded', { content: this }); -}; - -},{"../core":9,"./ImageLoader":87}],87:[function(require,module,exports){ -var core = require('../core'); - -/** - * The image loader class is responsible for loading images file formats ('jpeg', 'jpg', 'png' and 'gif') - * Once the image has been loaded it is stored in the PIXI texture cache and can be accessed though Texture.fromFrame() and Sprite.fromFrame() - * When loaded this class will dispatch a 'loaded' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the image - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function ImageLoader(url, crossorigin) { - /** - * The texture being loaded - * - * @member {Texture} - */ - this.texture = core.Texture.fromImage(url, crossorigin); - - /** - * if the image is loaded with loadFramedSpriteSheet - * frames will contain the sprite sheet frames - * - * @member {Array} - * @readOnly - */ - this.frames = []; -} - -// constructor -ImageLoader.prototype.constructor = ImageLoader; -module.exports = ImageLoader; - -core.utils.eventTarget.mixin(ImageLoader.prototype); - -/** - * Loads image or takes it from cache - * - */ -ImageLoader.prototype.load = function () { - if (!this.texture.baseTexture.hasLoaded) { - this.texture.baseTexture.on('loaded', this.onLoaded.bind(this)); - this.texture.baseTexture.on('error', this.onError.bind(this)); - } - else { - this.onLoaded(); - } -}; - -/** - * Invoked when image file is loaded or it is already cached and ready to use - * - * @private - */ -ImageLoader.prototype.onLoaded = function () { - this.emit('loaded', { content: this }); -}; - -/** - * Invoked when image file failed loading - * - * @method onError - * @private - */ -ImageLoader.prototype.onError = function () { - this.emit('error', { content: this }); -}; - -/** - * Loads image and split it to uniform sized frames - * - * @param frameWidth {number} width of each frame - * @param frameHeight {number} height of each frame - * @param textureName {String} if given, the frames will be cached in - format - */ -ImageLoader.prototype.loadFramedSpriteSheet = function (frameWidth, frameHeight, textureName) { - this.frames = []; - - var cols = Math.floor(this.texture.width / frameWidth); - var rows = Math.floor(this.texture.height / frameHeight); - - var i=0; - for (var y = 0; y < rows; ++y) { - for (var x = 0; x < cols; ++x, ++i) { - var texture = new core.Texture( - this.texture.baseTexture, - new core.math.Rectangle( - x * frameWidth, - y * frameHeight, - frameWidth, - frameHeight - ) - ); - - this.frames.push(texture); - - if (textureName) { - core.utils.TextureCache[textureName + '-' + i] = texture; - } - } - } - - this.load(); -}; - -},{"../core":9}],88:[function(require,module,exports){ -var core = require('../core'), - spine = require('../spine/SpineRuntime'), - ImageLoader = require('./ImageLoader'), - SpineTextureLoader = require('./SpineTextureLoader'); - -/** - * The json file loader is used to load in JSON data and parse it - * When loaded this class will dispatch a 'loaded' event - * If loading fails this class will dispatch an 'error' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function JsonLoader(url, crossorigin) { - /** - * The url of the bitmap font data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * The base url of the bitmap font data - * - * @member {String} - * @readOnly - */ - this.baseUrl = url.replace(/[^\/]*$/, ''); - - /** - * Whether the data has loaded yet - * - * @member {boolean} - * @readOnly - */ - this.loaded = false; -} - -// constructor -JsonLoader.prototype.constructor = JsonLoader; -module.exports = JsonLoader; - -core.utils.eventTarget.mixin(JsonLoader.prototype); - -/** - * Loads the JSON data - * - */ -JsonLoader.prototype.load = function () { - if (window.XDomainRequest && this.crossorigin) { - this.ajaxRequest = new window.XDomainRequest(); - - // XDomainRequest has a few quirks. Occasionally it will abort requests - // A way to avoid this is to make sure ALL callbacks are set even if not used - // More info here: http://stackoverflow.com/questions/15786966/xdomainrequest-aborts-post-on-ie-9 - this.ajaxRequest.timeout = 3000; - - this.ajaxRequest.onerror = this.onError.bind(this); - this.ajaxRequest.ontimeout = this.onError.bind(this); - - this.ajaxRequest.onprogress = function () {}; - - this.ajaxRequest.onload = this.onJSONLoaded.bind(this); - } - else { - if (window.XMLHttpRequest) { - this.ajaxRequest = new window.XMLHttpRequest(); - } - else { - this.ajaxRequest = new window.ActiveXObject('Microsoft.XMLHTTP'); - } - - this.ajaxRequest.onreadystatechange = this.onReadyStateChanged.bind(this); - } - - this.ajaxRequest.open('GET',this.url,true); - - this.ajaxRequest.send(); -}; - -/** - * Bridge function to be able to use the more reliable onreadystatechange in XMLHttpRequest. - * - * @private - */ -JsonLoader.prototype.onReadyStateChanged = function () { - if (this.ajaxRequest.readyState === 4 && (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1)) { - this.onJSONLoaded(); - } -}; - -/** - * Invoke when JSON file is loaded - * - * @private - */ -JsonLoader.prototype.onJSONLoaded = function () { - if (!this.ajaxRequest.responseText) { - this.onError(); - return; - } - - this.json = JSON.parse(this.ajaxRequest.responseText); - - if (this.json.frames) { - // sprite sheet - var textureUrl = this.baseUrl + this.json.meta.image; - var image = new ImageLoader(textureUrl, this.crossorigin); - var frameData = this.json.frames; - - this.texture = image.texture.baseTexture; - image.addEventListener('loaded', this.onLoaded.bind(this)); - image.addEventListener('error', this.onError.bind(this)); - - for (var i in frameData) { - var rect = frameData[i].frame; - - if (rect) { - var textureSize = new core.math.Rectangle(rect.x, rect.y, rect.w, rect.h); - var crop = textureSize.clone(); - var trim = null; - - // Check to see if the sprite is trimmed - if (frameData[i].trimmed) { - var actualSize = frameData[i].sourceSize; - var realSize = frameData[i].spriteSourceSize; - trim = new core.math.Rectangle(realSize.x, realSize.y, actualSize.w, actualSize.h); - } - core.utils.TextureCache[i] = new core.Texture(this.texture, textureSize, crop, trim); - } - } - - image.load(); - - } - else if (this.json.bones) { - // check if the json was loaded before - if (core.utils.AnimCache[this.url]) { - this.onLoaded(); - } - else { - /** - * use a bit of hackery to load the atlas file, here we assume that the .json, .atlas and .png files - * that correspond to the spine file are in the same base URL and that the .json and .atlas files - * have the same name - */ - var atlasPath = this.url.substr(0, this.url.lastIndexOf('.')) + '.atlas'; - var atlasLoader = new JsonLoader(atlasPath, this.crossorigin); - // save a copy of the current object for future reference // - var originalLoader = this; - // before loading the file, replace the "onJSONLoaded" function for our own // - atlasLoader.onJSONLoaded = function () { - // at this point "this" points at the atlasLoader (JsonLoader) instance // - if (!this.ajaxRequest.responseText) { - this.onError(); // FIXME: hmm, this is funny because we are not responding to errors yet - return; - } - // create a new instance of a spine texture loader for this spine object // - var textureLoader = new SpineTextureLoader(this.url.substring(0, this.url.lastIndexOf('/'))); - // create a spine atlas using the loaded text and a spine texture loader instance // - var spineAtlas = new spine.Atlas(this.ajaxRequest.responseText, textureLoader); - // now we use an atlas attachment loader // - var attachmentLoader = new spine.AtlasAttachmentLoader(spineAtlas); - // spine animation - var spineJsonParser = new spine.SkeletonJson(attachmentLoader); - var skeletonData = spineJsonParser.readSkeletonData(originalLoader.json); - core.utils.AnimCache[originalLoader.url] = skeletonData; - originalLoader.spine = skeletonData; - originalLoader.spineAtlas = spineAtlas; - originalLoader.spineAtlasLoader = atlasLoader; - // wait for textures to finish loading if needed - if (textureLoader.loadingCount > 0) { - textureLoader.addEventListener('loadedBaseTexture', function (evt){ - if (evt.content.content.loadingCount <= 0) { - originalLoader.onLoaded(); - } - }); - } - else { - originalLoader.onLoaded(); - } - }; - // start the loading // - atlasLoader.load(); - } - } - else { - this.onLoaded(); - } -}; - -/** - * Invoke when json file loaded - * - * @private - */ -JsonLoader.prototype.onLoaded = function () { - this.loaded = true; - this.dispatchEvent({ - type: 'loaded', - content: this - }); -}; - -/** - * Invoke when error occured - * - * @private - */ -JsonLoader.prototype.onError = function () { - - this.dispatchEvent({ - type: 'error', - content: this - }); -}; - -},{"../core":9,"../spine/SpineRuntime":94,"./ImageLoader":87,"./SpineTextureLoader":90}],89:[function(require,module,exports){ -var core = require('../core'), - JsonLoader = require('./JsonLoader'); - -/** - * @author Mat Groves http://matgroves.com/ @Doormat23 - * based on pixi impact spine implementation made by Eemeli Kelokorpi (@ekelokorpi) https://github.com/ekelokorpi - * - * Awesome JS run time provided by EsotericSoftware - * https://github.com/EsotericSoftware/spine-runtimes - * - */ - -/** - * The Spine loader is used to load in JSON spine data - * To generate the data you need to use http://esotericsoftware.com/ and export in the "JSON" format - * Due to a clash of names You will need to change the extension of the spine file from *.json to *.anim for it to load - * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source - * You will need to generate a sprite sheet to accompany the spine data - * When loaded this class will dispatch a "loaded" event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function SpineLoader(url, crossorigin) { - /** - * The url of the bitmap font data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * Whether the data has loaded yet - * - * @member {boolean} - * @readOnly - */ - this.loaded = false; -} - -SpineLoader.prototype.constructor = SpineLoader; -module.exports = SpineLoader; - -core.utils.eventTarget.mixin(SpineLoader.prototype); - -/** - * Loads the JSON data - * - */ -SpineLoader.prototype.load = function () { - var scope = this; - var jsonLoader = new JsonLoader(this.url, this.crossorigin); - - jsonLoader.on('loaded', function (event) { - scope.json = event.data.content.json; - scope.onLoaded(); - }); - - jsonLoader.load(); -}; - -/** - * Invoked when JSON file is loaded. - * - * @private - */ -SpineLoader.prototype.onLoaded = function () { - this.loaded = true; - this.emit('loaded', { content: this }); -}; - -},{"../core":9,"./JsonLoader":88}],90:[function(require,module,exports){ -var core = require('../core'); - -/** - * Supporting class to load images from spine atlases as per spine spec. - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param basePath {string} Tha base path where to look for the images to be loaded - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function SpineTextureLoader(basePath, crossorigin) { - this.basePath = basePath; - this.crossorigin = crossorigin; - this.loadingCount = 0; -} - -SpineTextureLoader.prototype.constructor = SpineTextureLoader; -module.exports = SpineTextureLoader; - -core.utils.eventTarget.mixin(SpineTextureLoader.prototype); - -/** - * Starts loading a base texture as per spine specification - * - * @param page {spine.AtlasPage} Atlas page to which texture belongs - * @param file {string} The file to load, this is just the file path relative to the base path configured in the constructor - */ -SpineTextureLoader.prototype.load = function (page, file) { - page.rendererObject = core.BaseTexture.fromImage(this.basePath + '/' + file, this.crossorigin); - if (!page.rendererObject.hasLoaded) { - var scope = this; - ++scope.loadingCount; - page.rendererObject.addEventListener('loaded', function (){ - --scope.loadingCount; - scope.dispatchEvent({ - type: 'loadedBaseTexture', - content: scope - }); - }); - } -}; - -/** - * Unloads a previously loaded texture as per spine specification - * - * @param texture {BaseTexture} Texture object to destroy - */ -SpineTextureLoader.prototype.unload = function (texture) { - texture.destroy(true); -}; - -},{"../core":9}],91:[function(require,module,exports){ -var core = require('../core'), - JsonLoader = require('./JsonLoader'); - -/** - * The sprite sheet loader is used to load in JSON sprite sheet data - * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format - * There is a free version so thats nice, although the paid version is great value for money. - * It is highly recommended to use Sprite sheets (also know as a 'texture atlas') as it means sprites can be batched and drawn together for highly increased rendering speed. - * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId() - * This loader will load the image file that the Spritesheet points to as well as the data. - * When loaded this class will dispatch a 'loaded' event - * - * @class - * @mixes eventTarget - * @namespace PIXI - * @param url {String} The url of the sprite sheet JSON file - * @param crossorigin {boolean} Whether requests should be treated as crossorigin - */ -function SpriteSheetLoader(url, crossorigin) { - - /** - * The url of the atlas data - * - * @member {String} - */ - this.url = url; - - /** - * Whether the requests should be treated as cross origin - * - * @member {boolean} - */ - this.crossorigin = crossorigin; - - /** - * The base url of the bitmap font data - * - * @member {String} - * @readOnly - */ - this.baseUrl = url.replace(/[^\/]*$/, ''); - - /** - * The texture being loaded - * - * @member {Texture} - */ - this.texture = null; - - /** - * The frames of the sprite sheet - * - * @member {object} - */ - this.frames = {}; -} - -// constructor -SpriteSheetLoader.prototype.constructor = SpriteSheetLoader; -module.exports = SpriteSheetLoader; - -core.utils.eventTarget.mixin(SpriteSheetLoader.prototype); - -/** - * This will begin loading the JSON file - * - */ -SpriteSheetLoader.prototype.load = function () { - var scope = this; - var jsonLoader = new JsonLoader(this.url, this.crossorigin); - - jsonLoader.on('loaded', function (event) { - scope.json = event.data.content.json; - scope.onLoaded(); - }); - - jsonLoader.load(); -}; - -/** - * Invoke when all files are loaded (json and texture) - * - * @private - */ -SpriteSheetLoader.prototype.onLoaded = function () { - this.emit('loaded', { - content: this - }); -}; - -},{"../core":9,"./JsonLoader":88}],92:[function(require,module,exports){ -/** - * @file Main export of the PIXI loaders library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - AtlasLoader: require('./AtlasLoader'), - BitmapFontLoader: require('./BitmapFontLoader'), - ImageLoader: require('./ImageLoader'), - JsonLoader: require('./JsonLoader'), - SpineLoader: require('./SpineLoader'), - SpriteSheetLoader: require('./SpriteSheetLoader') -}; - -},{"./AtlasLoader":85,"./BitmapFontLoader":86,"./ImageLoader":87,"./JsonLoader":88,"./SpineLoader":89,"./SpriteSheetLoader":91}],93:[function(require,module,exports){ -var core = require('../core'), - spine = require('./SpineRuntime'); - -/* Esoteric Software SPINE wrapper for pixi.js */ - -spine.Bone.yDown = true; - -/** - * A class that enables the you to import and run your spine animations in pixi. - * Spine animation data needs to be loaded using the AssetLoader or SpineLoader before it can be used by this class - * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - * @param url {string} The url of the spine anim file to be used - */ -function Spine(url) { - core.DisplayObjectContainer.call(this); - - this.spineData = core.utils.AnimCache[url]; - - if (!this.spineData) { - throw new Error('Spine data must be preloaded using SpineLoader or AssetLoader: ' + url); - } - - this.skeleton = new spine.Skeleton(this.spineData); - this.skeleton.updateWorldTransform(); - - this.stateData = new spine.AnimationStateData(this.spineData); - this.state = new spine.AnimationState(this.stateData); - - this.slotContainers = []; - - for (var i = 0, n = this.skeleton.drawOrder.length; i < n; i++) { - var slot = this.skeleton.drawOrder[i]; - var attachment = slot.attachment; - var slotContainer = new core.DisplayObjectContainer(); - this.slotContainers.push(slotContainer); - this.addChild(slotContainer); - - if (attachment instanceof spine.RegionAttachment) { - var spriteName = attachment.rendererObject.name; - var sprite = this.createSprite(slot, attachment); - slot.currentSprite = sprite; - slot.currentSpriteName = spriteName; - slotContainer.addChild(sprite); - } - else if (attachment instanceof spine.MeshAttachment) { - var mesh = this.createMesh(slot, attachment); - slot.currentMesh = mesh; - slot.currentMeshName = attachment.name; - slotContainer.addChild(mesh); - } - else { - continue; - } - - } - - this.autoUpdate = true; -} - -Spine.prototype = Object.create(core.DisplayObjectContainer.prototype); -Spine.prototype.constructor = Spine; -module.exports = Spine; - -Object.defineProperties(Spine.prototype, { - /** - * If this flag is set to true, the spine animation will be autoupdated every time - * the object id drawn. The down side of this approach is that the delta time is - * automatically calculated and you could miss out on cool effects like slow motion, - * pause, skip ahead and the sorts. Most of these effects can be achieved even with - * autoupdate enabled but are harder to achieve. - * - * @member {boolean} - * @memberof Spine# - * @default true - */ - autoUpdate: { - get: function () { - return (this.updateTransform === Spine.prototype.autoUpdateTransform); - }, - - set: function (value) { - this.updateTransform = value ? Spine.prototype.autoUpdateTransform : core.DisplayObjectContainer.prototype.updateTransform; - } - } -}); - -/** - * Update the spine skeleton and its animations by delta time (dt) - * - * @param dt {number} Delta time. Time by which the animation should be updated - */ -Spine.prototype.update = function (dt) { - this.state.update(dt); - this.state.apply(this.skeleton); - this.skeleton.updateWorldTransform(); - - var drawOrder = this.skeleton.drawOrder; - for (var i = 0, n = drawOrder.length; i < n; i++) { - var slot = drawOrder[i]; - var attachment = slot.attachment; - var slotContainer = this.slotContainers[i]; - - if (!attachment) { - slotContainer.visible = false; - continue; - } - - var type = attachment.type; - if (type === spine.AttachmentType.region) { - if (attachment.rendererObject) { - if (!slot.currentSpriteName || slot.currentSpriteName !== attachment.rendererObject.name) { - var spriteName = attachment.rendererObject.name; - if (slot.currentSprite !== undefined) { - slot.currentSprite.visible = false; - } - slot.sprites = slot.sprites || {}; - if (slot.sprites[spriteName] !== undefined) { - slot.sprites[spriteName].visible = true; - } - else { - var sprite = this.createSprite(slot, attachment); - slotContainer.addChild(sprite); - } - slot.currentSprite = slot.sprites[spriteName]; - slot.currentSpriteName = spriteName; - } - } - - var bone = slot.bone; - - slotContainer.position.x = bone.worldX + attachment.x * bone.m00 + attachment.y * bone.m01; - slotContainer.position.y = bone.worldY + attachment.x * bone.m10 + attachment.y * bone.m11; - slotContainer.scale.x = bone.worldScaleX; - slotContainer.scale.y = bone.worldScaleY; - - slotContainer.rotation = -(slot.bone.worldRotation * spine.degRad); - - slot.currentSprite.tint = core.utils.rgb2hex([slot.r,slot.g,slot.b]); - } - else if (type === spine.AttachmentType.skinnedmesh) { - if (!slot.currentMeshName || slot.currentMeshName !== attachment.name) { - var meshName = attachment.name; - if (slot.currentMesh !== undefined) { - slot.currentMesh.visible = false; - } - - slot.meshes = slot.meshes || {}; - - if (slot.meshes[meshName] !== undefined) { - slot.meshes[meshName].visible = true; - } - else { - var mesh = this.createMesh(slot, attachment); - slotContainer.addChild(mesh); - } - - slot.currentMesh = slot.meshes[meshName]; - slot.currentMeshName = meshName; - } - - attachment.computeWorldVertices(slot.bone.skeleton.x, slot.bone.skeleton.y, slot, slot.currentMesh.vertices); - - } - else { - slotContainer.visible = false; - continue; - } - slotContainer.visible = true; - - slotContainer.alpha = slot.a; - } -}; - -/** - * When autoupdate is set to yes this function is used as pixi's updateTransform function - * - * @private - */ -Spine.prototype.autoUpdateTransform = function () { - this.lastTime = this.lastTime || Date.now(); - var timeDelta = (Date.now() - this.lastTime) * 0.001; - this.lastTime = Date.now(); - - this.update(timeDelta); - - core.DisplayObjectContainer.prototype.updateTransform.call(this); -}; - -/** - * Create a new sprite to be used with spine.RegionAttachment - * - * @param slot {spine.Slot} The slot to which the attachment is parented - * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent - * @private - */ -Spine.prototype.createSprite = function (slot, attachment) { - var descriptor = attachment.rendererObject; - var baseTexture = descriptor.page.rendererObject; - var spriteRect = new core.math.Rectangle(descriptor.x, - descriptor.y, - descriptor.rotate ? descriptor.height : descriptor.width, - descriptor.rotate ? descriptor.width : descriptor.height); - var spriteTexture = new core.Texture(baseTexture, spriteRect); - var sprite = new core.Sprite(spriteTexture); - - var baseRotation = descriptor.rotate ? Math.PI * 0.5 : 0.0; - sprite.scale.set(descriptor.width / descriptor.originalWidth, descriptor.height / descriptor.originalHeight); - sprite.rotation = baseRotation - (attachment.rotation * spine.degRad); - sprite.anchor.x = sprite.anchor.y = 0.5; - - slot.sprites = slot.sprites || {}; - slot.sprites[descriptor.name] = sprite; - return sprite; -}; - -/** - * - * @param slot {spine.Slot} The slot to which the attachment is parented - * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent - * @private - */ -Spine.prototype.createMesh = function (slot, attachment) { - var descriptor = attachment.rendererObject; - var baseTexture = descriptor.page.rendererObject; - var texture = new core.Texture(baseTexture); - - var strip = new core.Strip(texture); - strip.drawMode = core.Strip.DrawModes.TRIANGLES; - strip.canvasPadding = 1.5; - - strip.vertices = new Float32Array(attachment.uvs.length); - strip.uvs = attachment.uvs; - strip.indices = attachment.triangles; - - slot.meshes = slot.meshes || {}; - slot.meshes[attachment.name] = strip; - - return strip; -}; - -},{"../core":9,"./SpineRuntime":94}],94:[function(require,module,exports){ -/****************************************************************************** - * Spine Runtimes Software License - * Version 2.1 - * - * Copyright (c) 2013, Esoteric Software - * All rights reserved. - * - * You are granted a perpetual, non-exclusive, non-sublicensable and - * non-transferable license to install, execute and perform the Spine Runtimes - * Software (the "Software") solely for internal use. Without the written - * permission of Esoteric Software (typically granted by licensing Spine), you - * may not (a) modify, translate, adapt or otherwise create derivative works, - * improvements of the Software or develop new applications using the Software - * or (b) remove, delete, alter or obscure any trademarks or any copyright, - * trademark, patent or other intellectual property or proprietary rights - * notices on or in the Software, including any copy thereof. Redistributions - * in binary or source form must include this license and terms. - * - * THIS SOFTWARE IS PROVIDED BY ESOTERIC SOFTWARE "AS IS" AND ANY EXPRESS OR - * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF - * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO - * EVENT SHALL ESOTERIC SOFTARE BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, - * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, - * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; - * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, - * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR - * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF - * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. - *****************************************************************************/ - -var spine = module.exports = { - radDeg: 180 / Math.PI, - degRad: Math.PI / 180, - temp: [], - Float32Array: (typeof(Float32Array) === 'undefined') ? Array : Float32Array, - Uint16Array: (typeof(Uint16Array) === 'undefined') ? Array : Uint16Array -}; - -spine.BoneData = function (name, parent) { - this.name = name; - this.parent = parent; -}; -spine.BoneData.prototype = { - length: 0, - x: 0, y: 0, - rotation: 0, - scaleX: 1, scaleY: 1, - inheritScale: true, - inheritRotation: true, - flipX: false, flipY: false -}; - -spine.SlotData = function (name, boneData) { - this.name = name; - this.boneData = boneData; -}; -spine.SlotData.prototype = { - r: 1, g: 1, b: 1, a: 1, - attachmentName: null, - additiveBlending: false -}; - -spine.IkConstraintData = function (name) { - this.name = name; - this.bones = []; -}; -spine.IkConstraintData.prototype = { - target: null, - bendDirection: 1, - mix: 1 -}; - -spine.Bone = function (boneData, skeleton, parent) { - this.data = boneData; - this.skeleton = skeleton; - this.parent = parent; - this.setToSetupPose(); -}; -spine.Bone.yDown = false; -spine.Bone.prototype = { - x: 0, y: 0, - rotation: 0, rotationIK: 0, - scaleX: 1, scaleY: 1, - flipX: false, flipY: false, - m00: 0, m01: 0, worldX: 0, // a b x - m10: 0, m11: 0, worldY: 0, // c d y - worldRotation: 0, - worldScaleX: 1, worldScaleY: 1, - worldFlipX: false, worldFlipY: false, - updateWorldTransform: function () { - var parent = this.parent; - if (parent) { - this.worldX = this.x * parent.m00 + this.y * parent.m01 + parent.worldX; - this.worldY = this.x * parent.m10 + this.y * parent.m11 + parent.worldY; - if (this.data.inheritScale) { - this.worldScaleX = parent.worldScaleX * this.scaleX; - this.worldScaleY = parent.worldScaleY * this.scaleY; - } else { - this.worldScaleX = this.scaleX; - this.worldScaleY = this.scaleY; - } - this.worldRotation = this.data.inheritRotation ? (parent.worldRotation + this.rotationIK) : this.rotationIK; - this.worldFlipX = parent.worldFlipX != this.flipX; - this.worldFlipY = parent.worldFlipY != this.flipY; - } else { - var skeletonFlipX = this.skeleton.flipX, skeletonFlipY = this.skeleton.flipY; - this.worldX = skeletonFlipX ? -this.x : this.x; - this.worldY = (skeletonFlipY != spine.Bone.yDown) ? -this.y : this.y; - this.worldScaleX = this.scaleX; - this.worldScaleY = this.scaleY; - this.worldRotation = this.rotationIK; - this.worldFlipX = skeletonFlipX != this.flipX; - this.worldFlipY = skeletonFlipY != this.flipY; - } - var radians = this.worldRotation * spine.degRad; - var cos = Math.cos(radians); - var sin = Math.sin(radians); - if (this.worldFlipX) { - this.m00 = -cos * this.worldScaleX; - this.m01 = sin * this.worldScaleY; - } else { - this.m00 = cos * this.worldScaleX; - this.m01 = -sin * this.worldScaleY; - } - if (this.worldFlipY != spine.Bone.yDown) { - this.m10 = -sin * this.worldScaleX; - this.m11 = -cos * this.worldScaleY; - } else { - this.m10 = sin * this.worldScaleX; - this.m11 = cos * this.worldScaleY; - } - }, - setToSetupPose: function () { - var data = this.data; - this.x = data.x; - this.y = data.y; - this.rotation = data.rotation; - this.rotationIK = this.rotation; - this.scaleX = data.scaleX; - this.scaleY = data.scaleY; - this.flipX = data.flipX; - this.flipY = data.flipY; - }, - worldToLocal: function (world) { - var dx = world[0] - this.worldX, dy = world[1] - this.worldY; - var m00 = this.m00, m10 = this.m10, m01 = this.m01, m11 = this.m11; - if (this.worldFlipX != (this.worldFlipY != spine.Bone.yDown)) { - m00 = -m00; - m11 = -m11; - } - var invDet = 1 / (m00 * m11 - m01 * m10); - world[0] = dx * m00 * invDet - dy * m01 * invDet; - world[1] = dy * m11 * invDet - dx * m10 * invDet; - }, - localToWorld: function (local) { - var localX = local[0], localY = local[1]; - local[0] = localX * this.m00 + localY * this.m01 + this.worldX; - local[1] = localX * this.m10 + localY * this.m11 + this.worldY; - } -}; - -spine.Slot = function (slotData, bone) { - this.data = slotData; - this.bone = bone; - this.setToSetupPose(); -}; -spine.Slot.prototype = { - r: 1, g: 1, b: 1, a: 1, - _attachmentTime: 0, - attachment: null, - attachmentVertices: [], - setAttachment: function (attachment) { - this.attachment = attachment; - this._attachmentTime = this.bone.skeleton.time; - this.attachmentVertices.length = 0; - }, - setAttachmentTime: function (time) { - this._attachmentTime = this.bone.skeleton.time - time; - }, - getAttachmentTime: function () { - return this.bone.skeleton.time - this._attachmentTime; - }, - setToSetupPose: function () { - var data = this.data; - this.r = data.r; - this.g = data.g; - this.b = data.b; - this.a = data.a; - - var slotDatas = this.bone.skeleton.data.slots; - for (var i = 0, n = slotDatas.length; i < n; i++) { - if (slotDatas[i] == data) { - this.setAttachment(!data.attachmentName ? null : this.bone.skeleton.getAttachmentBySlotIndex(i, data.attachmentName)); - break; - } - } - } -}; - -spine.IkConstraint = function (data, skeleton) { - this.data = data; - this.mix = data.mix; - this.bendDirection = data.bendDirection; - - this.bones = []; - for (var i = 0, n = data.bones.length; i < n; i++) - this.bones.push(skeleton.findBone(data.bones[i].name)); - this.target = skeleton.findBone(data.target.name); -}; -spine.IkConstraint.prototype = { - apply: function () { - var target = this.target; - var bones = this.bones; - switch (bones.length) { - case 1: - spine.IkConstraint.apply1(bones[0], target.worldX, target.worldY, this.mix); - break; - case 2: - spine.IkConstraint.apply2(bones[0], bones[1], target.worldX, target.worldY, this.bendDirection, this.mix); - break; - } - } -}; -/** Adjusts the bone rotation so the tip is as close to the target position as possible. The target is specified in the world - * coordinate system. */ -spine.IkConstraint.apply1 = function (bone, targetX, targetY, alpha) { - var parentRotation = (!bone.data.inheritRotation || !bone.parent) ? 0 : bone.parent.worldRotation; - var rotation = bone.rotation; - var rotationIK = Math.atan2(targetY - bone.worldY, targetX - bone.worldX) * spine.radDeg - parentRotation; - bone.rotationIK = rotation + (rotationIK - rotation) * alpha; -}; -/** Adjusts the parent and child bone rotations so the tip of the child is as close to the target position as possible. The - * target is specified in the world coordinate system. - * @param child Any descendant bone of the parent. */ -spine.IkConstraint.apply2 = function (parent, child, targetX, targetY, bendDirection, alpha) { - var childRotation = child.rotation, parentRotation = parent.rotation; - if (!alpha) { - child.rotationIK = childRotation; - parent.rotationIK = parentRotation; - return; - } - var positionX, positionY, tempPosition = spine.temp; - var parentParent = parent.parent; - if (parentParent) { - tempPosition[0] = targetX; - tempPosition[1] = targetY; - parentParent.worldToLocal(tempPosition); - targetX = (tempPosition[0] - parent.x) * parentParent.worldScaleX; - targetY = (tempPosition[1] - parent.y) * parentParent.worldScaleY; - } else { - targetX -= parent.x; - targetY -= parent.y; - } - if (child.parent == parent) { - positionX = child.x; - positionY = child.y; - } else { - tempPosition[0] = child.x; - tempPosition[1] = child.y; - child.parent.localToWorld(tempPosition); - parent.worldToLocal(tempPosition); - positionX = tempPosition[0]; - positionY = tempPosition[1]; - } - var childX = positionX * parent.worldScaleX, childY = positionY * parent.worldScaleY; - var offset = Math.atan2(childY, childX); - var len1 = Math.sqrt(childX * childX + childY * childY), len2 = child.data.length * child.worldScaleX; - // Based on code by Ryan Juckett with permission: Copyright (c) 2008-2009 Ryan Juckett, http://www.ryanjuckett.com/ - var cosDenom = 2 * len1 * len2; - if (cosDenom < 0.0001) { - child.rotationIK = childRotation + (Math.atan2(targetY, targetX) * spine.radDeg - parentRotation - childRotation) * alpha; - return; - } - var cos = (targetX * targetX + targetY * targetY - len1 * len1 - len2 * len2) / cosDenom; - if (cos < -1) - cos = -1; - else if (cos > 1) - cos = 1; - var childAngle = Math.acos(cos) * bendDirection; - var adjacent = len1 + len2 * cos, opposite = len2 * Math.sin(childAngle); - var parentAngle = Math.atan2(targetY * adjacent - targetX * opposite, targetX * adjacent + targetY * opposite); - var rotation = (parentAngle - offset) * spine.radDeg - parentRotation; - if (rotation > 180) - rotation -= 360; - else if (rotation < -180) // - rotation += 360; - parent.rotationIK = parentRotation + rotation * alpha; - rotation = (childAngle + offset) * spine.radDeg - childRotation; - if (rotation > 180) - rotation -= 360; - else if (rotation < -180) // - rotation += 360; - child.rotationIK = childRotation + (rotation + parent.worldRotation - child.parent.worldRotation) * alpha; -}; - -spine.Skin = function (name) { - this.name = name; - this.attachments = {}; -}; -spine.Skin.prototype = { - addAttachment: function (slotIndex, name, attachment) { - this.attachments[slotIndex + ":" + name] = attachment; - }, - getAttachment: function (slotIndex, name) { - return this.attachments[slotIndex + ":" + name]; - }, - _attachAll: function (skeleton, oldSkin) { - for (var key in oldSkin.attachments) { - var colon = key.indexOf(":"); - var slotIndex = parseInt(key.substring(0, colon)); - var name = key.substring(colon + 1); - var slot = skeleton.slots[slotIndex]; - if (slot.attachment && slot.attachment.name == name) { - var attachment = this.getAttachment(slotIndex, name); - if (attachment) slot.setAttachment(attachment); - } - } - } -}; - -spine.Animation = function (name, timelines, duration) { - this.name = name; - this.timelines = timelines; - this.duration = duration; -}; -spine.Animation.prototype = { - apply: function (skeleton, lastTime, time, loop, events) { - if (loop && this.duration != 0) { - time %= this.duration; - lastTime %= this.duration; - } - var timelines = this.timelines; - for (var i = 0, n = timelines.length; i < n; i++) - timelines[i].apply(skeleton, lastTime, time, events, 1); - }, - mix: function (skeleton, lastTime, time, loop, events, alpha) { - if (loop && this.duration != 0) { - time %= this.duration; - lastTime %= this.duration; - } - var timelines = this.timelines; - for (var i = 0, n = timelines.length; i < n; i++) - timelines[i].apply(skeleton, lastTime, time, events, alpha); - } -}; -spine.Animation.binarySearch = function (values, target, step) { - var low = 0; - var high = Math.floor(values.length / step) - 2; - if (!high) return step; - var current = high >>> 1; - while (true) { - if (values[(current + 1) * step] <= target) - low = current + 1; - else - high = current; - if (low == high) return (low + 1) * step; - current = (low + high) >>> 1; - } -}; -spine.Animation.binarySearch1 = function (values, target) { - var low = 0; - var high = values.length - 2; - if (!high) return 1; - var current = high >>> 1; - while (true) { - if (values[current + 1] <= target) - low = current + 1; - else - high = current; - if (low == high) return low + 1; - current = (low + high) >>> 1; - } -}; -spine.Animation.linearSearch = function (values, target, step) { - for (var i = 0, last = values.length - step; i <= last; i += step) - if (values[i] > target) return i; - return -1; -}; - -spine.Curves = function (frameCount) { - this.curves = []; // type, x, y, ... - //this.curves.length = (frameCount - 1) * 19/*BEZIER_SIZE*/; -}; -spine.Curves.prototype = { - setLinear: function (frameIndex) { - this.curves[frameIndex * 19/*BEZIER_SIZE*/] = 0/*LINEAR*/; - }, - setStepped: function (frameIndex) { - this.curves[frameIndex * 19/*BEZIER_SIZE*/] = 1/*STEPPED*/; - }, - /** Sets the control handle positions for an interpolation bezier curve used to transition from this keyframe to the next. - * cx1 and cx2 are from 0 to 1, representing the percent of time between the two keyframes. cy1 and cy2 are the percent of - * the difference between the keyframe's values. */ - setCurve: function (frameIndex, cx1, cy1, cx2, cy2) { - var subdiv1 = 1 / 10/*BEZIER_SEGMENTS*/, subdiv2 = subdiv1 * subdiv1, subdiv3 = subdiv2 * subdiv1; - var pre1 = 3 * subdiv1, pre2 = 3 * subdiv2, pre4 = 6 * subdiv2, pre5 = 6 * subdiv3; - var tmp1x = -cx1 * 2 + cx2, tmp1y = -cy1 * 2 + cy2, tmp2x = (cx1 - cx2) * 3 + 1, tmp2y = (cy1 - cy2) * 3 + 1; - var dfx = cx1 * pre1 + tmp1x * pre2 + tmp2x * subdiv3, dfy = cy1 * pre1 + tmp1y * pre2 + tmp2y * subdiv3; - var ddfx = tmp1x * pre4 + tmp2x * pre5, ddfy = tmp1y * pre4 + tmp2y * pre5; - var dddfx = tmp2x * pre5, dddfy = tmp2y * pre5; - - var i = frameIndex * 19/*BEZIER_SIZE*/; - var curves = this.curves; - curves[i++] = 2/*BEZIER*/; - - var x = dfx, y = dfy; - for (var n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) { - curves[i] = x; - curves[i + 1] = y; - dfx += ddfx; - dfy += ddfy; - ddfx += dddfx; - ddfy += dddfy; - x += dfx; - y += dfy; - } - }, - getCurvePercent: function (frameIndex, percent) { - percent = percent < 0 ? 0 : (percent > 1 ? 1 : percent); - var curves = this.curves; - var i = frameIndex * 19/*BEZIER_SIZE*/; - var type = curves[i]; - if (type === 0/*LINEAR*/) return percent; - if (type == 1/*STEPPED*/) return 0; - i++; - var x = 0; - for (var start = i, n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) { - x = curves[i]; - if (x >= percent) { - var prevX, prevY; - if (i == start) { - prevX = 0; - prevY = 0; - } else { - prevX = curves[i - 2]; - prevY = curves[i - 1]; - } - return prevY + (curves[i + 1] - prevY) * (percent - prevX) / (x - prevX); - } - } - var y = curves[i - 1]; - return y + (1 - y) * (percent - x) / (1 - x); // Last point is 1,1. - } -}; - -spine.RotateTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, angle, ... - this.frames.length = frameCount * 2; -}; -spine.RotateTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 2; - }, - setFrame: function (frameIndex, time, angle) { - frameIndex *= 2; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = angle; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var bone = skeleton.bones[this.boneIndex]; - - if (time >= frames[frames.length - 2]) { // Time is after last frame. - var amount = bone.data.rotation + frames[frames.length - 1] - bone.rotation; - while (amount > 180) - amount -= 360; - while (amount < -180) - amount += 360; - bone.rotation += amount * alpha; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 2); - var prevFrameValue = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex - 2/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 2 - 1, percent); - - var amount = frames[frameIndex + 1/*FRAME_VALUE*/] - prevFrameValue; - while (amount > 180) - amount -= 360; - while (amount < -180) - amount += 360; - amount = bone.data.rotation + (prevFrameValue + amount * percent) - bone.rotation; - while (amount > 180) - amount -= 360; - while (amount < -180) - amount += 360; - bone.rotation += amount * alpha; - } -}; - -spine.TranslateTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, x, y, ... - this.frames.length = frameCount * 3; -}; -spine.TranslateTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 3; - }, - setFrame: function (frameIndex, time, x, y) { - frameIndex *= 3; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = x; - this.frames[frameIndex + 2] = y; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var bone = skeleton.bones[this.boneIndex]; - - if (time >= frames[frames.length - 3]) { // Time is after last frame. - bone.x += (bone.data.x + frames[frames.length - 2] - bone.x) * alpha; - bone.y += (bone.data.y + frames[frames.length - 1] - bone.y) * alpha; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 3); - var prevFrameX = frames[frameIndex - 2]; - var prevFrameY = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent); - - bone.x += (bone.data.x + prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent - bone.x) * alpha; - bone.y += (bone.data.y + prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent - bone.y) * alpha; - } -}; - -spine.ScaleTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, x, y, ... - this.frames.length = frameCount * 3; -}; -spine.ScaleTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 3; - }, - setFrame: function (frameIndex, time, x, y) { - frameIndex *= 3; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = x; - this.frames[frameIndex + 2] = y; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var bone = skeleton.bones[this.boneIndex]; - - if (time >= frames[frames.length - 3]) { // Time is after last frame. - bone.scaleX += (bone.data.scaleX * frames[frames.length - 2] - bone.scaleX) * alpha; - bone.scaleY += (bone.data.scaleY * frames[frames.length - 1] - bone.scaleY) * alpha; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 3); - var prevFrameX = frames[frameIndex - 2]; - var prevFrameY = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent); - - bone.scaleX += (bone.data.scaleX * (prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent) - bone.scaleX) * alpha; - bone.scaleY += (bone.data.scaleY * (prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent) - bone.scaleY) * alpha; - } -}; - -spine.ColorTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, r, g, b, a, ... - this.frames.length = frameCount * 5; -}; -spine.ColorTimeline.prototype = { - slotIndex: 0, - getFrameCount: function () { - return this.frames.length / 5; - }, - setFrame: function (frameIndex, time, r, g, b, a) { - frameIndex *= 5; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = r; - this.frames[frameIndex + 2] = g; - this.frames[frameIndex + 3] = b; - this.frames[frameIndex + 4] = a; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var r, g, b, a; - if (time >= frames[frames.length - 5]) { - // Time is after last frame. - var i = frames.length - 1; - r = frames[i - 3]; - g = frames[i - 2]; - b = frames[i - 1]; - a = frames[i]; - } else { - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 5); - var prevFrameR = frames[frameIndex - 4]; - var prevFrameG = frames[frameIndex - 3]; - var prevFrameB = frames[frameIndex - 2]; - var prevFrameA = frames[frameIndex - 1]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex - 5/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 5 - 1, percent); - - r = prevFrameR + (frames[frameIndex + 1/*FRAME_R*/] - prevFrameR) * percent; - g = prevFrameG + (frames[frameIndex + 2/*FRAME_G*/] - prevFrameG) * percent; - b = prevFrameB + (frames[frameIndex + 3/*FRAME_B*/] - prevFrameB) * percent; - a = prevFrameA + (frames[frameIndex + 4/*FRAME_A*/] - prevFrameA) * percent; - } - var slot = skeleton.slots[this.slotIndex]; - if (alpha < 1) { - slot.r += (r - slot.r) * alpha; - slot.g += (g - slot.g) * alpha; - slot.b += (b - slot.b) * alpha; - slot.a += (a - slot.a) * alpha; - } else { - slot.r = r; - slot.g = g; - slot.b = b; - slot.a = a; - } - } -}; - -spine.AttachmentTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, ... - this.frames.length = frameCount; - this.attachmentNames = []; - this.attachmentNames.length = frameCount; -}; -spine.AttachmentTimeline.prototype = { - slotIndex: 0, - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, attachmentName) { - this.frames[frameIndex] = time; - this.attachmentNames[frameIndex] = attachmentName; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) { - if (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0); - return; - } else if (lastTime > time) // - lastTime = -1; - - var frameIndex = time >= frames[frames.length - 1] ? frames.length - 1 : spine.Animation.binarySearch1(frames, time) - 1; - if (frames[frameIndex] < lastTime) return; - - var attachmentName = this.attachmentNames[frameIndex]; - skeleton.slots[this.slotIndex].setAttachment( - !attachmentName ? null : skeleton.getAttachmentBySlotIndex(this.slotIndex, attachmentName)); - } -}; - -spine.EventTimeline = function (frameCount) { - this.frames = []; // time, ... - this.frames.length = frameCount; - this.events = []; - this.events.length = frameCount; -}; -spine.EventTimeline.prototype = { - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, event) { - this.frames[frameIndex] = time; - this.events[frameIndex] = event; - }, - /** Fires events for frames > lastTime and <= time. */ - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - if (!firedEvents) return; - - var frames = this.frames; - var frameCount = frames.length; - - if (lastTime > time) { // Fire events after last time for looped animations. - this.apply(skeleton, lastTime, Number.MAX_VALUE, firedEvents, alpha); - lastTime = -1; - } else if (lastTime >= frames[frameCount - 1]) // Last time is after last frame. - return; - if (time < frames[0]) return; // Time is before first frame. - - var frameIndex; - if (lastTime < frames[0]) - frameIndex = 0; - else { - frameIndex = spine.Animation.binarySearch1(frames, lastTime); - var frame = frames[frameIndex]; - while (frameIndex > 0) { // Fire multiple events with the same frame. - if (frames[frameIndex - 1] != frame) break; - frameIndex--; - } - } - var events = this.events; - for (; frameIndex < frameCount && time >= frames[frameIndex]; frameIndex++) - firedEvents.push(events[frameIndex]); - } -}; - -spine.DrawOrderTimeline = function (frameCount) { - this.frames = []; // time, ... - this.frames.length = frameCount; - this.drawOrders = []; - this.drawOrders.length = frameCount; -}; -spine.DrawOrderTimeline.prototype = { - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, drawOrder) { - this.frames[frameIndex] = time; - this.drawOrders[frameIndex] = drawOrder; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var frameIndex; - if (time >= frames[frames.length - 1]) // Time is after last frame. - frameIndex = frames.length - 1; - else - frameIndex = spine.Animation.binarySearch1(frames, time) - 1; - - var drawOrder = skeleton.drawOrder; - var slots = skeleton.slots; - var drawOrderToSetupIndex = this.drawOrders[frameIndex]; - if (!drawOrderToSetupIndex) { - for (var i = 0, n = slots.length; i < n; i++) - drawOrder[i] = slots[i]; - } else { - for (var i = 0, n = drawOrderToSetupIndex.length; i < n; i++) - drawOrder[i] = skeleton.slots[drawOrderToSetupIndex[i]]; - } - - } -}; - -spine.FfdTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; - this.frames.length = frameCount; - this.frameVertices = []; - this.frameVertices.length = frameCount; -}; -spine.FfdTimeline.prototype = { - slotIndex: 0, - attachment: 0, - getFrameCount: function () { - return this.frames.length; - }, - setFrame: function (frameIndex, time, vertices) { - this.frames[frameIndex] = time; - this.frameVertices[frameIndex] = vertices; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var slot = skeleton.slots[this.slotIndex]; - if (slot.attachment != this.attachment) return; - - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var frameVertices = this.frameVertices; - var vertexCount = frameVertices[0].length; - - var vertices = slot.attachmentVertices; - if (vertices.length != vertexCount) alpha = 1; - vertices.length = vertexCount; - - if (time >= frames[frames.length - 1]) { // Time is after last frame. - var lastVertices = frameVertices[frames.length - 1]; - if (alpha < 1) { - for (var i = 0; i < vertexCount; i++) - vertices[i] += (lastVertices[i] - vertices[i]) * alpha; - } else { - for (var i = 0; i < vertexCount; i++) - vertices[i] = lastVertices[i]; - } - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch1(frames, time); - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex - 1] - frameTime); - percent = this.curves.getCurvePercent(frameIndex - 1, percent < 0 ? 0 : (percent > 1 ? 1 : percent)); - - var prevVertices = frameVertices[frameIndex - 1]; - var nextVertices = frameVertices[frameIndex]; - - if (alpha < 1) { - for (var i = 0; i < vertexCount; i++) { - var prev = prevVertices[i]; - vertices[i] += (prev + (nextVertices[i] - prev) * percent - vertices[i]) * alpha; - } - } else { - for (var i = 0; i < vertexCount; i++) { - var prev = prevVertices[i]; - vertices[i] = prev + (nextVertices[i] - prev) * percent; - } - } - } -}; - -spine.IkConstraintTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, mix, bendDirection, ... - this.frames.length = frameCount * 3; -}; -spine.IkConstraintTimeline.prototype = { - ikConstraintIndex: 0, - getFrameCount: function () { - return this.frames.length / 3; - }, - setFrame: function (frameIndex, time, mix, bendDirection) { - frameIndex *= 3; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = mix; - this.frames[frameIndex + 2] = bendDirection; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) return; // Time is before first frame. - - var ikConstraint = skeleton.ikConstraints[this.ikConstraintIndex]; - - if (time >= frames[frames.length - 3]) { // Time is after last frame. - ikConstraint.mix += (frames[frames.length - 2] - ikConstraint.mix) * alpha; - ikConstraint.bendDirection = frames[frames.length - 1]; - return; - } - - // Interpolate between the previous frame and the current frame. - var frameIndex = spine.Animation.binarySearch(frames, time, 3); - var prevFrameMix = frames[frameIndex + -2/*PREV_FRAME_MIX*/]; - var frameTime = frames[frameIndex]; - var percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime); - percent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent); - - var mix = prevFrameMix + (frames[frameIndex + 1/*FRAME_MIX*/] - prevFrameMix) * percent; - ikConstraint.mix += (mix - ikConstraint.mix) * alpha; - ikConstraint.bendDirection = frames[frameIndex + -1/*PREV_FRAME_BEND_DIRECTION*/]; - } -}; - -spine.FlipXTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, flip, ... - this.frames.length = frameCount * 2; -}; -spine.FlipXTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 2; - }, - setFrame: function (frameIndex, time, flip) { - frameIndex *= 2; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = flip ? 1 : 0; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) { - if (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0); - return; - } else if (lastTime > time) // - lastTime = -1; - var frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2; - if (frames[frameIndex] < lastTime) return; - skeleton.bones[boneIndex].flipX = frames[frameIndex + 1] != 0; - } -}; - -spine.FlipYTimeline = function (frameCount) { - this.curves = new spine.Curves(frameCount); - this.frames = []; // time, flip, ... - this.frames.length = frameCount * 2; -}; -spine.FlipYTimeline.prototype = { - boneIndex: 0, - getFrameCount: function () { - return this.frames.length / 2; - }, - setFrame: function (frameIndex, time, flip) { - frameIndex *= 2; - this.frames[frameIndex] = time; - this.frames[frameIndex + 1] = flip ? 1 : 0; - }, - apply: function (skeleton, lastTime, time, firedEvents, alpha) { - var frames = this.frames; - if (time < frames[0]) { - if (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0); - return; - } else if (lastTime > time) // - lastTime = -1; - var frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2; - if (frames[frameIndex] < lastTime) return; - skeleton.bones[boneIndex].flipY = frames[frameIndex + 1] != 0; - } -}; - -spine.SkeletonData = function () { - this.bones = []; - this.slots = []; - this.skins = []; - this.events = []; - this.animations = []; - this.ikConstraints = []; -}; -spine.SkeletonData.prototype = { - name: null, - defaultSkin: null, - width: 0, height: 0, - version: null, hash: null, - /** @return May be null. */ - findBone: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].name == boneName) return bones[i]; - return null; - }, - /** @return -1 if the bone was not found. */ - findBoneIndex: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].name == boneName) return i; - return -1; - }, - /** @return May be null. */ - findSlot: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) { - if (slots[i].name == slotName) return slot[i]; - } - return null; - }, - /** @return -1 if the bone was not found. */ - findSlotIndex: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) - if (slots[i].name == slotName) return i; - return -1; - }, - /** @return May be null. */ - findSkin: function (skinName) { - var skins = this.skins; - for (var i = 0, n = skins.length; i < n; i++) - if (skins[i].name == skinName) return skins[i]; - return null; - }, - /** @return May be null. */ - findEvent: function (eventName) { - var events = this.events; - for (var i = 0, n = events.length; i < n; i++) - if (events[i].name == eventName) return events[i]; - return null; - }, - /** @return May be null. */ - findAnimation: function (animationName) { - var animations = this.animations; - for (var i = 0, n = animations.length; i < n; i++) - if (animations[i].name == animationName) return animations[i]; - return null; - }, - /** @return May be null. */ - findIkConstraint: function (ikConstraintName) { - var ikConstraints = this.ikConstraints; - for (var i = 0, n = ikConstraints.length; i < n; i++) - if (ikConstraints[i].name == ikConstraintName) return ikConstraints[i]; - return null; - } -}; - -spine.Skeleton = function (skeletonData) { - this.data = skeletonData; - - this.bones = []; - for (var i = 0, n = skeletonData.bones.length; i < n; i++) { - var boneData = skeletonData.bones[i]; - var parent = !boneData.parent ? null : this.bones[skeletonData.bones.indexOf(boneData.parent)]; - this.bones.push(new spine.Bone(boneData, this, parent)); - } - - this.slots = []; - this.drawOrder = []; - for (var i = 0, n = skeletonData.slots.length; i < n; i++) { - var slotData = skeletonData.slots[i]; - var bone = this.bones[skeletonData.bones.indexOf(slotData.boneData)]; - var slot = new spine.Slot(slotData, bone); - this.slots.push(slot); - this.drawOrder.push(slot); - } - - this.ikConstraints = []; - for (var i = 0, n = skeletonData.ikConstraints.length; i < n; i++) - this.ikConstraints.push(new spine.IkConstraint(skeletonData.ikConstraints[i], this)); - - this.boneCache = []; - this.updateCache(); -}; -spine.Skeleton.prototype = { - x: 0, y: 0, - skin: null, - r: 1, g: 1, b: 1, a: 1, - time: 0, - flipX: false, flipY: false, - /** Caches information about bones and IK constraints. Must be called if bones or IK constraints are added or removed. */ - updateCache: function () { - var ikConstraints = this.ikConstraints; - var ikConstraintsCount = ikConstraints.length; - - var arrayCount = ikConstraintsCount + 1; - var boneCache = this.boneCache; - if (boneCache.length > arrayCount) boneCache.length = arrayCount; - for (var i = 0, n = boneCache.length; i < n; i++) - boneCache[i].length = 0; - while (boneCache.length < arrayCount) - boneCache[boneCache.length] = []; - - var nonIkBones = boneCache[0]; - var bones = this.bones; - - outer: - for (var i = 0, n = bones.length; i < n; i++) { - var bone = bones[i]; - var current = bone; - do { - for (var ii = 0; ii < ikConstraintsCount; ii++) { - var ikConstraint = ikConstraints[ii]; - var parent = ikConstraint.bones[0]; - var child= ikConstraint.bones[ikConstraint.bones.length - 1]; - while (true) { - if (current == child) { - boneCache[ii].push(bone); - boneCache[ii + 1].push(bone); - continue outer; - } - if (child == parent) break; - child = child.parent; - } - } - current = current.parent; - } while (current); - nonIkBones[nonIkBones.length] = bone; - } - }, - /** Updates the world transform for each bone. */ - updateWorldTransform: function () { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) { - var bone = bones[i]; - bone.rotationIK = bone.rotation; - } - var i = 0, last = this.boneCache.length - 1; - while (true) { - var cacheBones = this.boneCache[i]; - for (var ii = 0, nn = cacheBones.length; ii < nn; ii++) - cacheBones[ii].updateWorldTransform(); - if (i == last) break; - this.ikConstraints[i].apply(); - i++; - } - }, - /** Sets the bones and slots to their setup pose values. */ - setToSetupPose: function () { - this.setBonesToSetupPose(); - this.setSlotsToSetupPose(); - }, - setBonesToSetupPose: function () { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - bones[i].setToSetupPose(); - - var ikConstraints = this.ikConstraints; - for (var i = 0, n = ikConstraints.length; i < n; i++) { - var ikConstraint = ikConstraints[i]; - ikConstraint.bendDirection = ikConstraint.data.bendDirection; - ikConstraint.mix = ikConstraint.data.mix; - } - }, - setSlotsToSetupPose: function () { - var slots = this.slots; - var drawOrder = this.drawOrder; - for (var i = 0, n = slots.length; i < n; i++) { - drawOrder[i] = slots[i]; - slots[i].setToSetupPose(i); - } - }, - /** @return May return null. */ - getRootBone: function () { - return this.bones.length ? this.bones[0] : null; - }, - /** @return May be null. */ - findBone: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].data.name == boneName) return bones[i]; - return null; - }, - /** @return -1 if the bone was not found. */ - findBoneIndex: function (boneName) { - var bones = this.bones; - for (var i = 0, n = bones.length; i < n; i++) - if (bones[i].data.name == boneName) return i; - return -1; - }, - /** @return May be null. */ - findSlot: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) - if (slots[i].data.name == slotName) return slots[i]; - return null; - }, - /** @return -1 if the bone was not found. */ - findSlotIndex: function (slotName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) - if (slots[i].data.name == slotName) return i; - return -1; - }, - setSkinByName: function (skinName) { - var skin = this.data.findSkin(skinName); - if (!skin) throw "Skin not found: " + skinName; - this.setSkin(skin); - }, - /** Sets the skin used to look up attachments before looking in the {@link SkeletonData#getDefaultSkin() default skin}. - * Attachments from the new skin are attached if the corresponding attachment from the old skin was attached. If there was - * no old skin, each slot's setup mode attachment is attached from the new skin. - * @param newSkin May be null. */ - setSkin: function (newSkin) { - if (newSkin) { - if (this.skin) - newSkin._attachAll(this, this.skin); - else { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) { - var slot = slots[i]; - var name = slot.data.attachmentName; - if (name) { - var attachment = newSkin.getAttachment(i, name); - if (attachment) slot.setAttachment(attachment); - } - } - } - } - this.skin = newSkin; - }, - /** @return May be null. */ - getAttachmentBySlotName: function (slotName, attachmentName) { - return this.getAttachmentBySlotIndex(this.data.findSlotIndex(slotName), attachmentName); - }, - /** @return May be null. */ - getAttachmentBySlotIndex: function (slotIndex, attachmentName) { - if (this.skin) { - var attachment = this.skin.getAttachment(slotIndex, attachmentName); - if (attachment) return attachment; - } - if (this.data.defaultSkin) return this.data.defaultSkin.getAttachment(slotIndex, attachmentName); - return null; - }, - /** @param attachmentName May be null. */ - setAttachment: function (slotName, attachmentName) { - var slots = this.slots; - for (var i = 0, n = slots.length; i < n; i++) { - var slot = slots[i]; - if (slot.data.name == slotName) { - var attachment = null; - if (attachmentName) { - attachment = this.getAttachmentBySlotIndex(i, attachmentName); - if (!attachment) throw "Attachment not found: " + attachmentName + ", for slot: " + slotName; - } - slot.setAttachment(attachment); - return; - } - } - throw "Slot not found: " + slotName; - }, - /** @return May be null. */ - findIkConstraint: function (ikConstraintName) { - var ikConstraints = this.ikConstraints; - for (var i = 0, n = ikConstraints.length; i < n; i++) - if (ikConstraints[i].data.name == ikConstraintName) return ikConstraints[i]; - return null; - }, - update: function (delta) { - this.time += delta; - } -}; - -spine.EventData = function (name) { - this.name = name; -}; -spine.EventData.prototype = { - intValue: 0, - floatValue: 0, - stringValue: null -}; - -spine.Event = function (data) { - this.data = data; -}; -spine.Event.prototype = { - intValue: 0, - floatValue: 0, - stringValue: null -}; - -spine.AttachmentType = { - region: 0, - boundingbox: 1, - mesh: 2, - skinnedmesh: 3 -}; - -spine.RegionAttachment = function (name) { - this.name = name; - this.offset = []; - this.offset.length = 8; - this.uvs = []; - this.uvs.length = 8; -}; -spine.RegionAttachment.prototype = { - type: spine.AttachmentType.region, - x: 0, y: 0, - rotation: 0, - scaleX: 1, scaleY: 1, - width: 0, height: 0, - r: 1, g: 1, b: 1, a: 1, - path: null, - rendererObject: null, - regionOffsetX: 0, regionOffsetY: 0, - regionWidth: 0, regionHeight: 0, - regionOriginalWidth: 0, regionOriginalHeight: 0, - setUVs: function (u, v, u2, v2, rotate) { - var uvs = this.uvs; - if (rotate) { - uvs[2/*X2*/] = u; - uvs[3/*Y2*/] = v2; - uvs[4/*X3*/] = u; - uvs[5/*Y3*/] = v; - uvs[6/*X4*/] = u2; - uvs[7/*Y4*/] = v; - uvs[0/*X1*/] = u2; - uvs[1/*Y1*/] = v2; - } else { - uvs[0/*X1*/] = u; - uvs[1/*Y1*/] = v2; - uvs[2/*X2*/] = u; - uvs[3/*Y2*/] = v; - uvs[4/*X3*/] = u2; - uvs[5/*Y3*/] = v; - uvs[6/*X4*/] = u2; - uvs[7/*Y4*/] = v2; - } - }, - updateOffset: function () { - var regionScaleX = this.width / this.regionOriginalWidth * this.scaleX; - var regionScaleY = this.height / this.regionOriginalHeight * this.scaleY; - var localX = -this.width / 2 * this.scaleX + this.regionOffsetX * regionScaleX; - var localY = -this.height / 2 * this.scaleY + this.regionOffsetY * regionScaleY; - var localX2 = localX + this.regionWidth * regionScaleX; - var localY2 = localY + this.regionHeight * regionScaleY; - var radians = this.rotation * spine.degRad; - var cos = Math.cos(radians); - var sin = Math.sin(radians); - var localXCos = localX * cos + this.x; - var localXSin = localX * sin; - var localYCos = localY * cos + this.y; - var localYSin = localY * sin; - var localX2Cos = localX2 * cos + this.x; - var localX2Sin = localX2 * sin; - var localY2Cos = localY2 * cos + this.y; - var localY2Sin = localY2 * sin; - var offset = this.offset; - offset[0/*X1*/] = localXCos - localYSin; - offset[1/*Y1*/] = localYCos + localXSin; - offset[2/*X2*/] = localXCos - localY2Sin; - offset[3/*Y2*/] = localY2Cos + localXSin; - offset[4/*X3*/] = localX2Cos - localY2Sin; - offset[5/*Y3*/] = localY2Cos + localX2Sin; - offset[6/*X4*/] = localX2Cos - localYSin; - offset[7/*Y4*/] = localYCos + localX2Sin; - }, - computeVertices: function (x, y, bone, vertices) { - x += bone.worldX; - y += bone.worldY; - var m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11; - var offset = this.offset; - vertices[0/*X1*/] = offset[0/*X1*/] * m00 + offset[1/*Y1*/] * m01 + x; - vertices[1/*Y1*/] = offset[0/*X1*/] * m10 + offset[1/*Y1*/] * m11 + y; - vertices[2/*X2*/] = offset[2/*X2*/] * m00 + offset[3/*Y2*/] * m01 + x; - vertices[3/*Y2*/] = offset[2/*X2*/] * m10 + offset[3/*Y2*/] * m11 + y; - vertices[4/*X3*/] = offset[4/*X3*/] * m00 + offset[5/*X3*/] * m01 + x; - vertices[5/*X3*/] = offset[4/*X3*/] * m10 + offset[5/*X3*/] * m11 + y; - vertices[6/*X4*/] = offset[6/*X4*/] * m00 + offset[7/*Y4*/] * m01 + x; - vertices[7/*Y4*/] = offset[6/*X4*/] * m10 + offset[7/*Y4*/] * m11 + y; - } -}; - -spine.MeshAttachment = function (name) { - this.name = name; -}; -spine.MeshAttachment.prototype = { - type: spine.AttachmentType.mesh, - vertices: null, - uvs: null, - regionUVs: null, - triangles: null, - hullLength: 0, - r: 1, g: 1, b: 1, a: 1, - path: null, - rendererObject: null, - regionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false, - regionOffsetX: 0, regionOffsetY: 0, - regionWidth: 0, regionHeight: 0, - regionOriginalWidth: 0, regionOriginalHeight: 0, - edges: null, - width: 0, height: 0, - updateUVs: function () { - var width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV; - var n = this.regionUVs.length; - if (!this.uvs || this.uvs.length != n) { - this.uvs = new spine.Float32Array(n); - } - if (this.regionRotate) { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width; - this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height; - } - } else { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i] * width; - this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height; - } - } - }, - computeWorldVertices: function (x, y, slot, worldVertices) { - var bone = slot.bone; - x += bone.worldX; - y += bone.worldY; - var m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11; - var vertices = this.vertices; - var verticesCount = vertices.length; - if (slot.attachmentVertices.length == verticesCount) vertices = slot.attachmentVertices; - for (var i = 0; i < verticesCount; i += 2) { - var vx = vertices[i]; - var vy = vertices[i + 1]; - worldVertices[i] = vx * m00 + vy * m01 + x; - worldVertices[i + 1] = vx * m10 + vy * m11 + y; - } - } -}; - -spine.SkinnedMeshAttachment = function (name) { - this.name = name; -}; -spine.SkinnedMeshAttachment.prototype = { - type: spine.AttachmentType.skinnedmesh, - bones: null, - weights: null, - uvs: null, - regionUVs: null, - triangles: null, - hullLength: 0, - r: 1, g: 1, b: 1, a: 1, - path: null, - rendererObject: null, - regionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false, - regionOffsetX: 0, regionOffsetY: 0, - regionWidth: 0, regionHeight: 0, - regionOriginalWidth: 0, regionOriginalHeight: 0, - edges: null, - width: 0, height: 0, - updateUVs: function (u, v, u2, v2, rotate) { - var width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV; - var n = this.regionUVs.length; - if (!this.uvs || this.uvs.length != n) { - this.uvs = new spine.Float32Array(n); - } - if (this.regionRotate) { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width; - this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height; - } - } else { - for (var i = 0; i < n; i += 2) { - this.uvs[i] = this.regionU + this.regionUVs[i] * width; - this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height; - } - } - }, - computeWorldVertices: function (x, y, slot, worldVertices) { - var skeletonBones = slot.bone.skeleton.bones; - var weights = this.weights; - var bones = this.bones; - - var w = 0, v = 0, b = 0, f = 0, n = bones.length, nn; - var wx, wy, bone, vx, vy, weight; - if (!slot.attachmentVertices.length) { - for (; v < n; w += 2) { - wx = 0; - wy = 0; - nn = bones[v++] + v; - for (; v < nn; v++, b += 3) { - bone = skeletonBones[bones[v]]; - vx = weights[b]; - vy = weights[b + 1]; - weight = weights[b + 2]; - wx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight; - wy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight; - } - worldVertices[w] = wx + x; - worldVertices[w + 1] = wy + y; - } - } else { - var ffd = slot.attachmentVertices; - for (; v < n; w += 2) { - wx = 0; - wy = 0; - nn = bones[v++] + v; - for (; v < nn; v++, b += 3, f += 2) { - bone = skeletonBones[bones[v]]; - vx = weights[b] + ffd[f]; - vy = weights[b + 1] + ffd[f + 1]; - weight = weights[b + 2]; - wx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight; - wy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight; - } - worldVertices[w] = wx + x; - worldVertices[w + 1] = wy + y; - } - } - } -}; - -spine.BoundingBoxAttachment = function (name) { - this.name = name; - this.vertices = []; -}; -spine.BoundingBoxAttachment.prototype = { - type: spine.AttachmentType.boundingbox, - computeWorldVertices: function (x, y, bone, worldVertices) { - x += bone.worldX; - y += bone.worldY; - var m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11; - var vertices = this.vertices; - for (var i = 0, n = vertices.length; i < n; i += 2) { - var px = vertices[i]; - var py = vertices[i + 1]; - worldVertices[i] = px * m00 + py * m01 + x; - worldVertices[i + 1] = px * m10 + py * m11 + y; - } - } -}; - -spine.AnimationStateData = function (skeletonData) { - this.skeletonData = skeletonData; - this.animationToMixTime = {}; -}; -spine.AnimationStateData.prototype = { - defaultMix: 0, - setMixByName: function (fromName, toName, duration) { - var from = this.skeletonData.findAnimation(fromName); - if (!from) throw "Animation not found: " + fromName; - var to = this.skeletonData.findAnimation(toName); - if (!to) throw "Animation not found: " + toName; - this.setMix(from, to, duration); - }, - setMix: function (from, to, duration) { - this.animationToMixTime[from.name + ":" + to.name] = duration; - }, - getMix: function (from, to) { - var key = from.name + ":" + to.name; - return this.animationToMixTime.hasOwnProperty(key) ? this.animationToMixTime[key] : this.defaultMix; - } -}; - -spine.TrackEntry = function () {}; -spine.TrackEntry.prototype = { - next: null, previous: null, - animation: null, - loop: false, - delay: 0, time: 0, lastTime: -1, endTime: 0, - timeScale: 1, - mixTime: 0, mixDuration: 0, mix: 1, - onStart: null, onEnd: null, onComplete: null, onEvent: null -}; - -spine.AnimationState = function (stateData) { - this.data = stateData; - this.tracks = []; - this.events = []; -}; -spine.AnimationState.prototype = { - onStart: null, - onEnd: null, - onComplete: null, - onEvent: null, - timeScale: 1, - update: function (delta) { - delta *= this.timeScale; - for (var i = 0; i < this.tracks.length; i++) { - var current = this.tracks[i]; - if (!current) continue; - - current.time += delta * current.timeScale; - if (current.previous) { - var previousDelta = delta * current.previous.timeScale; - current.previous.time += previousDelta; - current.mixTime += previousDelta; - } - - var next = current.next; - if (next) { - next.time = current.lastTime - next.delay; - if (next.time >= 0) this.setCurrent(i, next); - } else { - // End non-looping animation when it reaches its end time and there is no next entry. - if (!current.loop && current.lastTime >= current.endTime) this.clearTrack(i); - } - } - }, - apply: function (skeleton) { - for (var i = 0; i < this.tracks.length; i++) { - var current = this.tracks[i]; - if (!current) continue; - - this.events.length = 0; - - var time = current.time; - var lastTime = current.lastTime; - var endTime = current.endTime; - var loop = current.loop; - if (!loop && time > endTime) time = endTime; - - var previous = current.previous; - if (!previous) { - if (current.mix == 1) - current.animation.apply(skeleton, current.lastTime, time, loop, this.events); - else - current.animation.mix(skeleton, current.lastTime, time, loop, this.events, current.mix); - } else { - var previousTime = previous.time; - if (!previous.loop && previousTime > previous.endTime) previousTime = previous.endTime; - previous.animation.apply(skeleton, previousTime, previousTime, previous.loop, null); - - var alpha = current.mixTime / current.mixDuration * current.mix; - if (alpha >= 1) { - alpha = 1; - current.previous = null; - } - current.animation.mix(skeleton, current.lastTime, time, loop, this.events, alpha); - } - - for (var ii = 0, nn = this.events.length; ii < nn; ii++) { - var event = this.events[ii]; - if (current.onEvent) current.onEvent(i, event); - if (this.onEvent) this.onEvent(i, event); - } - - // Check if completed the animation or a loop iteration. - if (loop ? (lastTime % endTime > time % endTime) : (lastTime < endTime && time >= endTime)) { - var count = Math.floor(time / endTime); - if (current.onComplete) current.onComplete(i, count); - if (this.onComplete) this.onComplete(i, count); - } - - current.lastTime = current.time; - } - }, - clearTracks: function () { - for (var i = 0, n = this.tracks.length; i < n; i++) - this.clearTrack(i); - this.tracks.length = 0; - }, - clearTrack: function (trackIndex) { - if (trackIndex >= this.tracks.length) return; - var current = this.tracks[trackIndex]; - if (!current) return; - - if (current.onEnd) current.onEnd(trackIndex); - if (this.onEnd) this.onEnd(trackIndex); - - this.tracks[trackIndex] = null; - }, - _expandToIndex: function (index) { - if (index < this.tracks.length) return this.tracks[index]; - while (index >= this.tracks.length) - this.tracks.push(null); - return null; - }, - setCurrent: function (index, entry) { - var current = this._expandToIndex(index); - if (current) { - var previous = current.previous; - current.previous = null; - - if (current.onEnd) current.onEnd(index); - if (this.onEnd) this.onEnd(index); - - entry.mixDuration = this.data.getMix(current.animation, entry.animation); - if (entry.mixDuration > 0) { - entry.mixTime = 0; - // If a mix is in progress, mix from the closest animation. - if (previous && current.mixTime / current.mixDuration < 0.5) - entry.previous = previous; - else - entry.previous = current; - } - } - - this.tracks[index] = entry; - - if (entry.onStart) entry.onStart(index); - if (this.onStart) this.onStart(index); - }, - setAnimationByName: function (trackIndex, animationName, loop) { - var animation = this.data.skeletonData.findAnimation(animationName); - if (!animation) throw "Animation not found: " + animationName; - return this.setAnimation(trackIndex, animation, loop); - }, - /** Set the current animation. Any queued animations are cleared. */ - setAnimation: function (trackIndex, animation, loop) { - var entry = new spine.TrackEntry(); - entry.animation = animation; - entry.loop = loop; - entry.endTime = animation.duration; - this.setCurrent(trackIndex, entry); - return entry; - }, - addAnimationByName: function (trackIndex, animationName, loop, delay) { - var animation = this.data.skeletonData.findAnimation(animationName); - if (!animation) throw "Animation not found: " + animationName; - return this.addAnimation(trackIndex, animation, loop, delay); - }, - /** Adds an animation to be played delay seconds after the current or last queued animation. - * @param delay May be <= 0 to use duration of previous animation minus any mix duration plus the negative delay. */ - addAnimation: function (trackIndex, animation, loop, delay) { - var entry = new spine.TrackEntry(); - entry.animation = animation; - entry.loop = loop; - entry.endTime = animation.duration; - - var last = this._expandToIndex(trackIndex); - if (last) { - while (last.next) - last = last.next; - last.next = entry; - } else - this.tracks[trackIndex] = entry; - - if (delay <= 0) { - if (last) - delay += last.endTime - this.data.getMix(last.animation, animation); - else - delay = 0; - } - entry.delay = delay; - - return entry; - }, - /** May be null. */ - getCurrent: function (trackIndex) { - if (trackIndex >= this.tracks.length) return null; - return this.tracks[trackIndex]; - } -}; - -spine.SkeletonJson = function (attachmentLoader) { - this.attachmentLoader = attachmentLoader; -}; -spine.SkeletonJson.prototype = { - scale: 1, - readSkeletonData: function (root, name) { - var skeletonData = new spine.SkeletonData(); - skeletonData.name = name; - - // Skeleton. - var skeletonMap = root["skeleton"]; - if (skeletonMap) { - skeletonData.hash = skeletonMap["hash"]; - skeletonData.version = skeletonMap["spine"]; - skeletonData.width = skeletonMap["width"] || 0; - skeletonData.height = skeletonMap["height"] || 0; - } - - // Bones. - var bones = root["bones"]; - for (var i = 0, n = bones.length; i < n; i++) { - var boneMap = bones[i]; - var parent = null; - if (boneMap["parent"]) { - parent = skeletonData.findBone(boneMap["parent"]); - if (!parent) throw "Parent bone not found: " + boneMap["parent"]; - } - var boneData = new spine.BoneData(boneMap["name"], parent); - boneData.length = (boneMap["length"] || 0) * this.scale; - boneData.x = (boneMap["x"] || 0) * this.scale; - boneData.y = (boneMap["y"] || 0) * this.scale; - boneData.rotation = (boneMap["rotation"] || 0); - boneData.scaleX = boneMap.hasOwnProperty("scaleX") ? boneMap["scaleX"] : 1; - boneData.scaleY = boneMap.hasOwnProperty("scaleY") ? boneMap["scaleY"] : 1; - boneData.inheritScale = boneMap.hasOwnProperty("inheritScale") ? boneMap["inheritScale"] : true; - boneData.inheritRotation = boneMap.hasOwnProperty("inheritRotation") ? boneMap["inheritRotation"] : true; - skeletonData.bones.push(boneData); - } - - // IK constraints. - var ik = root["ik"]; - if (ik) { - for (var i = 0, n = ik.length; i < n; i++) { - var ikMap = ik[i]; - var ikConstraintData = new spine.IkConstraintData(ikMap["name"]); - - var bones = ikMap["bones"]; - for (var ii = 0, nn = bones.length; ii < nn; ii++) { - var bone = skeletonData.findBone(bones[ii]); - if (!bone) throw "IK bone not found: " + bones[ii]; - ikConstraintData.bones.push(bone); - } - - ikConstraintData.target = skeletonData.findBone(ikMap["target"]); - if (!ikConstraintData.target) throw "Target bone not found: " + ikMap["target"]; - - ikConstraintData.bendDirection = (!ikMap.hasOwnProperty("bendPositive") || ikMap["bendPositive"]) ? 1 : -1; - ikConstraintData.mix = ikMap.hasOwnProperty("mix") ? ikMap["mix"] : 1; - - skeletonData.ikConstraints.push(ikConstraintData); - } - } - - // Slots. - var slots = root["slots"]; - for (var i = 0, n = slots.length; i < n; i++) { - var slotMap = slots[i]; - var boneData = skeletonData.findBone(slotMap["bone"]); - if (!boneData) throw "Slot bone not found: " + slotMap["bone"]; - var slotData = new spine.SlotData(slotMap["name"], boneData); - - var color = slotMap["color"]; - if (color) { - slotData.r = this.toColor(color, 0); - slotData.g = this.toColor(color, 1); - slotData.b = this.toColor(color, 2); - slotData.a = this.toColor(color, 3); - } - - slotData.attachmentName = slotMap["attachment"]; - slotData.additiveBlending = slotMap["additive"] && slotMap["additive"] == "true"; - - skeletonData.slots.push(slotData); - } - - // Skins. - var skins = root["skins"]; - for (var skinName in skins) { - if (!skins.hasOwnProperty(skinName)) continue; - var skinMap = skins[skinName]; - var skin = new spine.Skin(skinName); - for (var slotName in skinMap) { - if (!skinMap.hasOwnProperty(slotName)) continue; - var slotIndex = skeletonData.findSlotIndex(slotName); - var slotEntry = skinMap[slotName]; - for (var attachmentName in slotEntry) { - if (!slotEntry.hasOwnProperty(attachmentName)) continue; - var attachment = this.readAttachment(skin, attachmentName, slotEntry[attachmentName]); - if (attachment) skin.addAttachment(slotIndex, attachmentName, attachment); - } - } - skeletonData.skins.push(skin); - if (skin.name == "default") skeletonData.defaultSkin = skin; - } - - // Events. - var events = root["events"]; - for (var eventName in events) { - if (!events.hasOwnProperty(eventName)) continue; - var eventMap = events[eventName]; - var eventData = new spine.EventData(eventName); - eventData.intValue = eventMap["int"] || 0; - eventData.floatValue = eventMap["float"] || 0; - eventData.stringValue = eventMap["string"] || null; - skeletonData.events.push(eventData); - } - - // Animations. - var animations = root["animations"]; - for (var animationName in animations) { - if (!animations.hasOwnProperty(animationName)) continue; - this.readAnimation(animationName, animations[animationName], skeletonData); - } - - return skeletonData; - }, - readAttachment: function (skin, name, map) { - name = map["name"] || name; - - var type = spine.AttachmentType[map["type"] || "region"]; - var path = map["path"] || name; - - var scale = this.scale; - if (type == spine.AttachmentType.region) { - var region = this.attachmentLoader.newRegionAttachment(skin, name, path); - if (!region) return null; - region.path = path; - region.x = (map["x"] || 0) * scale; - region.y = (map["y"] || 0) * scale; - region.scaleX = map.hasOwnProperty("scaleX") ? map["scaleX"] : 1; - region.scaleY = map.hasOwnProperty("scaleY") ? map["scaleY"] : 1; - region.rotation = map["rotation"] || 0; - region.width = (map["width"] || 0) * scale; - region.height = (map["height"] || 0) * scale; - - var color = map["color"]; - if (color) { - region.r = this.toColor(color, 0); - region.g = this.toColor(color, 1); - region.b = this.toColor(color, 2); - region.a = this.toColor(color, 3); - } - - region.updateOffset(); - return region; - } else if (type == spine.AttachmentType.mesh) { - var mesh = this.attachmentLoader.newMeshAttachment(skin, name, path); - if (!mesh) return null; - mesh.path = path; - mesh.vertices = this.getFloatArray(map, "vertices", scale); - mesh.triangles = this.getIntArray(map, "triangles"); - mesh.regionUVs = this.getFloatArray(map, "uvs", 1); - mesh.updateUVs(); - - color = map["color"]; - if (color) { - mesh.r = this.toColor(color, 0); - mesh.g = this.toColor(color, 1); - mesh.b = this.toColor(color, 2); - mesh.a = this.toColor(color, 3); - } - - mesh.hullLength = (map["hull"] || 0) * 2; - if (map["edges"]) mesh.edges = this.getIntArray(map, "edges"); - mesh.width = (map["width"] || 0) * scale; - mesh.height = (map["height"] || 0) * scale; - return mesh; - } else if (type == spine.AttachmentType.skinnedmesh) { - var mesh = this.attachmentLoader.newSkinnedMeshAttachment(skin, name, path); - if (!mesh) return null; - mesh.path = path; - - var uvs = this.getFloatArray(map, "uvs", 1); - var vertices = this.getFloatArray(map, "vertices", 1); - var weights = []; - var bones = []; - for (var i = 0, n = vertices.length; i < n; ) { - var boneCount = vertices[i++] | 0; - bones[bones.length] = boneCount; - for (var nn = i + boneCount * 4; i < nn; ) { - bones[bones.length] = vertices[i]; - weights[weights.length] = vertices[i + 1] * scale; - weights[weights.length] = vertices[i + 2] * scale; - weights[weights.length] = vertices[i + 3]; - i += 4; - } - } - mesh.bones = bones; - mesh.weights = weights; - mesh.triangles = this.getIntArray(map, "triangles"); - mesh.regionUVs = uvs; - mesh.updateUVs(); - - color = map["color"]; - if (color) { - mesh.r = this.toColor(color, 0); - mesh.g = this.toColor(color, 1); - mesh.b = this.toColor(color, 2); - mesh.a = this.toColor(color, 3); - } - - mesh.hullLength = (map["hull"] || 0) * 2; - if (map["edges"]) mesh.edges = this.getIntArray(map, "edges"); - mesh.width = (map["width"] || 0) * scale; - mesh.height = (map["height"] || 0) * scale; - return mesh; - } else if (type == spine.AttachmentType.boundingbox) { - var attachment = this.attachmentLoader.newBoundingBoxAttachment(skin, name); - var vertices = map["vertices"]; - for (var i = 0, n = vertices.length; i < n; i++) - attachment.vertices.push(vertices[i] * scale); - return attachment; - } - throw "Unknown attachment type: " + type; - }, - readAnimation: function (name, map, skeletonData) { - var timelines = []; - var duration = 0; - - var slots = map["slots"]; - for (var slotName in slots) { - if (!slots.hasOwnProperty(slotName)) continue; - var slotMap = slots[slotName]; - var slotIndex = skeletonData.findSlotIndex(slotName); - - for (var timelineName in slotMap) { - if (!slotMap.hasOwnProperty(timelineName)) continue; - var values = slotMap[timelineName]; - if (timelineName == "color") { - var timeline = new spine.ColorTimeline(values.length); - timeline.slotIndex = slotIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var color = valueMap["color"]; - var r = this.toColor(color, 0); - var g = this.toColor(color, 1); - var b = this.toColor(color, 2); - var a = this.toColor(color, 3); - timeline.setFrame(frameIndex, valueMap["time"], r, g, b, a); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 5 - 5]); - - } else if (timelineName == "attachment") { - var timeline = new spine.AttachmentTimeline(values.length); - timeline.slotIndex = slotIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - timeline.setFrame(frameIndex++, valueMap["time"], valueMap["name"]); - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]); - - } else - throw "Invalid timeline type for a slot: " + timelineName + " (" + slotName + ")"; - } - } - - var bones = map["bones"]; - for (var boneName in bones) { - if (!bones.hasOwnProperty(boneName)) continue; - var boneIndex = skeletonData.findBoneIndex(boneName); - if (boneIndex == -1) throw "Bone not found: " + boneName; - var boneMap = bones[boneName]; - - for (var timelineName in boneMap) { - if (!boneMap.hasOwnProperty(timelineName)) continue; - var values = boneMap[timelineName]; - if (timelineName == "rotate") { - var timeline = new spine.RotateTimeline(values.length); - timeline.boneIndex = boneIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - timeline.setFrame(frameIndex, valueMap["time"], valueMap["angle"]); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]); - - } else if (timelineName == "translate" || timelineName == "scale") { - var timeline; - var timelineScale = 1; - if (timelineName == "scale") - timeline = new spine.ScaleTimeline(values.length); - else { - timeline = new spine.TranslateTimeline(values.length); - timelineScale = this.scale; - } - timeline.boneIndex = boneIndex; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var x = (valueMap["x"] || 0) * timelineScale; - var y = (valueMap["y"] || 0) * timelineScale; - timeline.setFrame(frameIndex, valueMap["time"], x, y); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 3 - 3]); - - } else if (timelineName == "flipX" || timelineName == "flipY") { - var x = timelineName == "flipX"; - var timeline = x ? new spine.FlipXTimeline(values.length) : new spine.FlipYTimeline(values.length); - timeline.boneIndex = boneIndex; - - var field = x ? "x" : "y"; - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - timeline.setFrame(frameIndex, valueMap["time"], valueMap[field] || false); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]); - } else - throw "Invalid timeline type for a bone: " + timelineName + " (" + boneName + ")"; - } - } - - var ikMap = map["ik"]; - for (var ikConstraintName in ikMap) { - if (!ikMap.hasOwnProperty(ikConstraintName)) continue; - var ikConstraint = skeletonData.findIkConstraint(ikConstraintName); - var values = ikMap[ikConstraintName]; - var timeline = new spine.IkConstraintTimeline(values.length); - timeline.ikConstraintIndex = skeletonData.ikConstraints.indexOf(ikConstraint); - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var mix = valueMap.hasOwnProperty("mix") ? valueMap["mix"] : 1; - var bendDirection = (!valueMap.hasOwnProperty("bendPositive") || valueMap["bendPositive"]) ? 1 : -1; - timeline.setFrame(frameIndex, valueMap["time"], mix, bendDirection); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.frameCount * 3 - 3]); - } - - var ffd = map["ffd"]; - for (var skinName in ffd) { - var skin = skeletonData.findSkin(skinName); - var slotMap = ffd[skinName]; - for (slotName in slotMap) { - var slotIndex = skeletonData.findSlotIndex(slotName); - var meshMap = slotMap[slotName]; - for (var meshName in meshMap) { - var values = meshMap[meshName]; - var timeline = new spine.FfdTimeline(values.length); - var attachment = skin.getAttachment(slotIndex, meshName); - if (!attachment) throw "FFD attachment not found: " + meshName; - timeline.slotIndex = slotIndex; - timeline.attachment = attachment; - - var isMesh = attachment.type == spine.AttachmentType.mesh; - var vertexCount; - if (isMesh) - vertexCount = attachment.vertices.length; - else - vertexCount = attachment.weights.length / 3 * 2; - - var frameIndex = 0; - for (var i = 0, n = values.length; i < n; i++) { - var valueMap = values[i]; - var vertices; - if (!valueMap["vertices"]) { - if (isMesh) - vertices = attachment.vertices; - else { - vertices = []; - vertices.length = vertexCount; - } - } else { - var verticesValue = valueMap["vertices"]; - var vertices = []; - vertices.length = vertexCount; - var start = valueMap["offset"] || 0; - var nn = verticesValue.length; - if (this.scale == 1) { - for (var ii = 0; ii < nn; ii++) - vertices[ii + start] = verticesValue[ii]; - } else { - for (var ii = 0; ii < nn; ii++) - vertices[ii + start] = verticesValue[ii] * this.scale; - } - if (isMesh) { - var meshVertices = attachment.vertices; - for (var ii = 0, nn = vertices.length; ii < nn; ii++) - vertices[ii] += meshVertices[ii]; - } - } - - timeline.setFrame(frameIndex, valueMap["time"], vertices); - this.readCurve(timeline, frameIndex, valueMap); - frameIndex++; - } - timelines[timelines.length] = timeline; - duration = Math.max(duration, timeline.frames[timeline.frameCount - 1]); - } - } - } - - var drawOrderValues = map["drawOrder"]; - if (!drawOrderValues) drawOrderValues = map["draworder"]; - if (drawOrderValues) { - var timeline = new spine.DrawOrderTimeline(drawOrderValues.length); - var slotCount = skeletonData.slots.length; - var frameIndex = 0; - for (var i = 0, n = drawOrderValues.length; i < n; i++) { - var drawOrderMap = drawOrderValues[i]; - var drawOrder = null; - if (drawOrderMap["offsets"]) { - drawOrder = []; - drawOrder.length = slotCount; - for (var ii = slotCount - 1; ii >= 0; ii--) - drawOrder[ii] = -1; - var offsets = drawOrderMap["offsets"]; - var unchanged = []; - unchanged.length = slotCount - offsets.length; - var originalIndex = 0, unchangedIndex = 0; - for (var ii = 0, nn = offsets.length; ii < nn; ii++) { - var offsetMap = offsets[ii]; - var slotIndex = skeletonData.findSlotIndex(offsetMap["slot"]); - if (slotIndex == -1) throw "Slot not found: " + offsetMap["slot"]; - // Collect unchanged items. - while (originalIndex != slotIndex) - unchanged[unchangedIndex++] = originalIndex++; - // Set changed items. - drawOrder[originalIndex + offsetMap["offset"]] = originalIndex++; - } - // Collect remaining unchanged items. - while (originalIndex < slotCount) - unchanged[unchangedIndex++] = originalIndex++; - // Fill in unchanged items. - for (var ii = slotCount - 1; ii >= 0; ii--) - if (drawOrder[ii] == -1) drawOrder[ii] = unchanged[--unchangedIndex]; - } - timeline.setFrame(frameIndex++, drawOrderMap["time"], drawOrder); - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]); - } - - var events = map["events"]; - if (events) { - var timeline = new spine.EventTimeline(events.length); - var frameIndex = 0; - for (var i = 0, n = events.length; i < n; i++) { - var eventMap = events[i]; - var eventData = skeletonData.findEvent(eventMap["name"]); - if (!eventData) throw "Event not found: " + eventMap["name"]; - var event = new spine.Event(eventData); - event.intValue = eventMap.hasOwnProperty("int") ? eventMap["int"] : eventData.intValue; - event.floatValue = eventMap.hasOwnProperty("float") ? eventMap["float"] : eventData.floatValue; - event.stringValue = eventMap.hasOwnProperty("string") ? eventMap["string"] : eventData.stringValue; - timeline.setFrame(frameIndex++, eventMap["time"], event); - } - timelines.push(timeline); - duration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]); - } - - skeletonData.animations.push(new spine.Animation(name, timelines, duration)); - }, - readCurve: function (timeline, frameIndex, valueMap) { - var curve = valueMap["curve"]; - if (!curve) - timeline.curves.setLinear(frameIndex); - else if (curve == "stepped") - timeline.curves.setStepped(frameIndex); - else if (curve instanceof Array) - timeline.curves.setCurve(frameIndex, curve[0], curve[1], curve[2], curve[3]); - }, - toColor: function (hexString, colorIndex) { - if (hexString.length != 8) throw "Color hexidecimal length must be 8, recieved: " + hexString; - return parseInt(hexString.substring(colorIndex * 2, (colorIndex * 2) + 2), 16) / 255; - }, - getFloatArray: function (map, name, scale) { - var list = map[name]; - var values = new spine.Float32Array(list.length); - var i = 0, n = list.length; - if (scale == 1) { - for (; i < n; i++) - values[i] = list[i]; - } else { - for (; i < n; i++) - values[i] = list[i] * scale; - } - return values; - }, - getIntArray: function (map, name) { - var list = map[name]; - var values = new spine.Uint16Array(list.length); - for (var i = 0, n = list.length; i < n; i++) - values[i] = list[i] | 0; - return values; - } -}; - -spine.Atlas = function (atlasText, textureLoader) { - this.textureLoader = textureLoader; - this.pages = []; - this.regions = []; - - var reader = new spine.AtlasReader(atlasText); - var tuple = []; - tuple.length = 4; - var page = null; - while (true) { - var line = reader.readLine(); - if (line === null) break; - line = reader.trim(line); - if (!line.length) - page = null; - else if (!page) { - page = new spine.AtlasPage(); - page.name = line; - - if (reader.readTuple(tuple) == 2) { // size is only optional for an atlas packed with an old TexturePacker. - page.width = parseInt(tuple[0]); - page.height = parseInt(tuple[1]); - reader.readTuple(tuple); - } - page.format = spine.Atlas.Format[tuple[0]]; - - reader.readTuple(tuple); - page.minFilter = spine.Atlas.TextureFilter[tuple[0]]; - page.magFilter = spine.Atlas.TextureFilter[tuple[1]]; - - var direction = reader.readValue(); - page.uWrap = spine.Atlas.TextureWrap.clampToEdge; - page.vWrap = spine.Atlas.TextureWrap.clampToEdge; - if (direction == "x") - page.uWrap = spine.Atlas.TextureWrap.repeat; - else if (direction == "y") - page.vWrap = spine.Atlas.TextureWrap.repeat; - else if (direction == "xy") - page.uWrap = page.vWrap = spine.Atlas.TextureWrap.repeat; - - textureLoader.load(page, line, this); - - this.pages.push(page); - - } else { - var region = new spine.AtlasRegion(); - region.name = line; - region.page = page; - - region.rotate = reader.readValue() == "true"; - - reader.readTuple(tuple); - var x = parseInt(tuple[0]); - var y = parseInt(tuple[1]); - - reader.readTuple(tuple); - var width = parseInt(tuple[0]); - var height = parseInt(tuple[1]); - - region.u = x / page.width; - region.v = y / page.height; - if (region.rotate) { - region.u2 = (x + height) / page.width; - region.v2 = (y + width) / page.height; - } else { - region.u2 = (x + width) / page.width; - region.v2 = (y + height) / page.height; - } - region.x = x; - region.y = y; - region.width = Math.abs(width); - region.height = Math.abs(height); - - if (reader.readTuple(tuple) == 4) { // split is optional - region.splits = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])]; - - if (reader.readTuple(tuple) == 4) { // pad is optional, but only present with splits - region.pads = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])]; - - reader.readTuple(tuple); - } - } - - region.originalWidth = parseInt(tuple[0]); - region.originalHeight = parseInt(tuple[1]); - - reader.readTuple(tuple); - region.offsetX = parseInt(tuple[0]); - region.offsetY = parseInt(tuple[1]); - - region.index = parseInt(reader.readValue()); - - this.regions.push(region); - } - } -}; -spine.Atlas.prototype = { - findRegion: function (name) { - var regions = this.regions; - for (var i = 0, n = regions.length; i < n; i++) - if (regions[i].name == name) return regions[i]; - return null; - }, - dispose: function () { - var pages = this.pages; - for (var i = 0, n = pages.length; i < n; i++) - this.textureLoader.unload(pages[i].rendererObject); - }, - updateUVs: function (page) { - var regions = this.regions; - for (var i = 0, n = regions.length; i < n; i++) { - var region = regions[i]; - if (region.page != page) continue; - region.u = region.x / page.width; - region.v = region.y / page.height; - if (region.rotate) { - region.u2 = (region.x + region.height) / page.width; - region.v2 = (region.y + region.width) / page.height; - } else { - region.u2 = (region.x + region.width) / page.width; - region.v2 = (region.y + region.height) / page.height; - } - } - } -}; - -spine.Atlas.Format = { - alpha: 0, - intensity: 1, - luminanceAlpha: 2, - rgb565: 3, - rgba4444: 4, - rgb888: 5, - rgba8888: 6 -}; - -spine.Atlas.TextureFilter = { - nearest: 0, - linear: 1, - mipMap: 2, - mipMapNearestNearest: 3, - mipMapLinearNearest: 4, - mipMapNearestLinear: 5, - mipMapLinearLinear: 6 -}; - -spine.Atlas.TextureWrap = { - mirroredRepeat: 0, - clampToEdge: 1, - repeat: 2 -}; - -spine.AtlasPage = function () {}; -spine.AtlasPage.prototype = { - name: null, - format: null, - minFilter: null, - magFilter: null, - uWrap: null, - vWrap: null, - rendererObject: null, - width: 0, - height: 0 -}; - -spine.AtlasRegion = function () {}; -spine.AtlasRegion.prototype = { - page: null, - name: null, - x: 0, y: 0, - width: 0, height: 0, - u: 0, v: 0, u2: 0, v2: 0, - offsetX: 0, offsetY: 0, - originalWidth: 0, originalHeight: 0, - index: 0, - rotate: false, - splits: null, - pads: null -}; - -spine.AtlasReader = function (text) { - this.lines = text.split(/\r\n|\r|\n/); -}; -spine.AtlasReader.prototype = { - index: 0, - trim: function (value) { - return value.replace(/^\s+|\s+$/g, ""); - }, - readLine: function () { - if (this.index >= this.lines.length) return null; - return this.lines[this.index++]; - }, - readValue: function () { - var line = this.readLine(); - var colon = line.indexOf(":"); - if (colon == -1) throw "Invalid line: " + line; - return this.trim(line.substring(colon + 1)); - }, - /** Returns the number of tuple values read (1, 2 or 4). */ - readTuple: function (tuple) { - var line = this.readLine(); - var colon = line.indexOf(":"); - if (colon == -1) throw "Invalid line: " + line; - var i = 0, lastMatch = colon + 1; - for (; i < 3; i++) { - var comma = line.indexOf(",", lastMatch); - if (comma == -1) break; - tuple[i] = this.trim(line.substr(lastMatch, comma - lastMatch)); - lastMatch = comma + 1; - } - tuple[i] = this.trim(line.substring(lastMatch)); - return i + 1; - } -}; - -spine.AtlasAttachmentLoader = function (atlas) { - this.atlas = atlas; -}; -spine.AtlasAttachmentLoader.prototype = { - newRegionAttachment: function (skin, name, path) { - var region = this.atlas.findRegion(path); - if (!region) throw "Region not found in atlas: " + path + " (region attachment: " + name + ")"; - var attachment = new spine.RegionAttachment(name); - attachment.rendererObject = region; - attachment.setUVs(region.u, region.v, region.u2, region.v2, region.rotate); - attachment.regionOffsetX = region.offsetX; - attachment.regionOffsetY = region.offsetY; - attachment.regionWidth = region.width; - attachment.regionHeight = region.height; - attachment.regionOriginalWidth = region.originalWidth; - attachment.regionOriginalHeight = region.originalHeight; - return attachment; - }, - newMeshAttachment: function (skin, name, path) { - var region = this.atlas.findRegion(path); - if (!region) throw "Region not found in atlas: " + path + " (mesh attachment: " + name + ")"; - var attachment = new spine.MeshAttachment(name); - attachment.rendererObject = region; - attachment.regionU = region.u; - attachment.regionV = region.v; - attachment.regionU2 = region.u2; - attachment.regionV2 = region.v2; - attachment.regionRotate = region.rotate; - attachment.regionOffsetX = region.offsetX; - attachment.regionOffsetY = region.offsetY; - attachment.regionWidth = region.width; - attachment.regionHeight = region.height; - attachment.regionOriginalWidth = region.originalWidth; - attachment.regionOriginalHeight = region.originalHeight; - return attachment; - }, - newSkinnedMeshAttachment: function (skin, name, path) { - var region = this.atlas.findRegion(path); - if (!region) throw "Region not found in atlas: " + path + " (skinned mesh attachment: " + name + ")"; - var attachment = new spine.SkinnedMeshAttachment(name); - attachment.rendererObject = region; - attachment.regionU = region.u; - attachment.regionV = region.v; - attachment.regionU2 = region.u2; - attachment.regionV2 = region.v2; - attachment.regionRotate = region.rotate; - attachment.regionOffsetX = region.offsetX; - attachment.regionOffsetY = region.offsetY; - attachment.regionWidth = region.width; - attachment.regionHeight = region.height; - attachment.regionOriginalWidth = region.originalWidth; - attachment.regionOriginalHeight = region.originalHeight; - return attachment; - }, - newBoundingBoxAttachment: function (skin, name) { - return new spine.BoundingBoxAttachment(name); - } -}; - -spine.SkeletonBounds = function () { - this.polygonPool = []; - this.polygons = []; - this.boundingBoxes = []; -}; -spine.SkeletonBounds.prototype = { - minX: 0, minY: 0, maxX: 0, maxY: 0, - update: function (skeleton, updateAabb) { - var slots = skeleton.slots; - var slotCount = slots.length; - var x = skeleton.x, y = skeleton.y; - var boundingBoxes = this.boundingBoxes; - var polygonPool = this.polygonPool; - var polygons = this.polygons; - - boundingBoxes.length = 0; - for (var i = 0, n = polygons.length; i < n; i++) - polygonPool.push(polygons[i]); - polygons.length = 0; - - for (var i = 0; i < slotCount; i++) { - var slot = slots[i]; - var boundingBox = slot.attachment; - if (boundingBox.type != spine.AttachmentType.boundingbox) continue; - boundingBoxes.push(boundingBox); - - var poolCount = polygonPool.length, polygon; - if (poolCount > 0) { - polygon = polygonPool[poolCount - 1]; - polygonPool.splice(poolCount - 1, 1); - } else - polygon = []; - polygons.push(polygon); - - polygon.length = boundingBox.vertices.length; - boundingBox.computeWorldVertices(x, y, slot.bone, polygon); - } - - if (updateAabb) this.aabbCompute(); - }, - aabbCompute: function () { - var polygons = this.polygons; - var minX = Number.MAX_VALUE, minY = Number.MAX_VALUE, maxX = Number.MIN_VALUE, maxY = Number.MIN_VALUE; - for (var i = 0, n = polygons.length; i < n; i++) { - var vertices = polygons[i]; - for (var ii = 0, nn = vertices.length; ii < nn; ii += 2) { - var x = vertices[ii]; - var y = vertices[ii + 1]; - minX = Math.min(minX, x); - minY = Math.min(minY, y); - maxX = Math.max(maxX, x); - maxY = Math.max(maxY, y); - } - } - this.minX = minX; - this.minY = minY; - this.maxX = maxX; - this.maxY = maxY; - }, - /** Returns true if the axis aligned bounding box contains the point. */ - aabbContainsPoint: function (x, y) { - return x >= this.minX && x <= this.maxX && y >= this.minY && y <= this.maxY; - }, - /** Returns true if the axis aligned bounding box intersects the line segment. */ - aabbIntersectsSegment: function (x1, y1, x2, y2) { - var minX = this.minX, minY = this.minY, maxX = this.maxX, maxY = this.maxY; - if ((x1 <= minX && x2 <= minX) || (y1 <= minY && y2 <= minY) || (x1 >= maxX && x2 >= maxX) || (y1 >= maxY && y2 >= maxY)) - return false; - var m = (y2 - y1) / (x2 - x1); - var y = m * (minX - x1) + y1; - if (y > minY && y < maxY) return true; - y = m * (maxX - x1) + y1; - if (y > minY && y < maxY) return true; - var x = (minY - y1) / m + x1; - if (x > minX && x < maxX) return true; - x = (maxY - y1) / m + x1; - if (x > minX && x < maxX) return true; - return false; - }, - /** Returns true if the axis aligned bounding box intersects the axis aligned bounding box of the specified bounds. */ - aabbIntersectsSkeleton: function (bounds) { - return this.minX < bounds.maxX && this.maxX > bounds.minX && this.minY < bounds.maxY && this.maxY > bounds.minY; - }, - /** Returns the first bounding box attachment that contains the point, or null. When doing many checks, it is usually more - * efficient to only call this method if {@link #aabbContainsPoint(float, float)} returns true. */ - containsPoint: function (x, y) { - var polygons = this.polygons; - for (var i = 0, n = polygons.length; i < n; i++) - if (this.polygonContainsPoint(polygons[i], x, y)) return this.boundingBoxes[i]; - return null; - }, - /** Returns the first bounding box attachment that contains the line segment, or null. When doing many checks, it is usually - * more efficient to only call this method if {@link #aabbIntersectsSegment(float, float, float, float)} returns true. */ - intersectsSegment: function (x1, y1, x2, y2) { - var polygons = this.polygons; - for (var i = 0, n = polygons.length; i < n; i++) - if (polygons[i].intersectsSegment(x1, y1, x2, y2)) return this.boundingBoxes[i]; - return null; - }, - /** Returns true if the polygon contains the point. */ - polygonContainsPoint: function (polygon, x, y) { - var nn = polygon.length; - var prevIndex = nn - 2; - var inside = false; - for (var ii = 0; ii < nn; ii += 2) { - var vertexY = polygon[ii + 1]; - var prevY = polygon[prevIndex + 1]; - if ((vertexY < y && prevY >= y) || (prevY < y && vertexY >= y)) { - var vertexX = polygon[ii]; - if (vertexX + (y - vertexY) / (prevY - vertexY) * (polygon[prevIndex] - vertexX) < x) inside = !inside; - } - prevIndex = ii; - } - return inside; - }, - /** Returns true if the polygon contains the line segment. */ - polygonIntersectsSegment: function (polygon, x1, y1, x2, y2) { - var nn = polygon.length; - var width12 = x1 - x2, height12 = y1 - y2; - var det1 = x1 * y2 - y1 * x2; - var x3 = polygon[nn - 2], y3 = polygon[nn - 1]; - for (var ii = 0; ii < nn; ii += 2) { - var x4 = polygon[ii], y4 = polygon[ii + 1]; - var det2 = x3 * y4 - y3 * x4; - var width34 = x3 - x4, height34 = y3 - y4; - var det3 = width12 * height34 - height12 * width34; - var x = (det1 * width34 - width12 * det2) / det3; - if (((x >= x3 && x <= x4) || (x >= x4 && x <= x3)) && ((x >= x1 && x <= x2) || (x >= x2 && x <= x1))) { - var y = (det1 * height34 - height12 * det2) / det3; - if (((y >= y3 && y <= y4) || (y >= y4 && y <= y3)) && ((y >= y1 && y <= y2) || (y >= y2 && y <= y1))) return true; - } - x3 = x4; - y3 = y4; - } - return false; - }, - getPolygon: function (attachment) { - var index = this.boundingBoxes.indexOf(attachment); - return index == -1 ? null : this.polygons[index]; - }, - getWidth: function () { - return this.maxX - this.minX; - }, - getHeight: function () { - return this.maxY - this.minY; - } -}; - -},{}],95:[function(require,module,exports){ -/** - * @file Main export of the PIXI spine library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - Spine: require('./Spine') -}; - -},{"./Spine":93}],96:[function(require,module,exports){ -var core = require('../core'); - -/** - * A BitmapText object will create a line or multiple lines of text using bitmap font. To - * split a line you can use '\n', '\r' or '\r\n' in your string. You can generate the fnt files using: - * - * http://www.angelcode.com/products/bmfont/ for windows or - * http://www.bmglyph.com/ for mac. - * - * @class - * @extends DisplayObjectContainer - * @namespace PIXI - * @param text {string} The copy that you would like the text to display - * @param style {object} The style parameters - * @param style.font {string} The size (optional) and bitmap font id (required) eq 'Arial' or '20px Arial' (must have loaded previously) - * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text - */ -function BitmapText(text, style) { - core.DisplayObjectContainer.call(this); - - /** - * The width of the overall text, different from fontSize, - * which is defined in the style object - * - * @member {number} - * @readOnly - */ - this.textWidth = 0; - - /** - * The height of the overall text, different from fontSize, - * which is defined in the style object - * - * @member {number} - * @readOnly - */ - this.textHeight = 0; - - /** - * Private tracker for the letter sprite pool. - * - * @member {Sprite[]} - * @private - */ - this._pool = []; - - /** - * Private tracker for the current style. - * - * @member {object} - * @private - */ - this._style = { - tint: style.tint, - align: style.align, - fontName: null, - fontSize: 0 - }; - this.font = style.font; // run font setter - - /** - * Private tracker for the current text. - * - * @member {string} - * @private - */ - this._text = text; - - /** - * The dirty state of this object. - * - * @member {boolean} - */ - this.dirty = false; - - this.updateText(); -} - -// constructor -BitmapText.prototype = Object.create(core.DisplayObjectContainer.prototype); -BitmapText.prototype.constructor = BitmapText; -module.exports = BitmapText; - -Object.defineProperties(BitmapText.prototype, { - /** - * The tint of the BitmapText object - * - * @member {number} - * @memberof BitmapText# - */ - tint: { - get: function () { - return this._style.tint; - }, - set: function (value) { - this._style.tint = (typeof value === 'number' && value >= 0) ? value : 0xFFFFFF; - - this.dirty = true; - } - }, - - /** - * The tint of the BitmapText object - * - * @member {string} - * @default 'left' - * @memberof BitmapText# - */ - align: { - get: function () { - return this._style.align; - }, - set: function (value) { - this._style.align = value; - - this.dirty = true; - } - }, - - /** - * The tint of the BitmapText object - * - * @member {Font} - * @memberof BitmapText# - */ - font: { - get: function () { - return this._style.font; - }, - set: function (value) { - value = value.split(' '); - - // TODO - This should be object-based not string based like it has been. - this._style.fontName = value[value.length - 1]; - this._style.fontSize = value.length >= 2 ? parseInt(value[value.length - 2], 10) : BitmapText.fonts[this.fontName].size; - - this.dirty = true; - } - }, - - /** - * The text of the BitmapText object - * - * @member {string} - * @memberof BitmapText# - */ - text: { - get: function () { - return this._text; - }, - set: function (value) { - this._text = value; - - this.dirty = true; - } - } -}); - -/** - * Renders text and updates it when needed - * - * @private - */ -BitmapText.prototype.updateText = function () { - var data = BitmapText.fonts[this.fontName]; - var pos = new core.math.Point(); - var prevCharCode = null; - var chars = []; - var maxLineWidth = 0; - var lineWidths = []; - var line = 0; - var scale = this.fontSize / data.size; - - for (var i = 0; i < this.text.length; i++) { - var charCode = this.text.charCodeAt(i); - - if (/(?:\r\n|\r|\n)/.test(this.text.charAt(i))) { - lineWidths.push(pos.x); - maxLineWidth = Math.max(maxLineWidth, pos.x); - line++; - - pos.x = 0; - pos.y += data.lineHeight; - prevCharCode = null; - continue; - } - - var charData = data.chars[charCode]; - - if (!charData) { - continue; - } - - if (prevCharCode && charData.kerning[prevCharCode]) { - pos.x += charData.kerning[prevCharCode]; - } - - chars.push({texture:charData.texture, line: line, charCode: charCode, position: new core.math.Point(pos.x + charData.xOffset, pos.y + charData.yOffset)}); - pos.x += charData.xAdvance; - - prevCharCode = charCode; - } - - lineWidths.push(pos.x); - maxLineWidth = Math.max(maxLineWidth, pos.x); - - var lineAlignOffsets = []; - - for (i = 0; i <= line; i++) { - var alignOffset = 0; - - if (this.style.align === 'right') { - alignOffset = maxLineWidth - lineWidths[i]; - } - else if (this.style.align === 'center') { - alignOffset = (maxLineWidth - lineWidths[i]) / 2; - } - - lineAlignOffsets.push(alignOffset); - } - - var lenChildren = this.children.length; - var lenChars = chars.length; - var tint = this.tint; - - for (i = 0; i < lenChars; i++) { - var c = i < lenChildren ? this.children[i] : this._pool.pop(); // get old child if have. if not - take from pool. - - if (c) { - c.setTexture(chars[i].texture); // check if got one before. - } - else { - c = new core.Sprite(chars[i].texture); // if no create new one. - } - - c.position.x = (chars[i].position.x + lineAlignOffsets[chars[i].line]) * scale; - c.position.y = chars[i].position.y * scale; - c.scale.x = c.scale.y = scale; - c.tint = tint; - - if (!c.parent) { - this.addChild(c); - } - } - - // remove unnecessary children. - // and put their into the pool. - while(this.children.length > lenChars) { - var child = this.getChildAt(this.children.length - 1); - this._pool.push(child); - this.removeChild(child); - } - - this.textWidth = maxLineWidth * scale; - this.textHeight = (pos.y + data.lineHeight) * scale; -}; - -/** - * Updates the transform of this object - * - * @private - */ -BitmapText.prototype.updateTransform = function () { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - this.displayObjectContainerUpdateTransform(); -}; - -BitmapText.fonts = {}; - -},{"../core":9}],97:[function(require,module,exports){ -var core = require('../core'); - -/** - * A Text Object will create a line or multiple lines of text. To split a line you can use '\n' in your text string, - * or add a wordWrap property set to true and and wordWrapWidth property with a value in the style object. - * - * @class - * @extends Sprite - * @namespace PIXI - * @param text {string} The copy that you would like the text to display - * @param [style] {object} The style parameters - * @param [style.font] {string} default 'bold 20px Arial' The style and size of the font - * @param [style.fill='black'] {String|Number} A canvas fillstyle that will be used on the text e.g 'red', '#00FF00' - * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text - * @param [style.stroke] {String|Number} A canvas fillstyle that will be used on the text stroke e.g 'blue', '#FCFF00' - * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke) - * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used - * @param [style.wordWrapWidth=100] {number} The width at which text will wrap, it needs wordWrap to be set to true - * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text - * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00' - * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow - * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow - */ -function Text(text, style) { - /** - * The canvas element that everything is drawn to - * - * @member {HTMLCanvasElement} - */ - this.canvas = document.createElement('canvas'); - - /** - * The canvas 2d context that everything is drawn with - * @member {HTMLCanvasElement} - */ - this.context = this.canvas.getContext('2d'); - - /** - * The resolution of the canvas. - * @member {number} - */ - this.resolution = 1; - - core.Sprite.call(this, core.Texture.fromCanvas(this.canvas)); - - this.setText(text); - this.setStyle(style); -} - -// constructor -Text.prototype = Object.create(core.Sprite.prototype); -Text.prototype.constructor = Text; -module.exports = Text; - -Object.defineProperties(Text.prototype, { - /** - * The width of the Text, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof Text# - */ - width: { - get: function () { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - return this.scale.x * this.texture.frame.width; - }, - set: function (value) { - this.scale.x = value / this.texture.frame.width; - this._width = value; - } - }, - - /** - * The height of the Text, setting this will actually modify the scale to achieve the value set - * - * @member {number} - * @memberof Text# - */ - height: { - get: function () { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - return this.scale.y * this.texture.frame.height; - }, - set: function (value) { - this.scale.y = value / this.texture.frame.height; - this._height = value; - } - } -}); - -/** - * Set the style of the text - * - * @param [style] {object} The style parameters - * @param [style.font='bold 20pt Arial'] {string} The style and size of the font - * @param [style.fill='black'] {object} A canvas fillstyle that will be used on the text eg 'red', '#00FF00' - * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text - * @param [style.stroke='black'] {string} A canvas fillstyle that will be used on the text stroke eg 'blue', '#FCFF00' - * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke) - * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used - * @param [style.wordWrapWidth=100] {number} The width at which text will wrap - * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text - * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00' - * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow - * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow - */ -Text.prototype.setStyle = function (style) { - style = style || {}; - style.font = style.font || 'bold 20pt Arial'; - style.fill = style.fill || 'black'; - style.align = style.align || 'left'; - style.stroke = style.stroke || 'black'; //provide a default, see: https://github.com/GoodBoyDigital/pixi.js/issues/136 - style.strokeThickness = style.strokeThickness || 0; - style.wordWrap = style.wordWrap || false; - style.wordWrapWidth = style.wordWrapWidth || 100; - - style.dropShadow = style.dropShadow || false; - style.dropShadowAngle = style.dropShadowAngle || Math.PI / 6; - style.dropShadowDistance = style.dropShadowDistance || 4; - style.dropShadowColor = style.dropShadowColor || 'black'; - - this.style = style; - this.dirty = true; -}; - -/** - * Set the copy for the text object. To split a line you can use '\n'. - * - * @param text {string} The copy that you would like the text to display - */ -Text.prototype.setText = function (text) { - this.text = text.toString() || ' '; - this.dirty = true; -}; - -/** - * Renders text and updates it when needed - * - * @private - */ -Text.prototype.updateText = function () { - this.texture.baseTexture.resolution = this.resolution; - - this.context.font = this.style.font; - - var outputText = this.text; - - // word wrap - // preserve original text - if (this.style.wordWrap) { - outputText = this.wordWrap(this.text); - } - - //split text into lines - var lines = outputText.split(/(?:\r\n|\r|\n)/); - - //calculate text width - var lineWidths = []; - var maxLineWidth = 0; - var fontProperties = this.determineFontProperties(this.style.font); - for (var i = 0; i < lines.length; i++) { - var lineWidth = this.context.measureText(lines[i]).width; - lineWidths[i] = lineWidth; - maxLineWidth = Math.max(maxLineWidth, lineWidth); - } - - var width = maxLineWidth + this.style.strokeThickness; - if (this.style.dropShadow) { - width += this.style.dropShadowDistance; - } - - this.canvas.width = ( width + this.context.lineWidth ) * this.resolution; - - //calculate text height - var lineHeight = fontProperties.fontSize + this.style.strokeThickness; - - var height = lineHeight * lines.length; - if (this.style.dropShadow) { - height += this.style.dropShadowDistance; - } - - this.canvas.height = height * this.resolution; - - this.context.scale( this.resolution, this.resolution); - - if (navigator.isCocoonJS) { - this.context.clearRect(0, 0, this.canvas.width, this.canvas.height); - } - - // used for debugging.. - //this.context.fillStyle ="#FF0000" - //this.context.fillRect(0, 0, this.canvas.width,this.canvas.height); - - this.context.font = this.style.font; - this.context.strokeStyle = this.style.stroke; - this.context.lineWidth = this.style.strokeThickness; - this.context.textBaseline = 'alphabetic'; - //this.context.lineJoin = 'round'; - - var linePositionX; - var linePositionY; - - if (this.style.dropShadow) { - this.context.fillStyle = this.style.dropShadowColor; - - var xShadowOffset = Math.cos(this.style.dropShadowAngle) * this.style.dropShadowDistance; - var yShadowOffset = Math.sin(this.style.dropShadowAngle) * this.style.dropShadowDistance; - - for (i = 0; i < lines.length; i++) { - linePositionX = this.style.strokeThickness / 2; - linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent; - - if (this.style.align === 'right') { - linePositionX += maxLineWidth - lineWidths[i]; - } - else if (this.style.align === 'center') { - linePositionX += (maxLineWidth - lineWidths[i]) / 2; - } - - if (this.style.fill) { - this.context.fillText(lines[i], linePositionX + xShadowOffset, linePositionY + yShadowOffset); - } - - // if (dropShadow) - } - } - - //set canvas text styles - this.context.fillStyle = this.style.fill; - - //draw lines line by line - for (i = 0; i < lines.length; i++) { - linePositionX = this.style.strokeThickness / 2; - linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent; - - if (this.style.align === 'right') { - linePositionX += maxLineWidth - lineWidths[i]; - } - else if (this.style.align === 'center') { - linePositionX += (maxLineWidth - lineWidths[i]) / 2; - } - - if (this.style.stroke && this.style.strokeThickness) { - this.context.strokeText(lines[i], linePositionX, linePositionY); - } - - if (this.style.fill) { - this.context.fillText(lines[i], linePositionX, linePositionY); - } - - // if (dropShadow) - } - - this.updateTexture(); -}; - -/** - * Updates texture size based on canvas size - * - * @private - */ -Text.prototype.updateTexture = function () { - this.texture.baseTexture.width = this.canvas.width; - this.texture.baseTexture.height = this.canvas.height; - this.texture.crop.width = this.texture.frame.width = this.canvas.width; - this.texture.crop.height = this.texture.frame.height = this.canvas.height; - - this._width = this.canvas.width; - this._height = this.canvas.height; - - // update the dirty base textures - this.texture.baseTexture.dirty(); -}; - -/** - * Renders the object using the WebGL renderer -* - * @param renderer {WebGLRenderer} - */ -Text.prototype.renderWebGL = function (renderer) { - if (this.dirty) { - this.resolution = renderer.resolution; - - this.updateText(); - this.dirty = false; - } - - core.Sprite.prototype.renderWebGL.call(this, renderer); -}; - -/** - * Renders the object using the Canvas renderer -* - * @param renderer {CanvasRenderer} - */ -Text.prototype.renderCanvas = function (renderer) { - if (this.dirty) { - this.resolution = renderer.resolution; - - this.updateText(); - this.dirty = false; - } - - core.Sprite.prototype.renderCanvas.call(this, renderer); -}; - -/** - * Calculates the ascent, descent and fontSize of a given fontStyle -* - * @param fontStyle {object} - * @private - */ -Text.prototype.determineFontProperties = function (fontStyle) { - var properties = Text.fontPropertiesCache[fontStyle]; - - if (!properties) { - properties = {}; - - var canvas = Text.fontPropertiesCanvas; - var context = Text.fontPropertiesContext; - - context.font = fontStyle; - - var width = Math.ceil(context.measureText('|Mq').width); - var baseline = Math.ceil(context.measureText('M').width); - var height = 2 * baseline; - - baseline = baseline * 1.4 | 0; - - canvas.width = width; - canvas.height = height; - - context.fillStyle = '#f00'; - context.fillRect(0, 0, width, height); - - context.font = fontStyle; - - context.textBaseline = 'alphabetic'; - context.fillStyle = '#000'; - context.fillText('|MÉq', 0, baseline); - - var imagedata = context.getImageData(0, 0, width, height).data; - var pixels = imagedata.length; - var line = width * 4; - - var i, j; - - var idx = 0; - var stop = false; - - // ascent. scan from top to bottom until we find a non red pixel - for (i = 0; i < baseline; i++) { - for (j = 0; j < line; j += 4) { - if (imagedata[idx + j] !== 255) { - stop = true; - break; - } - } - if (!stop) { - idx += line; - } - else { - break; - } - } - - properties.ascent = baseline - i; - - idx = pixels - line; - stop = false; - - // descent. scan from bottom to top until we find a non red pixel - for (i = height; i > baseline; i--) { - for (j = 0; j < line; j += 4) { - if (imagedata[idx + j] !== 255) { - stop = true; - break; - } - } - if (!stop) { - idx -= line; - } - else { - break; - } - } - - properties.descent = i - baseline; - //TODO might need a tweak. kind of a temp fix! - properties.descent += 6; - properties.fontSize = properties.ascent + properties.descent; - - Text.fontPropertiesCache[fontStyle] = properties; - } - - return properties; -}; - -/** - * Applies newlines to a string to have it optimally fit into the horizontal - * bounds set by the Text object's wordWrapWidth property. - * - * @param text {string} - * @private - */ -Text.prototype.wordWrap = function (text) { - // Greedy wrapping algorithm that will wrap words as the line grows longer - // than its horizontal bounds. - var result = ''; - var lines = text.split('\n'); - for (var i = 0; i < lines.length; i++) { - var spaceLeft = this.style.wordWrapWidth; - var words = lines[i].split(' '); - for (var j = 0; j < words.length; j++) { - var wordWidth = this.context.measureText(words[j]).width; - var wordWidthWithSpace = wordWidth + this.context.measureText(' ').width; - if (j === 0 || wordWidthWithSpace > spaceLeft) { - // Skip printing the newline if it's the first word of the line that is - // greater than the word wrap width. - if (j > 0) { - result += '\n'; - } - result += words[j]; - spaceLeft = this.style.wordWrapWidth - wordWidth; - } - else { - spaceLeft -= wordWidthWithSpace; - result += ' ' + words[j]; - } - } - - if (i < lines.length-1) { - result += '\n'; - } - } - return result; -}; - -/** - * Returns the bounds of the Text as a rectangle. The bounds calculation takes the worldTransform into account. - * - * @param matrix {Matrix} the transformation matrix of the Text - * @return {Rectangle} the framing rectangle - */ -Text.prototype.getBounds = function (matrix) { - if (this.dirty) { - this.updateText(); - this.dirty = false; - } - - return core.Sprite.prototype.getBounds.call(this, matrix); -}; - -/** - * Destroys this text object. - * - * @param destroyBaseTexture {boolean} whether to destroy the base texture as well - */ -Text.prototype.destroy = function (destroyBaseTexture) { - // make sure to reset the the context and canvas.. dont want this hanging around in memory! - this.context = null; - this.canvas = null; - - this.texture.destroy(destroyBaseTexture === undefined ? true : destroyBaseTexture); -}; - -Text.fontPropertiesCache = {}; -Text.fontPropertiesCanvas = document.createElement('canvas'); -Text.fontPropertiesContext = Text.fontPropertiesCanvas.getContext('2d'); - -},{"../core":9}],98:[function(require,module,exports){ -/** - * @file Main export of the PIXI text library - * @author Mat Groves - * @copyright 2013-2015 GoodBoyDigital - * @license {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License} - */ - -/** - * @namespace PIXI - */ -module.exports = { - Text: require('./Text'), - BitmapText: require('./BitmapText') -}; - -},{"./BitmapText":96,"./Text":97}]},{},[1])(1) -}); -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","src/index","node_modules/webgl-enabled/index.js","package.json","src/core/const.js","src/core/display/DisplayObject.js","src/core/display/DisplayObjectContainer.js","src/core/display/Sprite.js","src/core/display/SpriteBatch.js","src/core/index.js","src/core/math/Matrix.js","src/core/math/Point.js","src/core/math/index.js","src/core/math/shapes/Circle.js","src/core/math/shapes/Ellipse.js","src/core/math/shapes/Polygon.js","src/core/math/shapes/Rectangle.js","src/core/math/shapes/RoundedRectangle.js","src/core/primitives/Graphics.js","src/core/primitives/GraphicsData.js","src/core/renderers/canvas/CanvasRenderer.js","src/core/renderers/canvas/utils/CanvasBuffer.js","src/core/renderers/canvas/utils/CanvasGraphics.js","src/core/renderers/canvas/utils/CanvasMaskManager.js","src/core/renderers/canvas/utils/CanvasTinter.js","src/core/renderers/webgl/WebGLRenderer.js","src/core/renderers/webgl/managers/WebGLBlendModeManager.js","src/core/renderers/webgl/managers/WebGLFilterManager.js","src/core/renderers/webgl/managers/WebGLManager.js","src/core/renderers/webgl/managers/WebGLMaskManager.js","src/core/renderers/webgl/managers/WebGLShaderManager.js","src/core/renderers/webgl/managers/WebGLStencilManager.js","src/core/renderers/webgl/shaders/ComplexPrimitiveShader.js","src/core/renderers/webgl/shaders/FastShader.js","src/core/renderers/webgl/shaders/PrimitiveShader.js","src/core/renderers/webgl/shaders/Shader.js","src/core/renderers/webgl/shaders/StripShader.js","src/core/renderers/webgl/utils/FilterTexture.js","src/core/renderers/webgl/utils/WebGLFastSpriteBatch.js","src/core/renderers/webgl/utils/WebGLGraphics.js","src/core/renderers/webgl/utils/WebGLGraphicsData.js","src/core/renderers/webgl/utils/WebGLSpriteBatch.js","src/core/textures/BaseTexture.js","src/core/textures/RenderTexture.js","src/core/textures/Texture.js","src/core/textures/TextureUvs.js","src/core/textures/VideoBaseTexture.js","src/core/utils/EventData.js","src/core/utils/PolyK.js","src/core/utils/eventTarget.js","src/core/utils/index.js","src/extras/MovieClip.js","src/extras/Rope.js","src/extras/Strip.js","src/extras/TilingSprite.js","src/extras/index.js","src/filters/AbstractFilter.js","src/filters/AlphaMaskFilter.js","src/filters/AsciiFilter.js","src/filters/BlurFilter.js","src/filters/BlurXFilter.js","src/filters/BlurYFilter.js","src/filters/ColorMatrixFilter.js","src/filters/ColorStepFilter.js","src/filters/ConvolutionFilter.js","src/filters/CrossHatchFilter.js","src/filters/DisplacementFilter.js","src/filters/DotScreenFilter.js","src/filters/FilterBlock.js","src/filters/GrayFilter.js","src/filters/InvertFilter.js","src/filters/NoiseFilter.js","src/filters/NormalMapFilter.js","src/filters/PixelateFilter.js","src/filters/RGBSplitFilter.js","src/filters/SepiaFilter.js","src/filters/SmartBlurFilter.js","src/filters/TiltShiftFilter.js","src/filters/TiltShiftXFilter.js","src/filters/TiltShiftYFilter.js","src/filters/TwistFilter.js","src/filters/index.js","src/interaction/InteractionData.js","src/interaction/InteractionManager.js","src/interaction/index.js","src/loaders/AtlasLoader.js","src/loaders/BitmapFontLoader.js","src/loaders/ImageLoader.js","src/loaders/JsonLoader.js","src/loaders/SpineLoader.js","src/loaders/SpineTextureLoader.js","src/loaders/SpriteSheetLoader.js","src/loaders/index.js","src/spine/Spine.js","src/spine/SpineRuntime.js","src/spine/index.js","src/text/BitmapText.js","src/text/Text.js","src/text/index.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/YA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5kBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjaA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3iCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5UA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/iBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1xBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3mBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3XA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzZA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtTA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACfA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpKA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjcA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC3DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7FA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACjGA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1DA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChxBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1GA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACzFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACnPA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClkFA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACbA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AChRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7dA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","module.exports = {\n    core:           require('./core'),\n    extras:         require('./extras'),\n    filters:        require('./filters'),\n    interaction:    require('./interaction'),\n    loaders:        require('./loaders'),\n    spine:          require('./spine'),\n    text:           require('./text')\n};\n","module.exports = function webglEnabled() {\n  try {\n    var canvas = document.createElement('canvas');\n    return !!(window.WebGLRenderingContext && (canvas.getContext('webgl') || canvas.getContext('experimental-webgl')));\n  } catch (e) {\n    return false;\n  }\n};\n","module.exports={\n  \"name\": \"pixi.js\",\n  \"version\": \"3.0.0\",\n  \"description\": \"Pixi.js is a fast lightweight 2D library that works across all devices.\",\n  \"author\": \"Mat Groves\",\n  \"contributors\": [\n    \"Chad Engler <chad@pantherdev.com>\",\n    \"Richard Davey <rdavey@gmail.com>\"\n  ],\n  \"main\": \"./src/index.js\",\n  \"homepage\": \"http://goodboydigital.com/\",\n  \"bugs\": \"https://github.com/GoodBoyDigital/pixi.js/issues\",\n  \"license\": \"MIT\",\n  \"repository\": {\n    \"type\": \"git\",\n    \"url\": \"https://github.com/GoodBoyDigital/pixi.js.git\"\n  },\n  \"scripts\": {\n    \"test\": \"gulp test\",\n    \"docs\": \"./node_modules/.bin/jsdoc -c ./gulp/util/jsdoc.conf.json\"\n  },\n  \"devDependencies\": {\n    \"browserify\": \"^8.0.2\",\n    \"chai\": \"^1.10.0\",\n    \"del\": \"^1.1.0\",\n    \"gulp\": \"^3.8.10\",\n    \"gulp-jshint\": \"^1.9.0\",\n    \"gulp-plumber\": \"^0.6.6\",\n    \"gulp-rename\": \"^1.2.0\",\n    \"gulp-uglify\": \"^1.0.2\",\n    \"gulp-util\": \"^3.0.1\",\n    \"ink-docstrap\": \"^0.4.12\",\n    \"jsdoc\": \"^3.3.0-alpha13\",\n    \"jshint-summary\": \"^0.4.0\",\n    \"karma\": \"^0.12.28\",\n    \"karma-firefox-launcher\": \"^0.1.0\",\n    \"karma-mocha\": \"^0.1.10\",\n    \"karma-spec-reporter\": \"^0.0.16\",\n    \"mocha\": \"^2.1.0\",\n    \"require-dir\": \"^0.1.0\",\n    \"run-sequence\": \"^1.0.2\",\n    \"vinyl-buffer\": \"^1.0.0\",\n    \"vinyl-source-stream\": \"^1.0.0\",\n    \"watchify\": \"^2.2.1\"\n  },\n  \"dependencies\": {\n    \"webgl-enabled\": \"^1.0.2\"\n  }\n}\n","/**\n * Constant values used in pixi\n *\n * @mixin const\n */\nmodule.exports = {\n    /**\n     * Constant to identify the WEBGL Renderer Type\n     *\n     * @static\n     * @constant\n     * @property {number} WEBGL_RENDERER\n     */\n    WEBGL_RENDERER: 1,\n\n    /**\n     * Constant to identify the CANVAS Renderer Type\n     *\n     * @static\n     * @constant\n     * @property {number} CANVAS_RENDERER\n     */\n    CANVAS_RENDERER: 2,\n\n    /**\n     * String of the current PIXI version\n     *\n     * @static\n     * @constant\n     * @property {string} VERSION\n     */\n    VERSION: require('../../package.json').version,\n\n    /**\n     * Various blend modes supported by PIXI. IMPORTANT - The WebGL renderer only supports\n     * the NORMAL, ADD, MULTIPLY and SCREEN blend modes. Anything else will silently act like\n     * NORMAL.\n     *\n     * @static\n     * @constant\n     * @property {object} blendModes\n     * @property {number} blendModes.NORMAL\n     * @property {number} blendModes.ADD\n     * @property {number} blendModes.MULTIPLY\n     * @property {number} blendModes.SCREEN\n     * @property {number} blendModes.OVERLAY\n     * @property {number} blendModes.DARKEN\n     * @property {number} blendModes.LIGHTEN\n     * @property {number} blendModes.COLOR_DODGE\n     * @property {number} blendModes.COLOR_BURN\n     * @property {number} blendModes.HARD_LIGHT\n     * @property {number} blendModes.SOFT_LIGHT\n     * @property {number} blendModes.DIFFERENCE\n     * @property {number} blendModes.EXCLUSION\n     * @property {number} blendModes.HUE\n     * @property {number} blendModes.SATURATION\n     * @property {number} blendModes.COLOR\n     * @property {number} blendModes.LUMINOSITY\n     */\n    blendModes: {\n        NORMAL:         0,\n        ADD:            1,\n        MULTIPLY:       2,\n        SCREEN:         3,\n        OVERLAY:        4,\n        DARKEN:         5,\n        LIGHTEN:        6,\n        COLOR_DODGE:    7,\n        COLOR_BURN:     8,\n        HARD_LIGHT:     9,\n        SOFT_LIGHT:     10,\n        DIFFERENCE:     11,\n        EXCLUSION:      12,\n        HUE:            13,\n        SATURATION:     14,\n        COLOR:          15,\n        LUMINOSITY:     16\n    },\n\n    /**\n     * The scale modes that are supported by pixi.\n     *\n     * The DEFAULT scale mode affects the default scaling mode of future operations.\n     * It can be re-assigned to either LINEAR or NEAREST, depending upon suitability.\n     *\n     * @static\n     * @constant\n     * @property {object} scaleModes\n     * @property {number} scaleModes.DEFAULT=LINEAR\n     * @property {number} scaleModes.LINEAR Smooth scaling\n     * @property {number} scaleModes.NEAREST Pixelating scaling\n     */\n    scaleModes: {\n        DEFAULT:    0,\n        LINEAR:     0,\n        NEAREST:    1\n    },\n\n    /**\n     * The prefix that denotes a URL is for a retina asset\n     *\n     * @static\n     * @constant\n     * @property {string} RETINA_PREFIX\n     */\n    RETINA_PREFIX: '@2x',\n\n    /**\n     * The default render options if none are supplied to {@link PIXI.WebGLRenderer}\n     * or {@link PIXI.CanvasRenderer}.\n     *\n     * @static\n     * @constant\n     * @property {object} defaultRenderOptions\n     * @property {HTMLCanvasElement} defaultRenderOptions.view=null\n     * @property {boolean} defaultRenderOptions.transparent=false\n     * @property {boolean} defaultRenderOptions.antialias=false\n     * @property {boolean} defaultRenderOptions.preserveDrawingBuffer=false\n     * @property {number} defaultRenderOptions.resolution=1\n     * @property {number} defaultRenderOptions.backgroundColor=0x000000\n     * @property {boolean} defaultRenderOptions.clearBeforeRender=true\n     * @property {boolean} defaultRenderOptions.autoResize=false\n     */\n    defaultRenderOptions: {\n        view: null,\n        resolution: 1,\n        antialias: false,\n        autoResize: false,\n        transparent: false,\n        backgroundColor: 0x000000,\n        clearBeforeRender: true,\n        preserveDrawingBuffer: false\n    },\n\n    /**\n     * Constants that identify shapes, mainly to prevent `instanceof` calls.\n     *\n     * @static\n     * @constant\n     * @property {object} SHAPES\n     * @property {object} SHAPES.POLY=0\n     * @property {object} SHAPES.RECT=1\n     * @property {object} SHAPES.CIRC=2\n     * @property {object} SHAPES.ELIP=3\n     * @property {object} SHAPES.RREC=4\n     */\n    SHAPES: {\n        POLY: 0,\n        RECT: 1,\n        CIRC: 2,\n        ELIP: 3,\n        RREC: 4\n    }\n};\n","var math = require('../math');\n\n/**\n * The base class for all objects that are rendered on the screen.\n * This is an abstract class and should not be used on its own rather it should be extended.\n *\n * @class\n * @namespace PIXI\n */\nfunction DisplayObject() {\n    /**\n     * The coordinate of the object relative to the local coordinates of the parent.\n     *\n     * @member {Point}\n     */\n    this.position = new math.Point();\n\n    /**\n     * The scale factor of the object.\n     *\n     * @member {Point}\n     */\n    this.scale = new math.Point(1, 1);\n\n    /**\n     * The pivot point of the displayObject that it rotates around\n     *\n     * @member {Point}\n     */\n    this.pivot = new math.Point(0, 0);\n\n    /**\n     * The rotation of the object in radians.\n     *\n     * @member {number}\n     */\n    this.rotation = 0;\n\n    /**\n     * The opacity of the object.\n     *\n     * @member {number}\n     */\n    this.alpha = 1;\n\n    /**\n     * The visibility of the object. If false the object will not be drawn, and\n     * the updateTransform function will not be called.\n     *\n     * @member {boolean}\n     */\n    this.visible = true;\n\n    /**\n     * Can this object be rendered, if false the object will not be drawn but the updateTransform\n     * methods will still be called.\n     *\n     * @member {boolean}\n     */\n    this.renderable = false;\n\n    /**\n     * The display object container that contains this display object.\n     *\n     * @member {DisplayObjectContainer}\n     * @readOnly\n     */\n    this.parent = null;\n\n    /**\n     * The multiplied alpha of the displayObject\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.worldAlpha = 1;\n\n    /**\n     * Current transform of the object based on world (parent) factors\n     *\n     * @member {Matrix}\n     * @readOnly\n     */\n    this.worldTransform = new math.Matrix();\n\n    /**\n     * The area the filter is applied to. This is used as more of an optimisation\n     * rather than figuring out the dimensions of the displayObject each frame you can set this rectangle\n     *\n     * @member {Rectangle}\n     */\n    this.filterArea = null;\n\n    /**\n     * cached sin rotation\n     *\n     * @member {number}\n     * @private\n     */\n    this._sr = 0;\n\n    /**\n     * cached cos rotation\n     *\n     * @member {number}\n     * @private\n     */\n    this._cr = 1;\n\n    /**\n     * The original, cached bounds of the object\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._bounds = new math.Rectangle(0, 0, 1, 1);\n\n    /**\n     * The most up-to-date bounds of the object\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._currentBounds = null;\n\n    /**\n     * The original, cached mask of the object\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._mask = null;\n\n    /**\n     * Cached internal flag.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this._cacheIsDirty = false;\n}\n\n// constructor\nDisplayObject.prototype.constructor = DisplayObject;\nmodule.exports = DisplayObject;\n\nObject.defineProperties(DisplayObject.prototype, {\n    /**\n     * The position of the displayObject on the x axis relative to the local coordinates of the parent.\n     *\n     * @member {number}\n     * @memberof DisplayObject#\n     */\n    x: {\n        get: function () {\n            return this.position.x;\n        },\n        set: function (value) {\n            this.position.x = value;\n        }\n    },\n\n    /**\n     * The position of the displayObject on the y axis relative to the local coordinates of the parent.\n     *\n     * @member {number}\n     * @memberof DisplayObject#\n     */\n    y: {\n        get: function () {\n            return this.position.y;\n        },\n        set: function (value) {\n            this.position.y = value;\n        }\n    },\n\n    /**\n     * Indicates if the sprite is globally visible.\n     *\n     * @member {boolean}\n     * @memberof DisplayObject#\n     * @readonly\n     */\n    worldVisible: {\n        get: function () {\n            var item = this;\n\n            do {\n                if (!item.visible) {\n                    return false;\n                }\n\n                item = item.parent;\n            } while(item);\n\n            return true;\n        }\n    },\n\n    /**\n     * Sets a mask for the displayObject. A mask is an object that limits the visibility of an object to the shape of the mask applied to it.\n     * In PIXI a regular mask must be a PIXI.Graphics object. This allows for much faster masking in canvas as it utilises shape clipping.\n     * To remove a mask, set this property to null.\n     *\n     * @member {Graphics}\n     * @memberof DisplayObject#\n     */\n    mask: {\n        get: function () {\n            return this._mask;\n        },\n        set: function (value) {\n            if (this._mask) {\n                this._mask.isMask = false;\n            }\n\n            this._mask = value;\n\n            if (this._mask) {\n                this._mask.isMask = true;\n            }\n        }\n    },\n\n    /**\n     * Sets the filters for the displayObject.\n     * * IMPORTANT: This is a webGL only feature and will be ignored by the canvas renderer.\n     * To remove filters simply set this property to 'null'\n     *\n     * @member {Filter[]}\n     * @memberof DisplayObject#\n     */\n    filters: {\n        get: function () {\n            return this._filters;\n        },\n        set: function (value) {\n            if (value) {\n                // now put all the passes in one place..\n                var passes = [];\n\n                for (var i = 0; i < value.length; i++) {\n                    var filterPasses = value[i].passes;\n\n                    for (var j = 0; j < filterPasses.length; j++) {\n                        passes.push(filterPasses[j]);\n                    }\n                }\n\n                // TODO change this as it is legacy\n                this._filterBlock = { target: this, filterPasses: passes };\n            }\n\n            this._filters = value;\n        }\n    }\n});\n\n/*\n * Updates the object transform for rendering\n *\n * TODO - Optimization pass!\n *\n * @private\n */\nDisplayObject.prototype.updateTransform = function () {\n    if (!this.parent) {\n        return;\n    }\n\n    // create some matrix refs for easy access\n    var pt = this.parent.worldTransform;\n    var wt = this.worldTransform;\n\n    // temporary matrix variables\n    var a, b, c, d, tx, ty;\n\n    // so if rotation is between 0 then we can simplify the multiplication process..\n    if (this.rotation % math.PI_2) {\n        // check to see if the rotation is the same as the previous render. This means we only need to use sin and cos when rotation actually changes\n        if (this.rotation !== this.rotationCache) {\n            this.rotationCache = this.rotation;\n            this._sr = Math.sin(this.rotation);\n            this._cr = Math.cos(this.rotation);\n        }\n\n        // get the matrix values of the displayobject based on its transform properties..\n        a  =  this._cr * this.scale.x;\n        b  =  this._sr * this.scale.x;\n        c  = -this._sr * this.scale.y;\n        d  =  this._cr * this.scale.y;\n        tx =  this.position.x;\n        ty =  this.position.y;\n\n        // check for pivot.. not often used so geared towards that fact!\n        if (this.pivot.x || this.pivot.y) {\n            tx -= this.pivot.x * a + this.pivot.y * c;\n            ty -= this.pivot.x * b + this.pivot.y * d;\n        }\n\n        // concat the parent matrix with the objects transform.\n        wt.a  = a  * pt.a + b  * pt.c;\n        wt.b  = a  * pt.b + b  * pt.d;\n        wt.c  = c  * pt.a + d  * pt.c;\n        wt.d  = c  * pt.b + d  * pt.d;\n        wt.tx = tx * pt.a + ty * pt.c + pt.tx;\n        wt.ty = tx * pt.b + ty * pt.d + pt.ty;\n    }\n    else {\n        // lets do the fast version as we know there is no rotation..\n        a  = this.scale.x;\n        d  = this.scale.y;\n\n        tx = this.position.x - this.pivot.x * a;\n        ty = this.position.y - this.pivot.y * d;\n\n        wt.a  = a  * pt.a;\n        wt.b  = a  * pt.b;\n        wt.c  = d  * pt.c;\n        wt.d  = d  * pt.d;\n        wt.tx = tx * pt.a + ty * pt.c + pt.tx;\n        wt.ty = tx * pt.b + ty * pt.d + pt.ty;\n    }\n\n    // multiply the alphas..\n    this.worldAlpha = this.alpha * this.parent.worldAlpha;\n};\n\n// performance increase to avoid using call.. (10x faster)\nDisplayObject.prototype.displayObjectUpdateTransform = DisplayObject.prototype.updateTransform;\n\n/**\n * Retrieves the bounds of the displayObject as a rectangle object\n *\n * @param matrix {Matrix}\n * @return {Rectangle} the rectangular bounding area\n */\nDisplayObject.prototype.getBounds = function (/* matrix */) {\n    return math.Rectangle.EMPTY;\n};\n\n/**\n * Retrieves the local bounds of the displayObject as a rectangle object\n *\n * @return {Rectangle} the rectangular bounding area\n */\nDisplayObject.prototype.getLocalBounds = function () {\n    return this.getBounds(math.Matrix.IDENTITY);\n};\n\n/**\n * Calculates the global position of the display object\n *\n * @param position {Point} The world origin to calculate from\n * @return {Point} A point object representing the position of this object\n */\nDisplayObject.prototype.toGlobal = function (position) {\n    // don't need to u[date the lot\n    this.displayObjectUpdateTransform();\n    return this.worldTransform.apply(position);\n};\n\n/**\n * Calculates the local position of the display object relative to another point\n *\n * @param position {Point} The world origin to calculate from\n * @param [from] {DisplayObject} The DisplayObject to calculate the global position from\n * @return {Point} A point object representing the position of this object\n */\nDisplayObject.prototype.toLocal = function (position, from) {\n    if (from) {\n        position = from.toGlobal(position);\n    }\n\n    // don't need to update the lot\n    this.displayObjectUpdateTransform();\n    return this.worldTransform.applyInverse(position);\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer} The renderer\n * @private\n */\nDisplayObject.prototype.renderWebGL = function (/* renderer */) {\n    // OVERWRITE;\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer} The renderer\n * @private\n */\nDisplayObject.prototype.renderCanvas = function (/* renderer */) {\n    // OVERWRITE;\n};\n","var math = require('../math'),\n    DisplayObject = require('./DisplayObject'),\n    RenderTexture = require('../textures/RenderTexture'),\n    // Sprite = require('./Sprite'),\n    _tempMatrix = new math.Matrix();\n\n/**\n * A DisplayObjectContainer represents a collection of display objects.\n * It is the base class of all display objects that act as a container for other objects.\n *\n * @class\n * @extends DisplayObject\n * @namespace PIXI\n */\nfunction DisplayObjectContainer() {\n    DisplayObject.call(this);\n\n    /**\n     * The array of children of this container.\n     *\n     * @member {DisplayObject[]}\n     * @readonly\n     */\n    this.children = [];\n\n    /**\n     * Cached internal flag.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this._cacheAsBitmap = false;\n\n    this._cachedSprite = null;\n}\n\n// constructor\nDisplayObjectContainer.prototype = Object.create(DisplayObject.prototype);\nDisplayObjectContainer.prototype.constructor = DisplayObjectContainer;\nmodule.exports = DisplayObjectContainer;\n\nObject.defineProperties(DisplayObjectContainer.prototype, {\n    /**\n     * The width of the displayObjectContainer, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof DisplayObjectContainer#\n     */\n    width: {\n        get: function () {\n            return this.scale.x * this.getLocalBounds().width;\n        },\n        set: function (value) {\n\n            var width = this.getLocalBounds().width;\n\n            if(width !== 0) {\n                this.scale.x = value / width;\n            }\n            else {\n                this.scale.x = 1;\n            }\n\n\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the displayObjectContainer, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof DisplayObjectContainer#\n     */\n    height: {\n        get: function () {\n            return  this.scale.y * this.getLocalBounds().height;\n        },\n        set: function (value) {\n\n            var height = this.getLocalBounds().height;\n\n            if (height !== 0) {\n                this.scale.y = value / height ;\n            }\n            else {\n                this.scale.y = 1;\n            }\n\n            this._height = value;\n        }\n    },\n\n    /**\n     * Set if this display object is cached as a bitmap.\n     * This basically takes a snap shot of the display object as it is at that moment. It can provide a performance benefit for complex static displayObjects.\n     * To remove simply set this property to 'null'\n     *\n     * @member {boolean}\n     * @memberof DisplayObject#\n     */\n    cacheAsBitmap: {\n        get: function () {\n            return this._cacheAsBitmap;\n        },\n        set: function (value) {\n            if (this._cacheAsBitmap === value) {\n                return;\n            }\n\n            if (value) {\n                this._generateCachedSprite();\n            }\n            else {\n                this._destroyCachedSprite();\n            }\n\n            this._cacheAsBitmap = value;\n        }\n    }\n});\n\n/**\n * Adds a child to the container.\n *\n * @param child {DisplayObject} The DisplayObject to add to the container\n * @return {DisplayObject} The child that was added.\n */\nDisplayObjectContainer.prototype.addChild = function (child) {\n    return this.addChildAt(child, this.children.length);\n};\n\n/**\n * Adds a child to the container at a specified index. If the index is out of bounds an error will be thrown\n *\n * @param child {DisplayObject} The child to add\n * @param index {Number} The index to place the child in\n * @return {DisplayObject} The child that was added.\n */\nDisplayObjectContainer.prototype.addChildAt = function (child, index) {\n    // prevent adding self as child\n    if (child === this) {\n        return;\n    }\n\n    if (index >= 0 && index <= this.children.length) {\n        if (child.parent) {\n            child.parent.removeChild(child);\n        }\n\n        child.parent = this;\n\n        this.children.splice(index, 0, child);\n\n        return child;\n    }\n    else {\n        throw new Error(child + 'addChildAt: The index '+ index +' supplied is out of bounds ' + this.children.length);\n    }\n};\n\n/**\n * Swaps the position of 2 Display Objects within this container.\n *\n * @param child {DisplayObject}\n * @param child2 {DisplayObject}\n */\nDisplayObjectContainer.prototype.swapChildren = function (child, child2) {\n    if (child === child2) {\n        return;\n    }\n\n    var index1 = this.getChildIndex(child);\n    var index2 = this.getChildIndex(child2);\n\n    if (index1 < 0 || index2 < 0) {\n        throw new Error('swapChildren: Both the supplied DisplayObjects must be a child of the caller.');\n    }\n\n    this.children[index1] = child2;\n    this.children[index2] = child;\n};\n\n/**\n * Returns the index position of a child DisplayObject instance\n *\n * @param child {DisplayObject} The DisplayObject instance to identify\n * @return {Number} The index position of the child display object to identify\n */\nDisplayObjectContainer.prototype.getChildIndex = function (child) {\n    var index = this.children.indexOf(child);\n\n    if (index === -1) {\n        throw new Error('The supplied DisplayObject must be a child of the caller');\n    }\n\n    return index;\n};\n\n/**\n * Changes the position of an existing child in the display object container\n *\n * @param child {DisplayObject} The child DisplayObject instance for which you want to change the index number\n * @param index {Number} The resulting index number for the child display object\n */\nDisplayObjectContainer.prototype.setChildIndex = function (child, index) {\n    if (index < 0 || index >= this.children.length) {\n        throw new Error('The supplied index is out of bounds');\n    }\n\n    var currentIndex = this.getChildIndex(child);\n\n    this.children.splice(currentIndex, 1); //remove from old position\n    this.children.splice(index, 0, child); //add at new position\n};\n\n/**\n * Returns the child at the specified index\n *\n * @param index {Number} The index to get the child from\n * @return {DisplayObject} The child at the given index, if any.\n */\nDisplayObjectContainer.prototype.getChildAt = function (index) {\n    if (index < 0 || index >= this.children.length) {\n        throw new Error('getChildAt: Supplied index ' + index + ' does not exist in the child list, or the supplied DisplayObject must be a child of the caller');\n    }\n\n    return this.children[index];\n};\n\n/**\n * Removes a child from the container.\n *\n * @param child {DisplayObject} The DisplayObject to remove\n * @return {DisplayObject} The child that was removed.\n */\nDisplayObjectContainer.prototype.removeChild = function (child) {\n    var index = this.children.indexOf(child);\n\n    if (index === -1) {\n        return;\n    }\n\n    return this.removeChildAt(index);\n};\n\n/**\n * Removes a child from the specified index position.\n *\n * @param index {Number} The index to get the child from\n * @return {DisplayObject} The child that was removed.\n */\nDisplayObjectContainer.prototype.removeChildAt = function (index) {\n    var child = this.getChildAt(index);\n\n    child.parent = null;\n    this.children.splice(index, 1);\n\n    return child;\n};\n\n/**\n * Removes all children from this container that are within the begin and end indexes.\n *\n * @param beginIndex {Number} The beginning position. Default value is 0.\n * @param endIndex {Number} The ending position. Default value is size of the container.\n */\nDisplayObjectContainer.prototype.removeChildren = function (beginIndex, endIndex) {\n    var begin = beginIndex || 0;\n    var end = typeof endIndex === 'number' ? endIndex : this.children.length;\n    var range = end - begin;\n\n    if (range > 0 && range <= end) {\n        var removed = this.children.splice(begin, range);\n\n        for (var i = 0; i < removed.length; ++i) {\n            removed[i].parent = null;\n        }\n\n        return removed;\n    }\n    else if (range === 0 && this.children.length === 0) {\n        return [];\n    }\n    else {\n        throw new RangeError('removeChildren: numeric values are outside the acceptable range.');\n    }\n};\n\n/**\n * Generates and updates the cached sprite for this object.\n *\n */\nDisplayObjectContainer.prototype.updateCachedSprite = function () {\n    this._generateCachedSprite();\n};\n\n/**\n * Useful function that returns a texture of the displayObject object that can then be used to create sprites\n * This can be quite useful if your displayObject is static / complicated and needs to be reused multiple times.\n *\n * @param resolution {Number} The resolution of the texture being generated\n * @param scaleMode {Number} See {{#crossLink \"PIXI/scaleModes:property\"}}PIXI.scaleModes{{/crossLink}} for possible values\n * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used to generate the texture.\n * @return {Texture} a texture of the graphics object\n */\nDisplayObjectContainer.prototype.generateTexture = function (resolution, scaleMode, renderer) {\n    var bounds = this.getLocalBounds();\n\n    var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0, renderer, scaleMode, resolution);\n\n    _tempMatrix.tx = -bounds.x;\n    _tempMatrix.ty = -bounds.y;\n\n    renderTexture.render(this, _tempMatrix);\n\n    return renderTexture;\n};\n\n/*\n * Updates the transform on all children of this container for rendering\n *\n * @private\n */\nDisplayObjectContainer.prototype.updateTransform = function () {\n    if (!this.visible) {\n        return;\n    }\n\n    this.displayObjectUpdateTransform();\n\n    if (this._cacheAsBitmap) {\n        return;\n    }\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].updateTransform();\n    }\n};\n\n// performance increase to avoid using call.. (10x faster)\nDisplayObjectContainer.prototype.displayObjectContainerUpdateTransform = DisplayObjectContainer.prototype.updateTransform;\n\n/**\n * Retrieves the bounds of the displayObjectContainer as a rectangle. The bounds calculation takes all visible children into consideration.\n *\n * @return {Rectangle} The rectangular bounding area\n */\nDisplayObjectContainer.prototype.getBounds = function () {\n    if (this.children.length === 0) {\n        return math.Rectangle.EMPTY;\n    }\n\n    // TODO the bounds have already been calculated this render session so return what we have\n\n    var minX = Infinity;\n    var minY = Infinity;\n\n    var maxX = -Infinity;\n    var maxY = -Infinity;\n\n    var childBounds;\n    var childMaxX;\n    var childMaxY;\n\n    var childVisible = false;\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        var child = this.children[i];\n\n        if (!child.visible) {\n            continue;\n        }\n\n        childVisible = true;\n\n        childBounds = this.children[i].getBounds();\n\n        minX = minX < childBounds.x ? minX : childBounds.x;\n        minY = minY < childBounds.y ? minY : childBounds.y;\n\n        childMaxX = childBounds.width + childBounds.x;\n        childMaxY = childBounds.height + childBounds.y;\n\n        maxX = maxX > childMaxX ? maxX : childMaxX;\n        maxY = maxY > childMaxY ? maxY : childMaxY;\n    }\n\n    if (!childVisible) {\n        return math.Rectangle.EMPTY;\n    }\n\n    this._bounds.x = minX;\n    this._bounds.y = minY;\n    this._bounds.width = maxX - minX;\n    this._bounds.height = maxY - minY;\n\n    // TODO: store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    //this._currentBounds = bounds;\n\n    return this._bounds;\n};\n\n/**\n * Retrieves the non-global local bounds of the displayObjectContainer as a rectangle.\n * The calculation takes all visible children into consideration.\n *\n * @return {Rectangle} The rectangular bounding area\n */\nDisplayObjectContainer.prototype.getLocalBounds = function () {\n    var matrixCache = this.worldTransform;\n\n    this.worldTransform = math.Matrix.IDENTITY;\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].updateTransform();\n    }\n\n    this.worldTransform = matrixCache;\n\n    return this.getBounds();\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * TODO - Optimization pass!\n *\n * @param renderer {WebGLRenderer} The renderer\n */\nDisplayObjectContainer.prototype.renderWebGL = function (renderer) {\n    // if the object is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        this._renderCachedSprite(renderer);\n        return;\n    }\n\n    var i, j;\n\n    // do a quick check to see if this element has a mask or a filter.\n    if (this._mask || this._filters) {\n        // push filter first as we need to ensure the stencil buffer is correct for any masking\n        if (this._filters) {\n            renderer.spriteBatch.flush();\n            renderer.filterManager.pushFilter(this._filterBlock);\n        }\n\n        if (this._mask) {\n            renderer.spriteBatch.stop();\n            renderer.maskManager.pushMask(this.mask, renderer);\n            renderer.spriteBatch.start();\n        }\n\n        // add this object to the batch, only rendered if it has a texture.\n        if (this.texture) {\n            renderer.spriteBatch.render(this);\n        }\n\n        // now loop through the children and make sure they get rendered\n        for (i = 0, j = this.children.length; i < j; i++) {\n            this.children[i].renderWebGL(renderer);\n        }\n\n        // time to stop the sprite batch as either a mask element or a filter draw will happen next\n        renderer.spriteBatch.stop();\n\n        if (this._mask) {\n            renderer.maskManager.popMask(this._mask, renderer);\n        }\n\n        if (this._filters) {\n            renderer.filterManager.popFilter();\n        }\n\n        renderer.spriteBatch.start();\n    }\n    else {\n        if (this.texture) {\n            renderer.spriteBatch.render(this);\n        }\n\n        // simple render children!\n        for (i = 0, j = this.children.length; i < j; ++i) {\n            this.children[i].renderWebGL(renderer);\n        }\n\n    }\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer} The renderer\n */\nDisplayObjectContainer.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        this._renderCachedSprite(renderer);\n        return;\n    }\n\n    if (this._mask) {\n        renderer.maskManager.pushMask(this._mask, renderer);\n    }\n\n    for (var i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].renderCanvas(renderer);\n    }\n\n    if (this._mask) {\n        renderer.maskManager.popMask(renderer);\n    }\n};\n\n/**\n * Internal method.\n *\n * @param renderer {WebGLRenderer|CanvasRenderer} The renderer\n * @private\n */\nDisplayObjectContainer.prototype._renderCachedSprite = function (renderer) {\n    this._cachedSprite.worldAlpha = this.worldAlpha;\n\n    if (renderer.gl) {\n        this._cachedSprite.renderWebGL(renderer);\n    }\n    else {\n        this._cachedSprite.renderCanvas(renderer);\n    }\n};\n\n/**\n * Internal method.\n *\n * @private\n */\nDisplayObjectContainer.prototype._generateCachedSprite = function () {\n    var bounds = this.getLocalBounds();\n\n    if (!this._cachedSprite) {\n        // TODO - RenderTexture now *requires* a renderer instance, so this is like broken\n        // because `renderer` isn't actually in scope here :P\n        var renderTexture = new RenderTexture(renderer, bounds.width | 0, bounds.height | 0);\n\n        this._cachedSprite = new Sprite(renderTexture);\n        this._cachedSprite.worldTransform = this.worldTransform;\n    }\n    else {\n        this._cachedSprite.texture.resize(bounds.width | 0, bounds.height | 0);\n    }\n\n    var tempFilters = this._filters;\n    this._filters = null;\n\n    this._cachedSprite.filters = tempFilters;\n\n    _tempMatrix.tx = -bounds.x;\n    _tempMatrix.ty = -bounds.y;\n\n    this._cachedSprite.texture.render(this, _tempMatrix, true);\n\n    this._cachedSprite.anchor.x = -(bounds.x / bounds.width);\n    this._cachedSprite.anchor.y = -(bounds.y / bounds.height);\n\n    this._filters = tempFilters;\n};\n\n/**\n * Destroys the cached sprite.\n *\n * @private\n */\nDisplayObjectContainer.prototype._destroyCachedSprite = function () {\n    if (!this._cachedSprite) {\n        return;\n    }\n\n    // TODO: Pool this sprite\n    this._cachedSprite.destroy(true, true);\n    this._cachedSprite = null;\n};\n","var math = require('../math'),\n    Texture = require('../textures/Texture'),\n    DisplayObjectContainer = require('./DisplayObjectContainer'),\n    CanvasTinter = require('../renderers/canvas/utils/CanvasTinter'),\n    utils = require('../utils'),\n    CONST = require('../const');\n\n/**\n * The Sprite object is the base for all textured objects that are rendered to the screen\n *\n * A sprite can be created directly from an image like this:\n *\n * ```js\n * var sprite = new Sprite.fromImage('assets/image.png');\n * ```\n *\n * @class Sprite\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param texture {Texture} The texture for this sprite\n */\nfunction Sprite(texture) {\n    DisplayObjectContainer.call(this);\n\n    /**\n     * The anchor sets the origin point of the texture.\n     * The default is 0,0 this means the texture's origin is the top left\n     * Setting than anchor to 0.5,0.5 means the textures origin is centered\n     * Setting the anchor to 1,1 would mean the textures origin points will be the bottom right corner\n     *\n     * @member {Point}\n     */\n    this.anchor = new math.Point();\n\n    /**\n     * The texture that the sprite is using\n     *\n     * @member {Texture}\n     * @private\n     */\n    this._texture = null;\n\n    /**\n     * The width of the sprite (this is initially set by the texture)\n     *\n     * @member {number}\n     * @private\n     */\n    this._width = 0;\n\n    /**\n     * The height of the sprite (this is initially set by the texture)\n     *\n     * @member {number}\n     * @private\n     */\n    this._height = 0;\n\n    /**\n     * The tint applied to the sprite. This is a hex value. A value of 0xFFFFFF will remove any tint effect.\n     *\n     * @member {number}\n     * @default 0xFFFFFF\n     */\n    this.tint = 0xFFFFFF;\n\n    /**\n     * The blend mode to be applied to the sprite. Set to CONST.blendModes.NORMAL to remove any blend mode.\n     *\n     * @member {number}\n     * @default CONST.blendModes.NORMAL;\n     */\n    this.blendMode = CONST.blendModes.NORMAL;\n\n    /**\n     * The shader that will be used to render the sprite. Set to null to remove a current shader.\n     *\n     * @member {AbstractFilter}\n     */\n    this.shader = null;\n\n    this.renderable = true;\n\n    // call texture setter\n    this.texture = texture || Texture.EMPTY;\n}\n\nSprite.prototype.destroy = function (destroyTexture, destroyBaseTexture) {\n    DisplayObjectContainer.prototype.destroy.call(this);\n\n    this.anchor = null;\n\n    if (destroyTexture) {\n        this._texture.destroy(destroyBaseTexture);\n    }\n\n    this._texture = null;\n    this.shader = null;\n};\n\n// constructor\nSprite.prototype = Object.create(DisplayObjectContainer.prototype);\nSprite.prototype.constructor = Sprite;\nmodule.exports = Sprite;\n\nObject.defineProperties(Sprite.prototype, {\n    /**\n     * The width of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member\n     * @memberof Sprite#\n     */\n    width: {\n        get: function () {\n            return this.scale.x * this.texture.frame.width;\n        },\n        set: function (value) {\n            this.scale.x = value / this.texture.frame.width;\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member\n     * @memberof Sprite#\n     */\n    height: {\n        get: function () {\n            return  this.scale.y * this.texture.frame.height;\n        },\n        set: function (value) {\n            this.scale.y = value / this.texture.frame.height;\n            this._height = value;\n        }\n    },\n\n    /**\n     * The height of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member\n     * @memberof Sprite#\n     */\n    texture: {\n        get: function () {\n            return  this._texture;\n        },\n        set: function (value) {\n            if (this._texture === value) {\n                return;\n            }\n\n            this._texture = value;\n            this.cachedTint = 0xFFFFFF;\n\n            if (value) {\n                // wait for the texture to load\n                if (value.baseTexture.hasLoaded) {\n                    this._onTextureUpdate();\n                }\n                else {\n                    value.once('update', this._onTextureUpdate.bind(this));\n                }\n            }\n        }\n    },\n});\n\n/**\n * When the texture is updated, this event will fire to update the scale and frame\n *\n * @private\n */\nSprite.prototype._onTextureUpdate = function () {\n    // so if _width is 0 then width was not set..\n    if (this._width) {\n        this.scale.x = this._width / this.texture.frame.width;\n    }\n\n    if (this._height) {\n        this.scale.y = this._height / this.texture.frame.height;\n    }\n};\n\n/**\n * Returns the bounds of the Sprite as a rectangle. The bounds calculation takes the worldTransform into account.\n *\n * @param matrix {Matrix} the transformation matrix of the sprite\n * @return {Rectangle} the framing rectangle\n */\nSprite.prototype.getBounds = function (matrix) {\n    var width = this.texture.frame.width;\n    var height = this.texture.frame.height;\n\n    var w0 = width * (1-this.anchor.x);\n    var w1 = width * -this.anchor.x;\n\n    var h0 = height * (1-this.anchor.y);\n    var h1 = height * -this.anchor.y;\n\n    var worldTransform = matrix || this.worldTransform ;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var minX,\n        maxX,\n        minY,\n        maxY;\n\n    if(b === 0 && c === 0)\n    {\n        // scale may be negative!\n        if (a < 0) {\n            a *= -1;\n        }\n\n        if (d < 0) {\n            d *= -1;\n        }\n\n        // this means there is no rotation going on right? RIGHT?\n        // if thats the case then we can avoid checking the bound values! yay\n        minX = a * w1 + tx;\n        maxX = a * w0 + tx;\n        minY = d * h1 + ty;\n        maxY = d * h0 + ty;\n    }\n    else\n    {\n        var x1 = a * w1 + c * h1 + tx;\n        var y1 = d * h1 + b * w1 + ty;\n\n        var x2 = a * w0 + c * h1 + tx;\n        var y2 = d * h1 + b * w0 + ty;\n\n        var x3 = a * w0 + c * h0 + tx;\n        var y3 = d * h0 + b * w0 + ty;\n\n        var x4 =  a * w1 + c * h0 + tx;\n        var y4 =  d * h0 + b * w1 + ty;\n\n        minX = x1;\n        minX = x2 < minX ? x2 : minX;\n        minX = x3 < minX ? x3 : minX;\n        minX = x4 < minX ? x4 : minX;\n\n        minY = y1;\n        minY = y2 < minY ? y2 : minY;\n        minY = y3 < minY ? y3 : minY;\n        minY = y4 < minY ? y4 : minY;\n\n        maxX = x1;\n        maxX = x2 > maxX ? x2 : maxX;\n        maxX = x3 > maxX ? x3 : maxX;\n        maxX = x4 > maxX ? x4 : maxX;\n\n        maxY = y1;\n        maxY = y2 > maxY ? y2 : maxY;\n        maxY = y3 > maxY ? y3 : maxY;\n        maxY = y4 > maxY ? y4 : maxY;\n    }\n\n    var bounds = this._bounds;\n\n    bounds.x = minX;\n    bounds.width = maxX - minX;\n\n    bounds.y = minY;\n    bounds.height = maxY - minY;\n\n    // store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    this._currentBounds = bounds;\n\n    return bounds;\n};\n\n/**\n* Renders the object using the Canvas renderer\n*\n* @param renderer {CanvasRenderer} The renderer\n*/\nSprite.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0 || this.texture.crop.width <= 0 || this.texture.crop.height <= 0) {\n        return;\n    }\n\n    if (this.blendMode !== renderer.currentBlendMode) {\n        renderer.currentBlendMode = this.blendMode;\n        renderer.context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode];\n    }\n\n    if (this._mask) {\n        renderer.maskManager.pushMask(this._mask, renderer);\n    }\n\n    //  Ignore null sources\n    if (this.texture.valid) {\n        var resolution = this.texture.baseTexture.resolution / renderer.resolution;\n\n        renderer.context.globalAlpha = this.worldAlpha;\n\n        // If smoothingEnabled is supported and we need to change the smoothing property for this texture\n        if (renderer.smoothProperty && renderer.scaleMode !== this.texture.baseTexture.scaleMode) {\n            renderer.scaleMode = this.texture.baseTexture.scaleMode;\n            renderer.context[renderer.smoothProperty] = (renderer.scaleMode === CONST.scaleModes.LINEAR);\n        }\n\n        // If the texture is trimmed we offset by the trim x/y, otherwise we use the frame dimensions\n        var dx = (this.texture.trim ? this.texture.trim.x : 0) - (this.anchor.x * this.texture.trim.width);\n        var dy = (this.texture.trim ? this.texture.trim.y : 0) - (this.anchor.y * this.texture.trim.height);\n\n        // Allow for pixel rounding\n        if (renderer.roundPixels) {\n            renderer.context.setTransform(\n                this.worldTransform.a,\n                this.worldTransform.b,\n                this.worldTransform.c,\n                this.worldTransform.d,\n                (this.worldTransform.tx * renderer.resolution) | 0,\n                (this.worldTransform.ty * renderer.resolution) | 0\n            );\n\n            dx = dx | 0;\n            dy = dy | 0;\n        }\n        else {\n            renderer.context.setTransform(\n                this.worldTransform.a,\n                this.worldTransform.b,\n                this.worldTransform.c,\n                this.worldTransform.d,\n                this.worldTransform.tx * renderer.resolution,\n                this.worldTransform.ty * renderer.resolution\n            );\n        }\n\n        if (this.tint !== 0xFFFFFF) {\n            if (this.cachedTint !== this.tint) {\n                this.cachedTint = this.tint;\n\n                // TODO clean up caching - how to clean up the caches?\n                this.tintedTexture = CanvasTinter.getTintedTexture(this, this.tint);\n            }\n\n            renderer.context.drawImage(\n                this.tintedTexture,\n                0,\n                0,\n                this.texture.crop.width,\n                this.texture.crop.height,\n                dx / resolution,\n                dy / resolution,\n                this.texture.crop.width / resolution,\n                this.texture.crop.height / resolution\n            );\n        }\n        else {\n            renderer.context.drawImage(\n                this.texture.baseTexture.source,\n                this.texture.crop.x,\n                this.texture.crop.y,\n                this.texture.crop.width,\n                this.texture.crop.height,\n                dx / resolution,\n                dy / resolution,\n                this.texture.crop.width / resolution,\n                this.texture.crop.height / resolution\n            );\n        }\n    }\n\n    for (var i = 0, j = this.children.length; i < j; i++) {\n        this.children[i].renderCanvas(renderer);\n    }\n\n    if (this._mask) {\n        renderer.maskManager.popMask(renderer);\n    }\n};\n\n// some helper functions..\n\n/**\n * Helper function that creates a sprite that will contain a texture from the TextureCache based on the frameId\n * The frame ids are created when a Texture packer file has been loaded\n *\n * @static\n * @param frameId {String} The frame Id of the texture in the cache\n * @return {Sprite} A new Sprite using a texture from the texture cache matching the frameId\n */\nSprite.fromFrame = function (frameId) {\n    var texture = utils.TextureCache[frameId];\n\n    if (!texture) {\n        throw new Error('The frameId \"' + frameId + '\" does not exist in the texture cache' + this);\n    }\n\n    return new Sprite(texture);\n};\n\n/**\n * Helper function that creates a sprite that will contain a texture based on an image url\n * If the image is not in the texture cache it will be loaded\n *\n * @static\n * @param imageId {String} The image url of the texture\n * @return {Sprite} A new Sprite using a texture from the texture cache matching the image id\n */\nSprite.fromImage = function (imageId, crossorigin, scaleMode) {\n    return new Sprite(Texture.fromImage(imageId, crossorigin, scaleMode));\n};\n","var DisplayObjectContainer = require('./DisplayObjectContainer'),\n    WebGLFastSpriteBatch = require('../renderers/webgl/utils/WebGLFastSpriteBatch');\n\n/**\n * The SpriteBatch class is a really fast version of the DisplayObjectContainer built solely for speed,\n * so use when you need a lot of sprites or particles. The tradeoff of the SpriteBatch is that advanced\n * functionality will not work. SpriteBatch implements only the basic object transform (position, scale, rotation).\n * Any other functionality like tinting, masking, etc will not work on sprites in this batch.\n *\n * It's extremely easy to use :\n *\n * ```js\n * var container = new SpriteBatch();\n *\n * for(var i = 0; i < 100; ++i) {\n *     var sprite = new PIXI.Sprite.fromImage(\"myImage.png\");\n *     container.addChild(sprite);\n * }\n * ```\n *\n * And here you have a hundred sprites that will be renderer at the speed of light.\n *\n * @class\n * @namespace PIXI\n */\n\n//TODO RENAME to PARTICLE CONTAINER?\nfunction SpriteBatch() {\n    DisplayObjectContainer.call(this);\n}\n\nSpriteBatch.prototype = Object.create(DisplayObjectContainer.prototype);\nSpriteBatch.prototype.constructor = SpriteBatch;\nmodule.exports = SpriteBatch;\n\n/**\n * Updates the object transform for rendering\n *\n * @private\n */\nSpriteBatch.prototype.updateTransform = function () {\n    // TODO don't need to!\n    this.displayObjectUpdateTransform();\n    //  PIXI.DisplayObjectContainer.prototype.updateTransform.call( this );\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer} The webgl renderer\n * @private\n */\nSpriteBatch.prototype.renderWebGL = function (renderer) {\n    if (!this.visible || this.alpha <= 0 || !this.children.length) {\n        return;\n    }\n\n    renderer.spriteBatch.stop();\n\n    renderer.shaderManager.setShader(renderer.shaderManager.fastShader);\n\n    renderer.fastSpriteBatch.begin(this);\n    renderer.fastSpriteBatch.render(this);\n\n    renderer.spriteBatch.start();\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer} The canvas renderer\n * @private\n */\nSpriteBatch.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0 || !this.children.length) {\n        return;\n    }\n\n    var context = renderer.context;\n    var transform = this.worldTransform;\n    var isRotated = true;\n\n    context.globalAlpha = this.worldAlpha;\n\n    this.displayObjectUpdateTransform();\n\n    for (var i = 0; i < this.children.length; ++i) {\n        var child = this.children[i];\n\n        if (!child.visible) {\n            continue;\n        }\n\n        var frame = child.texture.frame;\n\n        context.globalAlpha = this.worldAlpha * child.alpha;\n\n        if (child.rotation % (Math.PI * 2) === 0) {\n            // this is the fastest  way to optimise! - if rotation is 0 then we can avoid any kind of setTransform call\n            if (isRotated) {\n                context.setTransform(\n                    transform.a,\n                    transform.b,\n                    transform.c,\n                    transform.d,\n                    transform.tx,\n                    transform.ty\n                );\n\n                isRotated = false;\n            }\n\n            context.drawImage(\n                child.texture.baseTexture.source,\n                frame.x,\n                frame.y,\n                frame.width,\n                frame.height,\n                ((child.anchor.x) * (-frame.width * child.scale.x) + child.position.x  + 0.5) | 0,\n                ((child.anchor.y) * (-frame.height * child.scale.y) + child.position.y  + 0.5) | 0,\n                frame.width * child.scale.x,\n                frame.height * child.scale.y\n            );\n        }\n        else {\n            if (!isRotated) {\n                isRotated = true;\n            }\n\n            child.displayObjectUpdateTransform();\n\n            var childTransform = child.worldTransform;\n\n            if (renderer.roundPixels) {\n                context.setTransform(\n                    childTransform.a,\n                    childTransform.b,\n                    childTransform.c,\n                    childTransform.d,\n                    childTransform.tx | 0,\n                    childTransform.ty | 0\n                );\n            }\n            else {\n                context.setTransform(\n                    childTransform.a,\n                    childTransform.b,\n                    childTransform.c,\n                    childTransform.d,\n                    childTransform.tx,\n                    childTransform.ty\n                );\n            }\n\n            context.drawImage(\n                child.texture.baseTexture.source,\n                frame.x,\n                frame.y,\n                frame.width,\n                frame.height,\n                ((child.anchor.x) * (-frame.width) + 0.5) | 0,\n                ((child.anchor.y) * (-frame.height) + 0.5) | 0,\n                frame.width,\n                frame.height\n            );\n        }\n    }\n};\n","/**\n * @file        Main export of the PIXI core library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nvar core = module.exports = {\n    CONST: require('./const'),\n\n    // utils\n    utils: require('./utils'),\n    math: require('./math'),\n\n    // display\n    DisplayObject:          require('./display/DisplayObject'),\n    DisplayObjectContainer: require('./display/DisplayObjectContainer'),\n    Sprite:                 require('./display/Sprite'),\n    SpriteBatch:            require('./display/SpriteBatch'),\n\n    // primitives\n    Graphics:               require('./primitives/Graphics'),\n    GraphicsData:           require('./primitives/GraphicsData'),\n\n    // textures\n    Texture:                require('./textures/Texture'),\n    BaseTexture:            require('./textures/BaseTexture'),\n    RenderTexture:          require('./textures/RenderTexture'),\n    VideoBaseTexture:       require('./textures/VideoBaseTexture'),\n\n    // renderers - canvas\n    CanvasRenderer:         require('./renderers/canvas/CanvasRenderer'),\n    CanvasGraphics:         require('./renderers/canvas/utils/CanvasGraphics'),\n    CanvasBuffer:           require('./renderers/canvas/utils/CanvasBuffer'),\n\n    // renderers - webgl\n    WebGLRenderer:         require('./renderers/webgl/WebGLRenderer'),\n    WebGLGraphics:         require('./renderers/webgl/utils/WebGLGraphics'),\n\n    /**\n     * This helper function will automatically detect which renderer you should be using.\n     * WebGL is the preferred renderer as it is a lot faster. If webGL is not supported by\n     * the browser then this function will return a canvas renderer\n     *\n     * @param width=800 {number} the width of the renderers view\n     * @param height=600 {number} the height of the renderers view\n     * @param [options] {object} The optional renderer parameters\n     * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n     * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n     * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment)\n     * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you\n     *      need to call toDataUrl on the webgl context\n     * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n     * @param [noWebGL=false] {Boolean} prevents selection of WebGL renderer, even if such is present\n     *\n     * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer\n     */\n    autoDetectRenderer: function (width, height, options, noWebGL) {\n        width = width || 800;\n        height = height || 600;\n\n        if (!noWebGL && require('webgl-enabled')()) {\n            return new core.WebGLRenderer(width, height, options);\n        }\n\n        return new core.CanvasRenderer(width, height, options);\n    },\n\n    /**\n     * This helper function will automatically detect which renderer you should be using. This function is very\n     * similar to the autoDetectRenderer function except that is will return a canvas renderer for android.\n     * Even thought both android chrome supports webGL the canvas implementation perform better at the time of writing.\n     * This function will likely change and update as webGL performance improves on these devices.\n     *\n     * @param width=800 {number} the width of the renderers view\n     * @param height=600 {number} the height of the renderers view\n     * @param [options] {object} The optional renderer parameters\n     * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n     * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n     * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment)\n     * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you\n     *      need to call toDataUrl on the webgl context\n     * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n     *\n     * @return {WebGLRenderer|CanvasRenderer} Returns WebGL renderer if available, otherwise CanvasRenderer\n     */\n    autoDetectRecommendedRenderer: function (width, height, options) {\n        var isAndroid = /Android/i.test(navigator.userAgent);\n\n        return core.autoDetectRenderer(width, height, options, isAndroid);\n    }\n};\n","var Point = require('./Point');\n\n/**\n * The Matrix class is now an object, which makes it a lot faster,\n * here is a representation of it :\n * | a | b | tx|\n * | c | d | ty|\n * | 0 | 0 | 1 |\n *\n * @class\n * @namespace PIXI\n */\nfunction Matrix() {\n    /**\n     * @member {number}\n     * @default 1\n     */\n    this.a = 1;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.b = 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.c = 0;\n\n    /**\n     * @member {number}\n     * @default 1\n     */\n    this.d = 1;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.tx = 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.ty = 0;\n}\n\nMatrix.prototype.constructor = Matrix;\nmodule.exports = Matrix;\n\n/**\n * Creates a Matrix object based on the given array. The Element to Matrix mapping order is as follows:\n *\n * a = array[0]\n * b = array[1]\n * c = array[3]\n * d = array[4]\n * tx = array[2]\n * ty = array[5]\n *\n * @param array {number[]} The array that the matrix will be populated from.\n */\nMatrix.prototype.fromArray = function (array) {\n    this.a = array[0];\n    this.b = array[1];\n    this.c = array[3];\n    this.d = array[4];\n    this.tx = array[2];\n    this.ty = array[5];\n};\n\n/**\n * Creates an array from the current Matrix object.\n *\n * @param transpose {boolean} Whether we need to transpose the matrix or not\n * @return {number[]} the newly created array which contains the matrix\n */\nMatrix.prototype.toArray = function (transpose) {\n    if (!this.array) {\n        this.array = new Float32Array(9);\n    }\n\n    var array = this.array;\n\n    if (transpose) {\n        array[0] = this.a;\n        array[1] = this.b;\n        array[2] = 0;\n        array[3] = this.c;\n        array[4] = this.d;\n        array[5] = 0;\n        array[6] = this.tx;\n        array[7] = this.ty;\n        array[8] = 1;\n    }\n    else {\n        array[0] = this.a;\n        array[1] = this.c;\n        array[2] = this.tx;\n        array[3] = this.b;\n        array[4] = this.d;\n        array[5] = this.ty;\n        array[6] = 0;\n        array[7] = 0;\n        array[8] = 1;\n    }\n\n    return array;\n};\n\n/**\n * Get a new position with the current transformation applied.\n * Can be used to go from a child's coordinate space to the world coordinate space. (e.g. rendering)\n *\n * @param pos {Point} The origin\n * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input)\n * @return {Point} The new point, transformed through this matrix\n */\nMatrix.prototype.apply = function (pos, newPos) {\n    newPos = newPos || new Point();\n\n    newPos.x = this.a * pos.x + this.c * pos.y + this.tx;\n    newPos.y = this.b * pos.x + this.d * pos.y + this.ty;\n\n    return newPos;\n};\n\n/**\n * Get a new position with the inverse of the current transformation applied.\n * Can be used to go from the world coordinate space to a child's coordinate space. (e.g. input)\n *\n * @param pos {Point} The origin\n * @param [newPos] {Point} The point that the new position is assigned to (allowed to be same as input)\n * @return {Point} The new point, inverse-transformed through this matrix\n */\nMatrix.prototype.applyInverse = function (pos, newPos) {\n    newPos = newPos || new Point();\n\n    var id = 1 / (this.a * this.d + this.c * -this.b);\n\n    newPos.x = this.d * id * pos.x + -this.c * id * pos.y + (this.ty * this.c - this.tx * this.d) * id;\n    newPos.y = this.a * id * pos.y + -this.b * id * pos.x + (-this.ty * this.a + this.tx * this.b) * id;\n\n    return newPos;\n};\n\n/**\n * Translates the matrix on the x and y.\n *\n * @param {number} x\n * @param {number} y\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.translate = function (x, y) {\n    this.tx += x;\n    this.ty += y;\n\n    return this;\n};\n\n/**\n * Applies a scale transformation to the matrix.\n *\n * @param {number} x The amount to scale horizontally\n * @param {number} y The amount to scale vertically\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.scale = function (x, y) {\n    this.a *= x;\n    this.d *= y;\n    this.c *= x;\n    this.b *= y;\n    this.tx *= x;\n    this.ty *= y;\n\n    return this;\n};\n\n\n/**\n * Applies a rotation transformation to the matrix.\n *\n * @param {number} angle - The angle in radians.\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.rotate = function (angle) {\n    var cos = Math.cos( angle );\n    var sin = Math.sin( angle );\n\n    var a1 = this.a;\n    var c1 = this.c;\n    var tx1 = this.tx;\n\n    this.a = a1 * cos-this.b * sin;\n    this.b = a1 * sin+this.b * cos;\n    this.c = c1 * cos-this.d * sin;\n    this.d = c1 * sin+this.d * cos;\n    this.tx = tx1 * cos - this.ty * sin;\n    this.ty = tx1 * sin + this.ty * cos;\n\n    return this;\n};\n\n/**\n * Appends the given Matrix to this Matrix.\n *\n * @param {Matrix} matrix\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.append = function (matrix) {\n    var a1 = this.a;\n    var b1 = this.b;\n    var c1 = this.c;\n    var d1 = this.d;\n\n    this.a  = matrix.a * a1 + matrix.b * c1;\n    this.b  = matrix.a * b1 + matrix.b * d1;\n    this.c  = matrix.c * a1 + matrix.d * c1;\n    this.d  = matrix.c * b1 + matrix.d * d1;\n\n    this.tx = matrix.tx * a1 + matrix.ty * c1 + this.tx;\n    this.ty = matrix.tx * b1 + matrix.ty * d1 + this.ty;\n\n    return this;\n};\n\n/**\n * Resets this Matix to an identity (default) matrix.\n *\n * @return {Matrix} This matrix. Good for chaining method calls.\n */\nMatrix.prototype.identity = function () {\n    this.a = 1;\n    this.b = 0;\n    this.c = 0;\n    this.d = 1;\n    this.tx = 0;\n    this.ty = 0;\n\n    return this;\n};\n\nMatrix.IDENTITY = new Matrix();\n","/**\n * The Point object represents a location in a two-dimensional coordinate system, where x represents\n * the horizontal axis and y represents the vertical axis.\n *\n * @class\n * @namespace PIXI\n * @param [x=0] {number} position of the point on the x axis\n * @param [y=0] {number} position of the point on the y axis\n */\nfunction Point(x, y) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n}\n\nPoint.prototype.constructor = Point;\nmodule.exports = Point;\n\n/**\n * Creates a clone of this point\n *\n * @return {Point} a copy of the point\n */\nPoint.prototype.clone = function () {\n    return new Point(this.x, this.y);\n};\n\n/**\n * Sets the point to a new x and y position.\n * If y is omitted, both x and y will be set to x.\n *\n * @param [x=0] {number} position of the point on the x axis\n * @param [y=0] {number} position of the point on the y axis\n */\nPoint.prototype.set = function (x, y) {\n    this.x = x || 0;\n    this.y = y || ( (y !== 0) ? this.x : 0 ) ;\n};\n","/**\n * @namespace PIXI.math\n */\nmodule.exports = {\n    /**\n     * @property {number} PI_2 - Math.PI x 2\n     * @constant\n     * @static\n     */\n    PI_2: Math.PI * 2,\n\n    /**\n     * @property {number} RAD_TO_DEG - Constant conversion factor for converting radians to degrees\n     * @constant\n     * @static\n     */\n    RAD_TO_DEG: 180 / Math.PI,\n\n    /**\n     * @property {Number} DEG_TO_RAD - Constant conversion factor for converting degrees to radians\n     * @constant\n     * @static\n     */\n    DEG_TO_RAD: Math.PI / 180,\n\n    Point:      require('./Point'),\n    Matrix:     require('./Matrix'),\n\n    Circle:     require('./shapes/Circle'),\n    Ellipse:    require('./shapes/Ellipse'),\n    Polygon:    require('./shapes/Polygon'),\n    Rectangle:  require('./shapes/Rectangle'),\n    RoundedRectangle: require('./shapes/RoundedRectangle')\n};\n","var Rectangle = require('./Rectangle'),\n    CONST = require('../../const');\n\n/**\n * The Circle object can be used to specify a hit area for displayObjects\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the center of this circle\n * @param y {number} The Y coordinate of the center of this circle\n * @param radius {number} The radius of the circle\n */\nfunction Circle(x, y, radius) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.radius = radius || 0;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.CIRC;\n}\n\nCircle.prototype.constructor = Circle;\nmodule.exports = Circle;\n\n/**\n * Creates a clone of this Circle instance\n *\n * @method clone\n * @return {Circle} a copy of the Circle\n */\nCircle.prototype.clone = function () {\n    return new Circle(this.x, this.y, this.radius);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this circle\n *\n * @method contains\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this Circle\n */\nCircle.prototype.contains = function (x, y) {\n    if (this.radius <= 0) {\n        return false;\n    }\n\n    var dx = (this.x - x),\n        dy = (this.y - y),\n        r2 = this.radius * this.radius;\n\n    dx *= dx;\n    dy *= dy;\n\n    return (dx + dy <= r2);\n};\n\n/**\n* Returns the framing rectangle of the circle as a Rectangle object\n*\n* @method getBounds\n* @return {Rectangle} the framing rectangle\n*/\nCircle.prototype.getBounds = function () {\n    return new Rectangle(this.x - this.radius, this.y - this.radius, this.radius * 2, this.radius * 2);\n};\n","var Rectangle = require('./Rectangle'),\n    CONST = require('../../const');\n\n/**\n * The Ellipse object can be used to specify a hit area for displayObjects\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the center of the ellipse\n * @param y {number} The Y coordinate of the center of the ellipse\n * @param width {number} The half width of this ellipse\n * @param height {number} The half height of this ellipse\n */\nfunction Ellipse(x, y, width, height) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.width = width || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.height = height || 0;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.ELIP;\n}\n\nEllipse.prototype.constructor = Ellipse;\nmodule.exports = Ellipse;\n\n/**\n * Creates a clone of this Ellipse instance\n *\n * @method clone\n * @return {Ellipse} a copy of the ellipse\n */\nEllipse.prototype.clone = function () {\n    return new Ellipse(this.x, this.y, this.width, this.height);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this ellipse\n *\n * @method contains\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coords are within this ellipse\n */\nEllipse.prototype.contains = function (x, y) {\n    if (this.width <= 0 || this.height <= 0) {\n        return false;\n    }\n\n    //normalize the coords to an ellipse with center 0,0\n    var normx = ((x - this.x) / this.width),\n        normy = ((y - this.y) / this.height);\n\n    normx *= normx;\n    normy *= normy;\n\n    return (normx + normy <= 1);\n};\n\n/**\n* Returns the framing rectangle of the ellipse as a Rectangle object\n*\n* @method getBounds\n* @return {Rectangle} the framing rectangle\n*/\nEllipse.prototype.getBounds = function () {\n    return new Rectangle(this.x - this.width, this.y - this.height, this.width, this.height);\n};\n","var Point = require('../Point'),\n    CONST = require('../../const');\n\n/**\n * @class\n * @namespace PIXI\n * @param points* {Point[]|number[]|Point...|number...} This can be an array of Points that form the polygon,\n *      a flat array of numbers that will be interpreted as [x,y, x,y, ...], or the arguments passed can be\n *      all the points of the polygon e.g. `new Polygon(new Point(), new Point(), ...)`, or the\n *      arguments passed can be flat x,y values e.g. `new Polygon(x,y, x,y, x,y, ...)` where `x` and `y` are\n *      Numbers.\n */\nfunction Polygon(points) {\n    //if points isn't an array, use arguments as the array\n    if (!(points instanceof Array)) {\n        points = Array.prototype.slice.call(arguments);\n    }\n\n    //if this is a flat array of numbers, convert it to points\n    if (points[0] instanceof Point) {\n        var p = [];\n        for (var i = 0, il = points.length; i < il; i++) {\n            p.push(points[i].x, points[i].y);\n        }\n\n        points = p;\n    }\n\n    this.closed = true;\n\n    /**\n     * An array of the points of this polygon\n     *\n     * @member {Point[]}\n     */\n    this.points = points;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.POLY;\n}\n\nPolygon.prototype.constructor = Polygon;\nmodule.exports = Polygon;\n\n/**\n * Creates a clone of this polygon\n *\n * @return {Polygon} a copy of the polygon\n */\nPolygon.prototype.clone = function () {\n    return new Polygon(this.points.slice());\n};\n\n/**\n * Checks whether the x and y coordinates passed to this function are contained within this polygon\n *\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this polygon\n */\nPolygon.prototype.contains = function (x, y) {\n    var inside = false;\n\n    // use some raycasting to test hits\n    // https://github.com/substack/point-in-polygon/blob/master/index.js\n    var length = this.points.length / 2;\n\n    for (var i = 0, j = length - 1; i < length; j = i++) {\n        var xi = this.points[i * 2], yi = this.points[i * 2 + 1],\n            xj = this.points[j * 2], yj = this.points[j * 2 + 1],\n            intersect = ((yi > y) !== (yj > y)) && (x < (xj - xi) * (y - yi) / (yj - yi) + xi);\n\n        if (intersect) {\n            inside = !inside;\n        }\n    }\n\n    return inside;\n};\n","var CONST = require('../../const');\n\n/**\n * the Rectangle object is an area defined by its position, as indicated by its top-left corner point (x, y) and by its width and its height.\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the upper-left corner of the rectangle\n * @param y {number} The Y coordinate of the upper-left corner of the rectangle\n * @param width {number} The overall width of this rectangle\n * @param height {number} The overall height of this rectangle\n */\nfunction Rectangle(x, y, width, height) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.width = width || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.height = height || 0;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.RECT;\n}\n\nRectangle.prototype.constructor = Rectangle;\nmodule.exports = Rectangle;\n\n/**\n * A constant empty rectangle.\n *\n * @static\n * @constant\n */\nRectangle.EMPTY = new Rectangle(0, 0, 0, 0);\n\n\n/**\n * Creates a clone of this Rectangle\n *\n * @return {Rectangle} a copy of the rectangle\n */\nRectangle.prototype.clone = function () {\n    return new Rectangle(this.x, this.y, this.width, this.height);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this Rectangle\n *\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this Rectangle\n */\nRectangle.prototype.contains = function (x, y) {\n    if (this.width <= 0 || this.height <= 0) {\n        return false;\n    }\n\n    if (x >= this.x && x <= this.x + this.width) {\n        if (y >= this.y && y <= this.y + this.height) {\n            return true;\n        }\n    }\n\n    return false;\n};\n","var CONST = require('../../const');\n\n/**\n * The Rounded Rectangle object is an area defined by its position and has nice rounded corners, as indicated by its top-left corner point (x, y) and by its width and its height.\n *\n * @class\n * @namespace PIXI\n * @param x {number} The X coordinate of the upper-left corner of the rounded rectangle\n * @param y {number} The Y coordinate of the upper-left corner of the rounded rectangle\n * @param width {number} The overall width of this rounded rectangle\n * @param height {number} The overall height of this rounded rectangle\n * @param radius {number} Controls the radius of the rounded corners\n */\nfunction RoundedRectangle(x, y, width, height, radius) {\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.x = x || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.y = y || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.width = width || 0;\n\n    /**\n     * @member {number}\n     * @default 0\n     */\n    this.height = height || 0;\n\n    /**\n     * @member {number}\n     * @default 20\n     */\n    this.radius = radius || 20;\n\n    /**\n     * The type of the object, mainly used to avoid `instanceof` checks\n     *\n     * @member {number}\n     */\n    this.type = CONST.SHAPES.RREC;\n}\n\nRoundedRectangle.prototype.constructor = RoundedRectangle;\nmodule.exports = RoundedRectangle;\n\n/**\n * Creates a clone of this Rounded Rectangle\n *\n * @return {RoundedRectangle} a copy of the rounded rectangle\n */\nRoundedRectangle.prototype.clone = function () {\n    return new RoundedRectangle(this.x, this.y, this.width, this.height, this.radius);\n};\n\n/**\n * Checks whether the x and y coordinates given are contained within this Rounded Rectangle\n *\n * @param x {number} The X coordinate of the point to test\n * @param y {number} The Y coordinate of the point to test\n * @return {boolean} Whether the x/y coordinates are within this Rounded Rectangle\n */\nRoundedRectangle.prototype.contains = function (x, y) {\n    if (this.width <= 0 || this.height <= 0) {\n        return false;\n    }\n\n    if (x >= this.x && x <= this.x + this.width) {\n        if (y >= this.y && y <= this.y + this.height) {\n            return true;\n        }\n    }\n\n    return false;\n};\n","var DisplayObjectContainer = require('../display/DisplayObjectContainer'),\n    Sprite = require('../display/Sprite'),\n    Texture = require('../textures/Texture'),\n    CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'),\n    CanvasGraphics = require('../renderers/canvas/utils/CanvasGraphics'),\n    WebGLGraphics = require('../renderers/webgl/utils/WebGLGraphics'),\n    GraphicsData = require('./GraphicsData'),\n    math = require('../math'),\n    CONST = require('../const');\n\n/**\n * The Graphics class contains methods used to draw primitive shapes such as lines, circles and\n * rectangles to the display, and color and fill them.\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n */\nfunction Graphics() {\n    DisplayObjectContainer.call(this);\n\n    this.renderable = true;\n\n    /**\n     * The alpha value used when filling the Graphics object.\n     *\n     * @member {number}\n     * @default 1\n     */\n    this.fillAlpha = 1;\n\n    /**\n     * The width (thickness) of any lines drawn.\n     *\n     * @member {number}\n     * @default 0\n     */\n    this.lineWidth = 0;\n\n    /**\n     * The color of any lines drawn.\n     *\n     * @member {string}\n     * @default 0\n     */\n    this.lineColor = 0;\n\n    /**\n     * Graphics data\n     *\n     * @member {GraphicsData[]}\n     * @private\n     */\n    this.graphicsData = [];\n\n    /**\n     * The tint applied to the graphic shape. This is a hex value. Apply a value of 0xFFFFFF to reset the tint.\n     *\n     * @member {number}\n     * @default 0xFFFFFF\n     */\n    this.tint = 0xFFFFFF;\n\n    /**\n     * The blend mode to be applied to the graphic shape. Apply a value of blendModes.NORMAL to reset the blend mode.\n     *\n     * @member {number}\n     * @default CONST.blendModes.NORMAL;\n     */\n    this.blendMode = CONST.blendModes.NORMAL;\n\n    /**\n     * Current path\n     *\n     * @member {GraphicsData}\n     * @private\n     */\n    this.currentPath = null;\n\n    /**\n     * Array containing some WebGL-related properties used by the WebGL renderer.\n     *\n     * @member {object<number, object>}\n     * @private\n     */\n    // TODO - _webgl should use a prototype object, not a random undocumented object...\n    this._webGL = {};\n\n    /**\n     * Whether this shape is being used as a mask.\n     *\n     * @member {boolean}\n     */\n    this.isMask = false;\n\n    /**\n     * The bounds' padding used for bounds calculation.\n     *\n     * @member {number}\n     */\n    this.boundsPadding = 0;\n\n    /**\n     * A cache of the local bounds to prevent recalculation.\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._localBounds = new math.Rectangle(0,0,1,1);\n\n    /**\n     * Used to detect if the graphics object has changed. If this is set to true then the graphics\n     * object will be recalculated.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.dirty = true;\n\n    /**\n     * Used to detect if the WebGL graphics object has changed. If this is set to true then the\n     * graphics object will be recalculated.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.glDirty = false;\n\n    /**\n     * Used to detect if the cached sprite object needs to be updated.\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.cachedSpriteDirty = false;\n}\n\n// constructor\nGraphics.prototype = Object.create(DisplayObjectContainer.prototype);\nGraphics.prototype.constructor = Graphics;\nmodule.exports = Graphics;\n\nObject.defineProperties(Graphics.prototype, {\n    /**\n     * When cacheAsBitmap is set to true the graphics object will be rendered as if it was a sprite.\n     * This is useful if your graphics element does not change often, as it will speed up the rendering\n     * of the object in exchange for taking up texture memory. It is also useful if you need the graphics\n     * object to be anti-aliased, because it will be rendered using canvas. This is not recommended if\n     * you are constantly redrawing the graphics element.\n     *\n     * @member {boolean}\n     * @memberof Graphics#\n     * @default false\n     * @private\n     */\n    cacheAsBitmap: {\n        get: function () {\n            return this._cacheAsBitmap;\n        },\n        set: function (value) {\n            this._cacheAsBitmap = value;\n\n            if (this._cacheAsBitmap) {\n                this._generateCachedSprite();\n            }\n            else {\n                this.destroyCachedSprite();\n                this.dirty = true;\n            }\n        }\n    }\n});\n\n/**\n * Creates a new Graphics object with the same values as this one.\n *\n * @return {Graphics}\n */\nGraphicsData.prototype.clone = function () {\n    var clone = new Graphics();\n\n    clone.renderable    = this.renderable;\n    clone.fillAlpha     = this.fillAlpha;\n    clone.lineWidth     = this.lineWidth;\n    clone.lineColor     = this.lineColor;\n    clone.tint          = this.tint;\n    clone.blendMode     = this.blendMode;\n    clone.isMask        = this.isMask;\n    clone.boundsPadding = this.boundsPadding;\n    clone.dirty         = this.dirty;\n    clone.glDirty       = this.glDirty;\n    clone.cachedSpriteDirty = this.cachedSpriteDirty;\n\n    // copy graphics data\n    for (var i = 0; i < this.graphicsData.length; ++i) {\n        clone.graphicsData.push(this.graphicsData.clone());\n    }\n\n    clone.currentPath = clone.graphicsData[clone.graphicsData.length - 1];\n\n    clone.updateLocalBounds();\n\n    return clone;\n};\n\n/**\n * Specifies the line style used for subsequent calls to Graphics methods such as the lineTo() method or the drawCircle() method.\n *\n * @param lineWidth {number} width of the line to draw, will update the objects stored style\n * @param color {number} color of the line to draw, will update the objects stored style\n * @param alpha {number} alpha of the line to draw, will update the objects stored style\n * @return {Graphics}\n */\nGraphics.prototype.lineStyle = function (lineWidth, color, alpha) {\n    this.lineWidth = lineWidth || 0;\n    this.lineColor = color || 0;\n    this.lineAlpha = (arguments.length < 3) ? 1 : alpha;\n\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length) {\n            // halfway through a line? start a new one!\n            this.drawShape( new math.Polygon( this.currentPath.shape.points.slice(-2) ));\n        }\n        else {\n            // otherwise its empty so lets just set the line properties\n            this.currentPath.lineWidth = this.lineWidth;\n            this.currentPath.lineColor = this.lineColor;\n            this.currentPath.lineAlpha = this.lineAlpha;\n        }\n    }\n\n    return this;\n};\n\n/**\n * Moves the current drawing position to x, y.\n *\n * @param x {number} the X coordinate to move to\n * @param y {number} the Y coordinate to move to\n * @return {Graphics}\n  */\nGraphics.prototype.moveTo = function (x, y) {\n    this.drawShape(new math.Polygon([x,y]));\n\n    return this;\n};\n\n/**\n * Draws a line using the current line style from the current drawing position to (x, y);\n * The current drawing position is then set to (x, y).\n *\n * @param x {number} the X coordinate to draw to\n * @param y {number} the Y coordinate to draw to\n * @return {Graphics}\n */\nGraphics.prototype.lineTo = function (x, y) {\n    this.currentPath.shape.points.push(x, y);\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * Calculate the points for a quadratic bezier curve and then draws it.\n * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c\n *\n * @param cpX {number} Control point x\n * @param cpY {number} Control point y\n * @param toX {number} Destination point x\n * @param toY {number} Destination point y\n * @return {Graphics}\n */\nGraphics.prototype.quadraticCurveTo = function (cpX, cpY, toX, toY) {\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length === 0) {\n            this.currentPath.shape.points = [0, 0];\n        }\n    }\n    else {\n        this.moveTo(0,0);\n    }\n\n    var xa,\n        ya,\n        n = 20,\n        points = this.currentPath.shape.points;\n\n    if (points.length === 0) {\n        this.moveTo(0, 0);\n    }\n\n    var fromX = points[points.length-2];\n    var fromY = points[points.length-1];\n\n    var j = 0;\n    for (var i = 1; i <= n; ++i) {\n        j = i / n;\n\n        xa = fromX + ( (cpX - fromX) * j );\n        ya = fromY + ( (cpY - fromY) * j );\n\n        points.push( xa + ( ((cpX + ( (toX - cpX) * j )) - xa) * j ),\n                     ya + ( ((cpY + ( (toY - cpY) * j )) - ya) * j ) );\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * Calculate the points for a bezier curve and then draws it.\n *\n * @param cpX {number} Control point x\n * @param cpY {number} Control point y\n * @param cpX2 {number} Second Control point x\n * @param cpY2 {number} Second Control point y\n * @param toX {number} Destination point x\n * @param toY {number} Destination point y\n * @return {Graphics}\n */\nGraphics.prototype.bezierCurveTo = function (cpX, cpY, cpX2, cpY2, toX, toY) {\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length === 0) {\n            this.currentPath.shape.points = [0, 0];\n        }\n    }\n    else {\n        this.moveTo(0,0);\n    }\n\n    var n = 20,\n        dt,\n        dt2,\n        dt3,\n        t2,\n        t3,\n        points = this.currentPath.shape.points;\n\n    var fromX = points[points.length-2];\n    var fromY = points[points.length-1];\n\n    var j = 0;\n\n    for (var i = 1; i <= n; ++i) {\n        j = i / n;\n\n        dt = (1 - j);\n        dt2 = dt * dt;\n        dt3 = dt2 * dt;\n\n        t2 = j * j;\n        t3 = t2 * j;\n\n        points.push( dt3 * fromX + 3 * dt2 * j * cpX + 3 * dt * t2 * cpX2 + t3 * toX,\n                     dt3 * fromY + 3 * dt2 * j * cpY + 3 * dt * t2 * cpY2 + t3 * toY);\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * The arcTo() method creates an arc/curve between two tangents on the canvas.\n *\n * \"borrowed\" from https://code.google.com/p/fxcanvas/ - thanks google!\n *\n * @param x1 {number} The x-coordinate of the beginning of the arc\n * @param y1 {number} The y-coordinate of the beginning of the arc\n * @param x2 {number} The x-coordinate of the end of the arc\n * @param y2 {number} The y-coordinate of the end of the arc\n * @param radius {number} The radius of the arc\n * @return {Graphics}\n */\nGraphics.prototype.arcTo = function (x1, y1, x2, y2, radius) {\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length === 0) {\n            this.currentPath.shape.points.push(x1, y1);\n        }\n    }\n    else {\n        this.moveTo(x1, y1);\n    }\n\n    var points = this.currentPath.shape.points,\n        fromX = points[points.length-2],\n        fromY = points[points.length-1],\n        a1 = fromY - y1,\n        b1 = fromX - x1,\n        a2 = y2   - y1,\n        b2 = x2   - x1,\n        mm = Math.abs(a1 * b2 - b1 * a2);\n\n    if (mm < 1.0e-8 || radius === 0) {\n        if (points[points.length-2] !== x1 || points[points.length-1] !== y1) {\n            points.push(x1, y1);\n        }\n    }\n    else {\n        var dd = a1 * a1 + b1 * b1,\n            cc = a2 * a2 + b2 * b2,\n            tt = a1 * a2 + b1 * b2,\n            k1 = radius * Math.sqrt(dd) / mm,\n            k2 = radius * Math.sqrt(cc) / mm,\n            j1 = k1 * tt / dd,\n            j2 = k2 * tt / cc,\n            cx = k1 * b2 + k2 * b1,\n            cy = k1 * a2 + k2 * a1,\n            px = b1 * (k2 + j1),\n            py = a1 * (k2 + j1),\n            qx = b2 * (k1 + j2),\n            qy = a2 * (k1 + j2),\n            startAngle = Math.atan2(py - cy, px - cx),\n            endAngle   = Math.atan2(qy - cy, qx - cx);\n\n        this.arc(cx + x1, cy + y1, radius, startAngle, endAngle, b1 * a2 > b2 * a1);\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * The arc method creates an arc/curve (used to create circles, or parts of circles).\n *\n * @param cx {number} The x-coordinate of the center of the circle\n * @param cy {number} The y-coordinate of the center of the circle\n * @param radius {number} The radius of the circle\n * @param startAngle {number} The starting angle, in radians (0 is at the 3 o'clock position of the arc's circle)\n * @param endAngle {number} The ending angle, in radians\n * @param anticlockwise {boolean} Optional. Specifies whether the drawing should be counterclockwise or clockwise. False is default, and indicates clockwise, while true indicates counter-clockwise.\n * @return {Graphics}\n */\nGraphics.prototype.arc = function (cx, cy, radius, startAngle, endAngle, anticlockwise) {\n    var startX = cx + Math.cos(startAngle) * radius;\n    var startY = cy + Math.sin(startAngle) * radius;\n    var points;\n\n    // TODO - This if-else makes no sense. It uses currentPath in the else where it doesn't exist...\n    if (this.currentPath) {\n        points = this.currentPath.shape.points;\n\n        if (points.length === 0) {\n            points.push(startX, startY);\n        }\n        else if (points[points.length-2] !== startX || points[points.length-1] !== startY) {\n            points.push(startX, startY);\n        }\n    }\n    else {\n        this.moveTo(startX, startY);\n        points = this.currentPath.shape.points;\n    }\n\n    if (startAngle === endAngle) {\n        return this;\n    }\n\n    if (!anticlockwise && endAngle <= startAngle) {\n        endAngle += Math.PI * 2;\n    }\n    else if (anticlockwise && startAngle <= endAngle) {\n        startAngle += Math.PI * 2;\n    }\n\n    var sweep = anticlockwise ? (startAngle - endAngle) *-1 : (endAngle - startAngle);\n    var segs = (Math.abs(sweep)/ (Math.PI * 2)) * 40;\n\n    if (sweep === 0) {\n        return this;\n    }\n\n    var theta = sweep/(segs*2);\n    var theta2 = theta*2;\n\n    var cTheta = Math.cos(theta);\n    var sTheta = Math.sin(theta);\n\n    var segMinus = segs - 1;\n\n    var remainder = ( segMinus % 1 ) / segMinus;\n\n    for (var i = 0; i <= segMinus; ++i) {\n        var real =  i + remainder * i;\n        var angle = ((theta) + startAngle + (theta2 * real));\n\n        var c = Math.cos(angle);\n        var s = -Math.sin(angle);\n\n        points.push(( (cTheta *  c) + (sTheta * s) ) * radius + cx,\n                    ( (cTheta * -s) + (sTheta * c) ) * radius + cy);\n    }\n\n    this.dirty = true;\n\n    return this;\n};\n\n/**\n * Specifies a simple one-color fill that subsequent calls to other Graphics methods\n * (such as lineTo() or drawCircle()) use when drawing.\n *\n * @param color {number} the color of the fill\n * @param alpha {number} the alpha of the fill\n * @return {Graphics}\n */\nGraphics.prototype.beginFill = function (color, alpha) {\n    this.filling = true;\n    this.fillColor = color || 0;\n    this.fillAlpha = (alpha === undefined) ? 1 : alpha;\n\n    if (this.currentPath) {\n        if (this.currentPath.shape.points.length <= 2) {\n            this.currentPath.fill = this.filling;\n            this.currentPath.fillColor = this.fillColor;\n            this.currentPath.fillAlpha = this.fillAlpha;\n        }\n    }\n    return this;\n};\n\n/**\n * Applies a fill to the lines and shapes that were added since the last call to the beginFill() method.\n *\n * @return {Graphics}\n */\nGraphics.prototype.endFill = function () {\n    this.filling = false;\n    this.fillColor = null;\n    this.fillAlpha = 1;\n\n    return this;\n};\n\n/**\n *\n * @param x {number} The X coord of the top-left of the rectangle\n * @param y {number} The Y coord of the top-left of the rectangle\n * @param width {number} The width of the rectangle\n * @param height {number} The height of the rectangle\n * @return {Graphics}\n */\nGraphics.prototype.drawRect = function ( x, y, width, height ) {\n    this.drawShape(new math.Rectangle(x,y, width, height));\n\n    return this;\n};\n\n/**\n *\n * @param x {number} The X coord of the top-left of the rectangle\n * @param y {number} The Y coord of the top-left of the rectangle\n * @param width {number} The width of the rectangle\n * @param height {number} The height of the rectangle\n * @param radius {number} Radius of the rectangle corners\n */\nGraphics.prototype.drawRoundedRect = function ( x, y, width, height, radius ) {\n    this.drawShape(new math.RoundedRectangle(x, y, width, height, radius));\n\n    return this;\n};\n\n/**\n * Draws a circle.\n *\n * @param x {number} The X coordinate of the center of the circle\n * @param y {number} The Y coordinate of the center of the circle\n * @param radius {number} The radius of the circle\n * @return {Graphics}\n */\nGraphics.prototype.drawCircle = function (x, y, radius) {\n    this.drawShape(new math.Circle(x,y, radius));\n\n    return this;\n};\n\n/**\n * Draws an ellipse.\n *\n * @param x {number} The X coordinate of the center of the ellipse\n * @param y {number} The Y coordinate of the center of the ellipse\n * @param width {number} The half width of the ellipse\n * @param height {number} The half height of the ellipse\n * @return {Graphics}\n */\nGraphics.prototype.drawEllipse = function (x, y, width, height) {\n    this.drawShape(new math.Ellipse(x, y, width, height));\n\n    return this;\n};\n\n/**\n * Draws a polygon using the given path.\n *\n * @param path {Array} The path data used to construct the polygon.\n * @return {Graphics}\n */\nGraphics.prototype.drawPolygon = function (path) {\n    if (!(path instanceof Array)) {\n        path = Array.prototype.slice.call(arguments);\n    }\n\n    this.drawShape(new math.Polygon(path));\n\n    return this;\n};\n\n/**\n * Clears the graphics that were drawn to this Graphics object, and resets fill and line style settings.\n *\n * @return {Graphics}\n */\nGraphics.prototype.clear = function () {\n    this.lineWidth = 0;\n    this.filling = false;\n\n    this.dirty = true;\n    this.clearDirty = true;\n    this.graphicsData = [];\n\n    return this;\n};\n\n/**\n * Useful function that returns a texture of the graphics object that can then be used to create sprites\n * This can be quite useful if your geometry is complicated and needs to be reused multiple times.\n *\n * @param resolution {number} The resolution of the texture being generated\n * @param scaleMode {number} Should be one of the scaleMode consts\n * @return {Texture} a texture of the graphics object\n */\nGraphics.prototype.generateTexture = function (resolution, scaleMode) {\n    resolution = resolution || 1;\n\n    var bounds = this.getBounds();\n\n    var canvasBuffer = new CanvasBuffer(bounds.width * resolution, bounds.height * resolution);\n\n    var texture = Texture.fromCanvas(canvasBuffer.canvas, scaleMode);\n    texture.baseTexture.resolution = resolution;\n\n    canvasBuffer.context.scale(resolution, resolution);\n\n    canvasBuffer.context.translate(-bounds.x,-bounds.y);\n\n    CanvasGraphics.renderGraphics(this, canvasBuffer.context);\n\n    return texture;\n};\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer}\n */\nGraphics.prototype.renderWebGL = function (renderer) {\n    // if the sprite is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0 || this.isMask === true) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        if (this.dirty || this.cachedSpriteDirty) {\n            this._generateCachedSprite();\n\n            // we will also need to update the texture on the gpu too!\n            this.updateCachedSpriteTexture();\n\n            this.cachedSpriteDirty = false;\n            this.dirty = false;\n        }\n\n        this._cachedSprite.worldAlpha = this.worldAlpha;\n\n        Sprite.prototype.renderWebGL.call(this._cachedSprite, renderer);\n\n        return;\n    }\n    else {\n        renderer.spriteBatch.stop();\n        renderer.blendModeManager.setBlendMode(this.blendMode);\n\n        if (this._mask) {\n            renderer.maskManager.pushMask(this._mask, renderer);\n        }\n\n        if (this._filters) {\n            renderer.filterManager.pushFilter(this._filterBlock);\n        }\n\n        // check blend mode\n        if (this.blendMode !== renderer.spriteBatch.currentBlendMode) {\n            renderer.spriteBatch.currentBlendMode = this.blendMode;\n\n            var blendModeWebGL = renderer.blendModes[renderer.spriteBatch.currentBlendMode];\n\n            renderer.spriteBatch.gl.blendFunc(blendModeWebGL[0], blendModeWebGL[1]);\n        }\n\n        // check if the webgl graphic needs to be updated\n        if (this.glDirty) {\n            this.dirty = true;\n            this.glDirty = false;\n        }\n\n        WebGLGraphics.renderGraphics(this, renderer);\n\n        // only render if it has children!\n        if (this.children.length) {\n            renderer.spriteBatch.start();\n\n             // simple render children!\n            for (var i = 0, j = this.children.length; i < j; ++i) {\n                this.children[i].renderWebGL(renderer);\n            }\n\n            renderer.spriteBatch.stop();\n        }\n\n        if (this._filters) {\n            renderer.filterManager.popFilter();\n        }\n\n        if (this._mask) {\n            renderer.maskManager.popMask(this.mask, renderer);\n        }\n\n        renderer.drawCount++;\n\n        renderer.spriteBatch.start();\n    }\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer}\n * @private\n */\nGraphics.prototype.renderCanvas = function (renderer) {\n    // if the sprite is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0 || this.isMask === true) {\n        return;\n    }\n\n    if (this._cacheAsBitmap) {\n        if (this.dirty || this.cachedSpriteDirty) {\n            this._generateCachedSprite();\n\n            // we will also need to update the texture\n            this.updateCachedSpriteTexture();\n\n            this.cachedSpriteDirty = false;\n            this.dirty = false;\n        }\n\n        this._cachedSprite.alpha = this.alpha;\n\n        Sprite.prototype.renderCanvas.call(this._cachedSprite, renderer);\n\n        return;\n    }\n    else {\n        var context = renderer.context;\n        var transform = this.worldTransform;\n\n        if (this.blendMode !== renderer.currentBlendMode) {\n            renderer.currentBlendMode = this.blendMode;\n            context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode];\n        }\n\n        if (this._mask) {\n            renderer.maskManager.pushMask(this._mask, renderer);\n        }\n\n        var resolution = renderer.resolution;\n        context.setTransform(\n            transform.a * resolution,\n            transform.b * resolution,\n            transform.c * resolution,\n            transform.d * resolution,\n            transform.tx * resolution,\n            transform.ty * resolution\n        );\n\n        CanvasGraphics.renderGraphics(this, context);\n\n        for (var i = 0, j = this.children.length; i < j; ++i) {\n            this.children[i].renderCanvas(renderer);\n        }\n\n        if (this._mask) {\n            renderer.maskManager.popMask(renderer);\n        }\n    }\n};\n\n/**\n * Retrieves the bounds of the graphic shape as a rectangle object\n *\n * @return {Rectangle} the rectangular bounding area\n */\nGraphics.prototype.getBounds = function (matrix) {\n    // return an empty object if the item is a mask!\n    if (this.isMask) {\n        return math.Rectangle.EMPTY;\n    }\n\n    if (this.dirty) {\n        this.updateLocalBounds();\n\n        this.glDirty = true;\n        this.cachedSpriteDirty = true;\n        this.dirty = false;\n    }\n\n    var bounds = this._localBounds;\n\n    var w0 = bounds.x;\n    var w1 = bounds.width + bounds.x;\n\n    var h0 = bounds.y;\n    var h1 = bounds.height + bounds.y;\n\n    var worldTransform = matrix || this.worldTransform;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var x1 = a * w1 + c * h1 + tx;\n    var y1 = d * h1 + b * w1 + ty;\n\n    var x2 = a * w0 + c * h1 + tx;\n    var y2 = d * h1 + b * w0 + ty;\n\n    var x3 = a * w0 + c * h0 + tx;\n    var y3 = d * h0 + b * w0 + ty;\n\n    var x4 =  a * w1 + c * h0 + tx;\n    var y4 =  d * h0 + b * w1 + ty;\n\n    var maxX = x1;\n    var maxY = y1;\n\n    var minX = x1;\n    var minY = y1;\n\n    minX = x2 < minX ? x2 : minX;\n    minX = x3 < minX ? x3 : minX;\n    minX = x4 < minX ? x4 : minX;\n\n    minY = y2 < minY ? y2 : minY;\n    minY = y3 < minY ? y3 : minY;\n    minY = y4 < minY ? y4 : minY;\n\n    maxX = x2 > maxX ? x2 : maxX;\n    maxX = x3 > maxX ? x3 : maxX;\n    maxX = x4 > maxX ? x4 : maxX;\n\n    maxY = y2 > maxY ? y2 : maxY;\n    maxY = y3 > maxY ? y3 : maxY;\n    maxY = y4 > maxY ? y4 : maxY;\n\n    this._bounds.x = minX;\n    this._bounds.width = maxX - minX;\n\n    this._bounds.y = minY;\n    this._bounds.height = maxY - minY;\n\n    return this._bounds;\n};\n\n/**\n * Update the bounds of the object\n *\n */\nGraphics.prototype.updateLocalBounds = function () {\n    var minX = Infinity;\n    var maxX = -Infinity;\n\n    var minY = Infinity;\n    var maxY = -Infinity;\n\n    if (this.graphicsData.length) {\n        var shape, points, x, y, w, h;\n\n        for (var i = 0; i < this.graphicsData.length; i++) {\n            var data = this.graphicsData[i];\n            var type = data.type;\n            var lineWidth = data.lineWidth;\n            shape = data.shape;\n\n            if (type === CONST.SHAPES.RECT || type === CONST.SHAPES.RREC) {\n                x = shape.x - lineWidth/2;\n                y = shape.y - lineWidth/2;\n                w = shape.width + lineWidth;\n                h = shape.height + lineWidth;\n\n                minX = x < minX ? x : minX;\n                maxX = x + w > maxX ? x + w : maxX;\n\n                minY = y < minY ? y : minY;\n                maxY = y + h > maxY ? y + h : maxY;\n            }\n            else if (type === CONST.SHAPES.CIRC) {\n                x = shape.x;\n                y = shape.y;\n                w = shape.radius + lineWidth/2;\n                h = shape.radius + lineWidth/2;\n\n                minX = x - w < minX ? x - w : minX;\n                maxX = x + w > maxX ? x + w : maxX;\n\n                minY = y - h < minY ? y - h : minY;\n                maxY = y + h > maxY ? y + h : maxY;\n            }\n            else if (type === CONST.SHAPES.ELIP) {\n                x = shape.x;\n                y = shape.y;\n                w = shape.width + lineWidth/2;\n                h = shape.height + lineWidth/2;\n\n                minX = x - w < minX ? x - w : minX;\n                maxX = x + w > maxX ? x + w : maxX;\n\n                minY = y - h < minY ? y - h : minY;\n                maxY = y + h > maxY ? y + h : maxY;\n            }\n            else {\n                // POLY\n                points = shape.points;\n\n                for (var j = 0; j < points.length; j += 2) {\n                    x = points[j];\n                    y = points[j+1];\n\n                    minX = x-lineWidth < minX ? x-lineWidth : minX;\n                    maxX = x+lineWidth > maxX ? x+lineWidth : maxX;\n\n                    minY = y-lineWidth < minY ? y-lineWidth : minY;\n                    maxY = y+lineWidth > maxY ? y+lineWidth : maxY;\n                }\n            }\n        }\n    }\n    else {\n        minX = 0;\n        maxX = 0;\n        minY = 0;\n        maxY = 0;\n    }\n\n    var padding = this.boundsPadding;\n\n    this._localBounds.x = minX - padding;\n    this._localBounds.width = (maxX - minX) + padding * 2;\n\n    this._localBounds.y = minY - padding;\n    this._localBounds.height = (maxY - minY) + padding * 2;\n};\n\n/**\n * Generates the cached sprite when the sprite has cacheAsBitmap = true\n *\n * @private\n */\nGraphics.prototype._generateCachedSprite = function () {\n    var bounds = this.getLocalBounds();\n\n    if (!this._cachedSprite) {\n        var canvasBuffer = new CanvasBuffer(bounds.width, bounds.height);\n        var texture = Texture.fromCanvas(canvasBuffer.canvas);\n\n        this._cachedSprite = new Sprite(texture);\n        this._cachedSprite.buffer = canvasBuffer;\n\n        this._cachedSprite.worldTransform = this.worldTransform;\n    }\n    else {\n        this._cachedSprite.buffer.resize(bounds.width, bounds.height);\n    }\n\n    // leverage the anchor to account for the offset of the element\n    this._cachedSprite.anchor.x = -( bounds.x / bounds.width );\n    this._cachedSprite.anchor.y = -( bounds.y / bounds.height );\n\n    // this._cachedSprite.buffer.context.save();\n    this._cachedSprite.buffer.context.translate(-bounds.x,-bounds.y);\n\n    // make sure we set the alpha of the graphics to 1 for the render..\n    this.worldAlpha = 1;\n\n    // now render the graphic..\n    CanvasGraphics.renderGraphics(this, this._cachedSprite.buffer.context);\n\n    this._cachedSprite.alpha = this.alpha;\n};\n\n/**\n * Updates texture size based on canvas size\n *\n * @private\n */\nGraphics.prototype.updateCachedSpriteTexture = function () {\n    var cachedSprite = this._cachedSprite;\n    var texture = cachedSprite.texture;\n    var canvas = cachedSprite.buffer.canvas;\n\n    texture.baseTexture.width = canvas.width;\n    texture.baseTexture.height = canvas.height;\n    texture.crop.width = texture.frame.width = canvas.width;\n    texture.crop.height = texture.frame.height = canvas.height;\n\n    cachedSprite._width = canvas.width;\n    cachedSprite._height = canvas.height;\n\n    // update the dirty base textures\n    texture.baseTexture.dirty();\n};\n\n/**\n * Destroys a previous cached sprite.\n *\n */\nGraphics.prototype.destroyCachedSprite = function () {\n    this._cachedSprite.texture.destroy(true);\n\n    // let the gc collect the unused sprite\n    // TODO could be object pooled!\n    this._cachedSprite = null;\n};\n\n/**\n * Draws the given shape to this Graphics object. Can be any of Circle, Rectangle, Ellipse, Line or Polygon.\n *\n * @param {Circle|Rectangle|Ellipse|Line|Polygon} shape The Shape object to draw.\n * @return {GraphicsData} The generated GraphicsData object.\n */\nGraphics.prototype.drawShape = function (shape) {\n    if (this.currentPath) {\n        // check current path!\n        if (this.currentPath.shape.points.length <= 2) {\n            this.graphicsData.pop();\n        }\n    }\n\n    this.currentPath = null;\n\n    var data = new GraphicsData(this.lineWidth, this.lineColor, this.lineAlpha, this.fillColor, this.fillAlpha, this.filling, shape);\n\n    this.graphicsData.push(data);\n\n    if (data.type === CONST.SHAPES.POLY) {\n        data.shape.closed = this.filling;\n        this.currentPath = data;\n    }\n\n    this.dirty = true;\n\n    return data;\n};\n","/**\n * A GraphicsData object.\n *\n * @class\n * @namespace PIXI\n */\nfunction GraphicsData(lineWidth, lineColor, lineAlpha, fillColor, fillAlpha, fill, shape) {\n    this.lineWidth = lineWidth;\n    this.lineColor = lineColor;\n    this.lineAlpha = lineAlpha;\n    this._lineTint = lineColor;\n\n    this.fillColor = fillColor;\n    this.fillAlpha = fillAlpha;\n    this._fillTint = fillColor;\n    this.fill = fill;\n\n    this.shape = shape;\n    this.type = shape.type;\n}\n\nGraphicsData.prototype.constructor = GraphicsData;\nmodule.exports = GraphicsData;\n\n/**\n * Creates a new GraphicsData object with the same values as this one.\n *\n * @return {GraphicsData}\n */\nGraphicsData.prototype.clone = function () {\n    return new GraphicsData(\n        this.lineWidth,\n        this.lineColor,\n        this.lineAlpha,\n        this.fillColor,\n        this.fillAlpha,\n        this.fill,\n        this.shape\n    );\n};\n","var CanvasMaskManager = require('./utils/CanvasMaskManager'),\n    utils = require('../../utils'),\n    CONST = require('../../const');\n\n/**\n * The CanvasRenderer draws the scene and all its content onto a 2d canvas. This renderer should be used for browsers that do not support webGL.\n * Don't forget to add the CanvasRenderer.view to your DOM or you will not see anything :)\n *\n * @class\n * @namespace PIXI\n * @param [width=800] {number} the width of the canvas view\n * @param [height=600] {number} the height of the canvas view\n * @param [options] {object} The optional renderer parameters\n * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false\n * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n * @param [options.clearBeforeRender=true] {boolean} This sets if the CanvasRenderer will clear the canvas or not before the new render pass.\n */\nfunction CanvasRenderer(width, height, options) {\n    utils.sayHello('Canvas');\n\n    if (options) {\n        for (var i in CONST.defaultRenderOptions) {\n            if (typeof options[i] === 'undefined') {\n                options[i] = CONST.defaultRenderOptions[i];\n            }\n        }\n    }\n    else {\n        options = CONST.defaultRenderOptions;\n    }\n\n    /**\n     * The renderer type.\n     *\n     * @member {number}\n     */\n    this.type = CONST.CANVAS_RENDERER;\n\n    /**\n     * The resolution of the canvas.\n     *\n     * @member {number}\n     */\n    this.resolution = options.resolution;\n\n    /**\n     * This sets if the CanvasRenderer will clear the canvas or not before the new render pass.\n     * If the scene is NOT transparent Pixi will use a canvas sized fillRect operation every frame to set the canvas background color.\n     * If the scene is transparent Pixi will use clearRect to clear the canvas every frame.\n     * Disable this by setting this to false. For example if your game has a canvas filling background image you often don't need this set.\n     *\n     * @member {boolean}\n     * @default\n     */\n    this.clearBeforeRender = options.clearBeforeRender;\n\n    /**\n     * The background color as a number.\n     *\n     * @member {number}\n     * @private\n     */\n    this._backgroundColor = 0x000000;\n\n    /**\n     * The background color as a string.\n     *\n     * @member {string}\n     * @private\n     */\n    this._backgroundColorString = '#000000';\n\n    this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter\n\n    /**\n     * Whether the render view is transparent\n     *\n     * @member {boolean}\n     */\n    this.transparent = options.transparent;\n\n    /**\n     * Whether the render view should be resized automatically\n     *\n     * @member {boolean}\n     */\n    this.autoResize = options.autoResize || false;\n\n\n    /**\n     * The width of the canvas view\n     *\n     * @member {number}\n     * @default 800\n     */\n    this.width = width || 800;\n\n    /**\n     * The height of the canvas view\n     *\n     * @member {number}\n     * @default 600\n     */\n    this.height = height || 600;\n\n    this.width *= this.resolution;\n    this.height *= this.resolution;\n\n    /**\n     * The canvas element that everything is drawn to.\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.view = options.view || document.createElement('canvas');\n\n    /**\n     * The canvas 2d context that everything is drawn with\n     * @member {CanvasRenderingContext2D}\n     */\n    this.context = this.view.getContext('2d', { alpha: this.transparent });\n\n    /**\n     * Boolean flag controlling canvas refresh.\n     *\n     * @member {boolean}\n     */\n    this.refresh = true;\n\n    this.view.width = this.width * this.resolution;\n    this.view.height = this.height * this.resolution;\n\n    /**\n     * Internal var.\n     *\n     * @member {number}\n     */\n    this.count = 0;\n\n    /**\n     * Instance of a CanvasMaskManager, handles masking when using the canvas renderer\n     * @member {CanvasMaskManager}\n     */\n    this.maskManager = new CanvasMaskManager();\n\n    /**\n     * If true Pixi will Math.floor() x/y values when rendering, stopping pixel interpolation.\n     * Handy for crisp pixel art and speed on legacy devices.\n     *\n     * @member {boolean}\n     */\n    this.roundPixels = false;\n\n    this.scaleMode = null;\n\n    this.smoothProperty = null;\n\n    if (this.context.imageSmoothingEnabled) {\n        this.smoothProperty = 'imageSmoothingEnabled';\n    }\n    else if (this.context.webkitImageSmoothingEnabled) {\n        this.smoothProperty = 'webkitImageSmoothingEnabled';\n    }\n    else if (this.context.mozImageSmoothingEnabled) {\n        this.smoothProperty = 'mozImageSmoothingEnabled';\n    }\n    else if (this.context.oImageSmoothingEnabled) {\n        this.smoothProperty = 'oImageSmoothingEnabled';\n    }\n    else if (this.context.msImageSmoothingEnabled) {\n        this.smoothProperty = 'msImageSmoothingEnabled';\n    }\n\n    this.currentBlendMode = CONST.blendModes.NORMAL;\n\n    this.blendModes = null;\n\n    this._mapBlendModes();\n\n    this.resize(width, height);\n}\n\n// constructor\nCanvasRenderer.prototype.constructor = CanvasRenderer;\nmodule.exports = CanvasRenderer;\n\nObject.defineProperties(CanvasRenderer.prototype, {\n    /**\n     * The background color to fill if not transparent\n     *\n     * @member {number}\n     * @memberof CanvasRenderer#\n     */\n    backgroundColor: {\n        get: function () {\n            return this._backgroundColor;\n        },\n        set: function (val) {\n            this._backgroundColor = val;\n            this._backgroundColorString = utils.hex2string(val);\n        }\n    }\n});\n\n/**\n * Renders the object to this canvas view\n *\n * @param object {DisplayObject} the object to be rendered\n */\nCanvasRenderer.prototype.render = function (object) {\n    object.updateTransform();\n\n    this.context.setTransform(1,0,0,1,0,0);\n\n    this.context.globalAlpha = 1;\n\n    this.currentBlendMode = CONST.blendModes.NORMAL;\n    this.context.globalCompositeOperation = this.blendModes[CONST.blendModes.NORMAL];\n\n    if (navigator.isCocoonJS && this.view.screencanvas) {\n        this.context.fillStyle = 'black';\n        this.context.clear();\n    }\n\n    if (this.clearBeforeRender) {\n        if (this.transparent) {\n            this.context.clearRect(0, 0, this.width, this.height);\n        }\n        else {\n            this.context.fillStyle = this._backgroundColorString;\n            this.context.fillRect(0, 0, this.width , this.height);\n        }\n    }\n\n    this.renderDisplayObject(object);\n};\n\n/**\n * Removes everything from the renderer and optionally removes the Canvas DOM element.\n *\n * @param [removeView=false] {boolean} Removes the Canvas element from the DOM.\n */\nCanvasRenderer.prototype.destroy = function (removeView) {\n    if (removeView && this.view.parent) {\n        this.view.parent.removeChild(this.view);\n    }\n\n    this.view = null;\n    this.context = null;\n    this.maskManager = null;\n};\n\n/**\n * Resizes the canvas view to the specified width and height\n *\n * @param width {number} the new width of the canvas view\n * @param height {number} the new height of the canvas view\n */\nCanvasRenderer.prototype.resize = function (width, height) {\n    this.width = width * this.resolution;\n    this.height = height * this.resolution;\n\n    this.view.width = this.width;\n    this.view.height = this.height;\n\n    if (this.autoResize) {\n        this.view.style.width = this.width / this.resolution + 'px';\n        this.view.style.height = this.height / this.resolution + 'px';\n    }\n};\n\n/**\n * Renders a display object\n *\n * @param displayObject {DisplayObject} The displayObject to render\n * @private\n */\nCanvasRenderer.prototype.renderDisplayObject = function (displayObject) {\n    displayObject.renderCanvas(this);\n};\n\n/**\n * Maps Pixi blend modes to canvas blend modes.\n *\n * @private\n */\nCanvasRenderer.prototype._mapBlendModes = function () {\n    if (!this.blendModes) {\n        this.blendModes = {};\n\n        if (utils.canUseNewCanvasBlendModes()) {\n            this.blendModes[CONST.blendModes.NORMAL]        = 'source-over';\n            this.blendModes[CONST.blendModes.ADD]           = 'lighter'; //IS THIS OK???\n            this.blendModes[CONST.blendModes.MULTIPLY]      = 'multiply';\n            this.blendModes[CONST.blendModes.SCREEN]        = 'screen';\n            this.blendModes[CONST.blendModes.OVERLAY]       = 'overlay';\n            this.blendModes[CONST.blendModes.DARKEN]        = 'darken';\n            this.blendModes[CONST.blendModes.LIGHTEN]       = 'lighten';\n            this.blendModes[CONST.blendModes.COLOR_DODGE]   = 'color-dodge';\n            this.blendModes[CONST.blendModes.COLOR_BURN]    = 'color-burn';\n            this.blendModes[CONST.blendModes.HARD_LIGHT]    = 'hard-light';\n            this.blendModes[CONST.blendModes.SOFT_LIGHT]    = 'soft-light';\n            this.blendModes[CONST.blendModes.DIFFERENCE]    = 'difference';\n            this.blendModes[CONST.blendModes.EXCLUSION]     = 'exclusion';\n            this.blendModes[CONST.blendModes.HUE]           = 'hue';\n            this.blendModes[CONST.blendModes.SATURATION]    = 'saturation';\n            this.blendModes[CONST.blendModes.COLOR]         = 'color';\n            this.blendModes[CONST.blendModes.LUMINOSITY]    = 'luminosity';\n        }\n        else {\n            // this means that the browser does not support the cool new blend modes in canvas 'cough' ie 'cough'\n            this.blendModes[CONST.blendModes.NORMAL]        = 'source-over';\n            this.blendModes[CONST.blendModes.ADD]           = 'lighter'; //IS THIS OK???\n            this.blendModes[CONST.blendModes.MULTIPLY]      = 'source-over';\n            this.blendModes[CONST.blendModes.SCREEN]        = 'source-over';\n            this.blendModes[CONST.blendModes.OVERLAY]       = 'source-over';\n            this.blendModes[CONST.blendModes.DARKEN]        = 'source-over';\n            this.blendModes[CONST.blendModes.LIGHTEN]       = 'source-over';\n            this.blendModes[CONST.blendModes.COLOR_DODGE]   = 'source-over';\n            this.blendModes[CONST.blendModes.COLOR_BURN]    = 'source-over';\n            this.blendModes[CONST.blendModes.HARD_LIGHT]    = 'source-over';\n            this.blendModes[CONST.blendModes.SOFT_LIGHT]    = 'source-over';\n            this.blendModes[CONST.blendModes.DIFFERENCE]    = 'source-over';\n            this.blendModes[CONST.blendModes.EXCLUSION]     = 'source-over';\n            this.blendModes[CONST.blendModes.HUE]           = 'source-over';\n            this.blendModes[CONST.blendModes.SATURATION]    = 'source-over';\n            this.blendModes[CONST.blendModes.COLOR]         = 'source-over';\n            this.blendModes[CONST.blendModes.LUMINOSITY]    = 'source-over';\n        }\n    }\n};\n","/**\n * Creates a Canvas element of the given size.\n *\n * @class\n * @namespace PIXI\n * @param width {number} the width for the newly created canvas\n * @param height {number} the height for the newly created canvas\n */\nfunction CanvasBuffer(width, height) {\n    /**\n     * The Canvas object that belongs to this CanvasBuffer.\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.canvas = document.createElement('canvas');\n\n    /**\n     * A CanvasRenderingContext2D object representing a two-dimensional rendering context.\n     *\n     * @member {CanvasRenderingContext2D}\n     */\n    this.context = this.canvas.getContext('2d');\n\n    this.canvas.width = width;\n    this.canvas.height = height;\n}\n\nCanvasBuffer.prototype.constructor = CanvasBuffer;\nmodule.exports = CanvasBuffer;\n\nObject.defineProperties(CanvasBuffer.prototype, {\n    /**\n     * The width of the canvas buffer in pixels.\n     *\n     * @member {number}\n     * @memberof CanvasBuffer#\n     */\n    width: {\n        get: function () {\n            return this.canvas.width;\n        },\n        set: function (val) {\n            this.canvas.width = val;\n        }\n    },\n    /**\n     * The height of the canvas buffer in pixels.\n     *\n     * @member {number}\n     * @memberof CanvasBuffer#\n     */\n    height: {\n        get: function () {\n            return this.canvas.height;\n        },\n        set: function (val) {\n            this.canvas.height = val;\n        }\n    }\n});\n\n/**\n * Clears the canvas that was created by the CanvasBuffer class.\n *\n * @private\n */\nCanvasBuffer.prototype.clear = function () {\n    this.context.setTransform(1, 0, 0, 1, 0, 0);\n    this.context.clearRect(0,0, this.canvas.width, this.canvas.height);\n};\n\n/**\n * Resizes the canvas to the specified width and height.\n *\n * @param width {number} the new width of the canvas\n * @param height {number} the new height of the canvas\n */\nCanvasBuffer.prototype.resize = function (width, height) {\n    this.canvas.width = width;\n    this.canvas.height = height;\n};\n","var CONST = require('../../../const');\n\n/**\n * A set of functions used by the canvas renderer to draw the primitive graphics data.\n *\n * @namespace PIXI\n */\nvar CanvasGraphics = module.exports = {};\n\n/*\n * Renders a Graphics object to a canvas.\n *\n * @param graphics {Graphics} the actual graphics object to render\n * @param context {CanvasRenderingContext2D} the 2d drawing method of the canvas\n */\nCanvasGraphics.renderGraphics = function (graphics, context) {\n    var worldAlpha = graphics.worldAlpha;\n\n    if (graphics.dirty) {\n        this.updateGraphicsTint(graphics);\n        graphics.dirty = false;\n    }\n\n    for (var i = 0; i < graphics.graphicsData.length; i++) {\n        var data = graphics.graphicsData[i];\n        var shape = data.shape;\n\n        var fillColor = data._fillTint;\n        var lineColor = data._lineTint;\n\n        context.lineWidth = data.lineWidth;\n\n        if (data.type === CONST.SHAPES.POLY) {\n            context.beginPath();\n\n            var points = shape.points;\n\n            context.moveTo(points[0], points[1]);\n\n            for (var j=1; j < points.length/2; j++) {\n                context.lineTo(points[j * 2], points[j * 2 + 1]);\n            }\n\n            if (shape.closed) {\n                context.lineTo(points[0], points[1]);\n            }\n\n            // if the first and last point are the same close the path - much neater :)\n            if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) {\n                context.closePath();\n            }\n\n            if (data.fill) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n        else if (data.type === CONST.SHAPES.RECT) {\n\n            if (data.fillColor || data.fillColor === 0) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fillRect(shape.x, shape.y, shape.width, shape.height);\n\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.strokeRect(shape.x, shape.y, shape.width, shape.height);\n            }\n        }\n        else if (data.type === CONST.SHAPES.CIRC) {\n            // TODO - need to be Undefined!\n            context.beginPath();\n            context.arc(shape.x, shape.y, shape.radius,0,2*Math.PI);\n            context.closePath();\n\n            if (data.fill) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n        else if (data.type === CONST.SHAPES.ELIP) {\n            // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas\n\n            var w = shape.width * 2;\n            var h = shape.height * 2;\n\n            var x = shape.x - w/2;\n            var y = shape.y - h/2;\n\n            context.beginPath();\n\n            var kappa = 0.5522848,\n                ox = (w / 2) * kappa, // control point offset horizontal\n                oy = (h / 2) * kappa, // control point offset vertical\n                xe = x + w,           // x-end\n                ye = y + h,           // y-end\n                xm = x + w / 2,       // x-middle\n                ym = y + h / 2;       // y-middle\n\n            context.moveTo(x, ym);\n            context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y);\n            context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym);\n            context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye);\n            context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym);\n\n            context.closePath();\n\n            if (data.fill) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n        else if (data.type === CONST.SHAPES.RREC) {\n            var rx = shape.x;\n            var ry = shape.y;\n            var width = shape.width;\n            var height = shape.height;\n            var radius = shape.radius;\n\n            var maxRadius = Math.min(width, height) / 2 | 0;\n            radius = radius > maxRadius ? maxRadius : radius;\n\n            context.beginPath();\n            context.moveTo(rx, ry + radius);\n            context.lineTo(rx, ry + height - radius);\n            context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height);\n            context.lineTo(rx + width - radius, ry + height);\n            context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius);\n            context.lineTo(rx + width, ry + radius);\n            context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry);\n            context.lineTo(rx + radius, ry);\n            context.quadraticCurveTo(rx, ry, rx, ry + radius);\n            context.closePath();\n\n            if (data.fillColor || data.fillColor === 0) {\n                context.globalAlpha = data.fillAlpha * worldAlpha;\n                context.fillStyle = '#' + ('00000' + ( fillColor | 0).toString(16)).substr(-6);\n                context.fill();\n\n            }\n            if (data.lineWidth) {\n                context.globalAlpha = data.lineAlpha * worldAlpha;\n                context.strokeStyle = '#' + ('00000' + ( lineColor | 0).toString(16)).substr(-6);\n                context.stroke();\n            }\n        }\n    }\n};\n\n/*\n * Renders a graphics mask\n *\n * @private\n * @param graphics {Graphics} the graphics which will be used as a mask\n * @param context {CanvasRenderingContext2D} the context 2d method of the canvas\n */\nCanvasGraphics.renderGraphicsMask = function (graphics, context) {\n    var len = graphics.graphicsData.length;\n\n    if (len === 0) {\n        return;\n    }\n\n    context.beginPath();\n\n    for (var i = 0; i < len; i++) {\n        var data = graphics.graphicsData[i];\n        var shape = data.shape;\n\n        if (data.type === CONST.SHAPES.POLY) {\n\n            var points = shape.points;\n\n            context.moveTo(points[0], points[1]);\n\n            for (var j=1; j < points.length/2; j++) {\n                context.lineTo(points[j * 2], points[j * 2 + 1]);\n            }\n\n            // if the first and last point are the same close the path - much neater :)\n            if (points[0] === points[points.length-2] && points[1] === points[points.length-1]) {\n                context.closePath();\n            }\n\n        }\n        else if (data.type === CONST.SHAPES.RECT) {\n            context.rect(shape.x, shape.y, shape.width, shape.height);\n            context.closePath();\n        }\n        else if (data.type === CONST.SHAPES.CIRC) {\n            // TODO - need to be Undefined!\n            context.arc(shape.x, shape.y, shape.radius, 0, 2 * Math.PI);\n            context.closePath();\n        }\n        else if (data.type === CONST.SHAPES.ELIP) {\n\n            // ellipse code taken from: http://stackoverflow.com/questions/2172798/how-to-draw-an-oval-in-html5-canvas\n\n            var w = shape.width * 2;\n            var h = shape.height * 2;\n\n            var x = shape.x - w/2;\n            var y = shape.y - h/2;\n\n            var kappa = 0.5522848,\n                ox = (w / 2) * kappa, // control point offset horizontal\n                oy = (h / 2) * kappa, // control point offset vertical\n                xe = x + w,           // x-end\n                ye = y + h,           // y-end\n                xm = x + w / 2,       // x-middle\n                ym = y + h / 2;       // y-middle\n\n            context.moveTo(x, ym);\n            context.bezierCurveTo(x, ym - oy, xm - ox, y, xm, y);\n            context.bezierCurveTo(xm + ox, y, xe, ym - oy, xe, ym);\n            context.bezierCurveTo(xe, ym + oy, xm + ox, ye, xm, ye);\n            context.bezierCurveTo(xm - ox, ye, x, ym + oy, x, ym);\n            context.closePath();\n        }\n        else if (data.type === CONST.SHAPES.RREC) {\n\n            var rx = shape.x;\n            var ry = shape.y;\n            var width = shape.width;\n            var height = shape.height;\n            var radius = shape.radius;\n\n            var maxRadius = Math.min(width, height) / 2 | 0;\n            radius = radius > maxRadius ? maxRadius : radius;\n\n            context.moveTo(rx, ry + radius);\n            context.lineTo(rx, ry + height - radius);\n            context.quadraticCurveTo(rx, ry + height, rx + radius, ry + height);\n            context.lineTo(rx + width - radius, ry + height);\n            context.quadraticCurveTo(rx + width, ry + height, rx + width, ry + height - radius);\n            context.lineTo(rx + width, ry + radius);\n            context.quadraticCurveTo(rx + width, ry, rx + width - radius, ry);\n            context.lineTo(rx + radius, ry);\n            context.quadraticCurveTo(rx, ry, rx, ry + radius);\n            context.closePath();\n        }\n    }\n};\n\nCanvasGraphics.updateGraphicsTint = function (graphics) {\n    if (graphics.tint === 0xFFFFFF) {\n        return;\n    }\n\n    var tintR = (graphics.tint >> 16 & 0xFF) / 255;\n    var tintG = (graphics.tint >> 8 & 0xFF) / 255;\n    var tintB = (graphics.tint & 0xFF)/ 255;\n\n    for (var i = 0; i < graphics.graphicsData.length; i++) {\n        var data = graphics.graphicsData[i];\n\n        var fillColor = data.fillColor | 0;\n        var lineColor = data.lineColor | 0;\n\n        /*\n        var colorR = (fillColor >> 16 & 0xFF) / 255;\n        var colorG = (fillColor >> 8 & 0xFF) / 255;\n        var colorB = (fillColor & 0xFF) / 255;\n\n        colorR *= tintR;\n        colorG *= tintG;\n        colorB *= tintB;\n\n        fillColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255);\n\n        colorR = (lineColor >> 16 & 0xFF) / 255;\n        colorG = (lineColor >> 8 & 0xFF) / 255;\n        colorB = (lineColor & 0xFF) / 255;\n\n        colorR *= tintR;\n        colorG *= tintG;\n        colorB *= tintB;\n\n        lineColor = ((colorR*255 << 16) + (colorG*255 << 8) + colorB*255);\n        */\n\n        // super inline cos im an optimization NAZI :)\n        data._fillTint = (((fillColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((fillColor >> 8 & 0xFF) / 255 * tintG*255 << 8) +  (fillColor & 0xFF) / 255 * tintB*255);\n        data._lineTint = (((lineColor >> 16 & 0xFF) / 255 * tintR*255 << 16) + ((lineColor >> 8 & 0xFF) / 255 * tintG*255 << 8) +  (lineColor & 0xFF) / 255 * tintB*255);\n\n    }\n};\n\n","var CanvasGraphics = require('./CanvasGraphics');\n\n/**\n * A set of functions used to handle masking.\n *\n * @class\n * @namespace PIXI\n */\nfunction CanvasMaskManager() {}\n\nCanvasMaskManager.prototype.constructor = CanvasMaskManager;\nmodule.exports = CanvasMaskManager;\n\n/**\n * This method adds it to the current stack of masks.\n *\n * @param maskData {object} the maskData that will be pushed\n * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use.\n */\nCanvasMaskManager.prototype.pushMask = function (maskData, renderer) {\n    renderer.context.save();\n\n    var cacheAlpha = maskData.alpha;\n    var transform = maskData.worldTransform;\n    var resolution = renderer.resolution;\n\n    renderer.context.setTransform(\n        transform.a * resolution,\n        transform.b * resolution,\n        transform.c * resolution,\n        transform.d * resolution,\n        transform.tx * resolution,\n        transform.ty * resolution\n    );\n\n    CanvasGraphics.renderGraphicsMask(maskData, renderer.context);\n\n    renderer.context.clip();\n\n    maskData.worldAlpha = cacheAlpha;\n};\n\n/**\n * Restores the current drawing context to the state it was before the mask was applied.\n *\n * @param renderer {WebGLRenderer|CanvasRenderer} The renderer context to use.\n */\nCanvasMaskManager.prototype.popMask = function (renderer) {\n    renderer.context.restore();\n};\n","var utils = require('../../../utils');\n\n/**\n * Utility methods for Sprite/Texture tinting.\n *\n * @namespace PIXI\n */\nvar CanvasTinter = module.exports = {};\n\n/**\n * Basically this method just needs a sprite and a color and tints the sprite with the given color.\n *\n * @param sprite {Sprite} the sprite to tint\n * @param color {number} the color to use to tint the sprite with\n * @return {HTMLCanvasElement} The tinted canvas\n */\nCanvasTinter.getTintedTexture = function (sprite, color) {\n    var texture = sprite.texture;\n\n    color = CanvasTinter.roundColor(color);\n\n    var stringColor = '#' + ('00000' + ( color | 0).toString(16)).substr(-6);\n\n    texture.tintCache = texture.tintCache || {};\n\n    if (texture.tintCache[stringColor]) {\n        return texture.tintCache[stringColor];\n    }\n\n     // clone texture..\n    var canvas = CanvasTinter.canvas || document.createElement('canvas');\n\n    //CanvasTinter.tintWithPerPixel(texture, stringColor, canvas);\n    CanvasTinter.tintMethod(texture, color, canvas);\n\n    if (CanvasTinter.convertTintToImage) {\n        // is this better?\n        var tintImage = new Image();\n        tintImage.src = canvas.toDataURL();\n\n        texture.tintCache[stringColor] = tintImage;\n    }\n    else {\n        texture.tintCache[stringColor] = canvas;\n        // if we are not converting the texture to an image then we need to lose the reference to the canvas\n        CanvasTinter.canvas = null;\n    }\n\n    return canvas;\n};\n\n/**\n * Tint a texture using the 'multiply' operation.\n *\n * @param texture {Texture} the texture to tint\n * @param color {number} the color to use to tint the sprite with\n * @param canvas {HTMLCanvasElement} the current canvas\n */\nCanvasTinter.tintWithMultiply = function (texture, color, canvas) {\n    var context = canvas.getContext( '2d' );\n\n    var crop = texture.crop;\n\n    canvas.width = crop.width;\n    canvas.height = crop.height;\n\n    context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6);\n\n    context.fillRect(0, 0, crop.width, crop.height);\n\n    context.globalCompositeOperation = 'multiply';\n\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n\n    context.globalCompositeOperation = 'destination-atop';\n\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n};\n\n/**\n * Tint a texture using the 'overlay' operation.\n *\n * @param texture {Texture} the texture to tint\n * @param color {number} the color to use to tint the sprite with\n * @param canvas {HTMLCanvasElement} the current canvas\n */\nCanvasTinter.tintWithOverlay = function (texture, color, canvas) {\n    var context = canvas.getContext( '2d' );\n\n    var crop = texture.crop;\n\n    canvas.width = crop.width;\n    canvas.height = crop.height;\n\n    context.globalCompositeOperation = 'copy';\n    context.fillStyle = '#' + ('00000' + ( color | 0).toString(16)).substr(-6);\n    context.fillRect(0, 0, crop.width, crop.height);\n\n    context.globalCompositeOperation = 'destination-atop';\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n\n    // context.globalCompositeOperation = 'copy';\n};\n\n/**\n * Tint a texture pixel per pixel.\n *\n * @param texture {Texture} the texture to tint\n * @param color {number} the color to use to tint the sprite with\n * @param canvas {HTMLCanvasElement} the current canvas\n */\nCanvasTinter.tintWithPerPixel = function (texture, color, canvas) {\n    var context = canvas.getContext( '2d' );\n\n    var crop = texture.crop;\n\n    canvas.width = crop.width;\n    canvas.height = crop.height;\n\n    context.globalCompositeOperation = 'copy';\n    context.drawImage(\n        texture.baseTexture.source,\n        crop.x,\n        crop.y,\n        crop.width,\n        crop.height,\n        0,\n        0,\n        crop.width,\n        crop.height\n    );\n\n    var rgbValues = utils.hex2rgb(color);\n    var r = rgbValues[0], g = rgbValues[1], b = rgbValues[2];\n\n    var pixelData = context.getImageData(0, 0, crop.width, crop.height);\n\n    var pixels = pixelData.data;\n\n    for (var i = 0; i < pixels.length; i += 4) {\n        pixels[i+0] *= r;\n        pixels[i+1] *= g;\n        pixels[i+2] *= b;\n    }\n\n    context.putImageData(pixelData, 0, 0);\n};\n\n/**\n * Rounds the specified color according to the CanvasTinter.cacheStepsPerColorChannel.\n *\n * @param color {number} the color to round, should be a hex color\n */\nCanvasTinter.roundColor = function (color) {\n    var step = CanvasTinter.cacheStepsPerColorChannel;\n\n    var rgbValues = utils.hex2rgb(color);\n\n    rgbValues[0] = Math.min(255, (rgbValues[0] / step) * step);\n    rgbValues[1] = Math.min(255, (rgbValues[1] / step) * step);\n    rgbValues[2] = Math.min(255, (rgbValues[2] / step) * step);\n\n    return utils.rgb2hex(rgbValues);\n};\n\n/**\n * Number of steps which will be used as a cap when rounding colors.\n *\n * @member\n */\nCanvasTinter.cacheStepsPerColorChannel = 8;\n\n/**\n * Tint cache boolean flag.\n *\n * @member\n */\nCanvasTinter.convertTintToImage = false;\n\n/**\n * Whether or not the Canvas BlendModes are supported, consequently the ability to tint using the multiply method.\n *\n * @member\n */\nCanvasTinter.canUseMultiply = utils.canUseNewCanvasBlendModes();\n\n/**\n * The tinting method that will be used.\n *\n */\nCanvasTinter.tintMethod = CanvasTinter.canUseMultiply ? CanvasTinter.tintWithMultiply :  CanvasTinter.tintWithPerPixel;\n","var WebGLSpriteBatch = require('./utils/WebGLSpriteBatch'),\n    WebGLFastSpriteBatch = require('./utils/WebGLFastSpriteBatch'),\n    WebGLShaderManager = require('./managers/WebGLShaderManager'),\n    WebGLMaskManager = require('./managers/WebGLMaskManager'),\n    WebGLFilterManager = require('./managers/WebGLFilterManager'),\n    WebGLStencilManager = require('./managers/WebGLStencilManager'),\n    WebGLBlendModeManager = require('./managers/WebGLBlendModeManager'),\n    math = require('../../math'),\n    utils = require('../../utils'),\n    CONST = require('../../const');\n\n/**\n * The WebGLRenderer draws the scene and all its content onto a webGL enabled canvas. This renderer\n * should be used for browsers that support webGL. This Render works by automatically managing webGLBatchs.\n * So no need for Sprite Batches or Sprite Clouds.\n * Don't forget to add the view to your DOM or you will not see anything :)\n *\n * @class\n * @namespace PIXI\n * @param [width=0] {number} the width of the canvas view\n * @param [height=0] {number} the height of the canvas view\n * @param [options] {object} The optional renderer parameters\n * @param [options.view] {HTMLCanvasElement} the canvas to use as a view, optional\n * @param [options.transparent=false] {boolean} If the render view is transparent, default false\n * @param [options.autoResize=false] {boolean} If the render view is automatically resized, default false\n * @param [options.antialias=false] {boolean} sets antialias (only applicable in chrome at the moment)\n * @param [options.preserveDrawingBuffer=false] {boolean} enables drawing buffer preservation, enable this if you need to call toDataUrl on the webgl context\n * @param [options.resolution=1] {number} the resolution of the renderer retina would be 2\n */\nfunction WebGLRenderer(width, height, options) {\n    utils.sayHello('webGL');\n\n    if (options) {\n        for (var i in CONST.defaultRenderOptions) {\n            if (typeof options[i] === 'undefined') {\n                options[i] = CONST.defaultRenderOptions[i];\n            }\n        }\n    }\n    else {\n        options = CONST.defaultRenderOptions;\n    }\n\n    this.uuid = utils.uuid();\n\n    /**\n     * @member {number}\n     */\n    this.type = CONST.WEBGL_RENDERER;\n\n    /**\n     * The resolution of the renderer\n     *\n     * @member {number}\n     * @default 1\n     */\n    this.resolution = options.resolution;\n\n    // do a catch.. only 1 webGL renderer..\n\n    /**\n     * Whether the render view is transparent\n     *\n     * @member {boolean}\n     */\n    this.transparent = options.transparent;\n\n    /**\n     * The background color as a number.\n     *\n     * @member {number}\n     * @private\n     */\n    this._backgroundColor = 0x000000;\n\n    /**\n     * The background color as an [R, G, B] array.\n     *\n     * @member {number[]}\n     * @private\n     */\n    this._backgroundColorRgb = [0, 0, 0];\n\n    this.backgroundColor = options.backgroundColor || this._backgroundColor; // run bg color setter\n\n    /**\n     * Whether the render view should be resized automatically\n     *\n     * @member {boolean}\n     */\n    this.autoResize = options.autoResize || false;\n\n    /**\n     * The value of the preserveDrawingBuffer flag affects whether or not the contents of the stencil buffer is retained after rendering.\n     *\n     * @member {boolean}\n     */\n    this.preserveDrawingBuffer = options.preserveDrawingBuffer;\n\n    /**\n     * This sets if the WebGLRenderer will clear the context texture or not before the new render pass. If true:\n     * If the renderer is NOT transparent, Pixi will clear to alpha (0, 0, 0, 0).\n     * If the renderer is transparent, Pixi will clear to the target Stage's background color.\n     * Disable this by setting this to false. For example: if your game has a canvas filling background image, you often don't need this set.\n     *\n     * @member {boolean}\n     * @default\n     */\n    this.clearBeforeRender = options.clearBeforeRender;\n\n    /**\n     * The width of the canvas view\n     *\n     * @member {number}\n     * @default 800\n     */\n    this.width = width || 800;\n\n    /**\n     * The height of the canvas view\n     *\n     * @member {number}\n     * @default 600\n     */\n    this.height = height || 600;\n\n    /**\n     * The canvas element that everything is drawn to\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.view = options.view || document.createElement( 'canvas' );\n\n    // deal with losing context..\n\n    /**\n     * @member {Function}\n     */\n    this.contextLostBound = this.handleContextLost.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.contextRestoredBound = this.handleContextRestored.bind(this);\n\n    this.view.addEventListener('webglcontextlost', this.contextLostBound, false);\n    this.view.addEventListener('webglcontextrestored', this.contextRestoredBound, false);\n\n    /**\n     * @member {object}\n     * @private\n     */\n    this._contextOptions = {\n        alpha: this.transparent,\n        antialias: options.antialias, // SPEED UP??\n        premultipliedAlpha:this.transparent && this.transparent !== 'notMultiplied',\n        stencil:true,\n        preserveDrawingBuffer: options.preserveDrawingBuffer\n    };\n\n    /**\n     * @member {Point}\n     */\n    this.projection = new math.Point();\n\n    /**\n     * @member {Point}\n     */\n    this.offset = new math.Point(0, 0);\n\n    /**\n     * Counter for the number of draws made each frame\n     *\n     * @member {number}\n     */\n    this.drawCount = 0;\n\n    // time to create the render managers! each one focuses on managing a state in webGL\n\n    /**\n     * Deals with managing the shader programs and their attribs\n     * @member {WebGLShaderManager}\n     */\n    this.shaderManager = new WebGLShaderManager(this);\n\n    /**\n     * Manages the rendering of sprites\n     * @member {WebGLSpriteBatch}\n     */\n    this.spriteBatch = new WebGLSpriteBatch(this);\n\n    /**\n     * Manages the rendering of sprites\n     * @member {WebGLFastSpriteBatch}\n     */\n    this.fastSpriteBatch = new WebGLFastSpriteBatch(this);\n\n    /**\n     * Manages the masks using the stencil buffer\n     * @member {WebGLMaskManager}\n     */\n    this.maskManager = new WebGLMaskManager(this);\n\n    /**\n     * Manages the filters\n     * @member {WebGLFilterManager}\n     */\n    this.filterManager = new WebGLFilterManager(this);\n\n    /**\n     * Manages the stencil buffer\n     * @member {WebGLStencilManager}\n     */\n    this.stencilManager = new WebGLStencilManager(this);\n\n    /**\n     * Manages the blendModes\n     * @member {WebGLBlendModeManager}\n     */\n    this.blendModeManager = new WebGLBlendModeManager(this);\n\n    this.blendModes = null;\n\n    this._boundUpdateTexture = this.updateTexture.bind(this);\n    this._boundDestroyTexture = this.destroyTexture.bind(this);\n\n    // time init the context..\n    this._initContext();\n\n    // map some webGL blend modes..\n    this._mapBlendModes();\n}\n\n// constructor\nWebGLRenderer.prototype.constructor = WebGLRenderer;\nmodule.exports = WebGLRenderer;\n\nutils.eventTarget.mixin(WebGLRenderer.prototype);\n\nObject.defineProperties(WebGLRenderer.prototype, {\n    /**\n     * The background color to fill if not transparent\n     *\n     * @member {number}\n     * @memberof WebGLRenderer#\n     */\n    backgroundColor: {\n        get: function () {\n            return this._backgroundColor;\n        },\n        set: function (val) {\n            this._backgroundColor = val;\n            utils.hex2rgb(val, this._backgroundColorRgb);\n        }\n    }\n});\n\n/**\n *\n * @private\n */\nWebGLRenderer.prototype._initContext = function () {\n    var gl = this.view.getContext('webgl', this._contextOptions) || this.view.getContext('experimental-webgl', this._contextOptions);\n    this.gl = gl;\n\n    if (!gl) {\n        // fail, not able to get a context\n        throw new Error('This browser does not support webGL. Try using the canvas renderer');\n    }\n\n    this.glContextId = WebGLRenderer.glContextId++;\n    gl.id = this.glContextId;\n    gl.renderer = this;\n\n    // set up the default pixi settings..\n    gl.disable(gl.DEPTH_TEST);\n    gl.disable(gl.CULL_FACE);\n    gl.enable(gl.BLEND);\n\n    this.emit('context', gl);\n\n    // now resize and we are good to go!\n    this.resize(this.width, this.height);\n};\n\n/**\n * Renders the object to its webGL view\n *\n * @param object {DisplayObject} the object to be rendered\n */\nWebGLRenderer.prototype.render = function (object) {\n    // no point rendering if our context has been blown up!\n    if (this.gl.isContextLost()) {\n        return;\n    }\n\n    // update the scene graph\n    object.updateTransform();\n\n    var gl = this.gl;\n\n    // -- Does this need to be set every frame? -- //\n    gl.viewport(0, 0, this.width, this.height);\n\n    // make sure we are bound to the main frame buffer\n    gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n\n    if (this.clearBeforeRender) {\n        if (this.transparent) {\n            gl.clearColor(0, 0, 0, 0);\n        }\n        else {\n            gl.clearColor(this._backgroundColorRgb[0], this._backgroundColorRgb[1], this._backgroundColorRgb[2], 1);\n        }\n\n        gl.clear(gl.COLOR_BUFFER_BIT);\n    }\n\n    this.renderDisplayObject(object, this.projection);\n};\n\n/**\n * Renders a Display Object.\n *\n * @param displayObject {DisplayObject} The DisplayObject to render\n * @param projection {Point} The projection\n * @param buffer {Array} a standard WebGL buffer\n */\nWebGLRenderer.prototype.renderDisplayObject = function (displayObject, projection, buffer) {\n    this.blendModeManager.setBlendMode(CONST.blendModes.NORMAL);\n\n    // reset the render session data..\n    this.drawCount = 0;\n\n    // make sure to flip the Y if using a render texture..\n    this.flipY = buffer ? -1 : 1;\n\n    // set the default projection\n    this.projection = projection;\n\n    //set the default offset\n    this.offset = this.offset;\n\n    // start the sprite batch\n    this.spriteBatch.begin();\n\n    // start the filter manager\n    this.filterManager.begin(buffer);\n\n    // render the scene!\n    displayObject.renderWebGL(this);\n\n    // finish the sprite batch\n    this.spriteBatch.end();\n};\n\n/**\n * Resizes the webGL view to the specified width and height.\n *\n * @param width {number} the new width of the webGL view\n * @param height {number} the new height of the webGL view\n */\nWebGLRenderer.prototype.resize = function (width, height) {\n    this.width = width * this.resolution;\n    this.height = height * this.resolution;\n\n    this.view.width = this.width;\n    this.view.height = this.height;\n\n    if (this.autoResize) {\n        this.view.style.width = this.width / this.resolution + 'px';\n        this.view.style.height = this.height / this.resolution + 'px';\n    }\n\n    this.gl.viewport(0, 0, this.width, this.height);\n\n    this.projection.x =  this.width / 2 / this.resolution;\n    this.projection.y =  -this.height / 2 / this.resolution;\n};\n\n/**\n * Updates and/or Creates a WebGL texture for the renderer's context.\n *\n * @param texture {BaseTexture|Texture} the texture to update\n */\nWebGLRenderer.prototype.updateTexture = function (texture) {\n    texture = texture.baseTexture || texture;\n\n    if (!texture.hasLoaded) {\n        return;\n    }\n\n    var gl = this.gl;\n\n    if (!texture._glTextures[gl.id]) {\n        texture._glTextures[gl.id] = gl.createTexture();\n        texture.on('update', this._boundUpdateTexture);\n        texture.on('dispose', this._boundDestroyTexture);\n    }\n\n    gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]);\n\n    gl.pixelStorei(gl.UNPACK_PREMULTIPLY_ALPHA_WEBGL, texture.premultipliedAlpha);\n    gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA, gl.RGBA, gl.UNSIGNED_BYTE, texture.source);\n\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n\n\n    if (texture.mipmap && utils.isPowerOfTwo(texture.width, texture.height)) {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR_MIPMAP_LINEAR : gl.NEAREST_MIPMAP_NEAREST);\n        gl.generateMipmap(gl.TEXTURE_2D);\n    }\n    else {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, texture.scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n    }\n\n    if (!texture._powerOf2) {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n    }\n    else {\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.REPEAT);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.REPEAT);\n    }\n\n    return  texture._glTextures[gl.id];\n};\n\nWebGLRenderer.prototype.destroyTexture = function (texture) {\n    texture = texture.baseTexture || texture;\n\n    if (!texture.hasLoaded) {\n        return;\n    }\n\n    if (texture._glTextures[this.gl.id]) {\n        this.gl.deleteTexture(texture._glTextures[this.gl.id]);\n    }\n};\n\n/**\n * Handles a lost webgl context\n *\n * @param event {Event}\n * @private\n */\nWebGLRenderer.prototype.handleContextLost = function (event) {\n    event.preventDefault();\n};\n\n/**\n * Handles a restored webgl context\n *\n * @param event {Event}\n * @private\n */\nWebGLRenderer.prototype.handleContextRestored = function () {\n    this._initContext();\n\n    // empty all the ol gl textures as they are useless now\n    for (var key in utils.TextureCache) {\n        var texture = utils.TextureCache[key].baseTexture;\n        texture._glTextures = [];\n    }\n};\n\n/**\n * Removes everything from the renderer (event listeners, spritebatch, etc...)\n *\n * @param [removeView=false] {boolean} Removes the Canvas element from the DOM.\n */\nWebGLRenderer.prototype.destroy = function (removeView) {\n    if (removeView && this.view.parent) {\n        this.view.parent.removeChild(this.view);\n    }\n\n    // remove listeners\n    this.view.removeEventListener('webglcontextlost', this.contextLostBound);\n    this.view.removeEventListener('webglcontextrestored', this.contextRestoredBound);\n\n    // time to create the render managers! each one focuses on managine a state in webGL\n    this.shaderManager.destroy();\n    this.spriteBatch.destroy();\n    this.maskManager.destroy();\n    this.filterManager.destroy();\n\n\n    // this.uuid = utils.uuid();\n    // this.type = CONST.WEBGL_RENDERER;\n\n    // this.resolution = options.resolution;\n    // this.transparent = options.transparent;\n\n    this._backgroundColor = 0x000000;\n    this._backgroundColorRgb = null;\n\n    // this.backgroundColor = null;\n    // this.autoResize = options.autoResize || false;\n    // this.preserveDrawingBuffer = options.preserveDrawingBuffer;\n    // this.clearBeforeRender = options.clearBeforeRender;\n    // this.width = width || 800;\n    // this.height = height || 600;\n\n    this.view = null;\n\n    this.contextLostBound = null;\n    this.contextRestoredBound = null;\n\n    this._contextOptions = null;\n\n    this.projection = null;\n    this.offset = null;\n    this.drawCount = 0;\n\n    this.shaderManager = null;\n    this.spriteBatch = null;\n    this.maskManager = null;\n    this.filterManager = null;\n    this.stencilManager = null;\n    this.blendModeManager = null;\n\n    this.blendModes = null;\n\n    this.gl = null;\n    this.blendModes = null;\n};\n\n/**\n * Maps Pixi blend modes to WebGL blend modes.\n *\n * @private\n */\nWebGLRenderer.prototype._mapBlendModes = function () {\n    var gl = this.gl;\n\n    if (!this.blendModes) {\n        this.blendModes = {};\n\n        this.blendModes[CONST.blendModes.NORMAL]        = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.ADD]           = [gl.SRC_ALPHA, gl.DST_ALPHA];\n        this.blendModes[CONST.blendModes.MULTIPLY]      = [gl.DST_COLOR, gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.SCREEN]        = [gl.SRC_ALPHA, gl.ONE];\n        this.blendModes[CONST.blendModes.OVERLAY]       = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.DARKEN]        = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.LIGHTEN]       = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.COLOR_DODGE]   = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.COLOR_BURN]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.HARD_LIGHT]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.SOFT_LIGHT]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.DIFFERENCE]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.EXCLUSION]     = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.HUE]           = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.SATURATION]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.COLOR]         = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n        this.blendModes[CONST.blendModes.LUMINOSITY]    = [gl.ONE,       gl.ONE_MINUS_SRC_ALPHA];\n    }\n};\n\nWebGLRenderer.glContextId = 0;\n","var WebGLManager = require('./WebGLManager');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLBlendModeManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    /**\n     * @member {number}\n     */\n    this.currentBlendMode = 99999;\n}\n\nWebGLBlendModeManager.prototype = Object.create(WebGLManager.prototype);\nWebGLBlendModeManager.prototype.constructor = WebGLBlendModeManager;\nmodule.exports = WebGLBlendModeManager;\n\n/**\n * Sets-up the given blendMode from WebGL's point of view.\n *\n * @param blendMode {number} the blendMode, should be a Pixi const, such as BlendModes.ADD\n */\nWebGLBlendModeManager.prototype.setBlendMode = function (blendMode) {\n    if (this.currentBlendMode === blendMode) {\n        return false;\n    }\n\n    this.currentBlendMode = blendMode;\n\n    var mode = this.renderer.blendModes[this.currentBlendMode];\n    this.renderer.gl.blendFunc(mode[0], mode[1]);\n\n    return true;\n};\n","var WebGLManager = require('./WebGLManager'),\n    FilterTexture = require('../utils/FilterTexture'),\n    Shader = require('../shaders/Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLFilterManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    /**\n     * @member {any[]}\n     */\n    this.filterStack = [];\n\n    /**\n     * @member {any[]]}\n     */\n    this.texturePool = [];\n\n    /**\n     * @member {number}\n     */\n    this.offsetX = 0;\n\n    /**\n     * @member {number}\n     */\n    this.offsetY = 0;\n\n    // listen for context and update necessary buffers\n    var self = this;\n    this.renderer.on('context', function () {\n        self.texturePool.length = 0;\n        self.initShaderBuffers();\n    });\n}\n\nWebGLFilterManager.prototype = Object.create(WebGLManager.prototype);\nWebGLFilterManager.prototype.constructor = WebGLFilterManager;\nmodule.exports = WebGLFilterManager;\n\n/**\n * @param renderer {WebGLRenderer}\n * @param buffer {ArrayBuffer}\n */\nWebGLFilterManager.prototype.begin = function (buffer) {\n    this.defaultShader = this.renderer.shaderManager.defaultShader;\n\n    this.width = this.renderer.projection.x * 2;\n    this.height = -this.renderer.projection.y * 2;\n\n    this.buffer = buffer;\n};\n\n/**\n * Applies the filter and adds it to the current filter stack.\n *\n * @param filterBlock {object} the filter that will be pushed to the current filter stack\n */\nWebGLFilterManager.prototype.pushFilter = function (filterBlock) {\n    var gl = this.renderer.gl;\n\n    var projection = this.renderer.projection;\n    var offset = this.renderer.offset;\n\n    filterBlock._filterArea = filterBlock.target.filterArea || filterBlock.target.getBounds();\n\n    // filter program\n    // OPTIMISATION - the first filter is free if its a simple color change?\n    this.filterStack.push(filterBlock);\n\n    var filter = filterBlock.filterPasses[0];\n\n    this.offsetX += filterBlock._filterArea.x;\n    this.offsetY += filterBlock._filterArea.y;\n\n    var texture = this.texturePool.pop();\n    if (!texture) {\n        texture = new FilterTexture(this.renderer.gl, this.width, this.height);\n    }\n    else {\n        texture.resize(this.width, this.height);\n    }\n\n    gl.bindTexture(gl.TEXTURE_2D,  texture.texture);\n\n    var filterArea = filterBlock._filterArea;// filterBlock.target.getBounds();///filterBlock.target.filterArea;\n\n    var padding = filter.padding;\n    filterArea.x -= padding;\n    filterArea.y -= padding;\n    filterArea.width += padding * 2;\n    filterArea.height += padding * 2;\n\n    var localX = filterArea.x,\n        localY = filterArea.y;\n\n    if (filterArea.x < 0) {\n        filterArea.width += filterArea.x;\n        filterArea.x = 0;\n    }\n\n    if (filterArea.y < 0) {\n        filterArea.height += filterArea.y;\n        filterArea.y = 0;\n    }\n\n    if (localX + filterArea.width > this.width) {\n        filterArea.width = this.width - localX;\n    }\n\n    if (localY + filterArea.height > this.height) {\n        filterArea.height = this.height - localY;\n    }\n\n    if (filterArea.width < 0) {\n        filterArea.width = 0;\n    }\n\n    if (filterArea.height < 0) {\n        filterArea.height = 0;\n    }\n\n    //gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA,  filterArea.width, filterArea.height, 0, gl.RGBA, gl.UNSIGNED_BYTE, null);\n    gl.bindFramebuffer(gl.FRAMEBUFFER, texture.frameBuffer);\n\n    // set view port\n    gl.viewport(0, 0, filterArea.width, filterArea.height);\n\n    projection.x = filterArea.width/2;\n    projection.y = -filterArea.height/2;\n\n    offset.x = -filterArea.x;\n    offset.y = -filterArea.y;\n\n    // update projection\n    // now restore the regular shader..\n    // this.renderer.shaderManager.setShader(this.defaultShader);\n    //gl.uniform2f(this.defaultShader.projectionVector, filterArea.width/2, -filterArea.height/2);\n    //gl.uniform2f(this.defaultShader.offsetVector, -filterArea.x, -filterArea.y);\n\n    gl.colorMask(true, true, true, true);\n    gl.clearColor(0,0,0, 0);\n    gl.clear(gl.COLOR_BUFFER_BIT);\n\n    filterBlock._glFilterTexture = texture;\n\n};\n\n/**\n * Removes the last filter from the filter stack and doesn't return it.\n *\n */\nWebGLFilterManager.prototype.popFilter = function () {\n    var gl = this.renderer.gl;\n\n    var filterBlock = this.filterStack.pop();\n    var filterArea = filterBlock._filterArea;\n    var texture = filterBlock._glFilterTexture;\n    var projection = this.renderer.projection;\n    var offset = this.renderer.offset;\n\n    if (filterBlock.filterPasses.length > 1) {\n        gl.viewport(0, 0, filterArea.width, filterArea.height);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n\n        this.vertexArray[0] = 0;\n        this.vertexArray[1] = filterArea.height;\n\n        this.vertexArray[2] = filterArea.width;\n        this.vertexArray[3] = filterArea.height;\n\n        this.vertexArray[4] = 0;\n        this.vertexArray[5] = 0;\n\n        this.vertexArray[6] = filterArea.width;\n        this.vertexArray[7] = 0;\n\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n        // now set the uvs..\n        this.uvArray[2] = filterArea.width/this.width;\n        this.uvArray[5] = filterArea.height/this.height;\n        this.uvArray[6] = filterArea.width/this.width;\n        this.uvArray[7] = filterArea.height/this.height;\n\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray);\n\n        var inputTexture = texture;\n        var outputTexture = this.texturePool.pop();\n        if (!outputTexture) {\n            outputTexture = new FilterTexture(this.renderer.gl, this.width, this.height);\n        }\n        outputTexture.resize(this.width, this.height);\n\n        // need to clear this FBO as it may have some left over elements from a previous filter.\n        gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer );\n        gl.clear(gl.COLOR_BUFFER_BIT);\n\n        gl.disable(gl.BLEND);\n\n        for (var i = 0; i < filterBlock.filterPasses.length-1; i++) {\n            var filterPass = filterBlock.filterPasses[i];\n\n            gl.bindFramebuffer(gl.FRAMEBUFFER, outputTexture.frameBuffer );\n\n            // set texture\n            gl.activeTexture(gl.TEXTURE0);\n            gl.bindTexture(gl.TEXTURE_2D, inputTexture.texture);\n\n            // draw texture..\n            //filterPass.applyFilterPass(filterArea.width, filterArea.height);\n            this.applyFilterPass(filterPass, filterArea, filterArea.width, filterArea.height);\n\n            // swap the textures..\n            var temp = inputTexture;\n            inputTexture = outputTexture;\n            outputTexture = temp;\n        }\n\n        gl.enable(gl.BLEND);\n\n        texture = inputTexture;\n        this.texturePool.push(outputTexture);\n    }\n\n    var filter = filterBlock.filterPasses[filterBlock.filterPasses.length-1];\n\n    this.offsetX -= filterArea.x;\n    this.offsetY -= filterArea.y;\n\n    var sizeX = this.width;\n    var sizeY = this.height;\n\n    var offsetX = 0;\n    var offsetY = 0;\n\n    var buffer = this.buffer;\n\n    // time to render the filters texture to the previous scene\n    if (this.filterStack.length === 0) {\n        gl.colorMask(true, true, true, true);//this.transparent);\n    }\n    else {\n        var currentFilter = this.filterStack[this.filterStack.length-1];\n        filterArea = currentFilter._filterArea;\n\n        sizeX = filterArea.width;\n        sizeY = filterArea.height;\n\n        offsetX = filterArea.x;\n        offsetY = filterArea.y;\n\n        buffer =  currentFilter._glFilterTexture.frameBuffer;\n    }\n\n    // TODO need to remove these global elements..\n    projection.x = sizeX/2;\n    projection.y = -sizeY/2;\n\n    offset.x = offsetX;\n    offset.y = offsetY;\n\n    filterArea = filterBlock._filterArea;\n\n    var x = filterArea.x-offsetX;\n    var y = filterArea.y-offsetY;\n\n    // update the buffers..\n    // make sure to flip the y!\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n\n    this.vertexArray[0] = x;\n    this.vertexArray[1] = y + filterArea.height;\n\n    this.vertexArray[2] = x + filterArea.width;\n    this.vertexArray[3] = y + filterArea.height;\n\n    this.vertexArray[4] = x;\n    this.vertexArray[5] = y;\n\n    this.vertexArray[6] = x + filterArea.width;\n    this.vertexArray[7] = y;\n\n    gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertexArray);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n\n    this.uvArray[2] = filterArea.width/this.width;\n    this.uvArray[5] = filterArea.height/this.height;\n    this.uvArray[6] = filterArea.width/this.width;\n    this.uvArray[7] = filterArea.height/this.height;\n\n    gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.uvArray);\n\n    gl.viewport(0, 0, sizeX, sizeY);\n\n    // bind the buffer\n    gl.bindFramebuffer(gl.FRAMEBUFFER, buffer );\n\n    // set the blend mode!\n    //gl.blendFunc(gl.ONE, gl.ONE_MINUS_SRC_ALPHA)\n\n    // set texture\n    gl.activeTexture(gl.TEXTURE0);\n    gl.bindTexture(gl.TEXTURE_2D, texture.texture);\n\n    // apply!\n    this.applyFilterPass(filter, filterArea, sizeX, sizeY);\n\n    // now restore the regular shader.. should happen automatically now..\n    // this.renderer.shaderManager.setShader(this.defaultShader);\n    // gl.uniform2f(this.defaultShader.projectionVector, sizeX/2, -sizeY/2);\n    // gl.uniform2f(this.defaultShader.offsetVector, -offsetX, -offsetY);\n\n    // return the texture to the pool\n    this.texturePool.push(texture);\n    filterBlock._glFilterTexture = null;\n};\n\n\n/**\n * Applies the filter to the specified area.\n *\n * @param filter {AbstractFilter} the filter that needs to be applied\n * @param filterArea {Texture} TODO - might need an update\n * @param width {number} the horizontal range of the filter\n * @param height {number} the vertical range of the filter\n */\nWebGLFilterManager.prototype.applyFilterPass = function (filter, filterArea, width, height) {\n    // use program\n    var gl = this.renderer.gl;\n\n    var shader = filter.shaders[gl.id];\n\n    if (!shader) {\n        shader = new Shader(gl);\n\n        shader.fragmentSrc = filter.fragmentSrc;\n        shader.uniforms = filter.uniforms;\n        shader.init();\n\n        filter.shaders[gl.id] = shader;\n    }\n\n    // set the shader\n    this.renderer.shaderManager.setShader(shader);\n\n//    gl.useProgram(shader.program);\n\n    gl.uniform2f(shader.projectionVector, width/2, -height/2);\n    gl.uniform2f(shader.offsetVector, 0,0);\n\n    if (filter.uniforms.dimensions) {\n        filter.uniforms.dimensions.value[0] = this.width;//width;\n        filter.uniforms.dimensions.value[1] = this.height;//height;\n        filter.uniforms.dimensions.value[2] = this.vertexArray[0];\n        filter.uniforms.dimensions.value[3] = this.vertexArray[5];//filterArea.height;\n    }\n\n    shader.syncUniforms();\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n    gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer);\n    gl.vertexAttribPointer(shader.aColor, 2, gl.FLOAT, false, 0, 0);\n\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n\n    // draw the filter...\n    gl.drawElements(gl.TRIANGLES, 6, gl.UNSIGNED_SHORT, 0 );\n\n    this.renderer.drawCount++;\n};\n\n/**\n * Initialises the shader buffers.\n *\n */\nWebGLFilterManager.prototype.initShaderBuffers = function () {\n    var gl = this.renderer.gl;\n\n    // create some buffers\n    this.vertexBuffer = gl.createBuffer();\n    this.uvBuffer = gl.createBuffer();\n    this.colorBuffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n\n    // bind and upload the vertexs..\n    // keep a reference to the vertexFloatData..\n    this.vertexArray = new Float32Array([0.0, 0.0,\n                                         1.0, 0.0,\n                                         0.0, 1.0,\n                                         1.0, 1.0]);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertexArray, gl.STATIC_DRAW);\n\n    // bind and upload the uv buffer\n    this.uvArray = new Float32Array([0.0, 0.0,\n                                     1.0, 0.0,\n                                     0.0, 1.0,\n                                     1.0, 1.0]);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.uvBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.uvArray, gl.STATIC_DRAW);\n\n    this.colorArray = new Float32Array([1.0, 0xFFFFFF,\n                                        1.0, 0xFFFFFF,\n                                        1.0, 0xFFFFFF,\n                                        1.0, 0xFFFFFF]);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.colorBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.colorArray, gl.STATIC_DRAW);\n\n    // bind and upload the index\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, new Uint16Array([0, 1, 2, 1, 3, 2]), gl.STATIC_DRAW);\n\n};\n\n/**\n * Destroys the filter and removes it from the filter stack.\n *\n */\nWebGLFilterManager.prototype.destroy = function () {\n    var gl = this.renderer.gl;\n\n    this.filterStack = null;\n\n    this.offsetX = 0;\n    this.offsetY = 0;\n\n    // destroy textures\n    for (var i = 0; i < this.texturePool.length; i++) {\n        this.texturePool[i].destroy();\n    }\n\n    this.texturePool = null;\n\n    //destroy buffers..\n    gl.deleteBuffer(this.vertexBuffer);\n    gl.deleteBuffer(this.uvBuffer);\n    gl.deleteBuffer(this.colorBuffer);\n    gl.deleteBuffer(this.indexBuffer);\n\n    this.renderer = null;\n};\n","/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLManager(renderer) {\n    /**\n     * The renderer this manager works for.\n     *\n     * @member {WebGLRenderer}\n     */\n    this.renderer = renderer;\n}\n\nWebGLManager.prototype.constructor = WebGLManager;\nmodule.exports = WebGLManager;\n\nWebGLManager.prototype.destroy = function () {\n    this.renderer = null;\n};\n","var WebGLManager = require('./WebGLManager'),\n    WebGLGraphics = require('../utils/WebGLGraphics');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLMaskManager(renderer) {\n    WebGLManager.call(this, renderer);\n}\n\nWebGLMaskManager.prototype = Object.create(WebGLManager.prototype);\nWebGLMaskManager.prototype.constructor = WebGLMaskManager;\nmodule.exports = WebGLMaskManager;\n\n/**\n * Applies the Mask and adds it to the current filter stack.\n *\n * @param maskData {any[]}\n */\nWebGLMaskManager.prototype.pushMask = function (maskData) {\n    if (maskData.dirty) {\n        WebGLGraphics.updateGraphics(maskData, this.renderer.gl);\n    }\n\n    if (!maskData._webGL[this.renderer.gl.id].data.length) {\n        return;\n    }\n\n    this.renderer.stencilManager.pushStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer);\n};\n\n/**\n * Removes the last filter from the filter stack and doesn't return it.\n *\n * @param maskData {any[]}\n */\nWebGLMaskManager.prototype.popMask = function (maskData) {\n    this.renderer.stencilManager.popStencil(maskData, maskData._webGL[this.renderer.gl.id].data[0], this.renderer);\n};\n","var WebGLManager = require('./WebGLManager'),\n    PrimitiveShader = require('../shaders/PrimitiveShader'),\n    ComplexPrimitiveShader = require('../shaders/ComplexPrimitiveShader'),\n    Shader = require('../shaders/Shader'),\n    FastShader = require('../shaders/FastShader'),\n    StripShader = require('../shaders/StripShader');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLShaderManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    /**\n     * @member {number}\n     */\n    this.maxAttibs = 10;\n\n    /**\n     * @member {any[]}\n     */\n    this.attribState = [];\n\n    /**\n     * @member {any[]}\n     */\n    this.tempAttribState = [];\n\n    for (var i = 0; i < this.maxAttibs; i++) {\n        this.attribState[i] = false;\n    }\n\n    /**\n     * @member {any[]}\n     */\n    this.stack = [];\n\n    /**\n     * @member {number}\n     * @private\n     */\n    this._currentId = -1;\n\n    /**\n     * @member {Shader}\n     * @private\n     */\n    this.currentShader = null;\n\n    // this shader is used for rendering primitives\n    this.primitiveShader = null;\n\n    // this shader is used for rendering triangle strips\n    this.complexPrimitiveShader = null;\n\n    // this shader is used for the default sprite rendering\n    this.defaultShader = null;\n\n    // this shader is used for the fast sprite rendering\n    this.fastShader = null;\n\n    // the next one is used for rendering triangle strips\n    this.stripShader = null;\n\n    // listen for context and update necessary shaders\n    var self = this;\n    this.renderer.on('context', function (event) {\n        var gl = event.data;\n\n        // this shader is used for rendering primitives\n        self.primitiveShader = new PrimitiveShader(gl);\n\n        // this shader is used for rendering triangle strips\n        self.complexPrimitiveShader = new ComplexPrimitiveShader(gl);\n\n        // this shader is used for the default sprite rendering\n        self.defaultShader = new Shader(gl);\n\n        // this shader is used for the fast sprite rendering\n        self.fastShader = new FastShader(gl);\n\n        // the next one is used for rendering triangle strips\n        self.stripShader = new StripShader(gl);\n\n        self.setShader(self.defaultShader);\n    });\n}\n\nWebGLShaderManager.prototype = Object.create(WebGLManager.prototype);\nWebGLShaderManager.prototype.constructor = WebGLShaderManager;\nmodule.exports = WebGLShaderManager;\n\n/**\n * Takes the attributes given in parameters.\n *\n * @param attribs {Array} attribs\n */\nWebGLShaderManager.prototype.setAttribs = function (attribs) {\n    // reset temp state\n    var i;\n\n    for (i = 0; i < this.tempAttribState.length; i++) {\n        this.tempAttribState[i] = false;\n    }\n\n    // set the new attribs\n    for (var a in attribs) {\n        this.tempAttribState[attribs[a]] = true;\n    }\n\n    var gl = this.renderer.gl;\n\n    for (i = 0; i < this.attribState.length; i++) {\n        if (this.attribState[i] !== this.tempAttribState[i]) {\n            this.attribState[i] = this.tempAttribState[i];\n\n            if (this.attribState[i]) {\n                gl.enableVertexAttribArray(i);\n            }\n            else {\n                gl.disableVertexAttribArray(i);\n            }\n        }\n    }\n};\n\n/**\n * Sets the current shader.\n *\n * @param shader {Any}\n */\nWebGLShaderManager.prototype.setShader = function (shader) {\n    if (this._currentId === shader.uuid) {\n        return false;\n    }\n\n    this._currentId = shader.uuid;\n\n    this.currentShader = shader;\n\n    this.renderer.gl.useProgram(shader.program);\n    this.setAttribs(shader.attributes);\n\n    return true;\n};\n\n/**\n * Destroys this object.\n *\n */\nWebGLShaderManager.prototype.destroy = function () {\n    this.attribState = null;\n\n    this.tempAttribState = null;\n\n    this.primitiveShader.destroy();\n    this.primitiveShader = null;\n\n    this.complexPrimitiveShader.destroy();\n    this.complexPrimitiveShader = null;\n\n    this.defaultShader.destroy();\n    this.defaultShader = null;\n\n    this.fastShader.destroy();\n    this.fastShader = null;\n\n    this.stripShader.destroy();\n    this.stripShader = null;\n\n    this.renderer = null;\n};\n","var WebGLManager = require('./WebGLManager'),\n    utils = require('../../../utils');\n\n/**\n * @class\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this manager works for.\n */\nfunction WebGLStencilManager(renderer) {\n    WebGLManager.call(this, renderer);\n\n    this.stencilStack = [];\n    this.reverse = true;\n    this.count = 0;\n}\n\nWebGLStencilManager.prototype = Object.create(WebGLManager.prototype);\nWebGLStencilManager.prototype.constructor = WebGLStencilManager;\nmodule.exports = WebGLStencilManager;\n\n/**\n * Applies the Mask and adds it to the current filter stack.\n *\n * @param graphics {Graphics}\n * @param webGLData {any[]}\n */\nWebGLStencilManager.prototype.pushStencil = function (graphics, webGLData) {\n    var gl = this.renderer.gl;\n\n    this.bindGraphics(graphics, webGLData, this.renderer);\n\n    if (this.stencilStack.length === 0) {\n        gl.enable(gl.STENCIL_TEST);\n        gl.clear(gl.STENCIL_BUFFER_BIT);\n        this.reverse = true;\n        this.count = 0;\n    }\n\n    this.stencilStack.push(webGLData);\n\n    var level = this.count;\n\n    gl.colorMask(false, false, false, false);\n\n    gl.stencilFunc(gl.ALWAYS,0,0xFF);\n    gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT);\n\n    // draw the triangle strip!\n\n    if (webGLData.mode === 1) {\n        gl.drawElements(gl.TRIANGLE_FAN,  webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 );\n\n        if (this.reverse) {\n            gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n        }\n\n        // draw a quad to increment..\n        gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 );\n\n        if (this.reverse) {\n            gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n        }\n\n        this.reverse = !this.reverse;\n    }\n    else {\n        if (!this.reverse) {\n            gl.stencilFunc(gl.EQUAL, 0xFF - level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n        }\n\n        gl.drawElements(gl.TRIANGLE_STRIP,  webGLData.indices.length, gl.UNSIGNED_SHORT, 0 );\n\n        if (!this.reverse) {\n            gl.stencilFunc(gl.EQUAL,0xFF-(level+1), 0xFF);\n        }\n        else {\n            gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n        }\n    }\n\n    gl.colorMask(true, true, true, true);\n    gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP);\n\n    this.count++;\n};\n\n/**\n * TODO this does not belong here!\n *\n * @param graphics {Graphics}\n * @param webGLData {Array}\n */\nWebGLStencilManager.prototype.bindGraphics = function (graphics, webGLData) {\n    //if (this._currentGraphics === graphics)return;\n    this._currentGraphics = graphics;\n\n    var gl = this.renderer.gl;\n\n     // bind the graphics object..\n    var projection = this.renderer.projection,\n        offset = this.renderer.offset,\n        shader;// = this.renderer.shaderManager.primitiveShader;\n\n    if (webGLData.mode === 1) {\n        shader = this.renderer.shaderManager.complexPrimitiveShader;\n\n        this.renderer.shaderManager.setShader(shader);\n\n        gl.uniform1f(shader.flipY, this.renderer.flipY);\n\n        gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true));\n\n        gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n        gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n\n        gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint));\n        gl.uniform3fv(shader.color, webGLData.color);\n\n        gl.uniform1f(shader.alpha, graphics.worldAlpha * webGLData.alpha);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer);\n\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 2, 0);\n\n\n        // now do the rest..\n        // set the index buffer!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer);\n    }\n    else {\n        //this.renderer.shaderManager.activatePrimitiveShader();\n        shader = this.renderer.shaderManager.primitiveShader;\n        this.renderer.shaderManager.setShader( shader );\n\n        gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true));\n\n        gl.uniform1f(shader.flipY, this.renderer.flipY);\n        gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n        gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n\n        gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint));\n\n        gl.uniform1f(shader.alpha, graphics.worldAlpha);\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer);\n\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0);\n        gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4);\n\n        // set the index buffer!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer);\n    }\n};\n\n/**\n * @param graphics {Graphics}\n * @param webGLData {Array}\n */\nWebGLStencilManager.prototype.popStencil = function (graphics, webGLData) {\n\tvar gl = this.renderer.gl;\n\n    this.stencilStack.pop();\n\n    this.count--;\n\n    if (this.stencilStack.length === 0) {\n        // the stack is empty!\n        gl.disable(gl.STENCIL_TEST);\n\n    }\n    else {\n\n        var level = this.count;\n\n        this.bindGraphics(graphics, webGLData, this.renderer);\n\n        gl.colorMask(false, false, false, false);\n\n        if (webGLData.mode === 1) {\n            this.reverse = !this.reverse;\n\n            if (this.reverse) {\n                gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n            }\n\n            // draw a quad to increment..\n            gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 );\n\n            gl.stencilFunc(gl.ALWAYS,0,0xFF);\n            gl.stencilOp(gl.KEEP,gl.KEEP,gl.INVERT);\n\n            // draw the triangle strip!\n            gl.drawElements(gl.TRIANGLE_FAN,  webGLData.indices.length - 4, gl.UNSIGNED_SHORT, 0 );\n\n            if (!this.reverse) {\n                gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            }\n\n        }\n        else {\n          //  console.log(\"<<>>\")\n            if (!this.reverse) {\n                gl.stencilFunc(gl.EQUAL, 0xFF - (level+1), 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.INCR);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level+1, 0xFF);\n                gl.stencilOp(gl.KEEP,gl.KEEP,gl.DECR);\n            }\n\n            gl.drawElements(gl.TRIANGLE_STRIP,  webGLData.indices.length, gl.UNSIGNED_SHORT, 0 );\n\n            if (!this.reverse) {\n                gl.stencilFunc(gl.EQUAL,0xFF-(level), 0xFF);\n            }\n            else {\n                gl.stencilFunc(gl.EQUAL,level, 0xFF);\n            }\n        }\n\n        gl.colorMask(true, true, true, true);\n        gl.stencilOp(gl.KEEP,gl.KEEP,gl.KEEP);\n\n\n    }\n};\n\n/**\n * Destroys the mask stack.\n *\n */\nWebGLStencilManager.prototype.destroy = function () {\n    this.renderer = null;\n    this.stencilStack = null;\n};\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction ComplexPrimitiveShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            // 'attribute vec2 aTextureCoord;',\n            // 'attribute vec4 aColor;',\n\n            'uniform mat3 translationMatrix;',\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n\n            'uniform vec3 tint;',\n            'uniform float alpha;',\n            'uniform vec3 color;',\n            'uniform float flipY;',\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);',\n            '   v -= offsetVector.xyx;',\n            '   gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);',\n            '   vColor = vec4(color * alpha * tint, alpha);',//\" * vec4(tint * alpha, alpha);',\n            '}'\n        ].join('\\n'),\n        // fragment shader\n        [\n            'precision mediump float;',\n\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   gl_FragColor = vColor;',\n            '}'\n        ].join('\\n'),\n        // custom uniforms\n        {\n            tint:   { type: '3f', value: [0, 0, 0] },\n            flipY:  { type: '1f', value: 0 },\n            alpha:  { type: '1f', value: 0 },\n            translationMatrix: { type: 'mat3', value: new Float32Array(9) }\n        }\n    );\n}\n\nComplexPrimitiveShader.prototype = Object.create(Shader.prototype);\nComplexPrimitiveShader.prototype.constructor = ComplexPrimitiveShader;\nmodule.exports = ComplexPrimitiveShader;\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @extends Shader\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction FastShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            'attribute vec2 aTextureCoord;',\n            'attribute vec4 aColor;',\n\n            'attribute vec2 aPositionCoord;',\n            'attribute vec2 aScale;',\n            'attribute float aRotation;',\n\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n            'uniform mat3 uMatrix;',\n\n            'varying vec2 vTextureCoord;',\n            'varying vec4 vColor;',\n\n            'const vec2 center = vec2(-1.0, 1.0);',\n\n            'void main(void) {',\n            '   vec2 v;',\n            '   vec2 sv = aVertexPosition * aScale;',\n            '   v.x = (sv.x) * cos(aRotation) - (sv.y) * sin(aRotation);',\n            '   v.y = (sv.x) * sin(aRotation) + (sv.y) * cos(aRotation);',\n            '   v = ( uMatrix * vec3(v + aPositionCoord , 1.0) ).xy ;',\n            '   gl_Position = vec4( ( v / projectionVector) + center , 0.0, 1.0);',\n            '   vTextureCoord = aTextureCoord;',\n          //  '   vec3 color = mod(vec3(aColor.y/65536.0, aColor.y/256.0, aColor.y), 256.0) / 256.0;',\n            '   vColor = aColor;',\n            '}'\n        ].join('\\n'),\n        // fragment shader, use default\n        null,\n        // custom uniforms\n        {\n            uMatrix: { type: 'mat3', value: new Float32Array(9) }\n        },\n        // custom attributes\n        {\n            aPositionCoord: 0,\n            aRotation:      0,\n            aScale:         0\n        }\n    );\n}\n\nFastShader.prototype = Object.create(Shader.prototype);\nFastShader.prototype.constructor = FastShader;\nmodule.exports = FastShader;\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction PrimitiveShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            // 'attribute vec2 aTextureCoord;',\n            'attribute vec4 aColor;',\n\n            'uniform mat3 translationMatrix;',\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n            'uniform float alpha;',\n            'uniform float flipY;',\n            'uniform vec3 tint;',\n\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);',\n            '   v -= offsetVector.xyx;',\n            '   gl_Position = vec4( v.x / projectionVector.x -1.0, (v.y / projectionVector.y * -flipY) + flipY , 0.0, 1.0);',\n            '   vColor = aColor * vec4(tint * alpha, alpha);',\n            '}'\n        ].join('\\n'),\n        // fragment shader\n        [\n            'precision mediump float;',\n\n            'varying vec4 vColor;',\n\n            'void main(void) {',\n            '   gl_FragColor = vColor;',\n            '}'\n        ].join('\\n'),\n        // custom uniforms\n        {\n            tint:   { type: '3f', value: [0, 0, 0] },\n            flipY:  { type: '1f', value: 0 },\n            alpha:  { type: '1f', value: 0 },\n            translationMatrix: { type: 'mat3', value: new Float32Array(9) }\n        }\n    );\n}\n\nPrimitiveShader.prototype = Object.create(Shader.prototype);\nPrimitiveShader.prototype.constructor = PrimitiveShader;\nmodule.exports = PrimitiveShader;\n","var utils = require('../../../utils');\n\n/**\n * @class\n * @namespace PIXI\n * @param [fragmentSrc] {string} The source of the fragment shader.\n * @param [vertexSrc] {string} The source of the vertex shader.\n */\nfunction Shader(gl, vertexSrc, fragmentSrc, customUniforms, customAttributes) {\n    /**\n     * @member {number}\n     * @readonly\n     */\n    this.uuid = utils.uuid();\n\n    /**\n     * @member {WebGLContext}\n     * @readonly\n     */\n    this.gl = gl;\n\n    /**\n     * The WebGL program.\n     * @member {WebGLProgram}\n     * @readonly\n     */\n    this.program = null;\n\n    this.uniforms = {\n        uSampler:           { type: 'sampler2D', value: 0 },\n        projectionVector:   { type: '2f', value: { x: 0, y: 0 } },\n        offsetVector:       { type: '2f', value: { x: 0, y: 0 } },\n        dimensions:         { type: '4f', value: new Float32Array(4) }\n    };\n\n    for (var u in customUniforms) {\n        this.uniforms[u] = customUniforms[u];\n    }\n\n    this.attributes = {\n        aVertexPosition:    0,\n        aTextureCoord:      0,\n        aColor:             0\n    };\n\n    for (var a in customAttributes) {\n        this.attributes[a] = customAttributes[a];\n    }\n\n    this.textureCount = 0;\n\n    /**\n     * The vertex shader.\n     * @member {Array}\n     */\n    this.vertexSrc = vertexSrc || [\n        'attribute vec2 aVertexPosition;',\n        'attribute vec2 aTextureCoord;',\n        'attribute vec4 aColor;',\n\n        'uniform vec2 projectionVector;',\n        'uniform vec2 offsetVector;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'const vec2 center = vec2(-1.0, 1.0);',\n\n        'void main(void) {',\n        '   gl_Position = vec4( ((aVertexPosition + offsetVector) / projectionVector) + center , 0.0, 1.0);',\n        '   vTextureCoord = aTextureCoord;',\n        '   vColor = vec4(aColor.rgb * aColor.a, aColor.a);',\n        '}'\n    ].join('\\n');\n\n    /**\n     * The fragment shader.\n     * @member {Array}\n     */\n    this.fragmentSrc = fragmentSrc || [\n        'precision lowp float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord) * vColor ;',\n        '}'\n    ].join('\\n');\n\n    this.init();\n}\n\nShader.prototype.constructor = Shader;\nmodule.exports = Shader;\n\nShader.prototype.init = function () {\n    this.compile();\n\n    this.gl.useProgram(this.program);\n\n    this.cacheUniformLocations(Object.keys(this.uniforms));\n    this.cacheAttributeLocations(Object.keys(this.attributes));\n};\n\nShader.prototype.cacheUniformLocations = function (keys) {\n    for (var i = 0; i < keys.length; ++i) {\n        this.uniforms[keys[i]]._location = this.gl.getUniformLocation(this.program, keys[i]);\n    }\n};\n\nShader.prototype.cacheAttributeLocations = function (keys) {\n    for (var i = 0; i < keys.length; ++i) {\n        this.attributes[keys[i]] = this.gl.getAttribLocation(this.program, keys[i]);\n    }\n\n    // TODO: Check if this is needed anymore...\n\n    // Begin worst hack eva //\n\n    // WHY??? ONLY on my chrome pixel the line above returns -1 when using filters?\n    // maybe its something to do with the current state of the gl context.\n    // I'm convinced this is a bug in the chrome browser as there is NO reason why this should be returning -1 especially as it only manifests on my chrome pixel\n    // If theres any webGL people that know why could happen please help :)\n    // if (this.attributes.aColor === -1) {\n    //     this.attributes.aColor = 2;\n    // }\n\n    // End worst hack eva //\n};\n\nShader.prototype.compile = function () {\n    var gl = this.gl;\n\n    var glVertShader = this._glCompile(gl.VERTEX_SHADER, this.vertexSrc);\n    var glFragShader = this._glCompile(gl.FRAGMENT_SHADER, this.fragmentSrc);\n\n    var program = gl.createProgram();\n\n    gl.attachShader(program, glVertShader);\n    gl.attachShader(program, glFragShader);\n    gl.linkProgram(program);\n\n    // if linking fails, then log and cleanup\n    if (!gl.getProgramParameter(program, gl.LINK_STATUS)) {\n        window.console.error('Pixi.js Error: Could not initialize shader.');\n        window.console.error('gl.VALIDATE_STATUS', gl.getProgramParameter(program, gl.VALIDATE_STATUS));\n        window.console.error('gl.getError()', gl.getError());\n\n        // if there is a program info log, log it\n        if (gl.getProgramInfoLog(program) !== '') {\n            window.console.warn('Pixi.js Warning: gl.getProgramInfoLog()', gl.getProgramInfoLog(program));\n        }\n\n        gl.deleteProgram(program);\n        program = null;\n    }\n\n    // clean up some shaders\n    gl.deleteShader(glVertShader);\n    gl.deleteShader(glFragShader);\n\n    return (this.program = program);\n};\n\nShader.prototype.syncUniforms = function () {\n    var gl = this.gl;\n\n    this.textureCount = 1;\n\n    for (var key in this.uniforms) {\n        var uniform = this.uniforms[key],\n            location = uniform._location,\n            value = uniform.value,\n            i, il;\n\n        switch (uniform.type) {\n            case 'i':\n            case '1i':\n                gl.uniform1i(location, value);\n                break;\n\n            case 'f':\n            case '1f':\n                gl.uniform1f(location, value);\n                break;\n\n            case '2f':\n                gl.uniform2f(location, value[0], value[1]);\n                break;\n\n            case '3f':\n                gl.uniform3f(location, value[0], value[1], value[2]);\n                break;\n\n            case '4f':\n                gl.uniform4f(location, value[0], value[1], value[2], value[3]);\n                break;\n\n            // a 2D Point object\n            case 'v2':\n                gl.uniform2f(location, value.x, value.y);\n                break;\n\n            // a 3D Point object\n            case 'v3':\n                gl.uniform3f(location, value.x, value.y, value.z);\n                break;\n\n            // a 4D Point object\n            case 'v4':\n                gl.uniform4f(location, value.x, value.y, value.z, value.w);\n                break;\n\n            case '1iv':\n                gl.uniform1iv(location, value);\n                break;\n\n            case '3iv':\n                gl.uniform3iv(location, value);\n                break;\n\n            case '1fv':\n                gl.uniform1fv(location, value);\n                break;\n\n            case '2fv':\n                gl.uniform2fv(location, value);\n                break;\n\n            case '3fv':\n                gl.uniform3fv(location, value);\n                break;\n\n            case '4fv':\n                gl.uniform4fv(location, value);\n                break;\n\n            case 'm2':\n            case 'mat2':\n            case 'Matrix2fv':\n                gl.uniformMatrix2fv(location, uniform.transpose, value);\n                break;\n\n            case 'm3':\n            case 'mat3':\n            case 'Matrix3fv':\n                gl.uniformMatrix3fv(location, uniform.transpose, value);\n                break;\n\n            case 'm4':\n            case 'mat4':\n            case 'Matrix4fv':\n                gl.uniformMatrix4fv(location, uniform.transpose, value);\n                break;\n\n            // a Color Value\n            case 'c':\n                if (typeof value === 'number') {\n                    value = utils.hex2rgb(value);\n                }\n\n                gl.uniform3f(location, value[0], value[1], value[2]);\n                break;\n\n            // flat array of integers (JS or typed array)\n            case 'iv1':\n                gl.uniform1iv(location, value);\n                break;\n\n            // flat array of integers with 3 x N size (JS or typed array)\n            case 'iv':\n                gl.uniform3iv(location, value);\n                break;\n\n            // flat array of floats (JS or typed array)\n            case 'fv1':\n                gl.uniform1fv(location, value);\n                break;\n\n            // flat array of floats with 3 x N size (JS or typed array)\n            case 'fv':\n                gl.uniform3fv(location, value);\n                break;\n\n            // array of 2D Point objects\n            case 'v2v':\n                if (!uniform._array) {\n                    uniform._array = new Float32Array(2 * value.length);\n                }\n\n                for (i = 0, il = value.length; i < il; ++i) {\n                    uniform._array[i * 2]       = value[i].x;\n                    uniform._array[i * 2 + 1]   = value[i].y;\n                }\n\n                gl.uniform2fv(location, uniform._array);\n                break;\n\n            // array of 3D Point objects\n            case 'v3v':\n                if (!uniform._array) {\n                    uniform._array = new Float32Array(3 * value.length);\n                }\n\n                for (i = 0, il = value.length; i < il; ++i) {\n                    uniform._array[i * 3]       = value[i].x;\n                    uniform._array[i * 3 + 1]   = value[i].y;\n                    uniform._array[i * 3 + 2]   = value[i].z;\n\n                }\n\n                gl.uniform3fv(location, uniform._array);\n                break;\n\n            // array of 4D Point objects\n            case 'v4v':\n                if (!uniform._array) {\n                    uniform._array = new Float32Array(4 * value.length);\n                }\n\n                for (i = 0, il = value.length; i < il; ++i) {\n                    uniform._array[i * 4]       = value[i].x;\n                    uniform._array[i * 4 + 1]   = value[i].y;\n                    uniform._array[i * 4 + 2]   = value[i].z;\n                    uniform._array[i * 4 + 3]   = value[i].w;\n\n                }\n\n                gl.uniform4fv(location, uniform._array);\n                break;\n\n            case 't':\n            case 'sampler2D':\n                if (!uniform.value || !uniform.value.baseTexture || !uniform.value.baseTexture.hasLoaded) {\n                    break;\n                }\n\n                // activate this texture\n                gl.activeTexture(gl['TEXTURE' + this.textureCount]);\n\n                // bind the texture\n                gl.bindTexture(gl.TEXTURE_2D, uniform.value.baseTexture._glTextures[gl.id]);\n\n                // set uniform to texture index\n                gl.uniform1i(uniform._location, this.textureCount);\n\n                // increment next texture id\n                this.textureCount++;\n\n                // initialize the texture if we haven't yet\n                if (!uniform._init) {\n                    this.initSampler2D(uniform);\n\n                    uniform._init = true;\n                }\n                break;\n\n            default:\n                window.console.warn('Pixi.js Shader Warning: Unknown uniform type: ' + uniform.type);\n        }\n    }\n};\n\n\n/**\n * Initialises a Sampler2D uniform (which may only be available later on after initUniforms once the texture has loaded)\n *\n */\nShader.prototype.initSampler2D = function (uniform) {\n    var gl = this.gl;\n\n    //  Extended texture data\n    if (uniform.textureData) {\n        var data = uniform.textureData;\n\n        // GLTexture = mag linear, min linear_mipmap_linear, wrap repeat + gl.generateMipmap(gl.TEXTURE_2D);\n        // GLTextureLinear = mag/min linear, wrap clamp\n        // GLTextureNearestRepeat = mag/min NEAREST, wrap repeat\n        // GLTextureNearest = mag/min nearest, wrap clamp\n        // AudioTexture = whatever + luminance + width 512, height 2, border 0\n        // KeyTexture = whatever + luminance + width 256, height 2, border 0\n\n        //  magFilter can be: gl.LINEAR, gl.LINEAR_MIPMAP_LINEAR or gl.NEAREST\n        //  wrapS/T can be: gl.CLAMP_TO_EDGE or gl.REPEAT\n\n        var magFilter = (data.magFilter) ? data.magFilter : gl.LINEAR;\n        var minFilter = (data.minFilter) ? data.minFilter : gl.LINEAR;\n        var wrapS = (data.wrapS) ? data.wrapS : gl.CLAMP_TO_EDGE;\n        var wrapT = (data.wrapT) ? data.wrapT : gl.CLAMP_TO_EDGE;\n        var format = (data.luminance) ? gl.LUMINANCE : gl.RGBA;\n\n        if (data.repeat) {\n            wrapS = gl.REPEAT;\n            wrapT = gl.REPEAT;\n        }\n\n        gl.pixelStorei(gl.UNPACK_FLIP_Y_WEBGL, !!data.flipY);\n\n        if (data.width) {\n            var width = (data.width) ? data.width : 512;\n            var height = (data.height) ? data.height : 2;\n            var border = (data.border) ? data.border : 0;\n\n            // void texImage2D(GLenum target, GLint level, GLenum internalformat, GLsizei width, GLsizei height, GLint border, GLenum format, GLenum type, ArrayBufferView? pixels);\n            gl.texImage2D(gl.TEXTURE_2D, 0, format, width, height, border, format, gl.UNSIGNED_BYTE, null);\n        }\n        else {\n            //  void texImage2D(GLenum target, GLint level, GLenum internalformat, GLenum format, GLenum type, ImageData? pixels);\n            gl.texImage2D(gl.TEXTURE_2D, 0, format, gl.RGBA, gl.UNSIGNED_BYTE, uniform.value.baseTexture.source);\n        }\n\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, magFilter);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, minFilter);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, wrapS);\n        gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, wrapT);\n    }\n};\n\n/**\n * Destroys the shader.\n *\n */\nShader.prototype.destroy = function () {\n    this.gl.deleteProgram(this.program);\n\n    this.gl = null;\n    this.uniforms = null;\n    this.attributes = null;\n\n    this.vertexSrc = null;\n    this.fragmentSrc = null;\n};\n\nShader.prototype._glCompile = function (type, src) {\n    var shader = this.gl.createShader(type);\n\n    if (Array.isArray(src)) {\n        debugger;\n    }\n\n    this.gl.shaderSource(shader, src);\n    this.gl.compileShader(shader);\n\n    if (!this.gl.getShaderParameter(shader, this.gl.COMPILE_STATUS)) {\n        window.console.log(this.gl.getShaderInfoLog(shader));\n        return null;\n    }\n\n    return shader;\n};\n","var Shader = require('./Shader');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nfunction StripShader(gl) {\n    Shader.call(this,\n        gl,\n        // vertex shader\n        [\n            'attribute vec2 aVertexPosition;',\n            'attribute vec2 aTextureCoord;',\n            // 'attribute vec4 aColor;',\n\n            'uniform mat3 translationMatrix;',\n            'uniform vec2 projectionVector;',\n            'uniform vec2 offsetVector;',\n\n            'varying vec2 vTextureCoord;',\n\n            'void main(void) {',\n            '   vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);',\n            '   v -= offsetVector.xyx;',\n            '   gl_Position = vec4( v.x / projectionVector.x -1.0, v.y / -projectionVector.y + 1.0 , 0.0, 1.0);',\n            '   vTextureCoord = aTextureCoord;',\n            '}'\n        ].join('\\n'),\n        // fragment shader\n        [\n            'precision mediump float;',\n\n            'uniform float alpha;',\n            'uniform sampler2D uSampler;',\n\n            'varying vec2 vTextureCoord;',\n\n            'void main(void) {',\n            '   gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * alpha;',\n            '}'\n        ].join('\\n'),\n        // custom uniforms\n        {\n            alpha:  { type: '1f', value: 0 },\n            translationMatrix: { type: 'mat3', value: new Float32Array(9) }\n        }\n    );\n}\n\nStripShader.prototype = Object.create(Shader.prototype);\nStripShader.prototype.constructor = StripShader;\nmodule.exports = StripShader;\n","var CONST = require('../../../const');\n\n/**\n * @class\n * @namespace PIXI\n * @param gl {WebGLContext} the current WebGL drawing context\n * @param width {number} the horizontal range of the filter\n * @param height {number} the vertical range of the filter\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n */\nfunction FilterTexture(gl, width, height, scaleMode) {\n    /**\n     * @member {WebGLContext}\n     */\n    this.gl = gl;\n\n    // next time to create a frame buffer and texture\n\n    /**\n     * @member {Any}\n     */\n    this.frameBuffer = gl.createFramebuffer();\n\n    /**\n     * @member {Any}\n     */\n    this.texture = gl.createTexture();\n\n    /**\n     * @member {number}\n     */\n    scaleMode = scaleMode || CONST.scaleModes.DEFAULT;\n\n    gl.bindTexture(gl.TEXTURE_2D,  this.texture);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MAG_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_MIN_FILTER, scaleMode === CONST.scaleModes.LINEAR ? gl.LINEAR : gl.NEAREST);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_S, gl.CLAMP_TO_EDGE);\n    gl.texParameteri(gl.TEXTURE_2D, gl.TEXTURE_WRAP_T, gl.CLAMP_TO_EDGE);\n    gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer );\n\n    gl.bindFramebuffer(gl.FRAMEBUFFER, this.frameBuffer );\n    gl.framebufferTexture2D(gl.FRAMEBUFFER, gl.COLOR_ATTACHMENT0, gl.TEXTURE_2D, this.texture, 0);\n\n    // required for masking a mask??\n    this.renderBuffer = gl.createRenderbuffer();\n    gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer);\n    gl.framebufferRenderbuffer(gl.FRAMEBUFFER, gl.DEPTH_STENCIL_ATTACHMENT, gl.RENDERBUFFER, this.renderBuffer);\n\n    // reset render buffer\n    gl.bindRenderbuffer(gl.RENDERBUFFER, null);\n\n    this.resize(width, height);\n}\n\nFilterTexture.prototype.constructor = FilterTexture;\nmodule.exports = FilterTexture;\n\n/**\n * Clears the filter texture.\n *\n */\nFilterTexture.prototype.clear = function () {\n    var gl = this.gl;\n\n    gl.clearColor(0,0,0, 0);\n    gl.clear(gl.COLOR_BUFFER_BIT);\n};\n\n/**\n * Resizes the texture to the specified width and height\n *\n * @param width {number} the new width of the texture\n * @param height {number} the new height of the texture\n */\nFilterTexture.prototype.resize = function (width, height) {\n    if (this.width === width && this.height === height) {\n        return;\n    }\n\n    this.width = width;\n    this.height = height;\n\n    var gl = this.gl;\n\n    gl.bindTexture(gl.TEXTURE_2D,  this.texture);\n    gl.texImage2D(gl.TEXTURE_2D, 0, gl.RGBA,  width , height , 0, gl.RGBA, gl.UNSIGNED_BYTE, null);\n    // update the stencil buffer width and height\n    gl.bindRenderbuffer(gl.RENDERBUFFER, this.renderBuffer);\n    gl.renderbufferStorage(gl.RENDERBUFFER, gl.DEPTH_STENCIL, width , height);\n\n    // reset render buffer\n    gl.bindRenderbuffer(gl.RENDERBUFFER, null);\n};\n\n/**\n * Destroys the filter texture.\n *\n */\nFilterTexture.prototype.destroy = function () {\n    var gl = this.gl;\n    gl.deleteFramebuffer( this.frameBuffer );\n    gl.deleteTexture( this.texture );\n\n    this.frameBuffer = null;\n    this.texture = null;\n};\n","/**\n * @author Mat Groves\n *\n * Big thanks to the very clever Matt DesLauriers <mattdesl> https://github.com/mattdesl/\n * for creating the original pixi version!\n *\n * Heavily inspired by LibGDX's WebGLSpriteBatch:\n * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java\n */\n\n/**\n * @class\n * @private\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this sprite batch works for.\n */\nfunction WebGLFastSpriteBatch(renderer) {\n    /**\n     * The renderer instance this sprite batch operates on.\n     *\n     * @member {WebGLRenderer}\n     */\n    this.renderer = renderer;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.vertSize = 10;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.maxSize = 6000;//Math.pow(2, 16) /  this.vertSize;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.size = this.maxSize;\n\n    //the total number of floats in our batch\n    var numVerts = this.size * 4 *  this.vertSize;\n\n    //the total number of indices in our batch\n    var numIndices = this.maxSize * 6;\n\n    /**\n     * Vertex data\n     *\n     * @member {Float32Array}\n     */\n    this.vertices = new Float32Array(numVerts);\n\n    /**\n     * Index data\n     *\n     * @member {Uint16Array}\n     */\n    this.indices = new Uint16Array(numIndices);\n\n    /**\n     *\n     *\n     * @member {object}\n     */\n    this.vertexBuffer = null;\n\n    /**\n     *\n     *\n     * @member {object}\n     */\n    this.indexBuffer = null;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.lastIndexCount = 0;\n\n    for (var i=0, j=0; i < numIndices; i += 6, j += 4) {\n        this.indices[i + 0] = j + 0;\n        this.indices[i + 1] = j + 1;\n        this.indices[i + 2] = j + 2;\n        this.indices[i + 3] = j + 0;\n        this.indices[i + 4] = j + 2;\n        this.indices[i + 5] = j + 3;\n    }\n\n    /**\n     *\n     *\n     * @member {boolean}\n     */\n    this.drawing = false;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.currentBatchSize = 0;\n\n    /**\n     *\n     *\n     * @member {BaseTexture}\n     */\n    this.currentBaseTexture = null;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.currentBlendMode = 0;\n\n    /**\n     *\n     *\n     * @member {object}\n     */\n    this.shader = null;\n\n    /**\n     *\n     *\n     * @member {Matrix}\n     */\n    this.matrix = null;\n\n    // listen for context and update necessary buffers\n    var self = this;\n    this.renderer.on('context', function () {\n        self.setupContext();\n    });\n}\n\nWebGLFastSpriteBatch.prototype.constructor = WebGLFastSpriteBatch;\nmodule.exports = WebGLFastSpriteBatch;\n\n/**\n * Sets the WebGL Context.\n *\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nWebGLFastSpriteBatch.prototype.setupContext = function () {\n    var gl = this.renderer.gl;\n\n    // create a couple of buffers\n    this.vertexBuffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n\n    // 65535 is max index, so 65535 / 6 = 10922.\n\n    //upload the index data\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW);\n};\n\n/**\n * @param spriteBatch {SpriteBatch} The SpriteBatch container to prepare for.\n */\nWebGLFastSpriteBatch.prototype.begin = function (spriteBatch) {\n    this.shader = this.renderer.shaderManager.fastShader;\n\n    this.matrix = spriteBatch.worldTransform.toArray(true);\n\n    this.start();\n};\n\n/**\n */\nWebGLFastSpriteBatch.prototype.end = function () {\n    this.flush();\n};\n\n/**\n * @param spriteBatch {SpriteBatch} The SpriteBatch container to render.\n */\nWebGLFastSpriteBatch.prototype.render = function (spriteBatch) {\n    var children = spriteBatch.children;\n    var sprite = children[0];\n\n    // if the uvs have not updated then no point rendering just yet!\n\n    // check texture.\n    if (!sprite.texture._uvs) {\n        return;\n    }\n\n    this.currentBaseTexture = sprite.texture.baseTexture;\n\n    // check blend mode\n    if (sprite.blendMode !== this.renderer.blendModeManager.currentBlendMode) {\n        this.flush();\n        this.renderer.blendModeManager.setBlendMode(sprite.blendMode);\n    }\n\n    for (var i=0,j= children.length; i<j; i++) {\n        this.renderSprite(children[i]);\n    }\n\n    this.flush();\n};\n\n/**\n * @param sprite {Sprite} The Sprite to render.\n */\nWebGLFastSpriteBatch.prototype.renderSprite = function (sprite) {\n    //sprite = children[i];\n    if (!sprite.visible) {\n        return;\n    }\n\n    // TODO trim??\n    if (sprite.texture.baseTexture !== this.currentBaseTexture) {\n        this.flush();\n        this.currentBaseTexture = sprite.texture.baseTexture;\n\n        if (!sprite.texture._uvs) {\n            return;\n        }\n    }\n\n    var uvs, vertices = this.vertices, width, height, w0, w1, h0, h1, index;\n\n    uvs = sprite.texture._uvs;\n\n    width = sprite.texture.frame.width;\n    height = sprite.texture.frame.height;\n\n    if (sprite.texture.trim) {\n        // if the sprite is trimmed then we need to add the extra space before transforming the sprite coords..\n        var trim = sprite.texture.trim;\n\n        w1 = trim.x - sprite.anchor.x * trim.width;\n        w0 = w1 + sprite.texture.crop.width;\n\n        h1 = trim.y - sprite.anchor.y * trim.height;\n        h0 = h1 + sprite.texture.crop.height;\n    }\n    else {\n        w0 = (sprite.texture.frame.width ) * (1-sprite.anchor.x);\n        w1 = (sprite.texture.frame.width ) * -sprite.anchor.x;\n\n        h0 = sprite.texture.frame.height * (1-sprite.anchor.y);\n        h1 = sprite.texture.frame.height * -sprite.anchor.y;\n    }\n\n    index = this.currentBatchSize * 4 * this.vertSize;\n\n    // xy\n    vertices[index++] = w1;\n    vertices[index++] = h1;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n    //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x0;\n    vertices[index++] = uvs.y1;\n    // color\n    vertices[index++] = sprite.alpha;\n\n\n    // xy\n    vertices[index++] = w0;\n    vertices[index++] = h1;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n     //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x1;\n    vertices[index++] = uvs.y1;\n    // color\n    vertices[index++] = sprite.alpha;\n\n\n    // xy\n    vertices[index++] = w0;\n    vertices[index++] = h0;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n     //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x2;\n    vertices[index++] = uvs.y2;\n    // color\n    vertices[index++] = sprite.alpha;\n\n\n\n\n    // xy\n    vertices[index++] = w1;\n    vertices[index++] = h0;\n\n    vertices[index++] = sprite.position.x;\n    vertices[index++] = sprite.position.y;\n\n    //scale\n    vertices[index++] = sprite.scale.x;\n    vertices[index++] = sprite.scale.y;\n\n     //rotation\n    vertices[index++] = sprite.rotation;\n\n    // uv\n    vertices[index++] = uvs.x3;\n    vertices[index++] = uvs.y3;\n    // color\n    vertices[index++] = sprite.alpha;\n\n    // increment the batchs\n    this.currentBatchSize++;\n\n    if (this.currentBatchSize >= this.size) {\n        this.flush();\n    }\n};\n\n/**\n *\n */\nWebGLFastSpriteBatch.prototype.flush = function () {\n    // If the batch is length 0 then return as there is nothing to draw\n    if (this.currentBatchSize === 0) {\n        return;\n    }\n\n    var gl = this.renderer.gl;\n\n    // bind the current texture\n    if (!this.currentBaseTexture._glTextures[gl.id]) {\n        this.renderer.updateTexture(this.currentBaseTexture, gl);\n    }\n    //TODO-SHOUD THIS BE ELSE??!?!?!\n    else {\n        gl.bindTexture(gl.TEXTURE_2D, this.currentBaseTexture._glTextures[gl.id]);\n    }\n\n    // upload the verts to the buffer\n\n    if (this.currentBatchSize > ( this.size * 0.5 ) ) {\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices);\n    }\n    else {\n        var view = this.vertices.subarray(0, this.currentBatchSize * 4 * this.vertSize);\n\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, view);\n    }\n\n    // now draw those suckas!\n    gl.drawElements(gl.TRIANGLES, this.currentBatchSize * 6, gl.UNSIGNED_SHORT, 0);\n\n    // then reset the batch!\n    this.currentBatchSize = 0;\n\n    // increment the draw count\n    this.renderer.drawCount++;\n};\n\n\n/**\n * Ends the batch and flushes\n *\n */\nWebGLFastSpriteBatch.prototype.stop = function () {\n    this.flush();\n};\n\n/**\n *\n */\nWebGLFastSpriteBatch.prototype.start = function () {\n    var gl = this.renderer.gl;\n\n    // bind the main texture\n    gl.activeTexture(gl.TEXTURE0);\n\n    // bind the buffers\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n\n    // set the projection\n    var projection = this.renderer.projection;\n    gl.uniform2f(this.shader.projectionVector, projection.x, projection.y);\n\n    // set the matrix\n    gl.uniformMatrix3fv(this.shader.uMatrix, false, this.matrix);\n\n    // set the pointers\n    var stride =  this.vertSize * 4;\n\n    gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0);\n    gl.vertexAttribPointer(this.shader.attributes.aPositionCoord, 2, gl.FLOAT, false, stride, 2 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aScale, 2, gl.FLOAT, false, stride, 4 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aRotation, 1, gl.FLOAT, false, stride, 6 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 7 * 4);\n    gl.vertexAttribPointer(this.shader.attributes.aColor, 1, gl.FLOAT, false, stride, 9 * 4);\n};\n","var utils = require('../../../utils'),\n    math = require('../../../math'),\n    CONST = require('../../../const'),\n    WebGLGraphicsData = require('./WebGLGraphicsData');\n\n/**\n * A set of functions used by the webGL renderer to draw the primitive graphics data\n *\n * @namespace PIXI\n * @private\n */\nvar WebGLGraphics = module.exports = {};\n\n/**\n * Renders the graphics object\n *\n * @static\n * @private\n * @param graphics {Graphics}\n * @param renderer {WebGLRenderer}\n */\nWebGLGraphics.renderGraphics = function (graphics, renderer) {//projection, offset) {\n    var gl = renderer.gl;\n\n    var projection = renderer.projection,\n        offset = renderer.offset,\n        shader = renderer.shaderManager.primitiveShader,\n        webGLData;\n\n    if (graphics.dirty) {\n        WebGLGraphics.updateGraphics(graphics, gl);\n    }\n\n    var webGL = graphics._webGL[gl.id];\n\n    // This  could be speeded up for sure!\n\n    for (var i = 0; i < webGL.data.length; i++) {\n        if (webGL.data[i].mode === 1) {\n            webGLData = webGL.data[i];\n\n            renderer.stencilManager.pushStencil(graphics, webGLData, renderer);\n\n            // render quad..\n            gl.drawElements(gl.TRIANGLE_FAN, 4, gl.UNSIGNED_SHORT, ( webGLData.indices.length - 4 ) * 2 );\n\n            renderer.stencilManager.popStencil(graphics, webGLData, renderer);\n        }\n        else {\n            webGLData = webGL.data[i];\n\n\n            renderer.shaderManager.setShader( shader );//activatePrimitiveShader();\n            shader = renderer.shaderManager.primitiveShader;\n            gl.uniformMatrix3fv(shader.translationMatrix, false, graphics.worldTransform.toArray(true));\n\n            gl.uniform1f(shader.flipY, 1);\n\n            gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n            gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n\n            gl.uniform3fv(shader.tintColor, utils.hex2rgb(graphics.tint));\n\n            gl.uniform1f(shader.alpha, graphics.worldAlpha);\n\n\n            gl.bindBuffer(gl.ARRAY_BUFFER, webGLData.buffer);\n\n            gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 4 * 6, 0);\n            gl.vertexAttribPointer(shader.aColor, 4, gl.FLOAT, false,4 * 6, 2 * 4);\n\n            // set the index buffer!\n            gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, webGLData.indexBuffer);\n            gl.drawElements(gl.TRIANGLE_STRIP,  webGLData.indices.length, gl.UNSIGNED_SHORT, 0 );\n        }\n    }\n};\n\n/**\n * Updates the graphics object\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object to update\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nWebGLGraphics.updateGraphics = function (graphics, gl) {\n    // get the contexts graphics object\n    var webGL = graphics._webGL[gl.id];\n\n    // if the graphics object does not exist in the webGL context time to create it!\n    if (!webGL) {\n        webGL = graphics._webGL[gl.id] = {lastIndex:0, data:[], gl:gl};\n    }\n\n    // flag the graphics as not dirty as we are about to update it...\n    graphics.dirty = false;\n\n    var i;\n\n    // if the user cleared the graphics object we will need to clear every object\n    if (graphics.clearDirty) {\n        graphics.clearDirty = false;\n\n        // lop through and return all the webGLDatas to the object pool so than can be reused later on\n        for (i = 0; i < webGL.data.length; i++) {\n            var graphicsData = webGL.data[i];\n            graphicsData.reset();\n            WebGLGraphics.graphicsDataPool.push( graphicsData );\n        }\n\n        // clear the array and reset the index..\n        webGL.data = [];\n        webGL.lastIndex = 0;\n    }\n\n    var webGLData;\n\n    // loop through the graphics datas and construct each one..\n    // if the object is a complex fill then the new stencil buffer technique will be used\n    // other wise graphics objects will be pushed into a batch..\n    for (i = webGL.lastIndex; i < graphics.graphicsData.length; i++) {\n        var data = graphics.graphicsData[i];\n\n        if (data.type === CONST.SHAPES.POLY) {\n            // need to add the points the the graphics object..\n            data.points = data.shape.points.slice();\n            if (data.shape.closed) {\n                // close the poly if the value is true!\n                if (data.points[0] !== data.points[data.points.length-2] || data.points[1] !== data.points[data.points.length-1]) {\n                    data.points.push(data.points[0], data.points[1]);\n                }\n            }\n\n            // MAKE SURE WE HAVE THE CORRECT TYPE..\n            if (data.fill) {\n                if (data.points.length >= 6) {\n                    if (data.points.length < 6 * 2) {\n                        webGLData = WebGLGraphics.switchMode(webGL, 0);\n\n                        var canDrawUsingSimple = WebGLGraphics.buildPoly(data, webGLData);\n                   //     console.log(canDrawUsingSimple);\n\n                        if (!canDrawUsingSimple) {\n                        //    console.log(\"<>>>\")\n                            webGLData = WebGLGraphics.switchMode(webGL, 1);\n                            WebGLGraphics.buildComplexPoly(data, webGLData);\n                        }\n\n                    }\n                    else {\n                        webGLData = WebGLGraphics.switchMode(webGL, 1);\n                        WebGLGraphics.buildComplexPoly(data, webGLData);\n                    }\n                }\n            }\n\n            if (data.lineWidth > 0) {\n                webGLData = WebGLGraphics.switchMode(webGL, 0);\n                WebGLGraphics.buildLine(data, webGLData);\n\n            }\n        }\n        else {\n            webGLData = WebGLGraphics.switchMode(webGL, 0);\n\n            if (data.type === CONST.SHAPES.RECT) {\n                WebGLGraphics.buildRectangle(data, webGLData);\n            }\n            else if (data.type === CONST.SHAPES.CIRC || data.type === CONST.SHAPES.ELIP) {\n                WebGLGraphics.buildCircle(data, webGLData);\n            }\n            else if (data.type === CONST.SHAPES.RREC) {\n                WebGLGraphics.buildRoundedRectangle(data, webGLData);\n            }\n        }\n\n        webGL.lastIndex++;\n    }\n\n    // upload all the dirty data...\n    for (i = 0; i < webGL.data.length; i++) {\n        webGLData = webGL.data[i];\n\n        if (webGLData.dirty) {\n            webGLData.upload();\n        }\n    }\n};\n\n/**\n * @static\n * @private\n * @param webGL {WebGLContext}\n * @param type {number}\n */\nWebGLGraphics.switchMode = function (webGL, type) {\n    var webGLData;\n\n    if (!webGL.data.length) {\n        webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl);\n        webGLData.mode = type;\n        webGL.data.push(webGLData);\n    }\n    else {\n        webGLData = webGL.data[webGL.data.length-1];\n\n        if (webGLData.mode !== type || type === 1) {\n            webGLData = WebGLGraphics.graphicsDataPool.pop() || new WebGLGraphicsData(webGL.gl);\n            webGLData.mode = type;\n            webGL.data.push(webGLData);\n        }\n    }\n\n    webGLData.dirty = true;\n\n    return webGLData;\n};\n\n/**\n * Builds a rectangle to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildRectangle = function (graphicsData, webGLData) {\n    // --- //\n    // need to convert points to a nice regular data\n    //\n    var rectData = graphicsData.shape;\n    var x = rectData.x;\n    var y = rectData.y;\n    var width = rectData.width;\n    var height = rectData.height;\n\n    if (graphicsData.fill) {\n        var color = utils.hex2rgb(graphicsData.fillColor);\n        var alpha = graphicsData.fillAlpha;\n\n        var r = color[0] * alpha;\n        var g = color[1] * alpha;\n        var b = color[2] * alpha;\n\n        var verts = webGLData.points;\n        var indices = webGLData.indices;\n\n        var vertPos = verts.length/6;\n\n        // start\n        verts.push(x, y);\n        verts.push(r, g, b, alpha);\n\n        verts.push(x + width, y);\n        verts.push(r, g, b, alpha);\n\n        verts.push(x , y + height);\n        verts.push(r, g, b, alpha);\n\n        verts.push(x + width, y + height);\n        verts.push(r, g, b, alpha);\n\n        // insert 2 dead triangles..\n        indices.push(vertPos, vertPos, vertPos+1, vertPos+2, vertPos+3, vertPos+3);\n    }\n\n    if (graphicsData.lineWidth) {\n        var tempPoints = graphicsData.points;\n\n        graphicsData.points = [x, y,\n                  x + width, y,\n                  x + width, y + height,\n                  x, y + height,\n                  x, y];\n\n\n        WebGLGraphics.buildLine(graphicsData, webGLData);\n\n        graphicsData.points = tempPoints;\n    }\n};\n\n/**\n * Builds a rounded rectangle to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildRoundedRectangle = function (graphicsData, webGLData) {\n    var rrectData = graphicsData.shape;\n    var x = rrectData.x;\n    var y = rrectData.y;\n    var width = rrectData.width;\n    var height = rrectData.height;\n\n    var radius = rrectData.radius;\n\n    var recPoints = [];\n    recPoints.push(x, y + radius);\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x, y + height - radius, x, y + height, x + radius, y + height));\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width - radius, y + height, x + width, y + height, x + width, y + height - radius));\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + width, y + radius, x + width, y, x + width - radius, y));\n    recPoints = recPoints.concat(WebGLGraphics.quadraticBezierCurve(x + radius, y, x, y, x, y + radius));\n\n    if (graphicsData.fill) {\n        var color = utils.hex2rgb(graphicsData.fillColor);\n        var alpha = graphicsData.fillAlpha;\n\n        var r = color[0] * alpha;\n        var g = color[1] * alpha;\n        var b = color[2] * alpha;\n\n        var verts = webGLData.points;\n        var indices = webGLData.indices;\n\n        var vecPos = verts.length/6;\n\n        var triangles = utils.PolyK.Triangulate(recPoints);\n\n        //\n\n        var i = 0;\n        for (i = 0; i < triangles.length; i+=3) {\n            indices.push(triangles[i] + vecPos);\n            indices.push(triangles[i] + vecPos);\n            indices.push(triangles[i+1] + vecPos);\n            indices.push(triangles[i+2] + vecPos);\n            indices.push(triangles[i+2] + vecPos);\n        }\n\n\n        for (i = 0; i < recPoints.length; i++) {\n            verts.push(recPoints[i], recPoints[++i], r, g, b, alpha);\n        }\n    }\n\n    if (graphicsData.lineWidth) {\n        var tempPoints = graphicsData.points;\n\n        graphicsData.points = recPoints;\n\n        WebGLGraphics.buildLine(graphicsData, webGLData);\n\n        graphicsData.points = tempPoints;\n    }\n};\n\n/**\n * Calculate the points for a quadratic bezier curve. (helper function..)\n * Based on: https://stackoverflow.com/questions/785097/how-do-i-implement-a-bezier-curve-in-c\n *\n * @static\n * @private\n * @param fromX {number} Origin point x\n * @param fromY {number} Origin point x\n * @param cpX {number} Control point x\n * @param cpY {number} Control point y\n * @param toX {number} Destination point x\n * @param toY {number} Destination point y\n * @return {number[]}\n */\nWebGLGraphics.quadraticBezierCurve = function (fromX, fromY, cpX, cpY, toX, toY) {\n\n    var xa,\n        ya,\n        xb,\n        yb,\n        x,\n        y,\n        n = 20,\n        points = [];\n\n    function getPt(n1 , n2, perc) {\n        var diff = n2 - n1;\n\n        return n1 + ( diff * perc );\n    }\n\n    var j = 0;\n    for (var i = 0; i <= n; i++ ) {\n        j = i / n;\n\n        // The Green Line\n        xa = getPt( fromX , cpX , j );\n        ya = getPt( fromY , cpY , j );\n        xb = getPt( cpX , toX , j );\n        yb = getPt( cpY , toY , j );\n\n        // The Black Dot\n        x = getPt( xa , xb , j );\n        y = getPt( ya , yb , j );\n\n        points.push(x, y);\n    }\n    return points;\n};\n\n/**\n * Builds a circle to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object to draw\n * @param webGLData {object}\n */\nWebGLGraphics.buildCircle = function (graphicsData, webGLData) {\n    // need to convert points to a nice regular data\n    var circleData = graphicsData.shape;\n    var x = circleData.x;\n    var y = circleData.y;\n    var width;\n    var height;\n\n    // TODO - bit hacky??\n    if (graphicsData.type === CONST.SHAPES.CIRC) {\n        width = circleData.radius;\n        height = circleData.radius;\n    }\n    else {\n        width = circleData.width;\n        height = circleData.height;\n    }\n\n    var totalSegs = 40;\n    var seg = (Math.PI * 2) / totalSegs ;\n\n    var i = 0;\n\n    if (graphicsData.fill) {\n        var color = utils.hex2rgb(graphicsData.fillColor);\n        var alpha = graphicsData.fillAlpha;\n\n        var r = color[0] * alpha;\n        var g = color[1] * alpha;\n        var b = color[2] * alpha;\n\n        var verts = webGLData.points;\n        var indices = webGLData.indices;\n\n        var vecPos = verts.length/6;\n\n        indices.push(vecPos);\n\n        for (i = 0; i < totalSegs + 1 ; i++) {\n            verts.push(x,y, r, g, b, alpha);\n\n            verts.push(x + Math.sin(seg * i) * width,\n                       y + Math.cos(seg * i) * height,\n                       r, g, b, alpha);\n\n            indices.push(vecPos++, vecPos++);\n        }\n\n        indices.push(vecPos-1);\n    }\n\n    if (graphicsData.lineWidth) {\n        var tempPoints = graphicsData.points;\n\n        graphicsData.points = [];\n\n        for (i = 0; i < totalSegs + 1; i++) {\n            graphicsData.points.push(x + Math.sin(seg * i) * width,\n                                     y + Math.cos(seg * i) * height);\n        }\n\n        WebGLGraphics.buildLine(graphicsData, webGLData);\n\n        graphicsData.points = tempPoints;\n    }\n};\n\n/**\n * Builds a line to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildLine = function (graphicsData, webGLData) {\n    // TODO OPTIMISE!\n    var i = 0;\n    var points = graphicsData.points;\n\n    if (points.length === 0) {\n        return;\n    }\n\n    // if the line width is an odd number add 0.5 to align to a whole pixel\n    if (graphicsData.lineWidth%2) {\n        for (i = 0; i < points.length; i++) {\n            points[i] += 0.5;\n        }\n    }\n\n    // get first and last point.. figure out the middle!\n    var firstPoint = new math.Point(points[0], points[1]);\n    var lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]);\n\n    // if the first point is the last point - gonna have issues :)\n    if (firstPoint.x === lastPoint.x && firstPoint.y === lastPoint.y) {\n        // need to clone as we are going to slightly modify the shape..\n        points = points.slice();\n\n        points.pop();\n        points.pop();\n\n        lastPoint = new math.Point(points[points.length - 2], points[points.length - 1]);\n\n        var midPointX = lastPoint.x + (firstPoint.x - lastPoint.x) *0.5;\n        var midPointY = lastPoint.y + (firstPoint.y - lastPoint.y) *0.5;\n\n        points.unshift(midPointX, midPointY);\n        points.push(midPointX, midPointY);\n    }\n\n    var verts = webGLData.points;\n    var indices = webGLData.indices;\n    var length = points.length / 2;\n    var indexCount = points.length;\n    var indexStart = verts.length/6;\n\n    // DRAW the Line\n    var width = graphicsData.lineWidth / 2;\n\n    // sort color\n    var color = utils.hex2rgb(graphicsData.lineColor);\n    var alpha = graphicsData.lineAlpha;\n    var r = color[0] * alpha;\n    var g = color[1] * alpha;\n    var b = color[2] * alpha;\n\n    var px, py, p1x, p1y, p2x, p2y, p3x, p3y;\n    var perpx, perpy, perp2x, perp2y, perp3x, perp3y;\n    var a1, b1, c1, a2, b2, c2;\n    var denom, pdist, dist;\n\n    p1x = points[0];\n    p1y = points[1];\n\n    p2x = points[2];\n    p2y = points[3];\n\n    perpx = -(p1y - p2y);\n    perpy =  p1x - p2x;\n\n    dist = Math.sqrt(perpx*perpx + perpy*perpy);\n\n    perpx /= dist;\n    perpy /= dist;\n    perpx *= width;\n    perpy *= width;\n\n    // start\n    verts.push(p1x - perpx , p1y - perpy,\n                r, g, b, alpha);\n\n    verts.push(p1x + perpx , p1y + perpy,\n                r, g, b, alpha);\n\n    for (i = 1; i < length-1; i++) {\n        p1x = points[(i-1)*2];\n        p1y = points[(i-1)*2 + 1];\n\n        p2x = points[(i)*2];\n        p2y = points[(i)*2 + 1];\n\n        p3x = points[(i+1)*2];\n        p3y = points[(i+1)*2 + 1];\n\n        perpx = -(p1y - p2y);\n        perpy = p1x - p2x;\n\n        dist = Math.sqrt(perpx*perpx + perpy*perpy);\n        perpx /= dist;\n        perpy /= dist;\n        perpx *= width;\n        perpy *= width;\n\n        perp2x = -(p2y - p3y);\n        perp2y = p2x - p3x;\n\n        dist = Math.sqrt(perp2x*perp2x + perp2y*perp2y);\n        perp2x /= dist;\n        perp2y /= dist;\n        perp2x *= width;\n        perp2y *= width;\n\n        a1 = (-perpy + p1y) - (-perpy + p2y);\n        b1 = (-perpx + p2x) - (-perpx + p1x);\n        c1 = (-perpx + p1x) * (-perpy + p2y) - (-perpx + p2x) * (-perpy + p1y);\n        a2 = (-perp2y + p3y) - (-perp2y + p2y);\n        b2 = (-perp2x + p2x) - (-perp2x + p3x);\n        c2 = (-perp2x + p3x) * (-perp2y + p2y) - (-perp2x + p2x) * (-perp2y + p3y);\n\n        denom = a1*b2 - a2*b1;\n\n        if (Math.abs(denom) < 0.1 ) {\n\n            denom+=10.1;\n            verts.push(p2x - perpx , p2y - perpy,\n                r, g, b, alpha);\n\n            verts.push(p2x + perpx , p2y + perpy,\n                r, g, b, alpha);\n\n            continue;\n        }\n\n        px = (b1*c2 - b2*c1)/denom;\n        py = (a2*c1 - a1*c2)/denom;\n\n\n        pdist = (px -p2x) * (px -p2x) + (py -p2y) + (py -p2y);\n\n\n        if (pdist > 140 * 140) {\n            perp3x = perpx - perp2x;\n            perp3y = perpy - perp2y;\n\n            dist = Math.sqrt(perp3x*perp3x + perp3y*perp3y);\n            perp3x /= dist;\n            perp3y /= dist;\n            perp3x *= width;\n            perp3y *= width;\n\n            verts.push(p2x - perp3x, p2y -perp3y);\n            verts.push(r, g, b, alpha);\n\n            verts.push(p2x + perp3x, p2y +perp3y);\n            verts.push(r, g, b, alpha);\n\n            verts.push(p2x - perp3x, p2y -perp3y);\n            verts.push(r, g, b, alpha);\n\n            indexCount++;\n        }\n        else {\n\n            verts.push(px , py);\n            verts.push(r, g, b, alpha);\n\n            verts.push(p2x - (px-p2x), p2y - (py - p2y));\n            verts.push(r, g, b, alpha);\n        }\n    }\n\n    p1x = points[(length-2)*2];\n    p1y = points[(length-2)*2 + 1];\n\n    p2x = points[(length-1)*2];\n    p2y = points[(length-1)*2 + 1];\n\n    perpx = -(p1y - p2y);\n    perpy = p1x - p2x;\n\n    dist = Math.sqrt(perpx*perpx + perpy*perpy);\n    perpx /= dist;\n    perpy /= dist;\n    perpx *= width;\n    perpy *= width;\n\n    verts.push(p2x - perpx , p2y - perpy);\n    verts.push(r, g, b, alpha);\n\n    verts.push(p2x + perpx , p2y + perpy);\n    verts.push(r, g, b, alpha);\n\n    indices.push(indexStart);\n\n    for (i = 0; i < indexCount; i++) {\n        indices.push(indexStart++);\n    }\n\n    indices.push(indexStart-1);\n};\n\n/**\n * Builds a complex polygon to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildComplexPoly = function (graphicsData, webGLData) {\n    //TODO - no need to copy this as it gets turned into a FLoat32Array anyways..\n    var points = graphicsData.points.slice();\n\n    if (points.length < 6) {\n        return;\n    }\n\n    // get first and last point.. figure out the middle!\n    var indices = webGLData.indices;\n    webGLData.points = points;\n    webGLData.alpha = graphicsData.fillAlpha;\n    webGLData.color = utils.hex2rgb(graphicsData.fillColor);\n\n    // calclate the bounds..\n    var minX = Infinity;\n    var maxX = -Infinity;\n\n    var minY = Infinity;\n    var maxY = -Infinity;\n\n    var x,y;\n\n    // get size..\n    for (var i = 0; i < points.length; i+=2) {\n        x = points[i];\n        y = points[i+1];\n\n        minX = x < minX ? x : minX;\n        maxX = x > maxX ? x : maxX;\n\n        minY = y < minY ? y : minY;\n        maxY = y > maxY ? y : maxY;\n    }\n\n    // add a quad to the end cos there is no point making another buffer!\n    points.push(minX, minY,\n                maxX, minY,\n                maxX, maxY,\n                minX, maxY);\n\n    // push a quad onto the end..\n\n    //TODO - this aint needed!\n    var length = points.length / 2;\n    for (i = 0; i < length; i++) {\n        indices.push( i );\n    }\n\n};\n\n/**\n * Builds a polygon to draw\n *\n * @static\n * @private\n * @param graphicsData {Graphics} The graphics object containing all the necessary properties\n * @param webGLData {object}\n */\nWebGLGraphics.buildPoly = function (graphicsData, webGLData) {\n    var points = graphicsData.points;\n\n    if (points.length < 6) {\n        return;\n    }\n    // get first and last point.. figure out the middle!\n    var verts = webGLData.points;\n    var indices = webGLData.indices;\n\n    var length = points.length / 2;\n\n    // sort color\n    var color = utils.hex2rgb(graphicsData.fillColor);\n    var alpha = graphicsData.fillAlpha;\n    var r = color[0] * alpha;\n    var g = color[1] * alpha;\n    var b = color[2] * alpha;\n\n    var triangles = utils.PolyK.Triangulate(points);\n\n    if (!triangles) {\n        return false;\n    }\n\n    var vertPos = verts.length / 6;\n\n    var i = 0;\n\n    for (i = 0; i < triangles.length; i+=3) {\n        indices.push(triangles[i] + vertPos);\n        indices.push(triangles[i] + vertPos);\n        indices.push(triangles[i+1] + vertPos);\n        indices.push(triangles[i+2] +vertPos);\n        indices.push(triangles[i+2] + vertPos);\n    }\n\n    for (i = 0; i < length; i++) {\n        verts.push(points[i * 2], points[i * 2 + 1],\n                   r, g, b, alpha);\n    }\n\n    return true;\n};\n\nWebGLGraphics.graphicsDataPool = [];\n","/**\n * @class\n * @private\n */\nfunction WebGLGraphicsData(gl) {\n    this.gl = gl;\n\n    //TODO does this need to be split before uploding??\n    this.color = [0, 0, 0]; // color split!\n    this.points = [];\n    this.indices = [];\n    this.buffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n    this.mode = 1;\n    this.alpha = 1;\n    this.dirty = true;\n}\n\nWebGLGraphicsData.prototype.constructor = WebGLGraphicsData;\nmodule.exports = WebGLGraphicsData;\n\n/**\n *\n */\nWebGLGraphicsData.prototype.reset = function () {\n    this.points.length = 0;\n    this.indices.length = 0;\n};\n\n/**\n *\n */\nWebGLGraphicsData.prototype.upload = function () {\n    var gl = this.gl;\n\n//    this.lastIndex = graphics.graphicsData.length;\n    this.glPoints = new Float32Array(this.points);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.buffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.glPoints, gl.STATIC_DRAW);\n\n    this.glIndicies = new Uint16Array(this.indices);\n\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.glIndicies, gl.STATIC_DRAW);\n\n    this.dirty = false;\n};\n","var TextureUvs = require('../../../textures/TextureUvs'),\n    Shader = require('../shaders/Shader');\n\n/**\n * @author Mat Groves\n *\n * Big thanks to the very clever Matt DesLauriers <mattdesl> https://github.com/mattdesl/\n * for creating the original pixi version!\n * Also a thanks to https://github.com/bchevalier for tweaking the tint and alpha so that they now share 4 bytes on the vertex buffer\n *\n * Heavily inspired by LibGDX's WebGLSpriteBatch:\n * https://github.com/libgdx/libgdx/blob/master/gdx/src/com/badlogic/gdx/graphics/g2d/WebGLSpriteBatch.java\n */\n\n/**\n *\n * @class\n * @private\n * @namespace PIXI\n * @param renderer {WebGLRenderer} The renderer this sprite batch works for.\n */\nfunction WebGLSpriteBatch(renderer) {\n    /**\n     *\n     *\n     * @member {WebGLRenderer}\n     */\n    this.renderer = renderer;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.vertSize = 5;\n\n    /**\n     * The number of images in the SpriteBatch before it flushes.\n     *\n     * @member {number}\n     */\n    this.size = 2000;//Math.pow(2, 16) /  this.vertSize;\n\n    // the total number of bytes in our batch\n    var numVerts = this.size * 4 * 4 * this.vertSize;\n    // the total number of indices in our batch\n    var numIndices = this.size * 6;\n\n    /**\n     * Holds the vertices\n     *\n     * @member {ArrayBuffer}\n     */\n    this.vertices = new ArrayBuffer(numVerts);\n\n    /**\n     * View on the vertices as a Float32Array\n     *\n     * @member {Float32Array}\n     */\n    this.positions = new Float32Array(this.vertices);\n\n    /**\n     * View on the vertices as a Uint32Array\n     *\n     * @member {Uint32Array}\n     */\n    this.colors = new Uint32Array(this.vertices);\n\n    /**\n     * Holds the indices\n     *\n     * @member {Uint16Array}\n     */\n    this.indices = new Uint16Array(numIndices);\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.lastIndexCount = 0;\n\n    for (var i=0, j=0; i < numIndices; i += 6, j += 4) {\n        this.indices[i + 0] = j + 0;\n        this.indices[i + 1] = j + 1;\n        this.indices[i + 2] = j + 2;\n        this.indices[i + 3] = j + 0;\n        this.indices[i + 4] = j + 2;\n        this.indices[i + 5] = j + 3;\n    }\n\n    /**\n     *\n     *\n     * @member {boolean}\n     */\n    this.drawing = false;\n\n    /**\n     *\n     *\n     * @member {number}\n     */\n    this.currentBatchSize = 0;\n\n    /**\n     *\n     *\n     * @member {BaseTexture}\n     */\n    this.currentBaseTexture = null;\n\n    /**\n     *\n     *\n     * @member {boolean}\n     */\n    this.dirty = true;\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.textures = [];\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.blendModes = [];\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.shaders = [];\n\n    /**\n     *\n     *\n     * @member {Array}\n     */\n    this.sprites = [];\n\n    /**\n     * The default shader that is used if a sprite doesn't have a more specific one.\n     *\n     * @member {Shader}\n     */\n    this.shader = null;\n\n    // listen for context and update necessary buffers\n    var self = this;\n    this.renderer.on('context', function () {\n        self.setupContext();\n    });\n}\n\nWebGLSpriteBatch.prototype.constructor = WebGLSpriteBatch;\nmodule.exports = WebGLSpriteBatch;\n\n/**\n * @param gl {WebGLContext} the current WebGL drawing context\n */\nWebGLSpriteBatch.prototype.setupContext = function () {\n    var gl = this.renderer.gl;\n\n    // setup default shader\n    this.shader = new Shader(gl);\n\n    // create a couple of buffers\n    this.vertexBuffer = gl.createBuffer();\n    this.indexBuffer = gl.createBuffer();\n\n    // 65535 is max index, so 65535 / 6 = 10922.\n\n    //upload the index data\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW);\n\n    this.currentBlendMode = 99999;\n};\n\n/**\n *\n */\nWebGLSpriteBatch.prototype.begin = function () {\n    // this.shader = this.renderer.shaderManager.defaultShader;\n\n    this.start();\n};\n\n/**\n */\nWebGLSpriteBatch.prototype.end = function () {\n    this.flush();\n};\n\n/**\n * @param sprite {Sprite} the sprite to render when using this spritebatch\n */\nWebGLSpriteBatch.prototype.render = function (sprite) {\n    var texture = sprite.texture;\n\n    //TODO set blend modes..\n    // check texture..\n    if (this.currentBatchSize >= this.size) {\n        this.flush();\n        this.currentBaseTexture = texture.baseTexture;\n    }\n\n    // get the uvs for the texture\n    var uvs = texture._uvs;\n\n    // if the uvs have not updated then no point rendering just yet!\n    if (!uvs) {\n        return;\n    }\n\n    // TODO trim??\n    var aX = sprite.anchor.x;\n    var aY = sprite.anchor.y;\n\n    var w0, w1, h0, h1;\n\n    if (texture.trim) {\n        // if the sprite is trimmed then we need to add the extra space before transforming the sprite coords..\n        var trim = texture.trim;\n\n        w1 = trim.x - aX * trim.width;\n        w0 = w1 + texture.crop.width;\n\n        h1 = trim.y - aY * trim.height;\n        h0 = h1 + texture.crop.height;\n\n    }\n    else {\n        w0 = (texture.frame.width ) * (1-aX);\n        w1 = (texture.frame.width ) * -aX;\n\n        h0 = texture.frame.height * (1-aY);\n        h1 = texture.frame.height * -aY;\n    }\n\n    var index = this.currentBatchSize * 4 * this.vertSize;\n\n    var resolution = texture.baseTexture.resolution;\n\n    var worldTransform = sprite.worldTransform;\n\n    var a = worldTransform.a / resolution;\n    var b = worldTransform.b / resolution;\n    var c = worldTransform.c / resolution;\n    var d = worldTransform.d / resolution;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var colors = this.colors;\n    var positions = this.positions;\n\n    if (this.renderer.roundPixels) {\n        // xy\n        positions[index] = a * w1 + c * h1 + tx | 0;\n        positions[index+1] = d * h1 + b * w1 + ty | 0;\n\n        // xy\n        positions[index+5] = a * w0 + c * h1 + tx | 0;\n        positions[index+6] = d * h1 + b * w0 + ty | 0;\n\n         // xy\n        positions[index+10] = a * w0 + c * h0 + tx | 0;\n        positions[index+11] = d * h0 + b * w0 + ty | 0;\n\n        // xy\n        positions[index+15] = a * w1 + c * h0 + tx | 0;\n        positions[index+16] = d * h0 + b * w1 + ty | 0;\n    }\n    else {\n        // xy\n        positions[index] = a * w1 + c * h1 + tx;\n        positions[index+1] = d * h1 + b * w1 + ty;\n\n        // xy\n        positions[index+5] = a * w0 + c * h1 + tx;\n        positions[index+6] = d * h1 + b * w0 + ty;\n\n         // xy\n        positions[index+10] = a * w0 + c * h0 + tx;\n        positions[index+11] = d * h0 + b * w0 + ty;\n\n        // xy\n        positions[index+15] = a * w1 + c * h0 + tx;\n        positions[index+16] = d * h0 + b * w1 + ty;\n    }\n\n    // uv\n    positions[index+2] = uvs.x0;\n    positions[index+3] = uvs.y0;\n\n    // uv\n    positions[index+7] = uvs.x1;\n    positions[index+8] = uvs.y1;\n\n     // uv\n    positions[index+12] = uvs.x2;\n    positions[index+13] = uvs.y2;\n\n    // uv\n    positions[index+17] = uvs.x3;\n    positions[index+18] = uvs.y3;\n\n    // color and alpha\n    var tint = sprite.tint;\n    colors[index+4] = colors[index+9] = colors[index+14] = colors[index+19] = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (sprite.worldAlpha * 255 << 24);\n\n    // increment the batchsize\n    this.sprites[this.currentBatchSize++] = sprite;\n\n\n};\n\n/**\n * Renders a TilingSprite using the spriteBatch.\n *\n * @param sprite {TilingSprite} the tilingSprite to render\n */\nWebGLSpriteBatch.prototype.renderTilingSprite = function (tilingSprite) {\n    var texture = tilingSprite.tilingTexture;\n\n    // check texture..\n    if (this.currentBatchSize >= this.size) {\n        //return;\n        this.flush();\n        this.currentBaseTexture = texture.baseTexture;\n    }\n\n    // set the textures uvs temporarily\n    // TODO create a separate texture so that we can tile part of a texture\n\n    if (!tilingSprite._uvs) {\n        tilingSprite._uvs = new TextureUvs();\n    }\n\n    var uvs = tilingSprite._uvs;\n\n    tilingSprite.tilePosition.x %= texture.baseTexture.width * tilingSprite.tileScaleOffset.x;\n    tilingSprite.tilePosition.y %= texture.baseTexture.height * tilingSprite.tileScaleOffset.y;\n\n    var offsetX =  tilingSprite.tilePosition.x/(texture.baseTexture.width*tilingSprite.tileScaleOffset.x);\n    var offsetY =  tilingSprite.tilePosition.y/(texture.baseTexture.height*tilingSprite.tileScaleOffset.y);\n\n    var scaleX =  (tilingSprite.width / texture.baseTexture.width)  / (tilingSprite.tileScale.x * tilingSprite.tileScaleOffset.x);\n    var scaleY =  (tilingSprite.height / texture.baseTexture.height) / (tilingSprite.tileScale.y * tilingSprite.tileScaleOffset.y);\n\n    uvs.x0 = 0 - offsetX;\n    uvs.y0 = 0 - offsetY;\n\n    uvs.x1 = (1 * scaleX) - offsetX;\n    uvs.y1 = 0 - offsetY;\n\n    uvs.x2 = (1 * scaleX) - offsetX;\n    uvs.y2 = (1 * scaleY) - offsetY;\n\n    uvs.x3 = 0 - offsetX;\n    uvs.y3 = (1 * scaleY) - offsetY;\n\n    // get the tilingSprites current alpha and tint and combining them into a single color\n    var tint = tilingSprite.tint;\n    var color = (tint >> 16) + (tint & 0xff00) + ((tint & 0xff) << 16) + (tilingSprite.alpha * 255 << 24);\n\n    var positions = this.positions;\n    var colors = this.colors;\n\n    var width = tilingSprite.width;\n    var height = tilingSprite.height;\n\n    // TODO trim??\n    var aX = tilingSprite.anchor.x;\n    var aY = tilingSprite.anchor.y;\n    var w0 = width * (1-aX);\n    var w1 = width * -aX;\n\n    var h0 = height * (1-aY);\n    var h1 = height * -aY;\n\n    var index = this.currentBatchSize * 4 * this.vertSize;\n\n    var resolution = texture.baseTexture.resolution;\n\n    var worldTransform = tilingSprite.worldTransform;\n\n    var a = worldTransform.a / resolution;//[0];\n    var b = worldTransform.b / resolution;//[3];\n    var c = worldTransform.c / resolution;//[1];\n    var d = worldTransform.d / resolution;//[4];\n    var tx = worldTransform.tx;//[2];\n    var ty = worldTransform.ty;//[5];\n\n    // xy\n    positions[index++] = a * w1 + c * h1 + tx;\n    positions[index++] = d * h1 + b * w1 + ty;\n    // uv\n    positions[index++] = uvs.x0;\n    positions[index++] = uvs.y0;\n    // color\n    colors[index++] = color;\n\n    // xy\n    positions[index++] = (a * w0 + c * h1 + tx);\n    positions[index++] = d * h1 + b * w0 + ty;\n    // uv\n    positions[index++] = uvs.x1;\n    positions[index++] = uvs.y1;\n    // color\n    colors[index++] = color;\n\n    // xy\n    positions[index++] = a * w0 + c * h0 + tx;\n    positions[index++] = d * h0 + b * w0 + ty;\n    // uv\n    positions[index++] = uvs.x2;\n    positions[index++] = uvs.y2;\n    // color\n    colors[index++] = color;\n\n    // xy\n    positions[index++] = a * w1 + c * h0 + tx;\n    positions[index++] = d * h0 + b * w1 + ty;\n    // uv\n    positions[index++] = uvs.x3;\n    positions[index++] = uvs.y3;\n    // color\n    colors[index++] = color;\n\n    // increment the batchsize\n    this.sprites[this.currentBatchSize++] = tilingSprite;\n};\n\n/**\n * Renders the content and empties the current batch.\n *\n */\nWebGLSpriteBatch.prototype.flush = function () {\n    // If the batch is length 0 then return as there is nothing to draw\n    if (this.currentBatchSize === 0) {\n        return;\n    }\n\n    var gl = this.renderer.gl;\n    var shader;\n\n    if (this.dirty) {\n        this.dirty = false;\n        // bind the main texture\n        gl.activeTexture(gl.TEXTURE0);\n\n        // bind the buffers\n        gl.bindBuffer(gl.ARRAY_BUFFER, this.vertexBuffer);\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this.indexBuffer);\n\n        // this is the same for each shader?\n        var stride =  this.vertSize * 4;\n        gl.vertexAttribPointer(this.shader.attributes.aVertexPosition, 2, gl.FLOAT, false, stride, 0);\n        gl.vertexAttribPointer(this.shader.attributes.aTextureCoord, 2, gl.FLOAT, false, stride, 2 * 4);\n\n        // color attributes will be interpreted as unsigned bytes and normalized\n        gl.vertexAttribPointer(this.shader.attributes.aColor, 4, gl.UNSIGNED_BYTE, true, stride, 4 * 4);\n    }\n\n    // upload the verts to the buffer\n    if (this.currentBatchSize > ( this.size * 0.5 ) ) {\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices);\n    }\n    else {\n        var view = this.positions.subarray(0, this.currentBatchSize * 4 * this.vertSize);\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, view);\n    }\n\n    var nextTexture, nextBlendMode, nextShader;\n    var batchSize = 0;\n    var start = 0;\n\n    var currentBaseTexture = null;\n    var currentBlendMode = this.renderer.blendModeManager.currentBlendMode;\n    var currentShader = null;\n\n    var blendSwap = false;\n    var shaderSwap = false;\n    var sprite;\n\n    for (var i = 0, j = this.currentBatchSize; i < j; i++) {\n\n        sprite = this.sprites[i];\n\n        nextTexture = sprite.texture.baseTexture;\n        nextBlendMode = sprite.blendMode;\n        nextShader = sprite.shader || this.shader;\n\n        blendSwap = currentBlendMode !== nextBlendMode;\n        shaderSwap = currentShader !== nextShader; // should I use uuidS???\n\n        if (currentBaseTexture !== nextTexture || blendSwap || shaderSwap) {\n            this.renderBatch(currentBaseTexture, batchSize, start);\n\n            start = i;\n            batchSize = 0;\n            currentBaseTexture = nextTexture;\n\n            if (blendSwap) {\n                currentBlendMode = nextBlendMode;\n                this.renderer.blendModeManager.setBlendMode( currentBlendMode );\n            }\n\n            if (shaderSwap) {\n                currentShader = nextShader;\n\n                shader = currentShader.shaders ? currentShader.shaders[gl.id] : currentShader;\n\n                if (!shader) {\n                    shader = new Shader(gl, null, currentShader.fragmentSrc, currentShader.uniforms);\n                    currentShader.shaders[gl.id] = shader;\n                }\n\n                // set shader function???\n                this.renderer.shaderManager.setShader(shader);\n\n                if (shader.dirty) {\n                    shader.syncUniforms();\n                }\n\n                // both thease only need to be set if they are changing..\n                // set the projection\n                var projection = this.renderer.projection;\n                gl.uniform2f(shader.uniforms.projectionVector._location, projection.x, projection.y);\n\n                // TODO - this is temprorary!\n                var offsetVector = this.renderer.offset;\n                gl.uniform2f(shader.uniforms.offsetVector._location, offsetVector.x, offsetVector.y);\n\n                // set the pointers\n            }\n        }\n\n        batchSize++;\n    }\n\n    this.renderBatch(currentBaseTexture, batchSize, start);\n\n    // then reset the batch!\n    this.currentBatchSize = 0;\n};\n\n/**\n * @param texture {Texture}\n * @param size {number}\n * @param startIndex {number}\n */\nWebGLSpriteBatch.prototype.renderBatch = function (texture, size, startIndex) {\n    if (size === 0) {\n        return;\n    }\n\n    var gl = this.renderer.gl;\n\n    if (!texture._glTextures[gl.id]) {\n        this.renderer.updateTexture(texture);\n    }\n    else {\n        // bind the current texture\n        gl.bindTexture(gl.TEXTURE_2D, texture._glTextures[gl.id]);\n    }\n\n    // now draw those suckas!\n    gl.drawElements(gl.TRIANGLES, size * 6, gl.UNSIGNED_SHORT, startIndex * 6 * 2);\n\n    // increment the draw count\n    this.renderer.drawCount++;\n};\n\n/**\n *\n */\nWebGLSpriteBatch.prototype.stop = function () {\n    this.flush();\n    this.dirty = true;\n};\n\n/**\n *\n */\nWebGLSpriteBatch.prototype.start = function () {\n    this.dirty = true;\n};\n\n/**\n * Destroys the SpriteBatch.\n *\n */\nWebGLSpriteBatch.prototype.destroy = function () {\n    this.renderer.gl.deleteBuffer(this.vertexBuffer);\n    this.renderer.gl.deleteBuffer(this.indexBuffer);\n\n    this.vertices = null;\n    this.indices = null;\n\n    this.vertexBuffer = null;\n    this.indexBuffer = null;\n\n    this.currentBaseTexture = null;\n\n    this.renderer = null;\n};\n","var utils = require('../utils'),\n    CONST = require('../const');\n\n/**\n * A texture stores the information that represents an image. All textures have a base texture.\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param source {Image|Canvas} the source object of the texture.\n * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values\n */\nfunction BaseTexture(source, scaleMode) {\n    this.uuid = utils.uuid();\n\n    /**\n     * The Resolution of the texture.\n     *\n     * @member {number}\n     */\n    this.resolution = 1;\n\n    /**\n     * The width of the base texture set when the image has loaded\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.width = 100;\n\n    /**\n     * The height of the base texture set when the image has loaded\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.height = 100;\n\n    /**\n     * The scale mode to apply when scaling this texture\n     *\n     * @member {{number}}\n     * @default scaleModes.LINEAR\n     */\n    this.scaleMode = scaleMode || CONST.scaleModes.DEFAULT;\n\n    /**\n     * Set to true once the base texture has successfully loaded.\n     *\n     * This is never true if the underlying source fails to load or has no texture data.\n     *\n     * @member {boolean}\n     * @readOnly\n     */\n    this.hasLoaded = false;\n\n    /**\n     * Set to true if the source is currently loading.\n     *\n     * If an Image source is loading the 'loaded' or 'error' event will be\n     * dispatched when the operation ends. An underyling source that is\n     * immediately-available bypasses loading entirely.\n     *\n     * @member {boolean}\n     * @readonly\n     */\n    this.isLoading = false;\n\n    /**\n     * The image source that is used to create the texture.\n     *\n     * TODO: Make this a setter that calls loadSource();\n     *\n     * @member {Image|Canvas}\n     * @readonly\n     */\n    this.source = null; // set in loadSource, if at all\n\n    /**\n     * Controls if RGB channels should be pre-multiplied by Alpha  (WebGL only)\n     *\n     * @member {boolean}\n     * @default true\n     */\n    this.premultipliedAlpha = true;\n\n    /**\n     * @member {string}\n     */\n    this.imageUrl = null;\n\n    /**\n     * @member {boolean}\n     * @private\n     */\n    this._powerOf2 = false;\n\n    // used for webGL\n\n    /**\n     *\n     * Set this to true if a mipmap of this texture needs to be generated. This value needs to be set before the texture is used\n     * Also the texture must be a power of two size to work\n     *\n     * @member {boolean}\n     */\n    this.mipmap = false;\n\n    /**\n     * A map of renderer IDs to webgl textures\n     *\n     * @member {object<number, WebGLTexture>}\n     * @private\n     */\n    this._glTextures = {};\n\n    /**\n     * Does the texture on the GPU need to be updated?\n     *\n     * @member {boolean}\n     * @private\n     */\n    this._needsUpdate = false;\n\n    // if no source passed don't try to load\n    if (source) {\n        this.loadSource(source);\n    }\n\n    /**\n     * Fired when a not-immediately-available source finishes loading.\n     *\n     * @event loaded\n     * @protected\n     */\n\n    /**\n     * Fired when a not-immediately-available source fails to load.\n     *\n     * @event error\n     * @protected\n     */\n}\n\nBaseTexture.prototype.constructor = BaseTexture;\nmodule.exports = BaseTexture;\n\nutils.eventTarget.mixin(BaseTexture.prototype);\n\nObject.defineProperties(BaseTexture.prototype, {\n    needsUpdate: {\n        get: function () {\n            return this._needsUpdate;\n        },\n        set: function (val) {\n            this._needsUpdate = val;\n\n            if (val) {\n                this.emit('update', this);\n            }\n        }\n    }\n});\n\n/**\n * Load a source.\n *\n * If the source is not-immediately-available, such as an image that needs to be\n * downloaded, then the 'loaded' or 'error' event will be dispatched in the future\n * and `hasLoaded` will remain false after this call.\n *\n * The logic state after calling `loadSource` directly or indirectly (eg. `fromImage`, `new BaseTexture`) is:\n *\n *     if (texture.hasLoaded) {\n *        // texture ready for use\n *     } else if (texture.isLoading) {\n *        // listen to 'loaded' and/or 'error' events on texture\n *     } else {\n *        // not loading, not going to load UNLESS the source is reloaded\n *        // (it may still make sense to listen to the events)\n *     }\n *\n * @protected\n * @param source {Image|Canvas} the source object of the texture.\n */\nBaseTexture.prototype.loadSource = function (source) {\n    var wasLoading = this.isLoading;\n    this.hasLoaded = false;\n    this.isLoading = false;\n\n    if (wasLoading && this.source) {\n        this.source.onload = null;\n        this.source.onerror = null;\n    }\n\n    this.source = source;\n\n    // Apply source if loaded. Otherwise setup appropriate loading monitors.\n    if ((this.source.complete || this.source.getContext) && this.source.width && this.source.height) {\n        this._sourceLoaded();\n    }\n    else  if(!source.getContext) {\n        // Image fail / not ready\n        this.isLoading = true;\n\n        var scope = this;\n\n        source.onload = function () {\n            source.onload = null;\n            source.onerror = null;\n\n            if(!scope.isLoading) {\n                return;\n            }\n\n            scope.isLoading = false;\n            scope._sourceLoaded();\n\n            scope.emit('loaded', scope);\n        };\n\n        source.onerror = function () {\n            source.onload = null;\n            source.onerror = null;\n\n            if(!scope.isLoading) {\n                return;\n            }\n\n            scope.isLoading = false;\n            scope.emit('error', scope);\n        };\n\n        // Per http://www.w3.org/TR/html5/embedded-content-0.html#the-img-element\n        //   \"The value of `complete` can thus change while a script is executing.\"\n        // So complete needs to be re-checked after the callbacks have been added..\n        if (source.complete) {\n            this.isLoading = false;\n\n            // ..and if we're complete now, no need for callbacks\n            source.onload = null;\n            source.onerror = null;\n\n            if (source.width && source.height) {\n                this._sourceLoaded();\n\n                // If any previous subscribers possible\n                if (wasLoading) {\n                    this.emit('loaded', this);\n                }\n            }\n            else {\n                // If any previous subscribers possible\n                if (wasLoading) {\n                    this.emit('error', this);\n                }\n            }\n        }\n    }\n};\n\n/**\n * Used internally to update the width, height, and some other tracking vars once\n * a source has successfully loaded.\n *\n * @private\n */\nBaseTexture.prototype._sourceLoaded = function () {\n    this.hasLoaded = true;\n\n    this.width = this.source.naturalWidth || this.source.width;\n    this.height = this.source.naturalHeight || this.source.height;\n\n    this.needsUpdate = true;\n};\n\n/**\n * Destroys this base texture\n *\n */\nBaseTexture.prototype.destroy = function () {\n    if (this.imageUrl) {\n        delete utils.BaseTextureCache[this.imageUrl];\n        delete utils.TextureCache[this.imageUrl];\n        this.imageUrl = null;\n        if (!navigator.isCocoonJS) {\n            this.source.src = '';\n        }\n    }\n    else if (this.source && this.source._pixiId) {\n        delete utils.BaseTextureCache[this.source._pixiId];\n    }\n    this.source = null;\n\n    this.dispose();\n};\n\n/**\n * Frees the texture from WebGL memory without destroying this texture object.\n * This means you can still use the texture later which will upload it to GPU\n * memory again.\n *\n */\nBaseTexture.prototype.dispose = function () {\n    this.emit('dispose', this);\n};\n\n/**\n * Changes the source image of the texture.\n * The original source must be an Image element.\n *\n * @param newSrc {string} the path of the image\n */\nBaseTexture.prototype.updateSourceImage = function (newSrc) {\n    this.source.src = newSrc;\n    this.loadSource(this.source);\n};\n\n/**\n * Helper function that creates a base texture from the given image url.\n * If the image is not in the base texture cache it will be created and loaded.\n *\n * @static\n * @param imageUrl {string} The image url of the texture\n * @param [crossorigin=(auto)] {boolean} Should use anonymouse CORS? Defaults to true if the URL is not a data-URI.\n * @param [scaleMode=scaleModes.DEFAULT] {number} See {@link scaleModes} for possible values\n * @return BaseTexture\n */\nBaseTexture.fromImage = function (imageUrl, crossorigin, scaleMode) {\n    var baseTexture = utils.BaseTextureCache[imageUrl];\n\n    if (crossorigin === undefined && imageUrl.indexOf('data:') !== 0) {\n        crossorigin = true;\n    }\n\n    if (!baseTexture) {\n        // new Image() breaks tex loading in some versions of Chrome.\n        // See https://code.google.com/p/chromium/issues/detail?id=238071\n        var image = new Image();//document.createElement('img');\n        if (crossorigin) {\n            image.crossOrigin = '';\n        }\n\n        image.src = imageUrl;\n        baseTexture = new BaseTexture(image, scaleMode);\n        baseTexture.imageUrl = imageUrl;\n        utils.BaseTextureCache[imageUrl] = baseTexture;\n\n        // if there is an @2x at the end of the url we are going to assume its a highres image\n        if ( imageUrl.indexOf(CONST.RETINA_PREFIX + '.') !== -1) {\n            baseTexture.resolution = 2;\n        }\n    }\n\n    return baseTexture;\n};\n\n/**\n * Helper function that creates a base texture from the given canvas element.\n *\n * @static\n * @param canvas {Canvas} The canvas element source of the texture\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return BaseTexture\n */\nBaseTexture.fromCanvas = function (canvas, scaleMode) {\n    if (!canvas._pixiId) {\n        canvas._pixiId = 'canvas_' + utils.uuid();\n    }\n\n    var baseTexture = utils.BaseTextureCache[canvas._pixiId];\n\n    if (!baseTexture) {\n        baseTexture = new BaseTexture(canvas, scaleMode);\n        utils.BaseTextureCache[canvas._pixiId] = baseTexture;\n    }\n\n    return baseTexture;\n};\n","var BaseTexture = require('./BaseTexture'),\n    Texture = require('./Texture'),\n    FilterTexture = require('../renderers/webgl/utils/FilterTexture'),\n    CanvasBuffer = require('../renderers/canvas/utils/CanvasBuffer'),\n    math = require('../math'),\n    CONST = require('../const');\n\n/**\n * A RenderTexture is a special texture that allows any Pixi display object to be rendered to it.\n *\n * __Hint__: All DisplayObjects (i.e. Sprites) that render to a RenderTexture should be preloaded\n * otherwise black rectangles will be drawn instead.\n *\n * A RenderTexture takes a snapshot of any Display Object given to its render method. The position\n * and rotation of the given Display Objects is ignored. For example:\n *\n * ```js\n * var renderTexture = new PIXI.RenderTexture(800, 600);\n * var sprite = PIXI.Sprite.fromImage(\"spinObj_01.png\");\n *\n * sprite.position.x = 800/2;\n * sprite.position.y = 600/2;\n * sprite.anchor.x = 0.5;\n * sprite.anchor.y = 0.5;\n *\n * renderTexture.render(sprite);\n * ```\n *\n * The Sprite in this case will be rendered to a position of 0,0. To render this sprite at its actual\n * position a DisplayObjectContainer should be used:\n *\n * ```js\n * var doc = new DisplayObjectContainer();\n *\n * doc.addChild(sprite);\n *\n * renderTexture.render(doc);  // Renders to center of renderTexture\n * ```\n *\n * @class\n * @extends Texture\n * @namespace PIXI\n * @param renderer {CanvasRenderer|WebGLRenderer} The renderer used for this RenderTexture\n * @param [width=100] {number} The width of the render texture\n * @param [height=100] {number} The height of the render texture\n * @param [scaleMode] {number} See {@link scaleModes} for possible values\n * @param [resolution=1] {number} The resolution of the texture being generated\n */\nfunction RenderTexture(renderer, width, height, scaleMode, resolution) {\n    if (!renderer) {\n        throw new Error('Unable to create RenderTexture, you must pass a renderer into the constructor.');\n    }\n\n    /**\n     * The with of the render texture\n     *\n     * @member {number}\n     */\n    this.width = width || 100;\n\n    /**\n     * The height of the render texture\n     *\n     * @member {number}\n     */\n    this.height = height || 100;\n\n    /**\n     * The Resolution of the texture.\n     *\n     * @member {number}\n     */\n    this.resolution = resolution || 1;\n\n    /**\n     * The framing rectangle of the render texture\n     *\n     * @member {Rectangle}\n     */\n    this.frame = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution);\n\n    /**\n     * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering,\n     * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases)\n     *\n     * @member {Rectangle}\n     */\n    this.crop = new math.Rectangle(0, 0, this.width * this.resolution, this.height * this.resolution);\n\n    /**\n     * Draw/render the given DisplayObject onto the texture.\n     *\n     * The displayObject and descendents are transformed during this operation.\n     * If `restoreWorldTransform` is true then the transformations will be restored before the\n     * method returns. Otherwise it is up to the calling code to correctly use or reset\n     * the transformed display objects.\n     *\n     * The display object is always rendered with a worldAlpha value of 1.\n     *\n     * @method\n     * @param displayObject {DisplayObject} The display object to render this texture on\n     * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering.\n     * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn\n     * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children\n     *  transformations will be restored. Not restoring this information will be a little faster.\n     */\n    this.render = null;\n\n    /**\n     * The base texture object that this texture uses\n     *\n     * @member {BaseTexture}\n     */\n    this.baseTexture = new BaseTexture();\n    this.baseTexture.width = this.width * this.resolution;\n    this.baseTexture.height = this.height * this.resolution;\n    this.baseTexture._glTextures = [];\n    this.baseTexture.resolution = this.resolution;\n\n    this.baseTexture.scaleMode = scaleMode || CONST.scaleModes.DEFAULT;\n\n    this.baseTexture.hasLoaded = true;\n\n    Texture.call(this,\n        this.baseTexture,\n        new math.Rectangle(0, 0, this.width, this.height)\n    );\n\n    /**\n     * The renderer this RenderTexture uses. A RenderTexture can only belong to one renderer at the moment if its webGL.\n     *\n     * @member {CanvasRenderer|WebGLRenderer}\n     */\n    this.renderer = renderer;\n\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        var gl = this.renderer.gl;\n\n        this.textureBuffer = new FilterTexture(gl, this.width * this.resolution, this.height * this.resolution, this.baseTexture.scaleMode);\n        this.baseTexture._glTextures[gl.id] =  this.textureBuffer.texture;\n\n        this.render = this.renderWebGL;\n        this.projection = new math.Point(this.width*0.5, -this.height*0.5);\n    }\n    else {\n        this.render = this.renderCanvas;\n        this.textureBuffer = new CanvasBuffer(this.width* this.resolution, this.height* this.resolution);\n        this.baseTexture.source = this.textureBuffer.canvas;\n    }\n\n    /**\n     * @member {boolean}\n     */\n    this.valid = true;\n\n    this._updateUvs();\n}\n\nRenderTexture.prototype = Object.create(Texture.prototype);\nRenderTexture.prototype.constructor = RenderTexture;\nmodule.exports = RenderTexture;\n\n/**\n * Resizes the RenderTexture.\n *\n * @param width {number} The width to resize to.\n * @param height {number} The height to resize to.\n * @param updateBase {boolean} Should the baseTexture.width and height values be resized as well?\n */\nRenderTexture.prototype.resize = function (width, height, updateBase) {\n    if (width === this.width && height === this.height) {\n        return;\n    }\n\n    this.valid = (width > 0 && height > 0);\n\n    this.width = this.frame.width = this.crop.width = width;\n    this.height =  this.frame.height = this.crop.height = height;\n\n    if (updateBase) {\n        this.baseTexture.width = this.width;\n        this.baseTexture.height = this.height;\n    }\n\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        this.projection.x = this.width / 2;\n        this.projection.y = -this.height / 2;\n    }\n\n    if (!this.valid) {\n        return;\n    }\n\n    this.textureBuffer.resize(this.width * this.resolution, this.height * this.resolution);\n};\n\n/**\n * Clears the RenderTexture.\n *\n */\nRenderTexture.prototype.clear = function () {\n    if (!this.valid) {\n        return;\n    }\n\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        this.renderer.gl.bindFramebuffer(this.renderer.gl.FRAMEBUFFER, this.textureBuffer.frameBuffer);\n    }\n\n    this.textureBuffer.clear();\n};\n\n/**\n * Internal method assigned to the `render` property if using a CanvasRenderer.\n *\n * @private\n * @param displayObject {DisplayObject} The display object to render this texture on\n * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering.\n * @param [clear=false] {boolean} If true the texture will be cleared before the displayObject is drawn\n * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children\n *  transformations will be restored. Not restoring this information will be a little faster.\n */\nRenderTexture.prototype.renderWebGL = function (displayObject, matrix, clear, restoreWorldTransform) {\n    if (!this.valid) {\n        return;\n    }\n\n    if (typeof restoreWorldTransform === 'undefined') {\n        restoreWorldTransform = true;\n    }\n\n    var tempAlpha,\n        tempTransform;\n\n    if (restoreWorldTransform) {\n        tempAlpha = displayObject.worldAlpha;\n        tempTransform = displayObject.worldTransform.toArray();\n    }\n\n    //TOOD replace position with matrix..\n\n    //Lets create a nice matrix to apply to our display object. Frame buffers come in upside down so we need to flip the matrix\n    var wt = displayObject.worldTransform;\n\n    wt.identity();\n    wt.translate(0, this.projection.y * 2);\n\n    if (matrix) {\n        wt.append(matrix);\n    }\n\n    wt.scale(1,-1);\n\n    // setWorld Alpha to ensure that the object is renderer at full opacity\n    displayObject.worldAlpha = 1;\n\n    // Time to update all the children of the displayObject with the new matrix..\n    var children = displayObject.children;\n    var i, j;\n\n    for (i = 0, j = children.length; i < j; ++i) {\n        children[i].updateTransform();\n    }\n\n    // time for the webGL fun stuff!\n    var gl = this.renderer.gl;\n\n    gl.viewport(0, 0, this.width * this.resolution, this.height * this.resolution);\n\n    gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer );\n\n    if (clear) {\n        this.textureBuffer.clear();\n    }\n\n    this.renderer.spriteBatch.dirty = true;\n\n    this.renderer.renderDisplayObject(displayObject, this.projection, this.textureBuffer.frameBuffer);\n\n    this.renderer.spriteBatch.dirty = true;\n\n    if (restoreWorldTransform) {\n        displayObject.worldAlpha = tempAlpha;\n        displayObject.worldTransform.fromArray(tempTransform);\n\n        for (i = 0, j = children.length; i < j; ++i) {\n            children[i].updateTransform();\n        }\n    }\n};\n\n\n/**\n * Internal method assigned to the `render` property if using a CanvasRenderer.\n *\n * @private\n * @param displayObject {DisplayObject} The display object to render this texture on\n * @param [matrix] {Matrix} Optional matrix to apply to the display object before rendering.\n * @param [clear] {boolean} If true the texture will be cleared before the displayObject is drawn\n * @param [restoreWorldTransform=true] {boolean} If true the displayObject's worldTransform/worldAlpha and all children\n *  transformations will be restored. Not restoring this information will be a little faster.\n */\nRenderTexture.prototype.renderCanvas = function (displayObject, matrix, clear, restoreWorldTransform) {\n    if (!this.valid) {\n        return;\n    }\n\n    if (typeof restoreWorldTransform === 'undefined') {\n        restoreWorldTransform = true;\n    }\n\n    var tempAlpha,\n        tempTransform;\n\n    if (restoreWorldTransform) {\n        tempAlpha = displayObject.worldAlpha;\n        tempTransform = displayObject.worldTransform.toArray();\n    }\n\n    var wt = displayObject.worldTransform;\n    wt.identity();\n    if (matrix) {\n        wt.append(matrix);\n    }\n\n    // setWorld Alpha to ensure that the object is renderer at full opacity\n    displayObject.worldAlpha = 1;\n\n    // Time to update all the children of the displayObject with the new matrix..\n    var children = displayObject.children;\n    var i, j;\n\n    for (i = 0, j = children.length; i < j; ++i) {\n        children[i].updateTransform();\n    }\n\n    if (clear) {\n        this.textureBuffer.clear();\n    }\n\n    var context = this.textureBuffer.context;\n\n    var realResolution = this.renderer.resolution;\n\n    this.renderer.resolution = this.resolution;\n\n    this.renderer.renderDisplayObject(displayObject, context);\n\n    this.renderer.resolution = realResolution;\n\n    if (restoreWorldTransform) {\n        displayObject.worldAlpha = tempAlpha;\n        displayObject.worldTransform.fromArray(tempTransform);\n\n        for (i = 0, j = children.length; i < j; ++i) {\n            children[i].updateTransform();\n        }\n    }\n};\n\n/**\n * Will return a HTML Image of the texture\n *\n * @return {Image}\n */\nRenderTexture.prototype.getImage = function () {\n    var image = new Image();\n    image.src = this.getBase64();\n    return image;\n};\n\n/**\n * Will return a a base64 encoded string of this texture. It works by calling RenderTexture.getCanvas and then running toDataURL on that.\n *\n * @return {string} A base64 encoded string of the texture.\n */\nRenderTexture.prototype.getBase64 = function () {\n    return this.getCanvas().toDataURL();\n};\n\n/**\n * Creates a Canvas element, renders this RenderTexture to it and then returns it.\n *\n * @return {HTMLCanvasElement} A Canvas element with the texture rendered on.\n */\nRenderTexture.prototype.getCanvas = function () {\n    if (this.renderer.type === CONST.WEBGL_RENDERER) {\n        var gl =  this.renderer.gl;\n        var width = this.textureBuffer.width;\n        var height = this.textureBuffer.height;\n\n        var webGLPixels = new Uint8Array(4 * width * height);\n\n        gl.bindFramebuffer(gl.FRAMEBUFFER, this.textureBuffer.frameBuffer);\n        gl.readPixels(0, 0, width, height, gl.RGBA, gl.UNSIGNED_BYTE, webGLPixels);\n        gl.bindFramebuffer(gl.FRAMEBUFFER, null);\n\n        var tempCanvas = new CanvasBuffer(width, height);\n        var canvasData = tempCanvas.context.getImageData(0, 0, width, height);\n        canvasData.data.set(webGLPixels);\n\n        tempCanvas.context.putImageData(canvasData, 0, 0);\n\n        return tempCanvas.canvas;\n    }\n    else {\n        return this.textureBuffer.canvas;\n    }\n};\n","var BaseTexture = require('./BaseTexture'),\n    VideoBaseTexture = require('./VideoBaseTexture'),\n    TextureUvs = require('./TextureUvs'),\n    eventTarget = require('../utils/eventTarget'),\n    math = require('../math'),\n    utils = require('../utils');\n\n/**\n * A texture stores the information that represents an image or part of an image. It cannot be added\n * to the display list directly. Instead use it as the texture for a Sprite. If no frame is provided then the whole image is used.\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param baseTexture {BaseTexture} The base texture source to create the texture from\n * @param [frame] {Rectangle} The rectangle frame of the texture to show\n * @param [crop] {Rectangle} The area of original texture\n * @param [trim] {Rectangle} Trimmed texture rectangle\n */\nfunction Texture(baseTexture, frame, crop, trim) {\n    /**\n     * Does this Texture have any frame data assigned to it?\n     *\n     * @member {boolean}\n     */\n    this.noFrame = false;\n\n    if (!frame) {\n        this.noFrame = true;\n        frame = new math.Rectangle(0, 0, 1, 1);\n    }\n\n    if (baseTexture instanceof Texture) {\n        baseTexture = baseTexture.baseTexture;\n    }\n\n    /**\n     * The base texture that this texture uses.\n     *\n     * @member {BaseTexture}\n     */\n    this.baseTexture = baseTexture;\n\n    /**\n     * The frame specifies the region of the base texture that this texture uses\n     *\n     * @member {Rectangle}\n     * @private\n     */\n    this._frame = frame;\n\n    /**\n     * The texture trim data.\n     *\n     * @member {Rectangle}\n     */\n    this.trim = trim;\n\n    /**\n     * This will let the renderer know if the texture is valid. If it's not then it cannot be rendered.\n     *\n     * @member {boolean}\n     */\n    this.valid = false;\n\n    /**\n     * This will let a renderer know that a texture has been updated (used mainly for webGL uv updates)\n     *\n     * @member {boolean}\n     */\n    this.requiresUpdate = false;\n\n    /**\n     * The WebGL UV data cache.\n     *\n     * @member {object}\n     * @private\n     */\n    this._uvs = null;\n\n    /**\n     * The width of the Texture in pixels.\n     *\n     * @member {number}\n     */\n    this.width = 0;\n\n    /**\n     * The height of the Texture in pixels.\n     *\n     * @member {number}\n     */\n    this.height = 0;\n\n    /**\n     * This is the area of the BaseTexture image to actually copy to the Canvas / WebGL when rendering,\n     * irrespective of the actual frame size or placement (which can be influenced by trimmed texture atlases)\n     *\n     * @member {Rectangle}\n     */\n    this.crop = crop || new math.Rectangle(0, 0, 1, 1);\n\n    if (baseTexture.hasLoaded) {\n        if (this.noFrame) {\n            frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height);\n        }\n        this.frame = frame;\n    }\n    else {\n        baseTexture.addEventListener('loaded', this.onBaseTextureLoaded.bind(this));\n    }\n}\n\nTexture.prototype.constructor = Texture;\nmodule.exports = Texture;\n\neventTarget.mixin(Texture.prototype);\n\nObject.defineProperties(Texture.prototype, {\n    needsUpdate: {\n        get: function () {\n            return this.baseTexture.needsUpdate;\n        },\n        set: function (val) {\n            this.baseTexture.needsUpdate = val;\n        }\n    },\n\n    frame: {\n        get: function () {\n            return this._frame;\n        },\n        set: function (frame) {\n            this._frame = frame;\n\n            this.noFrame = false;\n\n            this.width = frame.width;\n            this.height = frame.height;\n\n            this.crop.x = frame.x;\n            this.crop.y = frame.y;\n            this.crop.width = frame.width;\n            this.crop.height = frame.height;\n\n            if (!this.trim && (frame.x + frame.width > this.baseTexture.width || frame.y + frame.height > this.baseTexture.height)) {\n                throw new Error('Texture Error: frame does not fit inside the base Texture dimensions ' + this);\n            }\n\n            this.valid = frame && frame.width && frame.height && this.baseTexture.source && this.baseTexture.hasLoaded;\n\n            if (this.trim) {\n                this.width = this.trim.width;\n                this.height = this.trim.height;\n                this._frame.width = this.trim.width;\n                this._frame.height = this.trim.height;\n            }\n\n            if (this.valid) {\n                this._updateUvs();\n            }\n        }\n    }\n});\n\n/**\n * Called when the base texture is loaded\n *\n * @private\n */\nTexture.prototype.onBaseTextureLoaded = function () {\n    var baseTexture = this.baseTexture;\n    baseTexture.removeEventListener('loaded', this.onLoaded);\n\n    if (this.noFrame) {\n        this.frame = new math.Rectangle(0, 0, baseTexture.width, baseTexture.height);\n    }\n\n    this.dispatchEvent( { type: 'update', content: this } );\n};\n\n/**\n * Destroys this texture\n *\n * @param destroyBase {boolean} Whether to destroy the base texture as well\n */\nTexture.prototype.destroy = function (destroyBase) {\n    if (destroyBase) {\n        this.baseTexture.destroy();\n    }\n\n    this.valid = false;\n};\n\n/**\n * Updates the internal WebGL UV cache.\n *\n * @private\n */\nTexture.prototype._updateUvs = function () {\n    if (!this._uvs) {\n        this._uvs = new TextureUvs();\n    }\n\n    var frame = this.crop;\n    var tw = this.baseTexture.width;\n    var th = this.baseTexture.height;\n\n    this._uvs.x0 = frame.x / tw;\n    this._uvs.y0 = frame.y / th;\n\n    this._uvs.x1 = (frame.x + frame.width) / tw;\n    this._uvs.y1 = frame.y / th;\n\n    this._uvs.x2 = (frame.x + frame.width) / tw;\n    this._uvs.y2 = (frame.y + frame.height) / th;\n\n    this._uvs.x3 = frame.x / tw;\n    this._uvs.y3 = (frame.y + frame.height) / th;\n};\n\n/**\n * Helper function that creates a Texture object from the given image url.\n * If the image is not in the texture cache it will be  created and loaded.\n *\n * @static\n * @param imageUrl {string} The image url of the texture\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return Texture\n */\nTexture.fromImage = function (imageUrl, crossorigin, scaleMode) {\n    var texture = utils.TextureCache[imageUrl];\n\n    if (!texture) {\n        texture = new Texture(BaseTexture.fromImage(imageUrl, crossorigin, scaleMode));\n        utils.TextureCache[imageUrl] = texture;\n    }\n\n    return texture;\n};\n\n/**\n * Helper function that returns a Texture objected based on the given frame id.\n * If the frame id is not in the texture cache an error will be thrown.\n *\n * @static\n * @param frameId {string} The frame id of the texture\n * @return Texture\n */\nTexture.fromFrame = function (frameId) {\n    var texture = utils.TextureCache[frameId];\n    if (!texture) {\n        throw new Error('The frameId \"' + frameId + '\" does not exist in the texture cache ');\n    }\n    return texture;\n};\n\n/**\n * Helper function that creates a new Texture based on the given canvas element.\n *\n * @static\n * @param canvas {Canvas} The canvas element source of the texture\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return {Texture}\n */\nTexture.fromCanvas = function (canvas, scaleMode) {\n    return new Texture(BaseTexture.fromCanvas(canvas, scaleMode));\n};\n\n/**\n * Helper function that creates a new Texture based on the given video element.\n *\n * @static\n * @param video {HTMLVideoElement}\n * @param scaleMode {number} See {{#crossLink \"PIXI/scaleModes:property\"}}scaleModes{{/crossLink}} for possible values\n * @return {Texture} A Texture\n */\nTexture.fromVideo = function (video, scaleMode) {\n    return new Texture(VideoBaseTexture.baseTextureFromVideo(video, scaleMode));\n};\n\n/**\n * Adds a texture to the global utils.TextureCache. This cache is shared across the whole PIXI object.\n *\n * @static\n * @param texture {Texture} The Texture to add to the cache.\n * @param id {string} The id that the texture will be stored against.\n */\nTexture.addTextureToCache = function (texture, id) {\n    utils.TextureCache[id] = texture;\n};\n\n/**\n * Remove a texture from the global utils.TextureCache.\n *\n * @static\n * @param id {string} The id of the texture to be removed\n * @return {Texture} The texture that was removed\n */\nTexture.removeTextureFromCache = function (id) {\n    var texture = utils.TextureCache[id];\n\n    delete utils.TextureCache[id];\n    delete utils.BaseTextureCache[id];\n\n    return texture;\n};\n\nTexture.emptyTexture = new Texture(new BaseTexture());\n","function TextureUvs() {\n    this.x0 = 0;\n    this.y0 = 0;\n\n    this.x1 = 0;\n    this.y1 = 0;\n\n    this.x2 = 0;\n    this.y2 = 0;\n\n    this.x3 = 0;\n    this.y3 = 0;\n}\n\nmodule.exports = TextureUvs;\n","var BaseTexture = require('./BaseTexture'),\n    utils = require('../utils');\n\n/**\n * A texture of a [playing] Video.\n *\n * See the [\"deus\" demo](http://www.goodboydigital.com/pixijs/examples/deus/).\n *\n * @class\n * @extends BaseTexture\n * @namespace PIXI\n * @param source {HTMLVideoElement}\n * @param [scaleMode] {number} See {@link scaleModes} for possible values\n */\nfunction VideoBaseTexture(source, scaleMode) {\n    if (!source){\n        throw new Error('No video source element specified.');\n    }\n\n    // hook in here to check if video is already available.\n    // BaseTexture looks for a source.complete boolean, plus width & height.\n\n    if ((source.readyState === source.HAVE_ENOUGH_DATA || source.readyState === source.HAVE_FUTURE_DATA) && source.width && source.height) {\n        source.complete = true;\n    }\n\n    BaseTexture.call(this, source, scaleMode);\n\n    this.autoUpdate = false;\n\n    this._boundOnUpdate = this._onUpdate.bind(this);\n    this._boundOnCanPlay = this._onCanPlay.bind(this);\n\n    if (!source.complete) {\n        source.addEventListener('canplay', this._boundOnCanPlay);\n        source.addEventListener('canplaythrough', this._boundOnCanPlay);\n\n        // started playing..\n        source.addEventListener('play', this._onPlayStart.bind(this));\n        source.addEventListener('pause', this._onPlayStop.bind(this));\n    }\n\n    this.__loaded = false;\n}\n\nVideoBaseTexture.prototype = Object.create(BaseTexture.prototype);\nVideoBaseTexture.prototype.constructor = VideoBaseTexture;\nmodule.exports = VideoBaseTexture;\n\nVideoBaseTexture.prototype._onUpdate = function () {\n    if (this.autoUpdate) {\n        window.requestAnimationFrame(this._boundOnUpdate);\n        this.needsUpdate = true;\n    }\n};\n\nVideoBaseTexture.prototype._onPlayStart = function () {\n    if (!this.autoUpdate) {\n        window.requestAnimationFrame(this._boundOnUpdate);\n        this.autoUpdate = true;\n    }\n};\n\nVideoBaseTexture.prototype._onPlayStop = function () {\n    this.autoUpdate = false;\n};\n\nVideoBaseTexture.prototype._onCanPlay = function () {\n    this.hasLoaded = true;\n\n    if (this.source) {\n        this.source.removeEventListener('canplay', this._boundOnCanPlay);\n        this.source.removeEventListener('canplaythrough', this._boundOnCanPlay);\n\n        this.width = this.source.videoWidth;\n        this.height = this.source.videoHeight;\n\n        this.source.play();\n\n        // prevent multiple loaded dispatches..\n        if(!this.__loaded){\n            this.__loaded = true;\n            this.emit('loaded', this);\n        }\n    }\n};\n\nVideoBaseTexture.prototype.destroy = function () {\n    if (this.source && this.source._pixiId) {\n        utils.BaseTextureCache[ this.source._pixiId ] = null;\n        delete utils.BaseTextureCache[ this.source._pixiId ];\n\n        this.source._pixiId = null;\n        delete this.source._pixiId;\n    }\n\n    BaseTexture.prototype.destroy.call(this);\n};\n\n/**\n * Mimic Pixi BaseTexture.from.... method.\n *\n * @static\n * @param video {HTMLVideoElement}\n * @param scaleMode {number} See {@link scaleModes} for possible values\n * @return {VideoBaseTexture}\n */\nVideoBaseTexture.fromVideo = function (video, scaleMode) {\n    if (!video._pixiId) {\n        video._pixiId = 'video_' + utils.uuid();\n    }\n\n    var baseTexture = utils.BaseTextureCache[video._pixiId];\n\n    if (!baseTexture) {\n        baseTexture = new VideoBaseTexture(video, scaleMode);\n        utils.BaseTextureCache[ video._pixiId ] = baseTexture;\n    }\n\n    return baseTexture;\n};\n\n/**\n * Mimic Pixi BaseTexture.from.... method.\n *\n * This can be used in a couple ways, such as:\n *\n * ```js\n * var texture = PIXI.VideoBaseTexture.fromUrl('http://mydomain.com/video.mp4');\n *\n * var texture = PIXI.VideoBaseTexture.fromUrl({ src: 'http://mydomain.com/video.mp4', mime: 'video/mp4' });\n *\n * var texture = PIXI.VideoBaseTexture.fromUrls(['/video.webm', '/video.mp4']);\n *\n * var texture = PIXI.VideoBaseTexture.fromUrls([\n *     { src: '/video.webm', mime: 'video/webm' },\n *     { src: '/video.mp4', mime: 'video/mp4' }\n * ]);\n * ```\n *\n * @alias fromUrls\n * @static\n * @param videoSrc {string|object|string[]|object[]} The URL(s) for the video.\n * @param [videoSrc.src] {string} One of the source urls for the video\n * @param [videoSrc.mime] {string} The mimetype of the video (e.g. 'video/mp4'). If not specified\n *  the url's extension will be used as the second part of the mime type.\n * @param scaleMode {number} See {@link scaleModes} for possible values\n * @return {VideoBaseTexture}\n */\nVideoBaseTexture.fromUrl = function (videoSrc, scaleMode) {\n    var video = document.createElement('video');\n\n    // array of objects or strings\n    if (Array.isArray(videoSrc)) {\n        for (var i = 0; i < videoSrc.length; ++i) {\n            video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime));\n        }\n    }\n    // single object or string\n    else {\n        video.appendChild(createSource(videoSrc.src || videoSrc, videoSrc.mime));\n    }\n\n    video.load();\n    video.play();\n\n    return VideoBaseTexture.textureFromVideo(video, scaleMode);\n};\n\nVideoBaseTexture.fromUrls = VideoBaseTexture.fromUrl;\n\nfunction createSource(path, type) {\n    if (!type) {\n        type = 'video/' + path.substr(path.lastIndexOf('.') + 1);\n    }\n\n    var source = document.createElement('source');\n\n    source.src = path;\n    source.type = type;\n\n    return source;\n}\n","/**\n * Creates an homogenous object for tracking events so users can know what to expect.\n *\n * @class\n * @namespace PIXI\n * @param target {object} The target object that the event is called on\n * @param name {string} The string name of the event that was triggered\n * @param data {object} Arbitrary event data to pass along\n */\nfunction EventData(target, name, data) {\n    // for duck typing in the \".on()\" function\n    this.__isEventObject = true;\n\n    /**\n     * Tracks the state of bubbling propagation. Do not\n     * set this directly, instead use `event.stopPropagation()`\n     *\n     * @member {boolean}\n     * @private\n     * @readonly\n     */\n    this.stopped = false;\n\n    /**\n     * Tracks the state of sibling listener propagation. Do not\n     * set this directly, instead use `event.stopImmediatePropagation()`\n     *\n     * @member {boolean}\n     * @private\n     * @readonly\n     */\n    this.stoppedImmediate = false;\n\n    /**\n     * The original target the event triggered on.\n     *\n     * @member {object}\n     * @readonly\n     */\n    this.target = target;\n\n    /**\n     * The string name of the event that this represents.\n     *\n     * @member {string}\n     * @readonly\n     */\n    this.type = name;\n\n    /**\n     * The data that was passed in with this event.\n     *\n     * @member {object}\n     * @readonly\n     */\n    this.data = data;\n\n    /**\n     * The timestamp when the event occurred.\n     *\n     * @member {number}\n     * @readonly\n     */\n    this.timeStamp = Date.now();\n}\n\nEventData.prototype.constructor = EventData;\nmodule.exports = EventData;\n\n/**\n * Stops the propagation of events up the scene graph (prevents bubbling).\n *\n */\nEventData.prototype.stopPropagation = function stopPropagation() {\n    this.stopped = true;\n};\n\n/**\n * Stops the propagation of events to sibling listeners (no longer calls any listeners).\n *\n */\nEventData.prototype.stopImmediatePropagation = function stopImmediatePropagation() {\n    this.stoppedImmediate = true;\n};\n","//TODO: Have Graphics use https://github.com/mattdesl/shape2d\n// and https://github.com/mattdesl/shape2d-triangulate instead of custom code.\n\n/*\n    PolyK library\n    url: http://polyk.ivank.net\n    Released under MIT licence.\n\n    Copyright (c) 2012 Ivan Kuckir\n\n    Permission is hereby granted, free of charge, to any person\n    obtaining a copy of this software and associated documentation\n    files (the \"Software\"), to deal in the Software without\n    restriction, including without limitation the rights to use,\n    copy, modify, merge, publish, distribute, sublicense, and/or sell\n    copies of the Software, and to permit persons to whom the\n    Software is furnished to do so, subject to the following\n    conditions:\n\n    The above copyright notice and this permission notice shall be\n    included in all copies or substantial portions of the Software.\n\n    THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND,\n    EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES\n    OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND\n    NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT\n    HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,\n    WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING\n    FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR\n    OTHER DEALINGS IN THE SOFTWARE.\n\n    This is an amazing lib!\n\n    Slightly modified by Mat Groves (matgroves.com);\n*/\n\n/**\n * Based on the Polyk library http://polyk.ivank.net released under MIT licence.\n * This is an amazing lib!\n * Slightly modified by Mat Groves (matgroves.com);\n *\n * @namespace PIXI\n */\nvar PolyK = module.exports = {};\n\n/**\n * Triangulates shapes for webGL graphic fills.\n *\n */\nPolyK.Triangulate = function (p) {\n    var sign = true;\n\n    var n = p.length >> 1;\n    if (n < 3) return [];\n\n    var tgs = [];\n    var avl = [];\n    for (var i = 0; i < n; i++) avl.push(i);\n\n    i = 0;\n    var al = n;\n    while(al > 3) {\n        var i0 = avl[(i+0)%al];\n        var i1 = avl[(i+1)%al];\n        var i2 = avl[(i+2)%al];\n\n        var ax = p[2*i0],  ay = p[2*i0+1];\n        var bx = p[2*i1],  by = p[2*i1+1];\n        var cx = p[2*i2],  cy = p[2*i2+1];\n\n        var earFound = false;\n        if (PolyK._convex(ax, ay, bx, by, cx, cy, sign)) {\n            earFound = true;\n            for (var j = 0; j < al; j++) {\n                var vi = avl[j];\n                if (vi === i0 || vi === i1 || vi === i2) continue;\n\n                if (PolyK._PointInTriangle(p[2*vi], p[2*vi+1], ax, ay, bx, by, cx, cy)) {\n                    earFound = false;\n                    break;\n                }\n            }\n        }\n\n        if (earFound) {\n            tgs.push(i0, i1, i2);\n            avl.splice((i+1)%al, 1);\n            al--;\n            i = 0;\n        }\n        else if (i++ > 3*al) {\n            // need to flip flip reverse it!\n            // reset!\n            if (sign) {\n                tgs = [];\n                avl = [];\n                for (i = 0; i < n; i++) avl.push(i);\n\n                i = 0;\n                al = n;\n\n                sign = false;\n            }\n            else {\n             //   window.console.log(\"PIXI Warning: shape too complex to fill\");\n                return null;\n            }\n        }\n    }\n\n    tgs.push(avl[0], avl[1], avl[2]);\n    return tgs;\n};\n\n/**\n * Checks whether a point is within a triangle\n *\n * @param px {number} x coordinate of the point to test\n * @param py {number} y coordinate of the point to test\n * @param ax {number} x coordinate of the a point of the triangle\n * @param ay {number} y coordinate of the a point of the triangle\n * @param bx {number} x coordinate of the b point of the triangle\n * @param by {number} y coordinate of the b point of the triangle\n * @param cx {number} x coordinate of the c point of the triangle\n * @param cy {number} y coordinate of the c point of the triangle\n * @private\n * @return {boolean}\n */\nPolyK._PointInTriangle = function (px, py, ax, ay, bx, by, cx, cy) {\n    var v0x = cx-ax;\n    var v0y = cy-ay;\n    var v1x = bx-ax;\n    var v1y = by-ay;\n    var v2x = px-ax;\n    var v2y = py-ay;\n\n    var dot00 = v0x*v0x+v0y*v0y;\n    var dot01 = v0x*v1x+v0y*v1y;\n    var dot02 = v0x*v2x+v0y*v2y;\n    var dot11 = v1x*v1x+v1y*v1y;\n    var dot12 = v1x*v2x+v1y*v2y;\n\n    var invDenom = 1 / (dot00 * dot11 - dot01 * dot01);\n    var u = (dot11 * dot02 - dot01 * dot12) * invDenom;\n    var v = (dot00 * dot12 - dot01 * dot02) * invDenom;\n\n    // Check if point is in triangle\n    return (u >= 0) && (v >= 0) && (u + v < 1);\n};\n\n/**\n * Checks whether a shape is convex\n *\n * @private\n * @return {boolean}\n */\nPolyK._convex = function (ax, ay, bx, by, cx, cy, sign) {\n    return ((ay-by)*(cx-bx) + (bx-ax)*(cy-by) >= 0) === sign;\n};\n","var EventData = require('./EventData');\n\n/**\n * Mixins event emitter functionality to an object.\n *\n * @mixin\n * @namespace PIXI\n * @example\n *      function MyEmitter() {}\n *\n *      eventTarget.mixin(MyEmitter.prototype);\n *\n *      var em = new MyEmitter();\n *      em.emit('eventName', 'some data', 'some more data', {}, null, ...);\n */\nfunction eventTarget(obj) {\n    /**\n     * Return a list of assigned event listeners.\n     *\n     * @param eventName {string} The events that should be listed.\n     * @return {Array} An array of listener functions\n     */\n    obj.listeners = function listeners(eventName) {\n        this._listeners = this._listeners || {};\n\n        return this._listeners[eventName] ? this._listeners[eventName].slice() : [];\n    };\n\n    /**\n     * Emit an event to all registered event listeners.\n     *\n     * @alias dispatchEvent\n     * @param eventName {string} The name of the event.\n     * @return {boolean} Indication if we've emitted an event.\n     */\n    obj.emit = obj.dispatchEvent = function emit(eventName, data) {\n        this._listeners = this._listeners || {};\n\n        // fast return when there are no listeners\n        if (!this._listeners[eventName]) {\n            return;\n        }\n\n        //backwards compat with old method \".emit({ type: 'something' })\"\n        if (typeof eventName === 'object') {\n            data = eventName;\n            eventName = eventName.type;\n        }\n\n        //ensure we are using a real pixi event\n        if (!data || data.__isEventObject !== true) {\n            data = new EventData(this, eventName, data);\n        }\n\n        //iterate the listeners\n        var listeners = this._listeners[eventName].slice(0),\n            length = listeners.length,\n            fn = listeners[0],\n            i;\n\n        for (i = 0; i < length; fn = listeners[++i]) {\n            //call the event listener\n            fn.call(this, data);\n\n            //if \"stopImmediatePropagation\" is called, stop calling sibling events\n            if (data.stoppedImmediate) {\n                return this;\n            }\n        }\n\n        //if \"stopPropagation\" is called then don't bubble the event\n        if (data.stopped) {\n            return this;\n        }\n\n        //bubble this event up the scene graph\n        if (this.parent && this.parent.emit) {\n            this.parent.emit.call(this.parent, eventName, data);\n        }\n\n        return this;\n    };\n\n    /**\n     * Register a new EventListener for the given event.\n     *\n     * @alias addEventListener\n     * @param eventName {string} Name of the event.\n     * @param callback {Functon} fn Callback function.\n     */\n    obj.on = obj.addEventListener = function on(eventName, fn) {\n        this._listeners = this._listeners || {};\n\n        (this._listeners[eventName] = this._listeners[eventName] || [])\n            .push(fn);\n\n        return this;\n    };\n\n    /**\n     * Add an EventListener that's only called once.\n     *\n     * @param eventName {string} Name of the event.\n     * @param callback {Function} Callback function.\n     */\n    obj.once = function once(eventName, fn) {\n        this._listeners = this._listeners || {};\n\n        var self = this;\n        function onceHandlerWrapper() {\n            fn.apply(self.off(eventName, onceHandlerWrapper), arguments);\n        }\n        onceHandlerWrapper._originalHandler = fn;\n\n        return this.on(eventName, onceHandlerWrapper);\n    };\n\n    /**\n     * Remove event listeners.\n     *\n     * @alias removeEventListener\n     * @param eventName {string} The event we want to remove.\n     * @param callback {Function} The listener that we need to find.\n     */\n    obj.off = obj.removeEventListener = function off(eventName, fn) {\n        this._listeners = this._listeners || {};\n\n        if (!this._listeners[eventName]) {\n            return this;\n        }\n\n        var list = this._listeners[eventName],\n            i = fn ? list.length : 0;\n\n        while(i-- > 0) {\n            if (list[i] === fn || list[i]._originalHandler === fn) {\n                list.splice(i, 1);\n            }\n        }\n\n        if (list.length === 0) {\n            delete this._listeners[eventName];\n        }\n\n        return this;\n    };\n\n    /**\n     * Remove all listeners or only the listeners for the specified event.\n     *\n     * @param eventName {string} The event you want to remove all listeners for.\n     */\n    obj.removeAllListeners = function removeAllListeners(eventName) {\n        this._listeners = this._listeners || {};\n\n        if (!this._listeners[eventName]) {\n            return this;\n        }\n\n        delete this._listeners[eventName];\n\n        return this;\n    };\n}\n\nmodule.exports = {\n    /**\n     * Mixes in the properties of the eventTarget prototype onto another object\n     *\n     * @param object {object} The obj to mix into\n     */\n    mixin: function mixin(obj) {\n        eventTarget(obj);\n    }\n};\n","var CONST = require('../const');\n\n/**\n * @namespace PIXI\n */\nvar utils = module.exports = {\n    _uid: 0,\n    _saidHello: false,\n\n    PolyK:      require('./PolyK'),\n    EventData:  require('./EventData'),\n    eventTarget: require('./eventTarget'),\n\n    /**\n     * Gets the next uuid\n     *\n     * @return {number} The next uuid to use.\n     */\n    uuid: function () {\n        return ++utils._uid;\n    },\n\n    /**\n     * Converts a hex color number to an [R, G, B] array\n     *\n     * @param hex {number}\n     * @return {number[]} An array representing the [R, G, B] of the color.\n     */\n    hex2rgb: function (hex, out) {\n        out = out || [];\n\n        out[0] = (hex >> 16 & 0xFF) / 255;\n        out[1] = (hex >> 8 & 0xFF) / 255;\n        out[2] = (hex & 0xFF) / 255;\n\n        return out;\n    },\n\n    /**\n     * Converts a hex color number to a string.\n     *\n     * @param hex {number}\n     * @return {string} The string color.\n     */\n    hex2string: function (hex) {\n        hex = hex.toString(16);\n        hex = '000000'.substr(0, 6 - hex.length) + hex;\n\n        return '#' + hex;\n    },\n\n    /**\n     * Converts a color as an [R, G, B] array to a hex number\n     *\n     * @param rgb {number[]}\n     * @return {number} The color number\n     */\n    rgb2hex: function (rgb) {\n        return ((rgb[0]*255 << 16) + (rgb[1]*255 << 8) + rgb[2]*255);\n    },\n\n    /**\n     * Checks whether the Canvas BlendModes are supported by the current browser\n     *\n     * @return {boolean} whether they are supported\n     */\n    canUseNewCanvasBlendModes: function () {\n        if (typeof document === 'undefined') {\n            return false;\n        }\n\n        var canvas = document.createElement('canvas'),\n            context = canvas.getContext('2d');\n\n        canvas.width = 1;\n        canvas.height = 1;\n\n        context.fillStyle = '#000';\n        context.fillRect(0, 0, 1, 1);\n\n        context.globalCompositeOperation = 'multiply';\n\n        context.fillStyle = '#fff';\n        context.fillRect(0, 0, 1, 1);\n\n        return context.getImageData(0,0,1,1).data[0] === 0;\n    },\n\n    /**\n     * Given a number, this function returns the closest number that is a power of two\n     * this function is taken from Starling Framework as its pretty neat ;)\n     *\n     * @param number {number}\n     * @return {number} the closest number that is a power of two\n     */\n    getNextPowerOfTwo: function (number) {\n        // see: http://en.wikipedia.org/wiki/Power_of_two#Fast_algorithm_to_check_if_a_positive_number_is_a_power_of_two\n        if (number > 0 && (number & (number - 1)) === 0) {\n            return number;\n        }\n        else {\n            var result = 1;\n\n            while (result < number) {\n                result <<= 1;\n            }\n\n            return result;\n        }\n    },\n\n    /**\n     * checks if the given width and height make a power of two rectangle\n     *\n     * @param width {number}\n     * @param height {number}\n     * @return {boolean}\n     */\n    isPowerOfTwo: function (width, height) {\n        return (width > 0 && (width & (width - 1)) === 0 && height > 0 && (height & (height - 1)) === 0);\n    },\n\n    /**\n     * Logs out the version and renderer information for this running instance of PIXI.\n     * If you don't want to see this message you can set `PIXI.utils._saidHello = true;`\n     * so the library thinks it already said it. Keep in mind that doing that will forever\n     * makes you a jerk face.\n     *\n     * @param {string} type - The string renderer type to log.\n     * @constant\n     * @static\n     */\n    sayHello: function (type) {\n        if (utils._saidHello) {\n            return;\n        }\n\n        if (navigator.userAgent.toLowerCase().indexOf('chrome') > -1) {\n            var args = [\n                '%c %c %c Pixi.js ' + CONST.VERSION + ' - ' + type + '  %c ' + ' %c ' + ' http://www.pixijs.com/  %c %c ♥%c♥%c♥ ',\n                'background: #ff66a5',\n                'background: #ff66a5',\n                'color: #ff66a5; background: #030307;',\n                'background: #ff66a5',\n                'background: #ffc3dc',\n                'background: #ff66a5',\n                'color: #ff2424; background: #fff',\n                'color: #ff2424; background: #fff',\n                'color: #ff2424; background: #fff'\n            ];\n\n            console.log.apply(console, args); //jshint ignore:line\n        }\n        else if (window.console) {\n            console.log('Pixi.js ' + CONST.VERSION + ' - ' + type + ' - http://www.pixijs.com/'); //jshint ignore:line\n        }\n\n        utils._saidHello = true;\n    },\n\n    /**\n     * A wrapper for ajax requests to be handled cross browser\n     *\n     * TODO: Replace this wil superagent\n     *\n     * @class\n     * @namespace PIXI\n     */\n    AjaxRequest: function () {\n        var activexmodes = ['Msxml2.XMLHTTP.6.0', 'Msxml2.XMLHTTP.3.0', 'Microsoft.XMLHTTP']; //activeX versions to check for in IE\n\n        if (window.ActiveXObject) { //Test for support for ActiveXObject in IE first (as XMLHttpRequest in IE7 is broken)\n            for (var i=0; i<activexmodes.length; i++) {\n                try{\n                    return new window.ActiveXObject(activexmodes[i]);\n                }\n                catch(e) {\n                    //suppress error\n                }\n            }\n        }\n        else if (window.XMLHttpRequest) // if Mozilla, Safari etc\n        {\n            return new window.XMLHttpRequest();\n        }\n        else {\n            return false;\n        }\n    },\n\n    // TODO: refactor out this\n    AnimCache: {},\n    FrameCache: {},\n    TextureCache: {},\n    BaseTextureCache: {}\n};\n","var core = require('../core');\n\n/**\n * A MovieClip is a simple way to display an animation depicted by a list of textures.\n *\n * @class\n * @extends Sprite\n * @namespace PIXI\n * @param textures {Texture[]} an array of {Texture} objects that make up the animation\n */\nfunction MovieClip(textures) {\n    core.Sprite.call(this, textures[0]);\n\n    /**\n     * The array of textures that make up the animation\n     *\n     * @member Texture[]\n     */\n    this.textures = textures;\n\n    /**\n     * The speed that the MovieClip will play at. Higher is faster, lower is slower\n     *\n     * @member number\n     * @default 1\n     */\n    this.animationSpeed = 1;\n\n    /**\n     * Whether or not the movie clip repeats after playing.\n     *\n     * @member boolean\n     * @default true\n     */\n    this.loop = true;\n\n    /**\n     * Function to call when a MovieClip finishes playing\n     *\n     * @method\n     * @memberof MovieClip#\n     */\n    this.onComplete = null;\n\n    /**\n     * The MovieClips current frame index (this may not have to be a whole number)\n     *\n     * @member number\n     * @default 0\n     * @readonly\n     */\n    this.currentFrame = 0;\n\n    /**\n     * Indicates if the MovieClip is currently playing\n     *\n     * @member boolean\n     * @readonly\n     */\n    this.playing = false;\n}\n\n// constructor\nMovieClip.prototype = Object.create(core.Sprite.prototype);\nMovieClip.prototype.constructor = MovieClip;\n\nObject.defineProperties(MovieClip.prototype, {\n    /**\n     * totalFrames is the total number of frames in the MovieClip. This is the same as number of textures\n     * assigned to the MovieClip.\n     *\n     * @member\n     * @memberof MovieClip#\n     * @default 0\n     * @readonly\n     */\n    totalFrames: {\n        get: function() {\n            return this.textures.length;\n        }\n    }\n});\n\n/**\n * Stops the MovieClip\n *\n */\nMovieClip.prototype.stop = function () {\n    this.playing = false;\n};\n\n/**\n * Plays the MovieClip\n *\n */\nMovieClip.prototype.play = function () {\n    this.playing = true;\n};\n\n/**\n * Stops the MovieClip and goes to a specific frame\n *\n * @param frameNumber {number} frame index to stop at\n */\nMovieClip.prototype.gotoAndStop = function (frameNumber) {\n    this.playing = false;\n    this.currentFrame = frameNumber;\n\n    var round = Math.round(this.currentFrame);\n    this.setTexture(this.textures[round % this.textures.length]);\n};\n\n/**\n * Goes to a specific frame and begins playing the MovieClip\n *\n * @param frameNumber {number} frame index to start at\n */\nMovieClip.prototype.gotoAndPlay = function (frameNumber) {\n    this.currentFrame = frameNumber;\n    this.playing = true;\n};\n\n/*\n * Updates the object transform for rendering\n *\n * @private\n */\nMovieClip.prototype.updateTransform = function () {\n    this.displayObjectContainerUpdateTransform();\n\n    if (!this.playing) {\n        return;\n    }\n\n    this.currentFrame += this.animationSpeed;\n\n    var round = Math.round(this.currentFrame);\n\n    if (round < 0) {\n        if (this.loop) {\n            this.currentFrame += this.textures.length;\n            this.texture = this.textures[this.currentFrame];\n        }\n        else {\n            this.gotoAndStop(0);\n\n            if (this.onComplete) {\n                this.onComplete();\n            }\n        }\n    }\n    else if (this.loop || round < this.textures.length) {\n        this.texture = this.textures[round % this.textures.length];\n    }\n    else if (round >= this.textures.length) {\n        this.gotoAndStop(this.textures.length - 1);\n\n        if (this.onComplete) {\n            this.onComplete();\n        }\n    }\n};\n\n/**\n * A short hand way of creating a movieclip from an array of frame ids\n *\n * @static\n * @param frames {string[]} the array of frames ids the movieclip will use as its texture frames\n */\nMovieClip.fromFrames = function (frames) {\n    var textures = [];\n\n    for (var i = 0; i < frames.length; ++i) {\n        textures.push(new core.Texture.fromFrame(frames[i]));\n    }\n\n    return new MovieClip(textures);\n};\n\n/**\n * A short hand way of creating a movieclip from an array of image ids\n *\n * @static\n * @param images {string[]} the array of image urls the movieclip will use as its texture frames\n */\nMovieClip.fromImages = function (images) {\n    var textures = [];\n\n    for (var i = 0; i < images.length; ++i) {\n        textures.push(new core.Texture.fromImage(images[i]));\n    }\n\n    return new MovieClip(textures);\n};\n","var Strip = require('./Strip');\n\n/**\n *\n * @class\n * @namespace PIXI\n * @extends Strip\n * @param {Texture} texture - The texture to use on the rope.\n * @param {Array} points - An array of {Point}.\n *\n */\nfunction Rope(texture, points) {\n    Strip.call(this, texture);\n    this.points = points;\n\n    this.vertices = new Float32Array(points.length * 4);\n    this.uvs = new Float32Array(points.length * 4);\n    this.colors = new Float32Array(points.length * 2);\n    this.indices = new Uint16Array(points.length * 2);\n\n    this.refresh();\n}\n\n\n// constructor\nRope.prototype = Object.create(Strip.prototype);\nRope.prototype.constructor = Rope;\nmodule.exports = Rope;\n\n/**\n * Refreshes\n *\n */\nRope.prototype.refresh = function () {\n    var points = this.points;\n\n    if (points.length < 1) {\n        return;\n    }\n\n    var uvs = this.uvs;\n\n    var indices = this.indices;\n    var colors = this.colors;\n\n    // this.count -= 0.2;\n\n    uvs[0] = 0;\n    uvs[1] = 0;\n    uvs[2] = 0;\n    uvs[3] = 1;\n\n    colors[0] = 1;\n    colors[1] = 1;\n\n    indices[0] = 0;\n    indices[1] = 1;\n\n    var total = points.length,\n        point, index, amount;\n\n    for (var i = 1; i < total; i++) {\n        point = points[i];\n        index = i * 4;\n        // time to do some smart drawing!\n        amount = i / (total-1);\n\n        if (i%2) {\n            uvs[index] = amount;\n            uvs[index+1] = 0;\n\n            uvs[index+2] = amount;\n            uvs[index+3] = 1;\n        }\n        else {\n            uvs[index] = amount;\n            uvs[index+1] = 0;\n\n            uvs[index+2] = amount;\n            uvs[index+3] = 1;\n        }\n\n        index = i * 2;\n        colors[index] = 1;\n        colors[index+1] = 1;\n\n        index = i * 2;\n        indices[index] = index;\n        indices[index + 1] = index + 1;\n    }\n};\n\n/*\n * Updates the object transform for rendering\n *\n * @private\n */\nRope.prototype.updateTransform = function () {\n    var points = this.points;\n\n    if (points.length < 1) {\n        return;\n    }\n\n    var lastPoint = points[0];\n    var nextPoint;\n    var perpX = 0;\n    var perpY = 0;\n\n    // this.count -= 0.2;\n\n    var vertices = this.vertices;\n    var total = points.length,\n        point, index, ratio, perpLength, num;\n\n    for (var i = 0; i < total; i++) {\n        point = points[i];\n        index = i * 4;\n\n        if (i < points.length-1) {\n            nextPoint = points[i+1];\n        }\n        else {\n            nextPoint = point;\n        }\n\n        perpY = -(nextPoint.x - lastPoint.x);\n        perpX = nextPoint.y - lastPoint.y;\n\n        ratio = (1 - (i / (total-1))) * 10;\n\n        if (ratio > 1) {\n            ratio = 1;\n        }\n\n        perpLength = Math.sqrt(perpX * perpX + perpY * perpY);\n        num = this.texture.height / 2; //(20 + Math.abs(Math.sin((i + this.count) * 0.3) * 50) )* ratio;\n        perpX /= perpLength;\n        perpY /= perpLength;\n\n        perpX *= num;\n        perpY *= num;\n\n        vertices[index] = point.x + perpX;\n        vertices[index+1] = point.y + perpY;\n        vertices[index+2] = point.x - perpX;\n        vertices[index+3] = point.y - perpY;\n\n        lastPoint = point;\n    }\n\n    this.displayObjectContainerUpdateTransform();\n};\n\n/**\n * Sets the texture that the Rope will use\n *\n * @param texture {Texture} the texture that will be used\n */\nRope.prototype.setTexture = function (texture) {\n    // stop current texture\n    this.texture = texture;\n    //this.updateFrame = true;\n};\n","var core = require('../core');\n\n/**\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param texture {Texture} The texture to use\n * @param width {number} the width\n * @param height {number} the height\n *\n */\nfunction Strip(texture) {\n    core.DisplayObjectContainer.call(this);\n\n    /**\n     * The texture of the strip\n     *\n     * @member {Texture}\n     */\n    this.texture = texture;\n\n    // set up the main bits..\n    this.uvs = new Float32Array([0, 1,\n                                 1, 1,\n                                 1, 0,\n                                 0, 1]);\n\n    this.vertices = new Float32Array([0, 0,\n                                      100, 0,\n                                      100, 100,\n                                      0, 100]);\n\n    this.colors = new Float32Array([1, 1, 1, 1]);\n\n    this.indices = new Uint16Array([0, 1, 2, 3]);\n\n    /**\n     * Whether the strip is dirty or not\n     *\n     * @member {boolean}\n     */\n    this.dirty = true;\n\n    /**\n     * The blend mode to be applied to the sprite. Set to blendModes.NORMAL to remove any blend mode.\n     *\n     * @member {number}\n     * @default CONST.blendModes.NORMAL;\n     */\n    this.blendMode = core.CONST.blendModes.NORMAL;\n\n    /**\n     * Triangles in canvas mode are automatically antialiased, use this value to force triangles to overlap a bit with each other.\n     *\n     * @member {number}\n     */\n    this.canvasPadding = 0;\n\n    this.drawMode = Strip.DrawModes.TRIANGLE_STRIP;\n}\n\n// constructor\nStrip.prototype = Object.create(core.DisplayObjectContainer.prototype);\nStrip.prototype.constructor = Strip;\nmodule.exports = Strip;\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer}\n */\nStrip.prototype.renderWebGL = function (renderer) {\n    // if the sprite is not visible or the alpha is 0 then no need to render this element\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    // render triangle strip..\n\n    renderer.spriteBatch.stop();\n\n    // init! init!\n    if (!this._vertexBuffer) {\n        this._initWebGL(renderer);\n    }\n\n    renderer.shaderManager.setShader(renderer.shaderManager.stripShader);\n\n    this._renderStrip(renderer);\n\n    ///renderer.shaderManager.activateDefaultShader();\n\n    renderer.spriteBatch.start();\n\n    //TODO check culling\n};\n\nStrip.prototype._initWebGL = function (renderer) {\n    // build the strip!\n    var gl = renderer.gl;\n\n    this._vertexBuffer = gl.createBuffer();\n    this._indexBuffer = gl.createBuffer();\n    this._uvBuffer = gl.createBuffer();\n    this._colorBuffer = gl.createBuffer();\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.DYNAMIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER,  this.uvs, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ARRAY_BUFFER, this._colorBuffer);\n    gl.bufferData(gl.ARRAY_BUFFER, this.colors, gl.STATIC_DRAW);\n\n    gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer);\n    gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n};\n\nStrip.prototype._renderStrip = function (renderer) {\n    var gl = renderer.gl;\n    var projection = renderer.projection,\n        offset = renderer.offset,\n        shader = renderer.shaderManager.stripShader;\n\n    var drawMode = this.drawMode === Strip.DrawModes.TRIANGLE_STRIP ? gl.TRIANGLE_STRIP : gl.TRIANGLES;\n\n    // gl.uniformMatrix4fv(shaderProgram.mvMatrixUniform, false, mat4Real);\n\n    renderer.blendModeManager.setBlendMode(this.blendMode);\n\n\n    // set uniforms\n    gl.uniformMatrix3fv(shader.translationMatrix, false, this.worldTransform.toArray(true));\n    gl.uniform2f(shader.projectionVector, projection.x, -projection.y);\n    gl.uniform2f(shader.offsetVector, -offset.x, -offset.y);\n    gl.uniform1f(shader.alpha, this.worldAlpha);\n\n    if (!this.dirty) {\n\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer);\n        gl.bufferSubData(gl.ARRAY_BUFFER, 0, this.vertices);\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0);\n\n        // update the uvs\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer);\n        gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0);\n\n        gl.activeTexture(gl.TEXTURE0);\n\n        // check if a texture is dirty..\n        if (this.texture.baseTexture._dirty[gl.id]) {\n            renderer.updateTexture(this.texture.baseTexture);\n        }\n        else {\n            // bind the current texture\n            gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]);\n        }\n\n        // dont need to upload!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer);\n\n\n    }\n    else {\n\n        this.dirty = false;\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._vertexBuffer);\n        gl.bufferData(gl.ARRAY_BUFFER, this.vertices, gl.STATIC_DRAW);\n        gl.vertexAttribPointer(shader.aVertexPosition, 2, gl.FLOAT, false, 0, 0);\n\n        // update the uvs\n        gl.bindBuffer(gl.ARRAY_BUFFER, this._uvBuffer);\n        gl.bufferData(gl.ARRAY_BUFFER, this.uvs, gl.STATIC_DRAW);\n        gl.vertexAttribPointer(shader.aTextureCoord, 2, gl.FLOAT, false, 0, 0);\n\n        gl.activeTexture(gl.TEXTURE0);\n\n        // check if a texture is dirty..\n        if (this.texture.baseTexture._dirty[gl.id]) {\n            renderer.updateTexture(this.texture.baseTexture);\n        }\n        else {\n            gl.bindTexture(gl.TEXTURE_2D, this.texture.baseTexture._glTextures[gl.id]);\n        }\n\n        // dont need to upload!\n        gl.bindBuffer(gl.ELEMENT_ARRAY_BUFFER, this._indexBuffer);\n        gl.bufferData(gl.ELEMENT_ARRAY_BUFFER, this.indices, gl.STATIC_DRAW);\n\n    }\n    //console.log(gl.TRIANGLE_STRIP)\n    //\n    //\n    gl.drawElements(drawMode, this.indices.length, gl.UNSIGNED_SHORT, 0);\n\n\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer}\n */\nStrip.prototype.renderCanvas = function (renderer) {\n    var context = renderer.context;\n\n    var transform = this.worldTransform;\n\n    if (renderer.roundPixels) {\n        context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx | 0, transform.ty | 0);\n    }\n    else {\n        context.setTransform(transform.a, transform.b, transform.c, transform.d, transform.tx, transform.ty);\n    }\n\n    if (this.drawMode === Strip.DrawModes.TRIANGLE_STRIP) {\n        this._renderCanvasTriangleStrip(context);\n    }\n    else {\n        this._renderCanvasTriangles(context);\n    }\n};\n\nStrip.prototype._renderCanvasTriangleStrip = function (context) {\n    // draw triangles!!\n    var vertices = this.vertices;\n    var uvs = this.uvs;\n\n    var length = vertices.length / 2;\n    // this.count++;\n\n    for (var i = 0; i < length - 2; i++) {\n        // draw some triangles!\n        var index = i * 2;\n        this._renderCanvasDrawTriangle(context, vertices, uvs, index, (index + 2), (index + 4));\n    }\n};\n\nStrip.prototype._renderCanvasTriangles = function (context) {\n    // draw triangles!!\n    var vertices = this.vertices;\n    var uvs = this.uvs;\n    var indices = this.indices;\n\n    var length = indices.length;\n    // this.count++;\n\n    for (var i = 0; i < length; i += 3) {\n        // draw some triangles!\n        var index0 = indices[i] * 2, index1 = indices[i + 1] * 2, index2 = indices[i + 2] * 2;\n        this._renderCanvasDrawTriangle(context, vertices, uvs, index0, index1, index2);\n    }\n};\n\nStrip.prototype._renderCanvasDrawTriangle = function (context, vertices, uvs, index0, index1, index2) {\n    var textureSource = this.texture.baseTexture.source;\n    var textureWidth = this.texture.width;\n    var textureHeight = this.texture.height;\n\n    var x0 = vertices[index0], x1 = vertices[index1], x2 = vertices[index2];\n    var y0 = vertices[index0 + 1], y1 = vertices[index1 + 1], y2 = vertices[index2 + 1];\n\n    var u0 = uvs[index0] * textureWidth, u1 = uvs[index1] * textureWidth, u2 = uvs[index2] * textureWidth;\n    var v0 = uvs[index0 + 1] * textureHeight, v1 = uvs[index1 + 1] * textureHeight, v2 = uvs[index2 + 1] * textureHeight;\n\n    if (this.canvasPadding > 0) {\n        var paddingX = this.canvasPadding / this.worldTransform.a;\n        var paddingY = this.canvasPadding / this.worldTransform.d;\n        var centerX = (x0 + x1 + x2) / 3;\n        var centerY = (y0 + y1 + y2) / 3;\n\n        var normX = x0 - centerX;\n        var normY = y0 - centerY;\n\n        var dist = Math.sqrt(normX * normX + normY * normY);\n        x0 = centerX + (normX / dist) * (dist + paddingX);\n        y0 = centerY + (normY / dist) * (dist + paddingY);\n\n        //\n\n        normX = x1 - centerX;\n        normY = y1 - centerY;\n\n        dist = Math.sqrt(normX * normX + normY * normY);\n        x1 = centerX + (normX / dist) * (dist + paddingX);\n        y1 = centerY + (normY / dist) * (dist + paddingY);\n\n        normX = x2 - centerX;\n        normY = y2 - centerY;\n\n        dist = Math.sqrt(normX * normX + normY * normY);\n        x2 = centerX + (normX / dist) * (dist + paddingX);\n        y2 = centerY + (normY / dist) * (dist + paddingY);\n    }\n\n    context.save();\n    context.beginPath();\n\n\n    context.moveTo(x0, y0);\n    context.lineTo(x1, y1);\n    context.lineTo(x2, y2);\n\n    context.closePath();\n\n    context.clip();\n\n    // Compute matrix transform\n    var delta =  (u0 * v1)      + (v0 * u2)      + (u1 * v2)      - (v1 * u2)      - (v0 * u1)      - (u0 * v2);\n    var deltaA = (x0 * v1)      + (v0 * x2)      + (x1 * v2)      - (v1 * x2)      - (v0 * x1)      - (x0 * v2);\n    var deltaB = (u0 * x1)      + (x0 * u2)      + (u1 * x2)      - (x1 * u2)      - (x0 * u1)      - (u0 * x2);\n    var deltaC = (u0 * v1 * x2) + (v0 * x1 * u2) + (x0 * u1 * v2) - (x0 * v1 * u2) - (v0 * u1 * x2) - (u0 * x1 * v2);\n    var deltaD = (y0 * v1)      + (v0 * y2)      + (y1 * v2)      - (v1 * y2)      - (v0 * y1)      - (y0 * v2);\n    var deltaE = (u0 * y1)      + (y0 * u2)      + (u1 * y2)      - (y1 * u2)      - (y0 * u1)      - (u0 * y2);\n    var deltaF = (u0 * v1 * y2) + (v0 * y1 * u2) + (y0 * u1 * v2) - (y0 * v1 * u2) - (v0 * u1 * y2) - (u0 * y1 * v2);\n\n    context.transform(deltaA / delta, deltaD / delta,\n        deltaB / delta, deltaE / delta,\n        deltaC / delta, deltaF / delta);\n\n    context.drawImage(textureSource, 0, 0);\n    context.restore();\n};\n\n\n\n/**\n * Renders a flat strip\n *\n * @param strip {Strip} The Strip to render\n * @private\n */\nStrip.prototype.renderStripFlat = function (strip) {\n    var context = this.context;\n    var vertices = strip.vertices;\n\n    var length = vertices.length/2;\n    // this.count++;\n\n    context.beginPath();\n    for (var i=1; i < length-2; i++) {\n        // draw some triangles!\n        var index = i*2;\n\n        var x0 = vertices[index],   x1 = vertices[index+2], x2 = vertices[index+4];\n        var y0 = vertices[index+1], y1 = vertices[index+3], y2 = vertices[index+5];\n\n        context.moveTo(x0, y0);\n        context.lineTo(x1, y1);\n        context.lineTo(x2, y2);\n    }\n\n    context.fillStyle = '#FF0000';\n    context.fill();\n    context.closePath();\n};\n\n/*\nStrip.prototype.setTexture = function (texture) {\n    //TODO SET THE TEXTURES\n    //TODO VISIBILITY\n\n    // stop current texture\n    this.texture = texture;\n    this.width   = texture.frame.width;\n    this.height  = texture.frame.height;\n    this.updateFrame = true;\n};\n */\n\n/**\n * When the texture is updated, this event will fire to update the scale and frame\n *\n * @param event\n * @private\n */\n\nStrip.prototype.onTextureUpdate = function () {\n    this.updateFrame = true;\n};\n\n/**\n * Returns the bounds of the mesh as a rectangle. The bounds calculation takes the worldTransform into account.\n *\n * @param matrix {Matrix} the transformation matrix of the sprite\n * @return {Rectangle} the framing rectangle\n */\nStrip.prototype.getBounds = function (matrix) {\n    var worldTransform = matrix || this.worldTransform;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var maxX = -Infinity;\n    var maxY = -Infinity;\n\n    var minX = Infinity;\n    var minY = Infinity;\n\n    var vertices = this.vertices;\n    for (var i = 0, n = vertices.length; i < n; i += 2) {\n        var rawX = vertices[i], rawY = vertices[i + 1];\n        var x = (a * rawX) + (c * rawY) + tx;\n        var y = (d * rawY) + (b * rawX) + ty;\n\n        minX = x < minX ? x : minX;\n        minY = y < minY ? y : minY;\n\n        maxX = x > maxX ? x : maxX;\n        maxY = y > maxY ? y : maxY;\n    }\n\n    if (minX === -Infinity || maxY === Infinity) {\n        return core.math.Rectangle.EMPTY;\n    }\n\n    var bounds = this._bounds;\n\n    bounds.x = minX;\n    bounds.width = maxX - minX;\n\n    bounds.y = minY;\n    bounds.height = maxY - minY;\n\n    // store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    this._currentBounds = bounds;\n\n    return bounds;\n};\n\n/**\n * Different drawing buffer modes supported\n *\n * @static\n * @constant\n * @property {object} DrawModes\n * @property {number} DrawModes.TRIANGLE_STRIP\n * @property {number} DrawModes.TRIANGLES\n */\nStrip.DrawModes = {\n    TRIANGLE_STRIP: 0,\n    TRIANGLES: 1\n};\n","var core = require('../core');\n\n/**\n * A tiling sprite is a fast way of rendering a tiling image\n *\n * @class\n * @extends Sprite\n * @namespace PIXI\n * @param texture {Texture} the texture of the tiling sprite\n * @param width {number}  the width of the tiling sprite\n * @param height {number} the height of the tiling sprite\n */\nfunction TilingSprite(texture, width, height) {\n    core.Sprite.call( this, texture);\n\n    /**\n     * The with of the tiling sprite\n     *\n     * @member {number}\n     */\n    this._width = width || 100;\n\n    /**\n     * The height of the tiling sprite\n     *\n     * @member {number}\n     */\n    this._height = height || 100;\n\n    /**\n     * The scaling of the image that is being tiled\n     *\n     * @member {Point}\n     */\n    this.tileScale = new core.math.Point(1,1);\n\n    /**\n     * A point that represents the scale of the texture object\n     *\n     * @member {Point}\n     */\n    this.tileScaleOffset = new core.math.Point(1,1);\n\n    /**\n     * The offset position of the image that is being tiled\n     *\n     * @member {Point}\n     */\n    this.tilePosition = new core.math.Point(0,0);\n\n    /**\n     * Whether this sprite is renderable or not\n     *\n     * @member {boolean}\n     * @default true\n     */\n    this.renderable = true;\n\n    /**\n     * The tint applied to the sprite. This is a hex value\n     *\n     * @member {number}\n     * @default 0xFFFFFF\n     */\n    this.tint = 0xFFFFFF;\n\n    /**\n     * The blend mode to be applied to the sprite\n     *\n     * @member {number}\n     * @default blendModes.NORMAL;\n     */\n    this.blendMode = core.CONST.blendModes.NORMAL;\n}\n\nTilingSprite.prototype = Object.create(core.Sprite.prototype);\nTilingSprite.prototype.constructor = TilingSprite;\nmodule.exports = TilingSprite;\n\n\nObject.defineProperties(TilingSprite.prototype, {\n    /**\n     * The width of the sprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof TilingSprite#\n     */\n    width: {\n        get: function () {\n            return this._width;\n        },\n        set: function (value) {\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the TilingSprite, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof TilingSprite#\n     */\n    height: {\n        get: function () {\n            return this._height;\n        },\n        set: function (value) {\n            this._height = value;\n        }\n    },\n\n    texture: {\n        get: function () {\n            return this._texture;\n        },\n        set: function (value) {\n            if (this._texture === value) {\n                return;\n            }\n\n            this._texture = value;\n            this.refreshTexture = true;\n            this.cachedTint = 0xFFFFFF;\n        }\n    }\n});\n\n/**\n * Renders the object using the WebGL renderer\n *\n * @param renderer {WebGLRenderer}\n */\nTilingSprite.prototype.renderWebGL = function (renderer) {\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    var i, j;\n\n    if (this._mask) {\n        renderer.spriteBatch.stop();\n        renderer.maskManager.pushMask(this.mask, renderer);\n        renderer.spriteBatch.start();\n    }\n\n    if (this._filters) {\n        renderer.spriteBatch.flush();\n        renderer.filterManager.pushFilter(this._filterBlock);\n    }\n\n\n\n    if (!this.tilingTexture || this.refreshTexture) {\n        this.generateTilingTexture(true);\n\n        if (this.tilingTexture && this.tilingTexture.needsUpdate) {\n            //TODO - tweaking\n            renderer.updateTexture(this.tilingTexture.baseTexture);\n            this.tilingTexture.needsUpdate = false;\n           // this.tilingTexture._uvs = null;\n        }\n    }\n    else {\n        renderer.spriteBatch.renderTilingSprite(this);\n    }\n    // simple render children!\n    for (i=0,j=this.children.length; i<j; i++) {\n        this.children[i].renderWebGL(renderer);\n    }\n\n    renderer.spriteBatch.stop();\n\n    if (this._filters) {\n        renderer.filterManager.popFilter();\n    }\n\n    if (this._mask) {\n        renderer.maskManager.popMask(this._mask, renderer);\n    }\n\n    renderer.spriteBatch.start();\n};\n\n/**\n * Renders the object using the Canvas renderer\n *\n * @param renderer {CanvasRenderer}\n */\nTilingSprite.prototype.renderCanvas = function (renderer) {\n    if (!this.visible || this.alpha <= 0) {\n        return;\n    }\n\n    var context = renderer.context;\n\n    if (this._mask) {\n        renderer.maskManager.pushMask(this._mask, context);\n    }\n\n    context.globalAlpha = this.worldAlpha;\n\n    var transform = this.worldTransform;\n\n    var i,j;\n\n    var resolution = renderer.resolution;\n\n    context.setTransform(transform.a * resolution,\n                         transform.b * resolution,\n                         transform.c * resolution,\n                         transform.d * resolution,\n                         transform.tx * resolution,\n                         transform.ty * resolution);\n\n    if (!this.__tilePattern ||  this.refreshTexture) {\n        this.generateTilingTexture(false);\n\n        if (this.tilingTexture) {\n            this.__tilePattern = context.createPattern(this.tilingTexture.baseTexture.source, 'repeat');\n        }\n        else {\n            return;\n        }\n    }\n\n    // check blend mode\n    if (this.blendMode !== renderer.currentBlendMode) {\n        renderer.currentBlendMode = this.blendMode;\n        context.globalCompositeOperation = renderer.blendModes[renderer.currentBlendMode];\n    }\n\n    var tilePosition = this.tilePosition;\n    var tileScale = this.tileScale;\n\n    tilePosition.x %= this.tilingTexture.baseTexture.width;\n    tilePosition.y %= this.tilingTexture.baseTexture.height;\n\n    // offset - make sure to account for the anchor point..\n    context.scale(tileScale.x,tileScale.y);\n    context.translate(tilePosition.x + (this.anchor.x * -this._width), tilePosition.y + (this.anchor.y * -this._height));\n\n    context.fillStyle = this.__tilePattern;\n\n    context.fillRect(-tilePosition.x,\n                    -tilePosition.y,\n                    this._width / tileScale.x,\n                    this._height / tileScale.y);\n\n    context.translate(-tilePosition.x + (this.anchor.x * this._width), -tilePosition.y + (this.anchor.y * this._height));\n    context.scale(1 / tileScale.x, 1 / tileScale.y);\n\n    if (this._mask) {\n        renderer.maskManager.popMask(renderer.context);\n    }\n\n    for (i = 0, j = this.children.length; i < j; ++i) {\n        this.children[i].renderCanvas(renderer);\n    }\n};\n\n\n/**\n * Returns the framing rectangle of the sprite as a Rectangle object\n*\n * @return {Rectangle} the framing rectangle\n */\nTilingSprite.prototype.getBounds = function () {\n    var width = this._width;\n    var height = this._height;\n\n    var w0 = width * (1-this.anchor.x);\n    var w1 = width * -this.anchor.x;\n\n    var h0 = height * (1-this.anchor.y);\n    var h1 = height * -this.anchor.y;\n\n    var worldTransform = this.worldTransform;\n\n    var a = worldTransform.a;\n    var b = worldTransform.b;\n    var c = worldTransform.c;\n    var d = worldTransform.d;\n    var tx = worldTransform.tx;\n    var ty = worldTransform.ty;\n\n    var x1 = a * w1 + c * h1 + tx;\n    var y1 = d * h1 + b * w1 + ty;\n\n    var x2 = a * w0 + c * h1 + tx;\n    var y2 = d * h1 + b * w0 + ty;\n\n    var x3 = a * w0 + c * h0 + tx;\n    var y3 = d * h0 + b * w0 + ty;\n\n    var x4 =  a * w1 + c * h0 + tx;\n    var y4 =  d * h0 + b * w1 + ty;\n\n    var minX,\n        maxX,\n        minY,\n        maxY;\n\n    minX = x1;\n    minX = x2 < minX ? x2 : minX;\n    minX = x3 < minX ? x3 : minX;\n    minX = x4 < minX ? x4 : minX;\n\n    minY = y1;\n    minY = y2 < minY ? y2 : minY;\n    minY = y3 < minY ? y3 : minY;\n    minY = y4 < minY ? y4 : minY;\n\n    maxX = x1;\n    maxX = x2 > maxX ? x2 : maxX;\n    maxX = x3 > maxX ? x3 : maxX;\n    maxX = x4 > maxX ? x4 : maxX;\n\n    maxY = y1;\n    maxY = y2 > maxY ? y2 : maxY;\n    maxY = y3 > maxY ? y3 : maxY;\n    maxY = y4 > maxY ? y4 : maxY;\n\n    var bounds = this._bounds;\n\n    bounds.x = minX;\n    bounds.width = maxX - minX;\n\n    bounds.y = minY;\n    bounds.height = maxY - minY;\n\n    // store a reference so that if this function gets called again in the render cycle we do not have to recalculate\n    this._currentBounds = bounds;\n\n    return bounds;\n};\n\n/**\n * When the texture is updated, this event will fire to update the scale and frame\n *\n * @param event\n * @private\n */\nTilingSprite.prototype.onTextureUpdate = function () {\n   // overriding the sprite version of this!\n};\n\n/**\n *\n * @param forcePowerOfTwo {boolean} Whether we want to force the texture to be a power of two\n */\nTilingSprite.prototype.generateTilingTexture = function (forcePowerOfTwo) {\n    if (!this.texture.baseTexture.hasLoaded) {\n        return;\n    }\n\n    var texture = this.originalTexture || this.texture;\n    var frame = texture.frame;\n    var targetWidth, targetHeight;\n\n    //  Check that the frame is the same size as the base texture.\n    var isFrame = frame.width !== texture.baseTexture.width || frame.height !== texture.baseTexture.height;\n\n    var newTextureRequired = false;\n\n    if (!forcePowerOfTwo) {\n        if (isFrame) {\n            targetWidth = frame.width;\n            targetHeight = frame.height;\n\n            newTextureRequired = true;\n        }\n    }\n    else {\n        targetWidth = core.utils.getNextPowerOfTwo(frame.width);\n        targetHeight = core.utils.getNextPowerOfTwo(frame.height);\n\n        //  If the BaseTexture dimensions don't match the texture frame then we need a new texture anyway because it's part of a texture atlas\n        if (frame.width !== targetWidth || frame.height !== targetHeight || texture.baseTexture.width !== targetWidth || texture.baseTexture.height || targetHeight) {\n            newTextureRequired = true;\n        }\n    }\n\n    if (newTextureRequired) {\n        var canvasBuffer;\n\n        if (this.tilingTexture && this.tilingTexture.isTiling) {\n            canvasBuffer = this.tilingTexture.canvasBuffer;\n            canvasBuffer.resize(targetWidth, targetHeight);\n            this.tilingTexture.baseTexture.width = targetWidth;\n            this.tilingTexture.baseTexture.height = targetHeight;\n            this.tilingTexture.needsUpdate = true;\n        }\n        else {\n            canvasBuffer = new core.CanvasBuffer(targetWidth, targetHeight);\n\n            this.tilingTexture = core.Texture.fromCanvas(canvasBuffer.canvas);\n            this.tilingTexture.canvasBuffer = canvasBuffer;\n            this.tilingTexture.isTiling = true;\n        }\n\n        canvasBuffer.context.drawImage(texture.baseTexture.source,\n                               texture.crop.x,\n                               texture.crop.y,\n                               texture.crop.width,\n                               texture.crop.height,\n                               0,\n                               0,\n                               targetWidth,\n                               targetHeight);\n\n        this.tileScaleOffset.x = frame.width / targetWidth;\n        this.tileScaleOffset.y = frame.height / targetHeight;\n    }\n    else {\n        //  TODO - switching?\n        if (this.tilingTexture && this.tilingTexture.isTiling) {\n            // destroy the tiling texture!\n            // TODO could store this somewhere?\n            this.tilingTexture.destroy(true);\n        }\n\n        this.tileScaleOffset.x = 1;\n        this.tileScaleOffset.y = 1;\n        this.tilingTexture = texture;\n    }\n\n    this.refreshTexture = false;\n\n    this.originalTexture = this.texture;\n    this.texture = this.tilingTexture;\n\n    this.tilingTexture.baseTexture._powerOf2 = true;\n};\n","/**\n * @file        Main export of the PIXI extras library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    MovieClip:      require('./MovieClip'),\n    Rope:           require('./Rope'),\n    Strip:          require('./Strip'),\n    TilingSprite:   require('./TilingSprite')\n};\n","/**\n * This is the base class for creating a PIXI filter. Currently only WebGL supports filters.\n * If you want to make a custom filter this should be your base class.\n *\n * @class\n * @namespace PIXI\n * @param fragmentSrc {string|string[]} The fragment source in an array of strings.\n * @param uniforms {object} An object containing the uniforms for this filter.\n */\nfunction AbstractFilter(fragmentSrc, uniforms) {\n    /**\n     * An array of passes - some filters contain a few steps this array simply stores the steps in a liniear fashion.\n     * For example the blur filter has two passes blurX and blurY.\n     *\n     * @member {AbstractFilter[]}\n     * @private\n     */\n    this.passes = [this];\n\n    /**\n     * @member {Shader[]}\n     * @private\n     */\n    this.shaders = [];\n\n    /**\n     * @member {number}\n     */\n    this.padding = 0;\n\n    /**\n     * @member {object}\n     * @private\n     */\n    this.uniforms = uniforms || {};\n\n    /**\n     * @member {string[]}\n     * @private\n     */\n    this.fragmentSrc = typeof fragmentSrc === 'string' ? fragmentSrc.split('') : (fragmentSrc || []);\n}\n\nAbstractFilter.prototype.constructor = AbstractFilter;\nmodule.exports = AbstractFilter;\n\n/**\n * Syncs the uniforms between the class object and the shaders.\n *\n */\nAbstractFilter.prototype.syncUniforms = function () {\n    for (var i = 0, j = this.shaders.length; i < j; ++i) {\n        this.shaders[i].dirty = true;\n    }\n};\n\n/*\nAbstractFilter.prototype.apply = function (frameBuffer) {\n    // TODO :)\n};\n*/\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The AlphaMaskFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object.\n * You can use this filter to apply all manor of crazy warping effects\n * Currently the r property of the texture is used to offset the x and the g property of the texture is used to offset the y.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment\n */\nfunction AlphaMaskFilter(texture) {\n    AbstractFilter.call(this);\n\n    texture.baseTexture._powerOf2 = true;\n\n    // set the uniforms\n    this.uniforms = {\n        mask:           { type: 'sampler2D',    value: texture },\n        mapDimensions:  { type: '2f',           value: { x: 1, y: 5112 } },\n        dimensions:     { type: '4fv',          value: [0, 0, 0, 0] },\n        offset:         { type: '2f',           value: { x: 0, y: 0 } }\n    };\n\n    if (texture.baseTexture.hasLoaded) {\n        this.uniforms.mask.value.x = texture.width;\n        this.uniforms.mask.value.y = texture.height;\n    }\n    else {\n        this.boundLoadedFunction = this.onTextureLoaded.bind(this);\n\n        texture.baseTexture.on('loaded', this.boundLoadedFunction);\n    }\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D uSampler;',\n        'uniform sampler2D mask;',\n        'uniform vec2 mapDimensions;',\n        'uniform vec4 dimensions;',\n        'uniform vec2 offset;',\n\n        'void main() {',\n        '   vec2 mapCords = vTextureCoord.xy;',\n        '   mapCords += (dimensions.zw + offset)/ dimensions.xy ;',\n        '   mapCords.y *= -1.0;',\n        '   mapCords.y += 1.0;',\n        '   mapCords *= dimensions.xy / mapDimensions;',\n\n        '   vec4 original =  texture2D(uSampler, vTextureCoord);',\n        '   float maskAlpha =  texture2D(mask, mapCords).r;',\n        '   original *= maskAlpha;',\n        //'   original.rgb *= maskAlpha;',\n        '   gl_FragColor =  original;',\n        //'   gl_FragColor = gl_FragColor;',\n        '}'\n    ];\n}\n\nAlphaMaskFilter.prototype = Object.create(AbstractFilter.prototype);\nAlphaMaskFilter.prototype.constructor = AlphaMaskFilter;\nmodule.exports = AlphaMaskFilter;\n\n/**\n * Sets the map dimensions uniforms when the texture becomes available.\n *\n */\nAlphaMaskFilter.prototype.onTextureLoaded = function () {\n    this.uniforms.mapDimensions.value.x = this.uniforms.mask.value.width;\n    this.uniforms.mapDimensions.value.y = this.uniforms.mask.value.height;\n\n    this.uniforms.mask.value.baseTexture.off('loaded', this.boundLoadedFunction);\n};\n\nObject.defineProperties(AlphaMaskFilter.prototype, {\n    /**\n     * The texture used for the displacement map. Must be power of 2 sized texture.\n     *\n     * @member {Texture}\n     * @memberof AlphaMaskFilter#\n     */\n    map: {\n        get: function () {\n            return this.uniforms.mask.value;\n        },\n        set: function (value) {\n            this.uniforms.mask.value = value;\n        }\n    },\n\n    /**\n     * The offset used to move the displacement map.\n     *\n     * @member {Point}\n     * @memberof AlphaMaskFilter#\n     */\n    offset: {\n        get: function() {\n            return this.uniforms.offset.value;\n        },\n        set: function(value) {\n            this.uniforms.offset.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original shader : https://www.shadertoy.com/view/lssGDj by @movAX13h\n */\n\n/**\n * An ASCII filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction AsciiFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        dimensions: { type: '4fv', value: new Float32Array([10000, 100, 10, 10]) },\n        pixelSize:  { type: '1f', value: 8}\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'uniform vec4 dimensions;',\n        'uniform float pixelSize;',\n        'uniform sampler2D uSampler;',\n\n        'float character(float n, vec2 p)',\n        '{',\n        '    p = floor(p*vec2(4.0, -4.0) + 2.5);',\n        '    if (clamp(p.x, 0.0, 4.0) == p.x && clamp(p.y, 0.0, 4.0) == p.y)',\n        '    {',\n        '        if (int(mod(n/exp2(p.x + 5.0*p.y), 2.0)) == 1) return 1.0;',\n        '    }',\n        '    return 0.0;',\n        '}',\n\n        'void main()',\n        '{',\n        '    vec2 uv = gl_FragCoord.xy;',\n        '    vec3 col = texture2D(uSampler, floor( uv / pixelSize ) * pixelSize / dimensions.xy).rgb;',\n\n        '    #ifdef HAS_GREENSCREEN',\n        '    float gray = (col.r + col.b)/2.0;',\n        '    #else',\n        '    float gray = (col.r + col.g + col.b)/3.0;',\n        '    #endif',\n\n        '    float n =  65536.0;             // .',\n        '    if (gray > 0.2) n = 65600.0;    // :',\n        '    if (gray > 0.3) n = 332772.0;   // *',\n        '    if (gray > 0.4) n = 15255086.0; // o',\n        '    if (gray > 0.5) n = 23385164.0; // &',\n        '    if (gray > 0.6) n = 15252014.0; // 8',\n        '    if (gray > 0.7) n = 13199452.0; // @',\n        '    if (gray > 0.8) n = 11512810.0; // #',\n\n        '    vec2 p = mod( uv / ( pixelSize * 0.5 ), 2.0) - vec2(1.0);',\n        '    col = col * character(n, p);',\n\n        '    gl_FragColor = vec4(col, 1.0);',\n        '}'\n    ];\n}\n\nAsciiFilter.prototype = Object.create(AbstractFilter.prototype);\nAsciiFilter.prototype.constructor = AsciiFilter;\nmodule.exports = AsciiFilter;\n\nObject.defineProperties(AsciiFilter.prototype, {\n    /**\n     * The pixel size used by the filter.\n     *\n     * @member {number}\n     * @memberof AsciiFilter#\n     */\n    size: {\n        get: function () {\n            return this.uniforms.pixelSize.value;\n        },\n        set: function (value) {\n            this.uniforms.pixelSize.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter'),\n    BlurXFilter = require('./BlurXFilter'),\n    BlurYFilter = require('./BlurYFilter');\n\n/**\n * The BlurFilter applies a Gaussian blur to an object.\n * The strength of the blur can be set for x- and y-axis separately.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction BlurFilter() {\n    AbstractFilter.call(this);\n\n    this.blurXFilter = new BlurXFilter();\n    this.blurYFilter = new BlurYFilter();\n\n    this.passes = [this.blurXFilter, this.blurYFilter];\n}\n\nBlurFilter.prototype = Object.create(AbstractFilter.prototype);\nBlurFilter.prototype.constructor = BlurFilter;\nmodule.exports = BlurFilter;\n\nObject.defineProperties(BlurFilter.prototype, {\n    /**\n     * Sets the strength of both the blurX and blurY properties simultaneously\n     *\n     * @member {number}\n     * @memberOf BlurFilter#\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.blurXFilter.blur;\n        },\n        set: function (value) {\n            this.blurXFilter.blur = this.blurYFilter.blur = value;\n        }\n    },\n\n    /**\n     * Sets the strength of the blurX property\n     *\n     * @member {number}\n     * @memberOf BlurFilter#\n     * @default 2\n     */\n    blurX: {\n        get: function () {\n            return this.blurXFilter.blur;\n        },\n        set: function (value) {\n            this.blurXFilter.blur = value;\n        }\n    },\n\n    /**\n     * Sets the strength of the blurY property\n     *\n     * @member {number}\n     * @memberOf BlurFilter#\n     * @default 2\n     */\n    blurY: {\n        get: function () {\n            return this.blurYFilter.blur;\n        },\n        set: function (value) {\n            this.blurYFilter.blur = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter'),\n    blurFactor = 1 / 7000;\n\n/**\n * The BlurXFilter applies a horizontal Gaussian blur to an object.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction BlurXFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur: { type: '1f', value: 1 / 512 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float blur;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec4 sum = vec4(0.0);',\n\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - 4.0*blur, vTextureCoord.y)) * 0.05;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - 3.0*blur, vTextureCoord.y)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - 2.0*blur, vTextureCoord.y)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x - blur, vTextureCoord.y)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + blur, vTextureCoord.y)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + 2.0*blur, vTextureCoord.y)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + 3.0*blur, vTextureCoord.y)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x + 4.0*blur, vTextureCoord.y)) * 0.05;',\n\n        '   gl_FragColor = sum;',\n        '}'\n    ];\n}\n\nBlurXFilter.prototype = Object.create(AbstractFilter.prototype);\nBlurXFilter.prototype.constructor = BlurXFilter;\nmodule.exports = BlurXFilter;\n\nObject.defineProperties(BlurXFilter.prototype, {\n    /**\n     * Sets the strength of both the blur.\n     *\n     * @member {number}\n     * @memberof BlurXFilter#\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value / blurFactor;\n        },\n        set: function (value) {\n            this.uniforms.blur.value = blurFactor * value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter'),\n    blurFactor = 1 / 7000;\n\n/**\n * The BlurYFilter applies a vertical Gaussian blur to an object.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction BlurYFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur: { type: '1f', value: 1 / 512 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float blur;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec4 sum = vec4(0.0);',\n\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 4.0*blur)) * 0.05;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 3.0*blur)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 2.0*blur)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - blur)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + blur)) * 0.15;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 2.0*blur)) * 0.12;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 3.0*blur)) * 0.09;',\n        '   sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 4.0*blur)) * 0.05;',\n\n        '   gl_FragColor = sum;',\n        '}'\n    ];\n}\n\nBlurYFilter.prototype = Object.create(AbstractFilter.prototype);\nBlurYFilter.prototype.constructor = BlurYFilter;\nmodule.exports = BlurYFilter;\n\nObject.defineProperties(BlurYFilter.prototype, {\n    /**\n     * Sets the strength of both the blur.\n     *\n     * @member {number}\n     * @memberof BlurYFilter\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value / blurFactor;\n        },\n        set: function (value) {\n            //this.padding = value;\n            this.uniforms.blur.value = blurFactor * value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The ColorMatrixFilter class lets you apply a 4x4 matrix transformation on the RGBA\n * color and alpha values of every pixel on your displayObject to produce a result\n * with a new set of RGBA color and alpha values. It's pretty powerful!\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction ColorMatrixFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        matrix: { type: 'mat4', value: [1, 0, 0, 0,\n                                        0, 1, 0, 0,\n                                        0, 0, 1, 0,\n                                        0, 0, 0, 1] }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float invert;',\n        'uniform mat4 matrix;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord) * matrix;',\n        '}'\n    ];\n}\n\nColorMatrixFilter.prototype = Object.create(AbstractFilter.prototype);\nColorMatrixFilter.prototype.constructor = ColorMatrixFilter;\nmodule.exports = ColorMatrixFilter;\n\nObject.defineProperties(ColorMatrixFilter.prototype, {\n    /**\n     * Sets the matrix of the color matrix filter\n     *\n     * @member {number[]}\n     * @memberof ColorMatrixFilter#\n     * @default [1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1, 0, 0, 0, 0, 1]\n     */\n    matrix: {\n        get: function () {\n            return this.uniforms.matrix.value;\n        },\n        set: function (value) {\n            this.uniforms.matrix.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This lowers the color depth of your image by the given amount, producing an image with a smaller palette.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction ColorStepFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        step: { type: '1f', value: 5 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n        'uniform sampler2D uSampler;',\n        'uniform float step;',\n\n        'void main(void) {',\n        '   vec4 color = texture2D(uSampler, vTextureCoord);',\n        '   color = floor(color * step) / step;',\n        '   gl_FragColor = color;',\n        '}'\n    ];\n}\n\nColorStepFilter.prototype = Object.create(AbstractFilter.prototype);\nColorStepFilter.prototype.constructor = ColorStepFilter;\nmodule.exports = ColorStepFilter;\n\nObject.defineProperties(ColorStepFilter.prototype, {\n    /**\n     * The number of steps to reduce the palette by.\n     *\n     * @member {number}\n     * @memberof ColorStepFilter#\n     */\n    step: {\n        get: function () {\n            return this.uniforms.step.value;\n        },\n        set: function (value) {\n            this.uniforms.step.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The ConvolutionFilter class applies a matrix convolution filter effect.\n * A convolution combines pixels in the input image with neighboring pixels to produce a new image.\n * A wide variety of image effects can be achieved through convolutions, including blurring, edge\n * detection, sharpening, embossing, and beveling. The matrix should be specified as a 9 point Array.\n * See http://docs.gimp.org/en/plug-in-convmatrix.html for more info.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param matrix {number[]} An array of values used for matrix transformation. Specified as a 9 point Array.\n * @param width {number} Width of the object you are transforming\n * @param height {number} Height of the object you are transforming\n */\nfunction ConvolutionFilter(matrix, width, height) {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        matrix:     { type: '1fv', value: new Float32Array(matrix) },\n        texelSizeX: { type: '1f', value: 1 / width },\n        texelSizeY: { type: '1f', value: 1 / height }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying mediump vec2 vTextureCoord;',\n\n        'uniform sampler2D texture;',\n        'uniform float texelSizeX;',\n        'uniform float texelSizeY;',\n        'uniform float matrix[9];',\n\n        'vec2 px = vec2(texelSizeX, texelSizeY);',\n\n        'void main(void) {',\n        '   vec4 c11 = texture2D(texture, vTextureCoord - px);', // top left\n        '   vec4 c12 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y - px.y));', // top center\n        '   vec4 c13 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y - px.y));', // top right\n\n        '   vec4 c21 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y) );', // mid left\n        '   vec4 c22 = texture2D(texture, vTextureCoord);', // mid center\n        '   vec4 c23 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y) );', // mid right\n\n        '   vec4 c31 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y + px.y) );', // bottom left\n        '   vec4 c32 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y + px.y) );', // bottom center\n        '   vec4 c33 = texture2D(texture, vTextureCoord + px );', // bottom right\n\n        '   gl_FragColor = ',\n        '       c11 * matrix[0] + c12 * matrix[1] + c13 * matrix[2] +',\n        '       c21 * matrix[3] + c22 * matrix[4] + c23 * matrix[5] +',\n        '       c31 * matrix[6] + c32 * matrix[7] + c33 * matrix[8];',\n        '   gl_FragColor.a = c22.a;',\n        '}'\n    ];\n}\n\nConvolutionFilter.prototype = Object.create(AbstractFilter.prototype);\nConvolutionFilter.prototype.constructor = ConvolutionFilter;\nmodule.exports = ConvolutionFilter;\n\nObject.defineProperties(ConvolutionFilter.prototype, {\n    /**\n     * An array of values used for matrix transformation. Specified as a 9 point Array.\n     *\n     * @member {number[]}\n     * @memberof ConvolutionFilter#\n     */\n    matrix: {\n        get: function () {\n            return this.uniforms.matrix.value;\n        },\n        set: function (value) {\n            this.uniforms.matrix.value = new Float32Array(value);\n        }\n    },\n\n    /**\n     * Width of the object you are transforming\n     *\n     * @member {number}\n     * @memberof ConvolutionFilter#\n     */\n    width: {\n        get: function () {\n            return this.uniforms.texelSizeX.value;\n        },\n        set: function (value) {\n            this.uniforms.texelSizeX.value = 1/value;\n        }\n    },\n\n    /**\n     * Height of the object you are transforming\n     *\n     * @member {number}\n     * @memberof ConvolutionFilter#\n     */\n    height: {\n        get: function () {\n            return this.uniforms.texelSizeY.value;\n        },\n        set: function (value) {\n            this.uniforms.texelSizeY.value = 1/value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * A Cross Hatch effect filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction CrossHatchFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur: { type: '1f', value: 1 / 512 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float blur;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '    float lum = length(texture2D(uSampler, vTextureCoord.xy).rgb);',\n\n        '    gl_FragColor = vec4(1.0, 1.0, 1.0, 1.0);',\n\n        '    if (lum < 1.00) {',\n        '        if (mod(gl_FragCoord.x + gl_FragCoord.y, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n\n        '    if (lum < 0.75) {',\n        '        if (mod(gl_FragCoord.x - gl_FragCoord.y, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n\n        '    if (lum < 0.50) {',\n        '        if (mod(gl_FragCoord.x + gl_FragCoord.y - 5.0, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n\n        '    if (lum < 0.3) {',\n        '        if (mod(gl_FragCoord.x - gl_FragCoord.y - 5.0, 10.0) == 0.0) {',\n        '            gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);',\n        '        }',\n        '    }',\n        '}'\n    ];\n}\n\nCrossHatchFilter.prototype = Object.create(AbstractFilter.prototype);\nCrossHatchFilter.prototype.constructor = CrossHatchFilter;\nmodule.exports = CrossHatchFilter;\n\nObject.defineProperties(CrossHatchFilter.prototype, {\n    /**\n     * Sets the strength of both the blur.\n     *\n     * @member {number}\n     * @memberof CrossHatchFilter#\n     * @default 2\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value / (1/7000);\n        },\n        set: function (value) {\n            //this.padding = value;\n            this.uniforms.blur.value = (1/7000) * value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The DisplacementFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object.\n * You can use this filter to apply all manor of crazy warping effects\n * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param texture {Texture} The texture used for the displacement map * must be power of 2 texture at the moment\n */\nfunction DisplacementFilter(texture) {\n    AbstractFilter.call(this);\n\n    texture.baseTexture._powerOf2 = true;\n\n    // set the uniforms\n    this.uniforms = {\n        displacementMap: { type: 'sampler2D', value: texture },\n        scale:           { type: '2f',  value: { x: 30, y: 30 } },\n        offset:          { type: '2f',  value: { x: 0,  y: 0 } },\n        mapDimensions:   { type: '2f',  value: { x: 1,  y: 5112 } },\n        dimensions:      { type: '4fv', value: [0, 0, 0, 0] }\n    };\n\n    if (texture.baseTexture.hasLoaded) {\n        this.onTextureLoaded();\n    }\n    else {\n        this.boundLoadedFunction = this.onTextureLoaded.bind(this);\n\n        texture.baseTexture.on('loaded', this.boundLoadedFunction);\n    }\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D displacementMap;',\n        'uniform sampler2D uSampler;',\n        'uniform vec2 scale;',\n        'uniform vec2 offset;',\n        'uniform vec4 dimensions;',\n        'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);',\n        // 'const vec2 textureDimensions = vec2(750.0, 750.0);',\n\n        'void main(void) {',\n        '   vec2 mapCords = vTextureCoord.xy;',\n        '   mapCords += (dimensions.zw + offset)/ dimensions.xy ;',\n        '   mapCords.y *= -1.0;',\n        '   mapCords.y += 1.0;',\n\n        '   vec2 matSample = texture2D(displacementMap, mapCords).xy;',\n        '   matSample -= 0.5;',\n        '   matSample *= scale;',\n        '   matSample /= mapDimensions;',\n\n        '   gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x + matSample.x, vTextureCoord.y + matSample.y));',\n\n        //TODO: Is this needed?\n        '   gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb, 1.0);',\n        '}'\n    ];\n}\n\nDisplacementFilter.prototype = Object.create(AbstractFilter.prototype);\nDisplacementFilter.prototype.constructor = DisplacementFilter;\nmodule.exports = DisplacementFilter;\n\n/**\n * Sets the map dimensions uniforms when the texture becomes available.\n *\n * @private\n */\nDisplacementFilter.prototype.onTextureLoaded = function () {\n    this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width;\n    this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height;\n\n    this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction);\n};\n\nObject.defineProperties(DisplacementFilter.prototype, {\n    /**\n     * The texture used for the displacement map. Must be power of 2 texture.\n     *\n     * @member {Texture}\n     * @memberof DisplacementFilter#\n     */\n    map: {\n        get: function () {\n            return this.uniforms.displacementMap.value;\n        },\n        set: function (value) {\n            this.uniforms.displacementMap.value = value;\n        }\n    },\n\n    /**\n     * The multiplier used to scale the displacement result from the map calculation.\n     *\n     * @member {Point}\n     * @memberof DisplacementFilter#\n     */\n    scale: {\n        get: function () {\n            return this.uniforms.scale.value;\n        },\n        set: function (value) {\n            this.uniforms.scale.value = value;\n        }\n    },\n\n    /**\n     * The offset used to move the displacement map.\n     *\n     * @member {Point}\n     * @memberof DisplacementFilter#\n     */\n    offset: {\n        get: function () {\n            return this.uniforms.offset.value;\n        },\n        set: function (value) {\n            this.uniforms.offset.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Mat Groves http://matgroves.com/ @Doormat23\n * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/fun/dotscreen.js\n */\n\n/**\n * This filter applies a dotscreen effect making display objects appear to be made out of\n * black and white halftone dots like an old printer.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction DotScreenFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        scale:      { type: '1f', value: 1 },\n        angle:      { type: '1f', value: 5 },\n        dimensions: { type: '4fv', value: [0, 0, 0, 0] }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform vec4 dimensions;',\n        'uniform sampler2D uSampler;',\n\n        'uniform float angle;',\n        'uniform float scale;',\n\n        'float pattern() {',\n        '   float s = sin(angle), c = cos(angle);',\n        '   vec2 tex = vTextureCoord * dimensions.xy;',\n        '   vec2 point = vec2(',\n        '       c * tex.x - s * tex.y,',\n        '       s * tex.x + c * tex.y',\n        '   ) * scale;',\n        '   return (sin(point.x) * sin(point.y)) * 4.0;',\n        '}',\n\n        'void main() {',\n        '   vec4 color = texture2D(uSampler, vTextureCoord);',\n        '   float average = (color.r + color.g + color.b) / 3.0;',\n        '   gl_FragColor = vec4(vec3(average * 10.0 - 5.0 + pattern()), color.a);',\n        '}'\n    ];\n}\n\nDotScreenFilter.prototype = Object.create(AbstractFilter.prototype);\nDotScreenFilter.prototype.constructor = DotScreenFilter;\nmodule.exports = DotScreenFilter;\n\nObject.defineProperties(DotScreenFilter.prototype, {\n    /**\n     * The scale of the effect.\n     * @member {number}\n     * @memberof DotScreenFilter#\n     */\n    scale: {\n        get: function () {\n            return this.uniforms.scale.value;\n        },\n        set: function (value) {\n            this.uniforms.scale.value = value;\n        }\n    },\n\n    /**\n     * The radius of the effect.\n     * @member {number}\n     * @memberof DotScreenFilter#\n     */\n    angle: {\n        get: function () {\n            return this.uniforms.angle.value;\n        },\n        set: function (value) {\n            this.uniforms.angle.value = value;\n        }\n    }\n});\n","/**\n * A target and pass info object for filters.\n *\n * @class\n * @namespace PIXI\n */\nfunction FilterBlock() {\n    /**\n     * The visible state of this FilterBlock.\n     *\n     * @member {boolean}\n     */\n    this.visible = true;\n\n    /**\n     * The renderable state of this FilterBlock.\n     *\n     * @member {boolean}\n     */\n    this.renderable = true;\n}\n\nFilterBlock.prototype.constructor = FilterBlock;\nmodule.exports = FilterBlock;\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This greyscales the palette of your Display Objects.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction GrayFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        gray: { type: '1f', value: 1 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform sampler2D uSampler;',\n        'uniform float gray;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord);',\n        '   gl_FragColor.rgb = mix(gl_FragColor.rgb, vec3(0.2126*gl_FragColor.r + 0.7152*gl_FragColor.g + 0.0722*gl_FragColor.b), gray);',\n     //   '   gl_FragColor = gl_FragColor;',\n        '}'\n    ];\n}\n\nGrayFilter.prototype = Object.create(AbstractFilter.prototype);\nGrayFilter.prototype.constructor = GrayFilter;\nmodule.exports = GrayFilter;\n\nObject.defineProperties(GrayFilter.prototype, {\n    /**\n     * The strength of the gray. 1 will make the object black and white, 0 will make the object its normal color.\n     *\n     * @member {number}\n     * @memberof GrayFilter#\n     */\n    gray: {\n        get: function () {\n            return this.uniforms.gray.value;\n        },\n        set: function (value) {\n            this.uniforms.gray.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This inverts your Display Objects colors.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction InvertFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        invert: { type: '1f', value: 1 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float invert;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord);',\n        '   gl_FragColor.rgb = mix( (vec3(1)-gl_FragColor.rgb) * gl_FragColor.a, gl_FragColor.rgb, 1.0 - invert);',\n        //'   gl_FragColor.rgb = gl_FragColor.rgb  * gl_FragColor.a;',\n      //  '   gl_FragColor = gl_FragColor * vColor;',\n        '}'\n    ];\n}\n\nInvertFilter.prototype = Object.create(AbstractFilter.prototype);\nInvertFilter.prototype.constructor = InvertFilter;\nmodule.exports = InvertFilter;\n\nObject.defineProperties(InvertFilter.prototype, {\n    /**\n     * The strength of the invert. 1 will fully invert the colors, 0 will make the object its normal color\n     *\n     * @member {number}\n     * @memberof InvertFilter#\n     */\n    invert: {\n        get: function () {\n            return this.uniforms.invert.value;\n        },\n        set: function (value) {\n            this.uniforms.invert.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter: https://github.com/evanw/glfx.js/blob/master/src/filters/adjust/noise.js\n */\n\n/**\n * A Noise effect filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction NoiseFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        noise: { type: '1f', value: 0.5 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float noise;',\n        'uniform sampler2D uSampler;',\n\n        'float rand(vec2 co) {',\n        '    return fract(sin(dot(co.xy ,vec2(12.9898,78.233))) * 43758.5453);',\n        '}',\n\n        'void main() {',\n        '    vec4 color = texture2D(uSampler, vTextureCoord);',\n\n        '    float diff = (rand(vTextureCoord) - 0.5) * noise;',\n        '    color.r += diff;',\n        '    color.g += diff;',\n        '    color.b += diff;',\n\n        '    gl_FragColor = color;',\n        '}'\n    ];\n}\n\nNoiseFilter.prototype = Object.create(AbstractFilter.prototype);\nNoiseFilter.prototype.constructor = NoiseFilter;\nmodule.exports = NoiseFilter;\n\nObject.defineProperties(NoiseFilter.prototype, {\n    /**\n     * The amount of noise to apply.\n     *\n     * @member {number}\n     * @memberof NoiseFilter#\n     * @default 0.5\n     */\n    noise: {\n        get: function () {\n            return this.uniforms.noise.value;\n        },\n        set: function (value) {\n            this.dirty = true;\n            this.uniforms.noise.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * The NormalMapFilter class uses the pixel values from the specified texture (called the displacement map) to perform a displacement of an object.\n * You can use this filter to apply all manor of crazy warping effects\n * Currently the r property of the texture is used offset the x and the g property of the texture is used to offset the y.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n * @param texture {Texture} The texture used for the normal map, must be power of 2 texture at the moment\n */\nfunction NormalMapFilter(texture) {\n    AbstractFilter.call(this);\n\n    texture.baseTexture._powerOf2 = true;\n\n    // set the uniforms\n    this.uniforms = {\n        displacementMap:  { type: 'sampler2D', value: texture },\n        scale:            { type: '2f', value: { x: 15, y: 15 } },\n        offset:           { type: '2f', value: { x: 0,  y: 0 } },\n        mapDimensions:    { type: '2f', value: { x: 1,  y: 1 } },\n        dimensions:       { type: '4f', value: [0, 0, 0, 0] },\n        // LightDir:         { type: 'f3', value: [0, 1, 0] },\n        LightPos:         { type: '3f', value: [0, 1, 0] }\n    };\n\n    if (texture.baseTexture.hasLoaded) {\n        this.onTextureLoaded();\n    }\n    else {\n        this.boundLoadedFunction = this.onTextureLoaded.bind(this);\n\n        texture.baseTexture.on('loaded', this.boundLoadedFunction);\n    }\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying float vColor;',\n\n        'uniform sampler2D displacementMap;',\n        'uniform sampler2D uSampler;',\n\n        'uniform vec4 dimensions;',\n\n        'const vec2 Resolution = vec2(1.0,1.0);',      //resolution of screen\n        'uniform vec3 LightPos;',    //light position, normalized\n        'const vec4 LightColor = vec4(1.0, 1.0, 1.0, 1.0);',      //light RGBA -- alpha is intensity\n        'const vec4 AmbientColor = vec4(1.0, 1.0, 1.0, 0.5);',    //ambient RGBA -- alpha is intensity\n        'const vec3 Falloff = vec3(0.0, 1.0, 0.2);',         //attenuation coefficients\n\n        'uniform vec3 LightDir;',//' = vec3(1.0, 0.0, 1.0);',\n\n\n        'uniform vec2 mapDimensions;',// = vec2(256.0, 256.0);',\n\n\n        'void main(void) {',\n        '   vec2 mapCords = vTextureCoord.xy;',\n\n        '   vec4 color = texture2D(uSampler, vTextureCoord.st);',\n        '   vec3 nColor = texture2D(displacementMap, vTextureCoord.st).rgb;',\n\n\n        '   mapCords *= vec2(dimensions.x/512.0, dimensions.y/512.0);',\n\n        '   mapCords.y *= -1.0;',\n        '   mapCords.y += 1.0;',\n\n        //RGBA of our diffuse color\n        '   vec4 DiffuseColor = texture2D(uSampler, vTextureCoord);',\n\n        //RGB of our normal map\n        '   vec3 NormalMap = texture2D(displacementMap, mapCords).rgb;',\n\n        //The delta position of light\n        //'vec3 LightDir = vec3(LightPos.xy - (gl_FragCoord.xy / Resolution.xy), LightPos.z);',\n        '   vec3 LightDir = vec3(LightPos.xy - (mapCords.xy), LightPos.z);',\n        //Correct for aspect ratio\n        // '   LightDir.x *= Resolution.x / Resolution.y;',\n\n        //Determine distance (used for attenuation) BEFORE we normalize our LightDir\n        '   float D = length(LightDir);',\n\n        //normalize our vectors\n        '   vec3 N = normalize(NormalMap * 2.0 - 1.0);',\n        '   vec3 L = normalize(LightDir);',\n\n        //Pre-multiply light color with intensity\n        //Then perform 'N dot L' to determine our diffuse term\n        '   vec3 Diffuse = (LightColor.rgb * LightColor.a) * max(dot(N, L), 0.0);',\n\n        //pre-multiply ambient color with intensity\n        '   vec3 Ambient = AmbientColor.rgb * AmbientColor.a;',\n\n        //calculate attenuation\n        '   float Attenuation = 1.0 / ( Falloff.x + (Falloff.y*D) + (Falloff.z*D*D) );',\n\n        //the calculation which brings it all together\n        '   vec3 Intensity = Ambient + Diffuse * Attenuation;',\n        '   vec3 FinalColor = DiffuseColor.rgb * Intensity;',\n        '   gl_FragColor = vColor * vec4(FinalColor, DiffuseColor.a);',\n        // '   gl_FragColor = vec4(1.0, 0.0, 0.0, Attenuation);',//vColor * vec4(FinalColor, DiffuseColor.a);',\n\n        /*// normalise color\n        '   vec3 normal = normalize(nColor * 2.0 - 1.0);',\n\n        '   vec3 deltaPos = vec3( (light.xy - gl_FragCoord.xy) / resolution.xy, light.z );',\n\n        '   float lambert = clamp(dot(normal, lightDir), 0.0, 1.0);',\n\n        '   float d = sqrt(dot(deltaPos, deltaPos));',\n        '   float att = 1.0 / ( attenuation.x + (attenuation.y*d) + (attenuation.z*d*d) );',\n\n        '   vec3 result = (ambientColor * ambientIntensity) + (lightColor.rgb * lambert) * att;',\n        '   result *= color.rgb;',\n\n        '   gl_FragColor = vec4(result, 1.0);',*/\n        '}'\n    ];\n}\n\nNormalMapFilter.prototype = Object.create(AbstractFilter.prototype);\nNormalMapFilter.prototype.constructor = NormalMapFilter;\nmodule.exports = NormalMapFilter;\n\n/**\n * Sets the map dimensions uniforms when the texture becomes available.\n *\n */\nNormalMapFilter.prototype.onTextureLoaded = function () {\n    this.uniforms.mapDimensions.value.x = this.uniforms.displacementMap.value.width;\n    this.uniforms.mapDimensions.value.y = this.uniforms.displacementMap.value.height;\n\n    this.uniforms.displacementMap.value.baseTexture.off('loaded', this.boundLoadedFunction);\n};\n\nObject.defineProperties(NormalMapFilter.prototype, {\n    /**\n     * The texture used for the displacement map. Must be power of 2 texture.\n     *\n     * @member {Texture}\n     * @memberof NormalMapFilter#\n     */\n    map: {\n        get: function () {\n            return this.uniforms.displacementMap.value;\n        },\n        set: function (value) {\n            this.uniforms.displacementMap.value = value;\n        }\n    },\n\n    /**\n     * The multiplier used to scale the displacement result from the map calculation.\n     *\n     * @member {Point}\n     * @memberof NormalMapFilter#\n     */\n    scale: {\n        get: function () {\n            return this.uniforms.scale.value;\n        },\n        set: function (value) {\n            this.uniforms.scale.value = value;\n        }\n    },\n\n    /**\n     * The offset used to move the displacement map.\n     *\n     * @member {Point}\n     * @memberof NormalMapFilter#\n     */\n    offset: {\n        get: function () {\n            return this.uniforms.offset.value;\n        },\n        set: function (value) {\n            this.uniforms.offset.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This filter applies a pixelate effect making display objects appear 'blocky'.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction PixelateFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        invert:     { type: '1f',   value: 0 },\n        dimensions: { type: '4fv',  value: new Float32Array([10000, 100, 10, 10]) },\n        pixelSize:  { type: '2f',   value: { x: 10, y: 10 } }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform vec2 testDim;',\n        'uniform vec4 dimensions;',\n        'uniform vec2 pixelSize;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec2 coord = vTextureCoord;',\n\n        '   vec2 size = dimensions.xy/pixelSize;',\n\n        '   vec2 color = floor( ( vTextureCoord * size ) ) / size + pixelSize/dimensions.xy * 0.5;',\n        '   gl_FragColor = texture2D(uSampler, color);',\n        '}'\n    ];\n}\n\nPixelateFilter.prototype = Object.create(AbstractFilter.prototype);\nPixelateFilter.prototype.constructor = PixelateFilter;\nmodule.exports = PixelateFilter;\n\nObject.defineProperties(PixelateFilter.prototype, {\n    /**\n     * This a point that describes the size of the blocks. x is the width of the block and y is the height.\n     *\n     * @member {Point}\n     * @memberof PixelateFilter#\n     */\n    size: {\n        get: function () {\n            return this.uniforms.pixelSize.value;\n        },\n        set: function (value) {\n            this.uniforms.pixelSize.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * An RGB Split Filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction RGBSplitFilter() {\n    AbstractFilter.call(this);\n\n    this.passes = [this];\n\n    // set the uniforms\n    this.uniforms = {\n        red:        { type: '2f', value: { x: 20, y: 20 } },\n        green:      { type: '2f', value: { x: -20, y: 20 } },\n        blue:       { type: '2f', value: { x: 20, y: -20 } },\n        dimensions: { type: '4fv', value: [0, 0, 0, 0] }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform vec2 red;',\n        'uniform vec2 green;',\n        'uniform vec2 blue;',\n        'uniform vec4 dimensions;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   gl_FragColor.r = texture2D(uSampler, vTextureCoord + red/dimensions.xy).r;',\n        '   gl_FragColor.g = texture2D(uSampler, vTextureCoord + green/dimensions.xy).g;',\n        '   gl_FragColor.b = texture2D(uSampler, vTextureCoord + blue/dimensions.xy).b;',\n        '   gl_FragColor.a = texture2D(uSampler, vTextureCoord).a;',\n        '}'\n    ];\n}\n\nRGBSplitFilter.prototype = Object.create(AbstractFilter.prototype);\nRGBSplitFilter.prototype.constructor = RGBSplitFilter;\nmodule.exports = RGBSplitFilter;\n\nObject.defineProperties(RGBSplitFilter.prototype, {\n    /**\n     * Red channel offset.\n     *\n     * @member {Point}\n     * @memberof RGBSplitFilter#\n     */\n    red: {\n        get: function () {\n            return this.uniforms.red.value;\n        },\n        set: function (value) {\n            this.uniforms.red.value = value;\n        }\n    },\n\n    /**\n     * Green channel offset.\n     *\n     * @member {Point}\n     * @memberof RGBSplitFilter#\n     */\n    green: {\n        get: function () {\n            return this.uniforms.green.value;\n        },\n        set: function (value) {\n            this.uniforms.green.value = value;\n        }\n    },\n\n    /**\n     * Blue offset.\n     *\n     * @member {Point}\n     * @memberof RGBSplitFilter#\n     */\n    blue: {\n        get: function () {\n            return this.uniforms.blue.value;\n        },\n        set: function (value) {\n            this.uniforms.blue.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This applies a sepia effect to your Display Objects.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction SepiaFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        sepia: { type: '1f', value: 1 }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float sepia;',\n        'uniform sampler2D uSampler;',\n\n        'const mat3 sepiaMatrix = mat3(0.3588, 0.7044, 0.1368, 0.2990, 0.5870, 0.1140, 0.2392, 0.4696, 0.0912);',\n\n        'void main(void) {',\n        '   gl_FragColor = texture2D(uSampler, vTextureCoord);',\n        '   gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb * sepiaMatrix, sepia);',\n        '}'\n    ];\n}\n\nSepiaFilter.prototype = Object.create(AbstractFilter.prototype);\nSepiaFilter.prototype.constructor = SepiaFilter;\nmodule.exports = SepiaFilter;\n\nObject.defineProperties(SepiaFilter.prototype, {\n    /**\n     * The strength of the sepia. 1 will apply the full sepia effect, 0 will make the object its normal color.\n     *\n     * @member {number}\n     * @memberof SepiaFilter#\n     */\n    sepia: {\n        get: function () {\n            return this.uniforms.sepia.value;\n        },\n        set: function (value) {\n            this.uniforms.sepia.value = value;\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * A Smart Blur Filter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction SmartBlurFilter() {\n    AbstractFilter.call(this);\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n\n        'uniform sampler2D uSampler;',\n        'const vec2 delta = vec2(1.0/10.0, 0.0);',\n\n        'float random(vec3 scale, float seed) {',\n        '   return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);',\n        '}',\n\n\n        'void main(void) {',\n        '   vec4 color = vec4(0.0);',\n        '   float total = 0.0;',\n\n        '   float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);',\n\n        '   for (float t = -30.0; t <= 30.0; t++) {',\n        '       float percent = (t + offset - 0.5) / 30.0;',\n        '       float weight = 1.0 - abs(percent);',\n        '       vec4 sample = texture2D(uSampler, vTextureCoord + delta * percent);',\n        '       sample.rgb *= sample.a;',\n        '       color += sample * weight;',\n        '       total += weight;',\n        '   }',\n\n        '   gl_FragColor = color / total;',\n        '   gl_FragColor.rgb /= gl_FragColor.a + 0.00001;',\n        '}'\n    ];\n}\n\nSmartBlurFilter.prototype = Object.create(AbstractFilter.prototype);\nSmartBlurFilter.prototype.constructor = SmartBlurFilter;\nmodule.exports = SmartBlurFilter;\n","var AbstractFilter = require('./AbstractFilter'),\n    TiltShiftXFilter = require('./TiltShiftXFilter'),\n    TiltShiftYFilter = require('./TiltShiftYFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/\n */\n\n/**\n * A TiltShift Filter. Manages the pass of both a TiltShiftXFilter and TiltShiftYFilter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TiltShiftFilter() {\n    AbstractFilter.call(this);\n\n    this.tiltShiftXFilter = new TiltShiftXFilter();\n    this.tiltShiftYFilter = new TiltShiftYFilter();\n\n    this.tiltShiftXFilter.updateDelta();\n    this.tiltShiftXFilter.updateDelta();\n\n    this.passes = [this.tiltShiftXFilter, this.tiltShiftYFilter];\n}\n\nTiltShiftFilter.prototype = Object.create(AbstractFilter.prototype);\nTiltShiftFilter.prototype.constructor = TiltShiftFilter;\nmodule.exports = TiltShiftFilter;\n\nObject.defineProperties(TiltShiftFilter.prototype, {\n    /**\n     * The strength of the blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    blur: {\n        get: function () {\n            return this.tiltShiftXFilter.blur;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.blur = this.tiltShiftYFilter.blur = value;\n        }\n    },\n\n    /**\n     * The strength of the gradient blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    gradientBlur: {\n        get: function () {\n            return this.tiltShiftXFilter.gradientBlur;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.gradientBlur = this.tiltShiftYFilter.gradientBlur = value;\n        }\n    },\n\n    /**\n     * The Y value to start the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    start: {\n        get: function () {\n            return this.tiltShiftXFilter.start;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.start = this.tiltShiftYFilter.start = value;\n        }\n    },\n\n    /**\n     * The Y value to end the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftFilter#\n     */\n    end: {\n        get: function () {\n            return this.tiltShiftXFilter.end;\n        },\n        set: function (value) {\n            this.tiltShiftXFilter.end = this.tiltShiftYFilter.end = value;\n        }\n    },\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/\n */\n\n/**\n * A TiltShiftXFilter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TiltShiftXFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur:           { type: '1f', value: 100 },\n        gradientBlur:   { type: '1f', value: 600 },\n        start:          { type: '2f', value: { x: 0,    y: window.screenHeight / 2 } },\n        end:            { type: '2f', value: { x: 600,  y: window.screenHeight / 2 } },\n        delta:          { type: '2f', value: { x: 30,   y: 30 } },\n        texSize:        { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } }\n    };\n\n    this.updateDelta();\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n\n        'uniform sampler2D uSampler;',\n        'uniform float blur;',\n        'uniform float gradientBlur;',\n        'uniform vec2 start;',\n        'uniform vec2 end;',\n        'uniform vec2 delta;',\n        'uniform vec2 texSize;',\n\n        'float random(vec3 scale, float seed) {',\n        '   return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);',\n        '}',\n\n        'void main(void) {',\n        '    vec4 color = vec4(0.0);',\n        '    float total = 0.0;',\n\n        '    float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);',\n        '    vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));',\n        '    float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;',\n\n        '    for (float t = -30.0; t <= 30.0; t++) {',\n        '        float percent = (t + offset - 0.5) / 30.0;',\n        '        float weight = 1.0 - abs(percent);',\n        '        vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);',\n        '        sample.rgb *= sample.a;',\n        '        color += sample * weight;',\n        '        total += weight;',\n        '    }',\n\n        '    gl_FragColor = color / total;',\n        '    gl_FragColor.rgb /= gl_FragColor.a + 0.00001;',\n        '}'\n    ];\n}\n\nTiltShiftXFilter.prototype = Object.create(AbstractFilter.prototype);\nTiltShiftXFilter.prototype.constructor = TiltShiftXFilter;\nmodule.exports = TiltShiftXFilter;\n\n/**\n * Updates the filter delta values.\n *\n */\nTiltShiftXFilter.prototype.updateDelta = function () {\n    var dx = this.uniforms.end.value.x - this.uniforms.start.value.x;\n    var dy = this.uniforms.end.value.y - this.uniforms.start.value.y;\n    var d = Math.sqrt(dx * dx + dy * dy);\n\n    this.uniforms.delta.value.x = dx / d;\n    this.uniforms.delta.value.y = dy / d;\n};\n\n\nObject.defineProperties(TiltShiftXFilter.prototype, {\n    /**\n     * The strength of the blur.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value;\n        },\n        set: function (value) {\n            this.uniforms.blur.value = value;\n        }\n    },\n\n    /**\n     * The strength of the gradient blur.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    gradientBlur: {\n        get: function () {\n            return this.uniforms.gradientBlur.value;\n        },\n        set: function (value) {\n            this.uniforms.gradientBlur.value = value;\n        }\n    },\n\n    /**\n     * The X value to start the effect at.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    start: {\n        get: function () {\n            return this.uniforms.start.value;\n        },\n        set: function (value) {\n            this.uniforms.start.value = value;\n            this.updateDelta();\n        }\n    },\n\n    /**\n     * The X value to end the effect at.\n     *\n     * @member {number}\n     * @memberof TilttShiftXFilter#\n     */\n    end: {\n        get: function () {\n            return this.uniforms.end.value;\n        },\n        set: function (value) {\n            this.uniforms.end.value = value;\n            this.updateDelta();\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * @author Vico @vicocotea\n * original filter https://github.com/evanw/glfx.js/blob/master/src/filters/blur/tiltshift.js by Evan Wallace : http://madebyevan.com/\n */\n\n/**\n * A TiltShiftYFilter.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TiltShiftYFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        blur:           { type: '1f', value: 100 },\n        gradientBlur:   { type: '1f', value: 600 },\n        start:          { type: '2f', value: { x: 0,    y: window.screenHeight / 2 } },\n        end:            { type: '2f', value: { x: 600,  y: window.screenHeight / 2 } },\n        delta:          { type: '2f', value: { x: 30,   y: 30 } },\n        texSize:        { type: '2f', value: { x: window.screenWidth, y: window.screenHeight } }\n    };\n\n    this.updateDelta();\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n\n        'uniform sampler2D uSampler;',\n        'uniform float blur;',\n        'uniform float gradientBlur;',\n        'uniform vec2 start;',\n        'uniform vec2 end;',\n        'uniform vec2 delta;',\n        'uniform vec2 texSize;',\n\n        'float random(vec3 scale, float seed) {',\n        '   return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);',\n        '}',\n\n        'void main(void) {',\n        '    vec4 color = vec4(0.0);',\n        '    float total = 0.0;',\n\n        '    float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);',\n        '    vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));',\n        '    float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;',\n\n        '    for (float t = -30.0; t <= 30.0; t++) {',\n        '        float percent = (t + offset - 0.5) / 30.0;',\n        '        float weight = 1.0 - abs(percent);',\n        '        vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);',\n        '        sample.rgb *= sample.a;',\n        '        color += sample * weight;',\n        '        total += weight;',\n        '    }',\n\n        '    gl_FragColor = color / total;',\n        '    gl_FragColor.rgb /= gl_FragColor.a + 0.00001;',\n        '}'\n    ];\n}\n\nTiltShiftYFilter.prototype = Object.create(AbstractFilter.prototype);\nTiltShiftYFilter.prototype.constructor = TiltShiftYFilter;\nmodule.exports = TiltShiftYFilter;\n\n/**\n * Updates the filter delta values.\n *\n */\nTiltShiftYFilter.prototype.updateDelta = function (){\n    var dx = this.uniforms.end.value.x - this.uniforms.start.value.x;\n    var dy = this.uniforms.end.value.y - this.uniforms.start.value.y;\n    var d = Math.sqrt(dx * dx + dy * dy);\n    this.uniforms.delta.value.x = -dy / d;\n    this.uniforms.delta.value.y = dx / d;\n};\n\nObject.defineProperties(TiltShiftYFilter.prototype, {\n    /**\n     * The strength of the blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    blur: {\n        get: function () {\n            return this.uniforms.blur.value;\n        },\n        set: function (value) {\n            this.uniforms.blur.value = value;\n        }\n    },\n\n    /**\n     * The strength of the gradient blur.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    gradientBlur: {\n        get: function () {\n            return this.uniforms.gradientBlur.value;\n        },\n        set: function (value) {\n            this.uniforms.gradientBlur.value = value;\n        }\n    },\n\n    /**\n     * The Y value to start the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    start: {\n        get: function () {\n            return this.uniforms.start.value;\n        },\n        set: function (value) {\n            this.uniforms.start.value = value;\n            this.updateDelta();\n        }\n    },\n\n    /**\n     * The Y value to end the effect at.\n     *\n     * @member {number}\n     * @memberof TiltShiftYFilter#\n     */\n    end: {\n        get: function () {\n            return this.uniforms.end.value;\n        },\n        set: function (value) {\n            this.uniforms.end.value = value;\n            this.updateDelta();\n        }\n    }\n});\n","var AbstractFilter = require('./AbstractFilter');\n\n/**\n * This filter applies a twist effect making display objects appear twisted in the given direction.\n *\n * @class\n * @extends AbstractFilter\n * @namespace PIXI\n */\nfunction TwistFilter() {\n    AbstractFilter.call(this);\n\n    // set the uniforms\n    this.uniforms = {\n        radius:     { type: '1f', value: 0.5},\n        angle:      { type: '1f', value: 5},\n        offset:     { type: '2f', value: { x: 0.5, y: 0.5 } }\n    };\n\n    this.fragmentSrc = [\n        'precision mediump float;',\n\n        'varying vec2 vTextureCoord;',\n        'varying vec4 vColor;',\n\n        'uniform float radius;',\n        'uniform float angle;',\n        'uniform vec2 offset;',\n        'uniform sampler2D uSampler;',\n\n        'void main(void) {',\n        '   vec2 coord = vTextureCoord - offset;',\n        '   float distance = length(coord);',\n\n        '   if (distance < radius) {',\n        '       float ratio = (radius - distance) / radius;',\n        '       float angleMod = ratio * ratio * angle;',\n        '       float s = sin(angleMod);',\n        '       float c = cos(angleMod);',\n        '       coord = vec2(coord.x * c - coord.y * s, coord.x * s + coord.y * c);',\n        '   }',\n\n        '   gl_FragColor = texture2D(uSampler, coord+offset);',\n        '}'\n    ];\n}\n\nTwistFilter.prototype = Object.create(AbstractFilter.prototype);\nTwistFilter.prototype.constructor = TwistFilter;\nmodule.exports = TwistFilter;\n\nObject.defineProperties(TwistFilter.prototype, {\n    /**\n     * This point describes the the offset of the twist.\n     *\n     * @member {Point}\n     * @memberof TwistFilter#\n     */\n    offset: {\n        get: function () {\n            return this.uniforms.offset.value;\n        },\n        set: function (value) {\n            this.uniforms.offset.value = value;\n        }\n    },\n\n    /**\n     * This radius of the twist.\n     *\n     * @member {number}\n     * @memberof TwistFilter#\n     */\n    radius: {\n        get: function () {\n            return this.uniforms.radius.value;\n        },\n        set: function (value) {\n            this.uniforms.radius.value = value;\n        }\n    },\n\n    /**\n     * This angle of the twist.\n     *\n     * @member {number}\n     * @memberof TwistFilter#\n     */\n    angle: {\n        get: function () {\n            return this.uniforms.angle.value;\n        },\n        set: function (value) {\n            this.uniforms.angle.value = value;\n        }\n    }\n});\n","/**\n * @file        Main export of the PIXI filters library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    AbstractFilter:     require('./AbstractFilter'),\n    AlphaMaskFilter:    require('./AlphaMaskFilter'),\n    AsciiFilter:        require('./AsciiFilter'),\n    BlurFilter:         require('./BlurFilter'),\n    BlurXFilter:        require('./BlurXFilter'),\n    BlurYFilter:        require('./BlurYFilter'),\n    ColorMatrixFilter:  require('./ColorMatrixFilter'),\n    ColorStepFilter:    require('./ColorStepFilter'),\n    ConvolutionFilter:  require('./ConvolutionFilter'),\n    CrossHatchFilter:   require('./CrossHatchFilter'),\n    DisplacementFilter: require('./DisplacementFilter'),\n    DotScreenFilter:    require('./DotScreenFilter'),\n    FilterBlock:        require('./FilterBlock'),\n    GrayFilter:         require('./GrayFilter'),\n    InvertFilter:       require('./InvertFilter'),\n    NoiseFilter:        require('./NoiseFilter'),\n    NormalMapFilter:    require('./NormalMapFilter'),\n    PixelateFilter:     require('./PixelateFilter'),\n    RGBSplitFilter:     require('./RGBSplitFilter'),\n    SepiaFilter:        require('./SepiaFilter'),\n    SmartBlurFilter:    require('./SmartBlurFilter'),\n    TiltShiftFilter:    require('./TiltShiftFilter'),\n    TiltShiftXFilter:   require('./TiltShiftXFilter'),\n    TiltShiftYFilter:   require('./TiltShiftYFilter'),\n    TwistFilter:        require('./TwistFilter')\n};\n","var core = require('../core');\n\n/**\n * Holds all information related to an Interaction event\n *\n * @class\n * @namespace PIXI\n */\nfunction InteractionData() {\n    /**\n     * This point stores the global coords of where the touch/mouse event happened\n     *\n     * @member {Point}\n     */\n    this.global = new core.math.Point();\n\n    /**\n     * The target Sprite that was interacted with\n     *\n     * @member {Sprite}\n     */\n    this.target = null;\n\n    /**\n     * When passed to an event handler, this will be the original DOM Event that was captured\n     *\n     * @member {Event}\n     */\n    this.originalEvent = null;\n}\n\nInteractionData.prototype.constructor = InteractionData;\nmodule.exports = InteractionData;\n\n/**\n * This will return the local coordinates of the specified displayObject for this InteractionData\n *\n * @param displayObject {DisplayObject} The DisplayObject that you would like the local coords off\n * @param [point] {Point} A Point object in which to store the value, optional (otherwise will create a new point)\n * @return {Point} A point containing the coordinates of the InteractionData position relative to the DisplayObject\n */\nInteractionData.prototype.getLocalPosition = function (displayObject, point) {\n    var worldTransform = displayObject.worldTransform;\n    var global = this.global;\n\n    // do a cheeky transform to get the mouse coords;\n    var a00 = worldTransform.a, a01 = worldTransform.c, a02 = worldTransform.tx,\n        a10 = worldTransform.b, a11 = worldTransform.d, a12 = worldTransform.ty,\n        id = 1 / (a00 * a11 + a01 * -a10);\n\n    point = point || new core.math.Point();\n\n    point.x = a11 * id * global.x + -a01 * id * global.y + (a12 * a01 - a02 * a11) * id;\n    point.y = a00 * id * global.y + -a10 * id * global.x + (-a12 * a00 + a02 * a10) * id;\n\n    // set the mouse coords...\n    return point;\n};\n","var core = require('../core'),\n    InteractionData = require('./InteractionData');\n\n// TODO: Obviously rewrite this...\nvar INTERACTION_FREQUENCY = 30;\nvar AUTO_PREVENT_DEFAULT = true;\n\n/**\n * The interaction manager deals with mouse and touch events. Any DisplayObject can be interactive\n * if its interactive parameter is set to true\n * This manager also supports multitouch.\n *\n * @class\n * @namespace PIXI\n * @param stage {Stage} The stage to handle interactions\n */\nfunction InteractionManager(stage) {\n    /**\n     * A reference to the stage\n     *\n     * @member {Stage}\n     */\n    this.stage = stage;\n\n    /**\n     * The mouse data\n     *\n     * @member {InteractionData}\n     */\n    this.mouse = new InteractionData();\n\n    /**\n     * An object that stores current touches (InteractionData) by id reference\n     *\n     * @member {object}\n     */\n    this.touches = {};\n\n    /**\n     * @member {Point}\n     * @private\n     */\n    this.tempPoint = new core.math.Point();\n\n    /**\n     * @member {boolean}\n     * @default\n     */\n    this.mouseoverEnabled = true;\n\n    /**\n     * Tiny little interactiveData pool !\n     *\n     * @member {Array}\n     */\n    this.pool = [];\n\n    /**\n     * An array containing all the iterative items from the our interactive tree\n     *\n     * @member {Array}\n     * @private\n     */\n    this.interactiveItems = [];\n\n    /**\n     * The DOM element to bind to.\n     *\n     * @member {HTMLElement}\n     * @private\n     */\n    this.interactionDOMElement = null;\n\n    /**\n     * Have events been attached to the dom element?\n     *\n     * @member {boolean}\n     * @private\n     */\n    this.eventsAdded = false;\n\n    //this will make it so that you don't have to call bind all the time\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseMove = this.onMouseMove.bind( this );\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseDown = this.onMouseDown.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseOut = this.onMouseOut.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onMouseUp = this.onMouseUp.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onTouchStart = this.onTouchStart.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onTouchEnd = this.onTouchEnd.bind(this);\n\n    /**\n     * @member {Function}\n     */\n    this.onTouchMove = this.onTouchMove.bind(this);\n\n    /**\n     * @member {number}\n     */\n    this.last = 0;\n\n    /**\n     * The css style of the cursor that is being used\n     * @member {string}\n     */\n    this.currentCursorStyle = 'inherit';\n\n    /**\n     * Is set to true when the mouse is moved out of the canvas\n     * @member {boolean}\n     */\n    this.mouseOut = false;\n\n    /**\n     * @member {number}\n     */\n    this.resolution = 1;\n\n    // used for hit testing\n    this._tempPoint = new core.math.Point();\n}\n\nInteractionManager.prototype.constructor = InteractionManager;\nmodule.exports = InteractionManager;\n\n/**\n * Collects an interactive sprite recursively to have their interactions managed\n *\n * @param displayObject {DisplayObject} the displayObject to collect\n * @param iParent {DisplayObject} the display object's parent\n * @private\n */\nInteractionManager.prototype.collectInteractiveSprite = function (displayObject, iParent) {\n    var children = displayObject.children;\n    var length = children.length;\n\n    // make an interaction tree... {item.__interactiveParent}\n    for (var i = length - 1; i >= 0; i--) {\n        var child = children[i];\n\n        // push all interactive bits\n        if (child._interactive) {\n            iParent.interactiveChildren = true;\n            //child.__iParent = iParent;\n            this.interactiveItems.push(child);\n\n            if (child.children.length > 0) {\n                this.collectInteractiveSprite(child, child);\n            }\n        }\n        else {\n            child.__iParent = null;\n            if (child.children.length > 0) {\n                this.collectInteractiveSprite(child, iParent);\n            }\n        }\n\n    }\n};\n\n/**\n * Sets the DOM element which will receive mouse/touch events. This is useful for when you have\n * other DOM elements on top of the renderers Canvas element. With this you'll be bale to deletegate\n * another DOM element to receive those events.\n *\n * @param element {HTMLElement} the DOM element which will receive mouse and touch events.\n * @param [resolution=1] {number} THe resolution of the new element (relative to the canvas).\n * @private\n */\nInteractionManager.prototype.setTargetElement = function (element, resolution) {\n    this.removeEvents();\n\n    this.interactionDOMElement = element;\n\n    this.resolution = resolution || 1;\n\n    this.addEvents();\n};\n\n/**\n *\n * @private\n */\nInteractionManager.prototype.addEvents = function () {\n    if (!this.interactionDOMElement) {\n        return;\n    }\n\n    if (window.navigator.msPointerEnabled) {\n        this.interactionDOMElement.style['-ms-content-zooming'] = 'none';\n        this.interactionDOMElement.style['-ms-touch-action'] = 'none';\n    }\n\n    this.interactionDOMElement.addEventListener('mousemove',    this.onMouseMove, true);\n    this.interactionDOMElement.addEventListener('mousedown',    this.onMouseDown, true);\n    this.interactionDOMElement.addEventListener('mouseout',     this.onMouseOut, true);\n\n    this.interactionDOMElement.addEventListener('touchstart',   this.onTouchStart, true);\n    this.interactionDOMElement.addEventListener('touchend',     this.onTouchEnd, true);\n    this.interactionDOMElement.addEventListener('touchmove',    this.onTouchMove, true);\n\n    window.addEventListener('mouseup',  this.onMouseUp, true);\n\n    this.eventsAdded = true;\n};\n\n/**\n *\n * @private\n */\nInteractionManager.prototype.removeEvents = function () {\n    if (!this.interactionDOMElement) {\n        return;\n    }\n\n    if (window.navigator.msPointerEnabled) {\n        this.interactionDOMElement.style['-ms-content-zooming'] = '';\n        this.interactionDOMElement.style['-ms-touch-action'] = '';\n    }\n\n    this.interactionDOMElement.removeEventListener('mousemove', this.onMouseMove, true);\n    this.interactionDOMElement.removeEventListener('mousedown', this.onMouseDown, true);\n    this.interactionDOMElement.removeEventListener('mouseout',  this.onMouseOut, true);\n\n    this.interactionDOMElement.removeEventListener('touchstart', this.onTouchStart, true);\n    this.interactionDOMElement.removeEventListener('touchend',  this.onTouchEnd, true);\n    this.interactionDOMElement.removeEventListener('touchmove', this.onTouchMove, true);\n\n    this.interactionDOMElement = null;\n\n    window.removeEventListener('mouseup',  this.onMouseUp, true);\n\n    this.eventsAdded = false;\n};\n\n/**\n * updates the state of interactive objects\n *\n * @private\n */\nInteractionManager.prototype.update = function () {\n    if (!this.interactionDOMElement) {\n        return;\n    }\n\n    // frequency of 30fps??\n    var now = Date.now();\n    var diff = now - this.last;\n    diff = (diff * INTERACTION_FREQUENCY ) / 1000;\n    if (diff < 1) {\n        return;\n    }\n\n    this.last = now;\n\n    var i = 0;\n\n    // ok.. so mouse events??\n    // yes for now :)\n    // OPTIMISE - how often to check??\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    // loop through interactive objects!\n    var length = this.interactiveItems.length;\n    var cursor = 'inherit';\n    var over = false;\n\n    for (i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        // OPTIMISATION - only calculate every time if the mousemove function exists..\n        // OK so.. does the object have any other interactive functions?\n        // hit-test the clip!\n       // if (item.mouseover || item.mouseout || item.buttonMode)\n       // {\n        // ok so there are some functions so lets hit test it..\n        item.__hit = this.hitTest(item, this.mouse);\n        this.mouse.target = item;\n        // ok so deal with interactions..\n        // looks like there was a hit!\n        if (item.__hit && !over) {\n            if (item.buttonMode) {\n                cursor = item.defaultCursor;\n            }\n\n            if (!item.interactiveChildren) {\n                over = true;\n            }\n\n            if (!item.__isOver) {\n                if (item.mouseover) {\n                    item.mouseover (this.mouse);\n                }\n                item.__isOver = true;\n            }\n        }\n        else {\n            if (item.__isOver) {\n                // roll out!\n                if (item.mouseout) {\n                    item.mouseout (this.mouse);\n                }\n                item.__isOver = false;\n            }\n        }\n    }\n\n    if (this.currentCursorStyle !== cursor) {\n        this.currentCursorStyle = cursor;\n        this.interactionDOMElement.style.cursor = cursor;\n    }\n};\n\n/**\n * @private\n */\nInteractionManager.prototype.rebuildInteractiveGraph = function () {\n    this.dirty = false;\n\n    var len = this.interactiveItems.length;\n\n    for (var i = 0; i < len; i++) {\n        this.interactiveItems[i].interactiveChildren = false;\n    }\n\n    this.interactiveItems.length = 0;\n\n    if (this.stage.interactive) {\n        this.interactiveItems.push(this.stage);\n    }\n\n    // Go through and collect all the objects that are interactive..\n    this.collectInteractiveSprite(this.stage, this.stage);\n};\n\n/**\n * Is called when the mouse moves across the renderer element\n *\n * @param event {Event} The DOM event of the mouse moving\n * @private\n */\nInteractionManager.prototype.onMouseMove = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    // TODO optimize by not check EVERY TIME! maybe half as often? //\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n\n    this.mouse.global.x = (event.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) / this.resolution;\n    this.mouse.global.y = (event.clientY - rect.top) * ( this.interactionDOMElement.height / rect.height) / this.resolution;\n\n    var length = this.interactiveItems.length;\n\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        // Call the function!\n        if (item.mousemove) {\n            item.mousemove(this.mouse);\n        }\n    }\n};\n\n/**\n * Is called when the mouse button is pressed down on the renderer element\n *\n * @param event {Event} The DOM event of a mouse button being pressed down\n * @private\n */\nInteractionManager.prototype.onMouseDown = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    if (AUTO_PREVENT_DEFAULT) {\n        this.mouse.originalEvent.preventDefault();\n    }\n\n    // loop through interaction tree...\n    // hit test each item! ->\n    // get interactive items under point??\n    //stage.__i\n    var length = this.interactiveItems.length;\n\n    var e = this.mouse.originalEvent;\n    var isRightButton = e.button === 2 || e.which === 3;\n    var downFunction = isRightButton ? 'rightdown' : 'mousedown';\n    var clickFunction = isRightButton ? 'rightclick' : 'click';\n    var buttonIsDown = isRightButton ? '__rightIsDown' : '__mouseIsDown';\n    var isDown = isRightButton ? '__isRightDown' : '__isDown';\n\n    // while\n    // hit test\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        if (item[downFunction] || item[clickFunction]) {\n            item[buttonIsDown] = true;\n            item.__hit = this.hitTest(item, this.mouse);\n\n            if (item.__hit) {\n                //call the function!\n                if (item[downFunction]) {\n                    item[downFunction](this.mouse);\n                }\n                item[isDown] = true;\n\n                // just the one!\n                if (!item.interactiveChildren) {\n                    break;\n                }\n            }\n        }\n    }\n};\n\n/**\n * Is called when the mouse is moved out of the renderer element\n *\n * @param event {Event} The DOM event of a mouse being moved out\n * @private\n */\nInteractionManager.prototype.onMouseOut = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    var length = this.interactiveItems.length;\n\n    this.interactionDOMElement.style.cursor = 'inherit';\n\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n        if (item.__isOver) {\n            this.mouse.target = item;\n            if (item.mouseout) {\n                item.mouseout(this.mouse);\n            }\n            item.__isOver = false;\n        }\n    }\n\n    this.mouseOut = true;\n\n    // move the mouse to an impossible position\n    this.mouse.global.x = -10000;\n    this.mouse.global.y = -10000;\n};\n\n/**\n * Is called when the mouse button is released on the renderer element\n *\n * @param event {Event} The DOM event of a mouse button being released\n * @private\n */\nInteractionManager.prototype.onMouseUp = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    this.mouse.originalEvent = event;\n\n    var length = this.interactiveItems.length;\n    var up = false;\n\n    var e = this.mouse.originalEvent;\n    var isRightButton = e.button === 2 || e.which === 3;\n\n    var upFunction = isRightButton ? 'rightup' : 'mouseup';\n    var clickFunction = isRightButton ? 'rightclick' : 'click';\n    var upOutsideFunction = isRightButton ? 'rightupoutside' : 'mouseupoutside';\n    var isDown = isRightButton ? '__isRightDown' : '__isDown';\n\n    for (var i = 0; i < length; i++) {\n        var item = this.interactiveItems[i];\n\n        if (item[clickFunction] || item[upFunction] || item[upOutsideFunction]) {\n            item.__hit = this.hitTest(item, this.mouse);\n\n            if (item.__hit && !up) {\n                //call the function!\n                if (item[upFunction]) {\n                    item[upFunction](this.mouse);\n                }\n                if (item[isDown]) {\n                    if (item[clickFunction]) {\n                        item[clickFunction](this.mouse);\n                    }\n                }\n\n                if (!item.interactiveChildren) {\n                    up = true;\n                }\n            }\n            else {\n                if (item[isDown]) {\n                    if (item[upOutsideFunction]) {\n                        item[upOutsideFunction](this.mouse);\n                    }\n                }\n            }\n\n            item[isDown] = false;\n        }\n    }\n};\n\n/**\n * Tests if the current mouse coordinates hit a sprite\n *\n * @param item {DisplayObject} The displayObject to test for a hit\n * @param interactionData {InteractionData} The interactionData object to update in the case there is a hit\n * @private\n */\nInteractionManager.prototype.hitTest = function (item, interactionData) {\n    var global = interactionData.global;\n\n    if (!item.worldVisible) {\n        return false;\n    }\n\n    // map the global point to local space.\n    item.worldTransform.applyInverse(global,  this._tempPoint);\n\n    var x = this._tempPoint.x,\n        y = this._tempPoint.y,\n        i;\n\n    interactionData.target = item;\n\n    //a sprite or display object with a hit area defined\n    if (item.hitArea && item.hitArea.contains) {\n        return item.hitArea.contains(x, y);\n    }\n    // a sprite with no hitarea defined\n    else if (item instanceof core.Sprite) {\n        var width = item.texture.frame.width;\n        var height = item.texture.frame.height;\n        var x1 = -width * item.anchor.x;\n        var y1;\n\n        if (x > x1 && x < x1 + width) {\n            y1 = -height * item.anchor.y;\n\n            if (y > y1 && y < y1 + height) {\n                // set the target property if a hit is true!\n                return true;\n            }\n        }\n    }\n    else if (item instanceof core.Graphics) {\n        var graphicsData = item.graphicsData;\n        for (i = 0; i < graphicsData.length; i++) {\n            var data = graphicsData[i];\n\n            if (!data.fill) {\n                continue;\n            }\n\n            // only deal with fills..\n            if (data.shape) {\n                if (data.shape.contains(x, y)) {\n                    //interactionData.target = item;\n                    return true;\n                }\n            }\n        }\n    }\n\n    var length = item.children.length;\n\n    for (i = 0; i < length; i++) {\n        var tempItem = item.children[i];\n        var hit = this.hitTest(tempItem, interactionData);\n        if (hit) {\n            // hmm.. TODO SET CORRECT TARGET?\n            interactionData.target = item;\n            return true;\n        }\n    }\n    return false;\n};\n\n/**\n * Is called when a touch is moved across the renderer element\n *\n * @param event {Event} The DOM event of a touch moving across the renderer view\n * @private\n */\nInteractionManager.prototype.onTouchMove = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n    var changedTouches = event.changedTouches;\n    var touchData;\n    var i = 0;\n\n    for (i = 0; i < changedTouches.length; i++) {\n        var touchEvent = changedTouches[i];\n        touchData = this.touches[touchEvent.identifier];\n        touchData.originalEvent = event;\n\n        // update the touch position\n        touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution;\n        touchData.global.y = ( (touchEvent.clientY - rect.top)  * (this.interactionDOMElement.height / rect.height) )  / this.resolution;\n        if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) {\n            //Support for CocoonJS fullscreen scale modes\n            touchData.global.x = touchEvent.clientX;\n            touchData.global.y = touchEvent.clientY;\n        }\n\n        for (var j = 0; j < this.interactiveItems.length; j++) {\n            var item = this.interactiveItems[j];\n            if (item.touchmove && item.__touchData && item.__touchData[touchEvent.identifier]) {\n                item.touchmove(touchData);\n            }\n        }\n    }\n};\n\n/**\n * Is called when a touch is started on the renderer element\n *\n * @param event {Event} The DOM event of a touch starting on the renderer view\n * @private\n */\nInteractionManager.prototype.onTouchStart = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n\n    if (AUTO_PREVENT_DEFAULT) {\n        event.preventDefault();\n    }\n\n    var changedTouches = event.changedTouches;\n    for (var i=0; i < changedTouches.length; i++) {\n        var touchEvent = changedTouches[i];\n\n        var touchData = this.pool.pop();\n        if (!touchData) {\n            touchData = new InteractionData();\n        }\n\n        touchData.originalEvent = event;\n\n        this.touches[touchEvent.identifier] = touchData;\n        touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution;\n        touchData.global.y = ( (touchEvent.clientY - rect.top)  * (this.interactionDOMElement.height / rect.height) ) / this.resolution;\n        if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) {\n            //Support for CocoonJS fullscreen scale modes\n            touchData.global.x = touchEvent.clientX;\n            touchData.global.y = touchEvent.clientY;\n        }\n\n        var length = this.interactiveItems.length;\n\n        for (var j = 0; j < length; j++) {\n            var item = this.interactiveItems[j];\n\n            if (item.touchstart || item.tap) {\n                item.__hit = this.hitTest(item, touchData);\n\n                if (item.__hit) {\n                    //call the function!\n                    if (item.touchstart) {\n                        item.touchstart(touchData);\n                    }\n\n                    item.__isDown = true;\n                    item.__touchData = item.__touchData || {};\n                    item.__touchData[touchEvent.identifier] = touchData;\n\n                    if (!item.interactiveChildren) {\n                        break;\n                    }\n                }\n            }\n        }\n    }\n};\n\n/**\n * Is called when a touch is ended on the renderer element\n *\n * @param event {Event} The DOM event of a touch ending on the renderer view\n * @private\n */\nInteractionManager.prototype.onTouchEnd = function (event) {\n    if (this.dirty) {\n        this.rebuildInteractiveGraph();\n    }\n\n    var rect = this.interactionDOMElement.getBoundingClientRect();\n    var changedTouches = event.changedTouches;\n\n    for (var i=0; i < changedTouches.length; i++) {\n        var touchEvent = changedTouches[i];\n        var touchData = this.touches[touchEvent.identifier];\n        var up = false;\n        touchData.global.x = ( (touchEvent.clientX - rect.left) * (this.interactionDOMElement.width / rect.width) ) / this.resolution;\n        touchData.global.y = ( (touchEvent.clientY - rect.top)  * (this.interactionDOMElement.height / rect.height) ) / this.resolution;\n        if (navigator.isCocoonJS && !rect.left && !rect.top && !event.target.style.width && !event.target.style.height) {\n            //Support for CocoonJS fullscreen scale modes\n            touchData.global.x = touchEvent.clientX;\n            touchData.global.y = touchEvent.clientY;\n        }\n\n        var length = this.interactiveItems.length;\n        for (var j = 0; j < length; j++) {\n            var item = this.interactiveItems[j];\n\n            if (item.__touchData && item.__touchData[touchEvent.identifier]) {\n\n                item.__hit = this.hitTest(item, item.__touchData[touchEvent.identifier]);\n\n                // so this one WAS down...\n                touchData.originalEvent = event;\n                // hitTest??\n\n                if (item.touchend || item.tap) {\n                    if (item.__hit && !up) {\n                        if (item.touchend) {\n                            item.touchend(touchData);\n                        }\n                        if (item.__isDown && item.tap) {\n                            item.tap(touchData);\n                        }\n                        if (!item.interactiveChildren) {\n                            up = true;\n                        }\n                    }\n                    else {\n                        if (item.__isDown && item.touchendoutside) {\n                            item.touchendoutside(touchData);\n                        }\n                    }\n\n                    item.__isDown = false;\n                }\n\n                item.__touchData[touchEvent.identifier] = null;\n            }\n        }\n        // remove the touch..\n        this.pool.push(touchData);\n        this.touches[touchEvent.identifier] = null;\n    }\n};\n","/**\n * @file        Main export of the PIXI interactions library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    InteractionData:    require('./InteractionData'),\n    InteractionManager: require('./InteractionManager')\n};\n","var core = require('../core'),\n    ImageLoader = require('./ImageLoader');\n\n/**\n * The atlas file loader is used to load in Texture Atlas data and parse it. When loaded this class will dispatch a 'loaded' event. If loading fails this class will dispatch an 'error' event.\n *\n * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format.\n *\n * It is highly recommended to use texture atlases (also know as 'sprite sheets') as it allowed sprites to be batched and drawn together for highly increased rendering speed.\n * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId()\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction AtlasLoader(url, crossorigin) {\n    this.url = url;\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n    this.crossorigin = crossorigin;\n    this.loaded = false;\n}\n\nAtlasLoader.prototype.constructor = AtlasLoader;\nmodule.exports = AtlasLoader;\n\ncore.utils.eventTarget.mixin(AtlasLoader.prototype);\n\n /**\n * Starts loading the JSON file\n *\n */\nAtlasLoader.prototype.load = function () {\n    this.ajaxRequest = new core.utils.AjaxRequest();\n    this.ajaxRequest.onreadystatechange = this.onAtlasLoaded.bind(this);\n\n    this.ajaxRequest.open('GET', this.url, true);\n\n    if (this.ajaxRequest.overrideMimeType) {\n        this.ajaxRequest.overrideMimeType('application/json');\n    }\n\n    this.ajaxRequest.send(null);\n};\n\n/**\n * Invoked when the Atlas has fully loaded. Parses the JSON and builds the texture frames.\n *\n * @private\n */\nAtlasLoader.prototype.onAtlasLoaded = function () {\n    if (this.ajaxRequest.readyState === 4) {\n        if (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1) {\n            this.atlas = {\n                meta : {\n                    image : []\n                },\n                frames : []\n            };\n            var result = this.ajaxRequest.responseText.split(/\\r?\\n/);\n            var lineCount = -3;\n\n            var currentImageId = 0;\n            var currentFrame = null;\n            var nameInNextLine = false;\n\n            var i = 0,\n                j = 0,\n                selfOnLoaded = this.onLoaded.bind(this);\n\n            // parser without rotation support yet!\n            for (i = 0; i < result.length; i++) {\n                result[i] = result[i].replace(/^\\s+|\\s+$/g, '');\n\n                if (result[i] === '') {\n                    nameInNextLine = i+1;\n                }\n\n                if (result[i].length > 0) {\n                    if (nameInNextLine === i) {\n                        this.atlas.meta.image.push(result[i]);\n                        currentImageId = this.atlas.meta.image.length - 1;\n                        this.atlas.frames.push({});\n                        lineCount = -3;\n                    } else if (lineCount > 0) {\n                        if (lineCount % 7 === 1) { // frame name\n                            if (currentFrame != null) { //jshint ignore:line\n                                this.atlas.frames[currentImageId][currentFrame.name] = currentFrame;\n                            }\n                            currentFrame = { name: result[i], frame : {} };\n                        } else {\n                            var text = result[i].split(' ');\n                            if (lineCount % 7 === 3) { // position\n                                currentFrame.frame.x = Number(text[1].replace(',', ''));\n                                currentFrame.frame.y = Number(text[2]);\n                            } else if (lineCount % 7 === 4) { // size\n                                currentFrame.frame.w = Number(text[1].replace(',', ''));\n                                currentFrame.frame.h = Number(text[2]);\n                            } else if (lineCount % 7 === 5) { // real size\n                                var realSize = {\n                                    x : 0,\n                                    y : 0,\n                                    w : Number(text[1].replace(',', '')),\n                                    h : Number(text[2])\n                                };\n\n                                if (realSize.w > currentFrame.frame.w || realSize.h > currentFrame.frame.h) {\n                                    currentFrame.trimmed = true;\n                                    currentFrame.realSize = realSize;\n                                } else {\n                                    currentFrame.trimmed = false;\n                                }\n                            }\n                        }\n                    }\n                    lineCount++;\n                }\n            }\n\n            if (currentFrame != null) { //jshint ignore:line\n                this.atlas.frames[currentImageId][currentFrame.name] = currentFrame;\n            }\n\n            if (this.atlas.meta.image.length > 0) {\n                this.images = [];\n                for (j = 0; j < this.atlas.meta.image.length; j++) {\n                    // sprite sheet\n                    var textureUrl = this.baseUrl + this.atlas.meta.image[j];\n                    var frameData = this.atlas.frames[j];\n                    this.images.push(new ImageLoader(textureUrl, this.crossorigin));\n\n                    for (i in frameData) {\n                        var rect = frameData[i].frame;\n                        if (rect) {\n                            core.utils.TextureCache[i] = new core.Texture(this.images[j].texture.baseTexture, {\n                                x: rect.x,\n                                y: rect.y,\n                                width: rect.w,\n                                height: rect.h\n                            });\n                            if (frameData[i].trimmed) {\n                                core.utils.TextureCache[i].realSize = frameData[i].realSize;\n                                // trim in pixi not supported yet, todo update trim properties if it is done ...\n                                core.utils.TextureCache[i].trim.x = 0;\n                                core.utils.TextureCache[i].trim.y = 0;\n                            }\n                        }\n                    }\n                }\n\n                this.currentImageId = 0;\n                for (j = 0; j < this.images.length; j++) {\n                    this.images[j].on('loaded', selfOnLoaded);\n                }\n                this.images[this.currentImageId].load();\n\n            } else {\n                this.onLoaded();\n            }\n\n        } else {\n            this.onError();\n        }\n    }\n};\n\n/**\n * Invoked when json file has loaded.\n *\n * @private\n */\nAtlasLoader.prototype.onLoaded = function () {\n    if (this.images.length - 1 > this.currentImageId) {\n        this.currentImageId++;\n        this.images[this.currentImageId].load();\n    } else {\n        this.loaded = true;\n        this.emit('loaded', { content: this });\n    }\n};\n\n/**\n * Invoked when an error occurs.\n *\n * @private\n */\nAtlasLoader.prototype.onError = function () {\n    this.emit('error', { content: this });\n};\n","var core = require('../core'),\n    ImageLoader = require('./ImageLoader');\n\n/**\n * The xml loader is used to load in XML bitmap font data ('xml' or 'fnt')\n * To generate the data you can use http://www.angelcode.com/products/bmfont/\n * This loader will also load the image file as the data.\n * When loaded this class will dispatch a 'loaded' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the sprite sheet JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction BitmapFontLoader(url, crossorigin) {\n    /**\n     * The url of the bitmap font data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * The base url of the bitmap font data\n     *\n     * @member {String}\n     * @readOnly\n     */\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n\n    /**\n     * The texture of the bitmap font\n     *\n     * @member {Texture}\n     */\n    this.texture = null;\n}\n\n// constructor\nBitmapFontLoader.prototype.constructor = BitmapFontLoader;\nmodule.exports = BitmapFontLoader;\n\ncore.utils.eventTarget.mixin(BitmapFontLoader.prototype);\n\n/**\n * Loads the XML font data\n *\n */\nBitmapFontLoader.prototype.load = function () {\n    this.ajaxRequest = new core.utils.AjaxRequest();\n    this.ajaxRequest.onreadystatechange = this.onXMLLoaded.bind(this);\n\n    this.ajaxRequest.open('GET', this.url, true);\n\n    if (this.ajaxRequest.overrideMimeType) {\n        this.ajaxRequest.overrideMimeType('application/xml');\n    }\n\n    this.ajaxRequest.send(null);\n};\n\n/**\n * Invoked when the XML file is loaded, parses the data.\n *\n * @private\n */\nBitmapFontLoader.prototype.onXMLLoaded = function () {\n    if (this.ajaxRequest.readyState === 4) {\n        if (this.ajaxRequest.status === 200 || window.location.protocol.indexOf('http') === -1) {\n            var responseXML = this.ajaxRequest.responseXML;\n            if (!responseXML || /MSIE 9/i.test(navigator.userAgent) || navigator.isCocoonJS) {\n                if (typeof(window.DOMParser) === 'function') {\n                    var domparser = new DOMParser();\n                    responseXML = domparser.parseFromString(this.ajaxRequest.responseText, 'text/xml');\n                } else {\n                    var div = document.createElement('div');\n                    div.innerHTML = this.ajaxRequest.responseText;\n                    responseXML = div;\n                }\n            }\n\n            var textureUrl = this.baseUrl + responseXML.getElementsByTagName('page')[0].getAttribute('file');\n            var image = new ImageLoader(textureUrl, this.crossorigin);\n            this.texture = image.texture.baseTexture;\n\n            var data = {};\n            var info = responseXML.getElementsByTagName('info')[0];\n            var common = responseXML.getElementsByTagName('common')[0];\n            data.font = info.getAttribute('face');\n            data.size = parseInt(info.getAttribute('size'), 10);\n            data.lineHeight = parseInt(common.getAttribute('lineHeight'), 10);\n            data.chars = {};\n\n            //parse letters\n            var letters = responseXML.getElementsByTagName('char');\n\n            for (var i = 0; i < letters.length; i++) {\n                var charCode = parseInt(letters[i].getAttribute('id'), 10);\n\n                var textureRect = new core.math.Rectangle(\n                    parseInt(letters[i].getAttribute('x'), 10),\n                    parseInt(letters[i].getAttribute('y'), 10),\n                    parseInt(letters[i].getAttribute('width'), 10),\n                    parseInt(letters[i].getAttribute('height'), 10)\n                );\n\n                data.chars[charCode] = {\n                    xOffset: parseInt(letters[i].getAttribute('xoffset'), 10),\n                    yOffset: parseInt(letters[i].getAttribute('yoffset'), 10),\n                    xAdvance: parseInt(letters[i].getAttribute('xadvance'), 10),\n                    kerning: {},\n                    texture: core.utils.TextureCache[charCode] = new core.Texture(this.texture, textureRect)\n\n                };\n            }\n\n            //parse kernings\n            var kernings = responseXML.getElementsByTagName('kerning');\n            for (i = 0; i < kernings.length; i++) {\n                var first = parseInt(kernings[i].getAttribute('first'), 10);\n                var second = parseInt(kernings[i].getAttribute('second'), 10);\n                var amount = parseInt(kernings[i].getAttribute('amount'), 10);\n\n                data.chars[second].kerning[first] = amount;\n\n            }\n\n            core.BitmapText.fonts[data.font] = data;\n\n            image.addEventListener('loaded', this.onLoaded.bind(this));\n            image.load();\n        }\n    }\n};\n\n/**\n * Invoked when all files are loaded (xml/fnt and texture)\n *\n * @private\n */\nBitmapFontLoader.prototype.onLoaded = function () {\n    this.emit('loaded', { content: this });\n};\n","var core = require('../core');\n\n/**\n * The image loader class is responsible for loading images file formats ('jpeg', 'jpg', 'png' and 'gif')\n * Once the image has been loaded it is stored in the PIXI texture cache and can be accessed though Texture.fromFrame() and Sprite.fromFrame()\n * When loaded this class will dispatch a 'loaded' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the image\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction ImageLoader(url, crossorigin) {\n    /**\n     * The texture being loaded\n     *\n     * @member {Texture}\n     */\n    this.texture = core.Texture.fromImage(url, crossorigin);\n\n    /**\n     * if the image is loaded with loadFramedSpriteSheet\n     * frames will contain the sprite sheet frames\n     *\n     * @member {Array}\n     * @readOnly\n     */\n    this.frames = [];\n}\n\n// constructor\nImageLoader.prototype.constructor = ImageLoader;\nmodule.exports = ImageLoader;\n\ncore.utils.eventTarget.mixin(ImageLoader.prototype);\n\n/**\n * Loads image or takes it from cache\n *\n */\nImageLoader.prototype.load = function () {\n    if (!this.texture.baseTexture.hasLoaded) {\n        this.texture.baseTexture.on('loaded', this.onLoaded.bind(this));\n        this.texture.baseTexture.on('error', this.onError.bind(this));\n    }\n    else {\n        this.onLoaded();\n    }\n};\n\n/**\n * Invoked when image file is loaded or it is already cached and ready to use\n *\n * @private\n */\nImageLoader.prototype.onLoaded = function () {\n    this.emit('loaded', { content: this });\n};\n\n/**\n * Invoked when image file failed loading\n *\n * @method onError\n * @private\n */\nImageLoader.prototype.onError = function () {\n    this.emit('error', { content: this });\n};\n\n/**\n * Loads image and split it to uniform sized frames\n *\n * @param frameWidth {number} width of each frame\n * @param frameHeight {number} height of each frame\n * @param textureName {String} if given, the frames will be cached in <textureName>-<ord> format\n */\nImageLoader.prototype.loadFramedSpriteSheet = function (frameWidth, frameHeight, textureName) {\n    this.frames = [];\n\n    var cols = Math.floor(this.texture.width / frameWidth);\n    var rows = Math.floor(this.texture.height / frameHeight);\n\n    var i=0;\n    for (var y = 0; y < rows; ++y) {\n        for (var x = 0; x < cols; ++x, ++i) {\n            var texture = new core.Texture(\n                this.texture.baseTexture,\n                new core.math.Rectangle(\n                    x * frameWidth,\n                    y * frameHeight,\n                    frameWidth,\n                    frameHeight\n                )\n            );\n\n            this.frames.push(texture);\n\n            if (textureName) {\n                core.utils.TextureCache[textureName + '-' + i] = texture;\n            }\n        }\n    }\n\n\tthis.load();\n};\n","var core = require('../core'),\n    spine = require('../spine/SpineRuntime'),\n    ImageLoader = require('./ImageLoader'),\n    SpineTextureLoader = require('./SpineTextureLoader');\n\n/**\n * The json file loader is used to load in JSON data and parse it\n * When loaded this class will dispatch a 'loaded' event\n * If loading fails this class will dispatch an 'error' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction JsonLoader(url, crossorigin) {\n    /**\n     * The url of the bitmap font data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * The base url of the bitmap font data\n     *\n     * @member {String}\n     * @readOnly\n     */\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n\n    /**\n     * Whether the data has loaded yet\n     *\n     * @member {boolean}\n     * @readOnly\n     */\n     this.loaded = false;\n}\n\n// constructor\nJsonLoader.prototype.constructor = JsonLoader;\nmodule.exports = JsonLoader;\n\ncore.utils.eventTarget.mixin(JsonLoader.prototype);\n\n/**\n * Loads the JSON data\n *\n */\nJsonLoader.prototype.load = function () {\n    if (window.XDomainRequest && this.crossorigin) {\n        this.ajaxRequest = new window.XDomainRequest();\n\n        // XDomainRequest has a few quirks. Occasionally it will abort requests\n        // A way to avoid this is to make sure ALL callbacks are set even if not used\n        // More info here: http://stackoverflow.com/questions/15786966/xdomainrequest-aborts-post-on-ie-9\n        this.ajaxRequest.timeout = 3000;\n\n        this.ajaxRequest.onerror = this.onError.bind(this);\n        this.ajaxRequest.ontimeout = this.onError.bind(this);\n\n        this.ajaxRequest.onprogress = function () {};\n\n        this.ajaxRequest.onload = this.onJSONLoaded.bind(this);\n    }\n    else {\n        if (window.XMLHttpRequest) {\n            this.ajaxRequest = new window.XMLHttpRequest();\n        }\n        else {\n            this.ajaxRequest = new window.ActiveXObject('Microsoft.XMLHTTP');\n        }\n\n        this.ajaxRequest.onreadystatechange = this.onReadyStateChanged.bind(this);\n    }\n\n    this.ajaxRequest.open('GET',this.url,true);\n\n    this.ajaxRequest.send();\n};\n\n/**\n * Bridge function to be able to use the more reliable onreadystatechange in XMLHttpRequest.\n *\n * @private\n */\nJsonLoader.prototype.onReadyStateChanged = function () {\n    if (this.ajaxRequest.readyState === 4 && (this.ajaxRequest.status === 200 || window.location.href.indexOf('http') === -1)) {\n        this.onJSONLoaded();\n    }\n};\n\n/**\n * Invoke when JSON file is loaded\n *\n * @private\n */\nJsonLoader.prototype.onJSONLoaded = function () {\n    if (!this.ajaxRequest.responseText) {\n        this.onError();\n        return;\n    }\n\n    this.json = JSON.parse(this.ajaxRequest.responseText);\n\n    if (this.json.frames) {\n        // sprite sheet\n        var textureUrl = this.baseUrl + this.json.meta.image;\n        var image = new ImageLoader(textureUrl, this.crossorigin);\n        var frameData = this.json.frames;\n\n        this.texture = image.texture.baseTexture;\n        image.addEventListener('loaded', this.onLoaded.bind(this));\n        image.addEventListener('error', this.onError.bind(this));\n\n        for (var i in frameData) {\n            var rect = frameData[i].frame;\n\n            if (rect) {\n                var textureSize = new core.math.Rectangle(rect.x, rect.y, rect.w, rect.h);\n                var crop = textureSize.clone();\n                var trim = null;\n\n                //  Check to see if the sprite is trimmed\n                if (frameData[i].trimmed) {\n                    var actualSize = frameData[i].sourceSize;\n                    var realSize = frameData[i].spriteSourceSize;\n                    trim = new core.math.Rectangle(realSize.x, realSize.y, actualSize.w, actualSize.h);\n                }\n                core.utils.TextureCache[i] = new core.Texture(this.texture, textureSize, crop, trim);\n            }\n        }\n\n        image.load();\n\n    }\n    else if (this.json.bones) {\n\t\t// check if the json was loaded before\n\t\tif (core.utils.AnimCache[this.url]) {\n\t\t\tthis.onLoaded();\n\t\t}\n\t\telse {\n\t\t\t/**\n             * use a bit of hackery to load the atlas file, here we assume that the .json, .atlas and .png files\n\t\t\t * that correspond to the spine file are in the same base URL and that the .json and .atlas files\n\t\t\t * have the same name\n\t\t\t */\n\t\t\tvar atlasPath = this.url.substr(0, this.url.lastIndexOf('.')) + '.atlas';\n\t\t\tvar atlasLoader = new JsonLoader(atlasPath, this.crossorigin);\n\t\t\t// save a copy of the current object for future reference //\n\t\t\tvar originalLoader = this;\n\t\t\t// before loading the file, replace the \"onJSONLoaded\" function for our own //\n\t\t\tatlasLoader.onJSONLoaded = function () {\n\t\t\t\t// at this point \"this\" points at the atlasLoader (JsonLoader) instance //\n\t\t\t\tif (!this.ajaxRequest.responseText) {\n\t\t\t\t\tthis.onError(); // FIXME: hmm, this is funny because we are not responding to errors yet\n\t\t\t\t\treturn;\n\t\t\t\t}\n\t\t\t\t// create a new instance of a spine texture loader for this spine object //\n\t\t\t\tvar textureLoader = new SpineTextureLoader(this.url.substring(0, this.url.lastIndexOf('/')));\n\t\t\t\t// create a spine atlas using the loaded text and a spine texture loader instance //\n\t\t\t\tvar spineAtlas = new spine.Atlas(this.ajaxRequest.responseText, textureLoader);\n\t\t\t\t// now we use an atlas attachment loader //\n\t\t\t\tvar attachmentLoader = new spine.AtlasAttachmentLoader(spineAtlas);\n\t\t\t\t// spine animation\n\t\t\t\tvar spineJsonParser = new spine.SkeletonJson(attachmentLoader);\n\t\t\t\tvar skeletonData = spineJsonParser.readSkeletonData(originalLoader.json);\n\t\t\t\tcore.utils.AnimCache[originalLoader.url] = skeletonData;\n\t\t\t\toriginalLoader.spine = skeletonData;\n\t\t\t\toriginalLoader.spineAtlas = spineAtlas;\n\t\t\t\toriginalLoader.spineAtlasLoader = atlasLoader;\n\t\t\t\t// wait for textures to finish loading if needed\n\t\t\t\tif (textureLoader.loadingCount > 0) {\n\t\t\t\t\ttextureLoader.addEventListener('loadedBaseTexture', function (evt){\n\t\t\t\t\t\tif (evt.content.content.loadingCount <= 0) {\n\t\t\t\t\t\t\toriginalLoader.onLoaded();\n\t\t\t\t\t\t}\n\t\t\t\t\t});\n\t\t\t\t}\n\t\t\t\telse {\n\t\t\t\t\toriginalLoader.onLoaded();\n\t\t\t\t}\n\t\t\t};\n\t\t\t// start the loading //\n\t\t\tatlasLoader.load();\n\t\t}\n    }\n    else {\n        this.onLoaded();\n    }\n};\n\n/**\n * Invoke when json file loaded\n *\n * @private\n */\nJsonLoader.prototype.onLoaded = function () {\n    this.loaded = true;\n    this.dispatchEvent({\n        type: 'loaded',\n        content: this\n    });\n};\n\n/**\n * Invoke when error occured\n *\n * @private\n */\nJsonLoader.prototype.onError = function () {\n\n    this.dispatchEvent({\n        type: 'error',\n        content: this\n    });\n};\n","var core = require('../core'),\n    JsonLoader = require('./JsonLoader');\n\n/**\n * @author Mat Groves http://matgroves.com/ @Doormat23\n * based on pixi impact spine implementation made by Eemeli Kelokorpi (@ekelokorpi) https://github.com/ekelokorpi\n *\n * Awesome JS run time provided by EsotericSoftware\n * https://github.com/EsotericSoftware/spine-runtimes\n *\n */\n\n/**\n * The Spine loader is used to load in JSON spine data\n * To generate the data you need to use http://esotericsoftware.com/ and export in the \"JSON\" format\n * Due to a clash of names  You will need to change the extension of the spine file from *.json to *.anim for it to load\n * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source\n * You will need to generate a sprite sheet to accompany the spine data\n * When loaded this class will dispatch a \"loaded\" event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction SpineLoader(url, crossorigin) {\n    /**\n     * The url of the bitmap font data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * Whether the data has loaded yet\n     *\n     * @member {boolean}\n     * @readOnly\n     */\n    this.loaded = false;\n}\n\nSpineLoader.prototype.constructor = SpineLoader;\nmodule.exports = SpineLoader;\n\ncore.utils.eventTarget.mixin(SpineLoader.prototype);\n\n/**\n * Loads the JSON data\n *\n */\nSpineLoader.prototype.load = function () {\n    var scope = this;\n    var jsonLoader = new JsonLoader(this.url, this.crossorigin);\n\n    jsonLoader.on('loaded', function (event) {\n        scope.json = event.data.content.json;\n        scope.onLoaded();\n    });\n\n    jsonLoader.load();\n};\n\n/**\n * Invoked when JSON file is loaded.\n *\n * @private\n */\nSpineLoader.prototype.onLoaded = function () {\n    this.loaded = true;\n    this.emit('loaded', { content: this });\n};\n","var core = require('../core');\n\n/**\n * Supporting class to load images from spine atlases as per spine spec.\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param basePath {string} Tha base path where to look for the images to be loaded\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction SpineTextureLoader(basePath, crossorigin) {\n    this.basePath = basePath;\n    this.crossorigin = crossorigin;\n    this.loadingCount = 0;\n}\n\nSpineTextureLoader.prototype.constructor = SpineTextureLoader;\nmodule.exports = SpineTextureLoader;\n\ncore.utils.eventTarget.mixin(SpineTextureLoader.prototype);\n\n/**\n * Starts loading a base texture as per spine specification\n *\n * @param page {spine.AtlasPage} Atlas page to which texture belongs\n * @param file {string} The file to load, this is just the file path relative to the base path configured in the constructor\n */\nSpineTextureLoader.prototype.load = function (page, file) {\n    page.rendererObject = core.BaseTexture.fromImage(this.basePath + '/' + file, this.crossorigin);\n    if (!page.rendererObject.hasLoaded) {\n        var scope = this;\n        ++scope.loadingCount;\n        page.rendererObject.addEventListener('loaded', function (){\n            --scope.loadingCount;\n            scope.dispatchEvent({\n                type: 'loadedBaseTexture',\n                content: scope\n            });\n        });\n    }\n};\n\n/**\n * Unloads a previously loaded texture as per spine specification\n *\n * @param texture {BaseTexture} Texture object to destroy\n */\nSpineTextureLoader.prototype.unload = function (texture) {\n    texture.destroy(true);\n};\n","var core = require('../core'),\n    JsonLoader = require('./JsonLoader');\n\n/**\n * The sprite sheet loader is used to load in JSON sprite sheet data\n * To generate the data you can use http://www.codeandweb.com/texturepacker and publish in the 'JSON' format\n * There is a free version so thats nice, although the paid version is great value for money.\n * It is highly recommended to use Sprite sheets (also know as a 'texture atlas') as it means sprites can be batched and drawn together for highly increased rendering speed.\n * Once the data has been loaded the frames are stored in the PIXI texture cache and can be accessed though Texture.fromFrameId() and Sprite.fromFrameId()\n * This loader will load the image file that the Spritesheet points to as well as the data.\n * When loaded this class will dispatch a 'loaded' event\n *\n * @class\n * @mixes eventTarget\n * @namespace PIXI\n * @param url {String} The url of the sprite sheet JSON file\n * @param crossorigin {boolean} Whether requests should be treated as crossorigin\n */\nfunction SpriteSheetLoader(url, crossorigin) {\n\n    /**\n     * The url of the atlas data\n     *\n     * @member {String}\n     */\n    this.url = url;\n\n    /**\n     * Whether the requests should be treated as cross origin\n     *\n     * @member {boolean}\n     */\n    this.crossorigin = crossorigin;\n\n    /**\n     * The base url of the bitmap font data\n     *\n     * @member {String}\n     * @readOnly\n     */\n    this.baseUrl = url.replace(/[^\\/]*$/, '');\n\n    /**\n     * The texture being loaded\n     *\n     * @member {Texture}\n     */\n    this.texture = null;\n\n    /**\n     * The frames of the sprite sheet\n     *\n     * @member {object}\n     */\n    this.frames = {};\n}\n\n// constructor\nSpriteSheetLoader.prototype.constructor = SpriteSheetLoader;\nmodule.exports = SpriteSheetLoader;\n\ncore.utils.eventTarget.mixin(SpriteSheetLoader.prototype);\n\n/**\n * This will begin loading the JSON file\n *\n */\nSpriteSheetLoader.prototype.load = function () {\n    var scope = this;\n    var jsonLoader = new JsonLoader(this.url, this.crossorigin);\n\n    jsonLoader.on('loaded', function (event) {\n        scope.json = event.data.content.json;\n        scope.onLoaded();\n    });\n\n    jsonLoader.load();\n};\n\n/**\n * Invoke when all files are loaded (json and texture)\n *\n * @private\n */\nSpriteSheetLoader.prototype.onLoaded = function () {\n    this.emit('loaded', {\n        content: this\n    });\n};\n","/**\n * @file        Main export of the PIXI loaders library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    AtlasLoader:        require('./AtlasLoader'),\n    BitmapFontLoader:   require('./BitmapFontLoader'),\n    ImageLoader:        require('./ImageLoader'),\n    JsonLoader:         require('./JsonLoader'),\n    SpineLoader:        require('./SpineLoader'),\n    SpriteSheetLoader:  require('./SpriteSheetLoader')\n};\n","var core = require('../core'),\n    spine = require('./SpineRuntime');\n\n/* Esoteric Software SPINE wrapper for pixi.js */\n\nspine.Bone.yDown = true;\n\n/**\n * A class that enables the you to import and run your spine animations in pixi.\n * Spine animation data needs to be loaded using the AssetLoader or SpineLoader before it can be used by this class\n * See example 12 (http://www.goodboydigital.com/pixijs/examples/12/) to see a working example and check out the source\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param url {string} The url of the spine anim file to be used\n */\nfunction Spine(url) {\n    core.DisplayObjectContainer.call(this);\n\n    this.spineData = core.utils.AnimCache[url];\n\n    if (!this.spineData) {\n        throw new Error('Spine data must be preloaded using SpineLoader or AssetLoader: ' + url);\n    }\n\n    this.skeleton = new spine.Skeleton(this.spineData);\n    this.skeleton.updateWorldTransform();\n\n    this.stateData = new spine.AnimationStateData(this.spineData);\n    this.state = new spine.AnimationState(this.stateData);\n\n    this.slotContainers = [];\n\n    for (var i = 0, n = this.skeleton.drawOrder.length; i < n; i++) {\n        var slot = this.skeleton.drawOrder[i];\n        var attachment = slot.attachment;\n        var slotContainer = new core.DisplayObjectContainer();\n        this.slotContainers.push(slotContainer);\n        this.addChild(slotContainer);\n\n        if (attachment instanceof spine.RegionAttachment) {\n            var spriteName = attachment.rendererObject.name;\n            var sprite = this.createSprite(slot, attachment);\n            slot.currentSprite = sprite;\n            slot.currentSpriteName = spriteName;\n            slotContainer.addChild(sprite);\n        }\n        else if (attachment instanceof spine.MeshAttachment) {\n            var mesh = this.createMesh(slot, attachment);\n            slot.currentMesh = mesh;\n            slot.currentMeshName = attachment.name;\n            slotContainer.addChild(mesh);\n        }\n        else {\n            continue;\n        }\n\n    }\n\n    this.autoUpdate = true;\n}\n\nSpine.prototype = Object.create(core.DisplayObjectContainer.prototype);\nSpine.prototype.constructor = Spine;\nmodule.exports = Spine;\n\nObject.defineProperties(Spine.prototype, {\n    /**\n     * If this flag is set to true, the spine animation will be autoupdated every time\n     * the object id drawn. The down side of this approach is that the delta time is\n     * automatically calculated and you could miss out on cool effects like slow motion,\n     * pause, skip ahead and the sorts. Most of these effects can be achieved even with\n     * autoupdate enabled but are harder to achieve.\n     *\n     * @member {boolean}\n     * @memberof Spine#\n     * @default true\n     */\n    autoUpdate: {\n        get: function () {\n            return (this.updateTransform === Spine.prototype.autoUpdateTransform);\n        },\n\n        set: function (value) {\n            this.updateTransform = value ? Spine.prototype.autoUpdateTransform : core.DisplayObjectContainer.prototype.updateTransform;\n        }\n    }\n});\n\n/**\n * Update the spine skeleton and its animations by delta time (dt)\n *\n * @param dt {number} Delta time. Time by which the animation should be updated\n */\nSpine.prototype.update = function (dt) {\n    this.state.update(dt);\n    this.state.apply(this.skeleton);\n    this.skeleton.updateWorldTransform();\n\n    var drawOrder = this.skeleton.drawOrder;\n    for (var i = 0, n = drawOrder.length; i < n; i++) {\n        var slot = drawOrder[i];\n        var attachment = slot.attachment;\n        var slotContainer = this.slotContainers[i];\n\n        if (!attachment) {\n            slotContainer.visible = false;\n            continue;\n        }\n\n        var type = attachment.type;\n        if (type === spine.AttachmentType.region) {\n            if (attachment.rendererObject) {\n                if (!slot.currentSpriteName || slot.currentSpriteName !== attachment.rendererObject.name) {\n                    var spriteName = attachment.rendererObject.name;\n                    if (slot.currentSprite !== undefined) {\n                        slot.currentSprite.visible = false;\n                    }\n                    slot.sprites = slot.sprites || {};\n                    if (slot.sprites[spriteName] !== undefined) {\n                        slot.sprites[spriteName].visible = true;\n                    }\n                    else {\n                        var sprite = this.createSprite(slot, attachment);\n                        slotContainer.addChild(sprite);\n                    }\n                    slot.currentSprite = slot.sprites[spriteName];\n                    slot.currentSpriteName = spriteName;\n                }\n            }\n\n            var bone = slot.bone;\n\n            slotContainer.position.x = bone.worldX + attachment.x * bone.m00 + attachment.y * bone.m01;\n            slotContainer.position.y = bone.worldY + attachment.x * bone.m10 + attachment.y * bone.m11;\n            slotContainer.scale.x = bone.worldScaleX;\n            slotContainer.scale.y = bone.worldScaleY;\n\n            slotContainer.rotation = -(slot.bone.worldRotation * spine.degRad);\n\n            slot.currentSprite.tint = core.utils.rgb2hex([slot.r,slot.g,slot.b]);\n        }\n        else if (type === spine.AttachmentType.skinnedmesh) {\n            if (!slot.currentMeshName || slot.currentMeshName !== attachment.name) {\n                var meshName = attachment.name;\n                if (slot.currentMesh !== undefined) {\n                    slot.currentMesh.visible = false;\n                }\n\n                slot.meshes = slot.meshes || {};\n\n                if (slot.meshes[meshName] !== undefined) {\n                    slot.meshes[meshName].visible = true;\n                }\n                else {\n                    var mesh = this.createMesh(slot, attachment);\n                    slotContainer.addChild(mesh);\n                }\n\n                slot.currentMesh = slot.meshes[meshName];\n                slot.currentMeshName = meshName;\n            }\n\n            attachment.computeWorldVertices(slot.bone.skeleton.x, slot.bone.skeleton.y, slot, slot.currentMesh.vertices);\n\n        }\n        else {\n            slotContainer.visible = false;\n            continue;\n        }\n        slotContainer.visible = true;\n\n        slotContainer.alpha = slot.a;\n    }\n};\n\n/**\n * When autoupdate is set to yes this function is used as pixi's updateTransform function\n *\n * @private\n */\nSpine.prototype.autoUpdateTransform = function () {\n    this.lastTime = this.lastTime || Date.now();\n    var timeDelta = (Date.now() - this.lastTime) * 0.001;\n    this.lastTime = Date.now();\n\n    this.update(timeDelta);\n\n    core.DisplayObjectContainer.prototype.updateTransform.call(this);\n};\n\n/**\n * Create a new sprite to be used with spine.RegionAttachment\n *\n * @param slot {spine.Slot} The slot to which the attachment is parented\n * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent\n * @private\n */\nSpine.prototype.createSprite = function (slot, attachment) {\n    var descriptor = attachment.rendererObject;\n    var baseTexture = descriptor.page.rendererObject;\n    var spriteRect = new core.math.Rectangle(descriptor.x,\n                                        descriptor.y,\n                                        descriptor.rotate ? descriptor.height : descriptor.width,\n                                        descriptor.rotate ? descriptor.width : descriptor.height);\n    var spriteTexture = new core.Texture(baseTexture, spriteRect);\n    var sprite = new core.Sprite(spriteTexture);\n\n    var baseRotation = descriptor.rotate ? Math.PI * 0.5 : 0.0;\n    sprite.scale.set(descriptor.width / descriptor.originalWidth, descriptor.height / descriptor.originalHeight);\n    sprite.rotation = baseRotation - (attachment.rotation * spine.degRad);\n    sprite.anchor.x = sprite.anchor.y = 0.5;\n\n    slot.sprites = slot.sprites || {};\n    slot.sprites[descriptor.name] = sprite;\n    return sprite;\n};\n\n/**\n *\n * @param slot {spine.Slot} The slot to which the attachment is parented\n * @param attachment {spine.RegionAttachment} The attachment that the sprite will represent\n * @private\n */\nSpine.prototype.createMesh = function (slot, attachment) {\n    var descriptor = attachment.rendererObject;\n    var baseTexture = descriptor.page.rendererObject;\n    var texture = new core.Texture(baseTexture);\n\n    var strip = new core.Strip(texture);\n    strip.drawMode = core.Strip.DrawModes.TRIANGLES;\n    strip.canvasPadding = 1.5;\n\n    strip.vertices = new Float32Array(attachment.uvs.length);\n    strip.uvs = attachment.uvs;\n    strip.indices = attachment.triangles;\n\n    slot.meshes = slot.meshes || {};\n    slot.meshes[attachment.name] = strip;\n\n    return strip;\n};\n","/******************************************************************************\n * Spine Runtimes Software License\n * Version 2.1\n *\n * Copyright (c) 2013, Esoteric Software\n * All rights reserved.\n *\n * You are granted a perpetual, non-exclusive, non-sublicensable and\n * non-transferable license to install, execute and perform the Spine Runtimes\n * Software (the \"Software\") solely for internal use. Without the written\n * permission of Esoteric Software (typically granted by licensing Spine), you\n * may not (a) modify, translate, adapt or otherwise create derivative works,\n * improvements of the Software or develop new applications using the Software\n * or (b) remove, delete, alter or obscure any trademarks or any copyright,\n * trademark, patent or other intellectual property or proprietary rights\n * notices on or in the Software, including any copy thereof. Redistributions\n * in binary or source form must include this license and terms.\n *\n * THIS SOFTWARE IS PROVIDED BY ESOTERIC SOFTWARE \"AS IS\" AND ANY EXPRESS OR\n * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF\n * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO\n * EVENT SHALL ESOTERIC SOFTARE BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,\n * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,\n * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;\n * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,\n * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR\n * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF\n * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.\n *****************************************************************************/\n\nvar spine = module.exports = {\n\tradDeg: 180 / Math.PI,\n\tdegRad: Math.PI / 180,\n\ttemp: [],\n    Float32Array: (typeof(Float32Array) === 'undefined') ? Array : Float32Array,\n    Uint16Array: (typeof(Uint16Array) === 'undefined') ? Array : Uint16Array\n};\n\nspine.BoneData = function (name, parent) {\n\tthis.name = name;\n\tthis.parent = parent;\n};\nspine.BoneData.prototype = {\n\tlength: 0,\n\tx: 0, y: 0,\n\trotation: 0,\n\tscaleX: 1, scaleY: 1,\n\tinheritScale: true,\n\tinheritRotation: true,\n\tflipX: false, flipY: false\n};\n\nspine.SlotData = function (name, boneData) {\n\tthis.name = name;\n\tthis.boneData = boneData;\n};\nspine.SlotData.prototype = {\n\tr: 1, g: 1, b: 1, a: 1,\n\tattachmentName: null,\n\tadditiveBlending: false\n};\n\nspine.IkConstraintData = function (name) {\n\tthis.name = name;\n\tthis.bones = [];\n};\nspine.IkConstraintData.prototype = {\n\ttarget: null,\n\tbendDirection: 1,\n\tmix: 1\n};\n\nspine.Bone = function (boneData, skeleton, parent) {\n\tthis.data = boneData;\n\tthis.skeleton = skeleton;\n\tthis.parent = parent;\n\tthis.setToSetupPose();\n};\nspine.Bone.yDown = false;\nspine.Bone.prototype = {\n\tx: 0, y: 0,\n\trotation: 0, rotationIK: 0,\n\tscaleX: 1, scaleY: 1,\n\tflipX: false, flipY: false,\n\tm00: 0, m01: 0, worldX: 0, // a b x\n\tm10: 0, m11: 0, worldY: 0, // c d y\n\tworldRotation: 0,\n\tworldScaleX: 1, worldScaleY: 1,\n\tworldFlipX: false, worldFlipY: false,\n\tupdateWorldTransform: function () {\n\t\tvar parent = this.parent;\n\t\tif (parent) {\n\t\t\tthis.worldX = this.x * parent.m00 + this.y * parent.m01 + parent.worldX;\n\t\t\tthis.worldY = this.x * parent.m10 + this.y * parent.m11 + parent.worldY;\n\t\t\tif (this.data.inheritScale) {\n\t\t\t\tthis.worldScaleX = parent.worldScaleX * this.scaleX;\n\t\t\t\tthis.worldScaleY = parent.worldScaleY * this.scaleY;\n\t\t\t} else {\n\t\t\t\tthis.worldScaleX = this.scaleX;\n\t\t\t\tthis.worldScaleY = this.scaleY;\n\t\t\t}\n\t\t\tthis.worldRotation = this.data.inheritRotation ? (parent.worldRotation + this.rotationIK) : this.rotationIK;\n\t\t\tthis.worldFlipX = parent.worldFlipX != this.flipX;\n\t\t\tthis.worldFlipY = parent.worldFlipY != this.flipY;\n\t\t} else {\n\t\t\tvar skeletonFlipX = this.skeleton.flipX, skeletonFlipY = this.skeleton.flipY;\n\t\t\tthis.worldX = skeletonFlipX ? -this.x : this.x;\n\t\t\tthis.worldY = (skeletonFlipY != spine.Bone.yDown) ? -this.y : this.y;\n\t\t\tthis.worldScaleX = this.scaleX;\n\t\t\tthis.worldScaleY = this.scaleY;\n\t\t\tthis.worldRotation = this.rotationIK;\n\t\t\tthis.worldFlipX = skeletonFlipX != this.flipX;\n\t\t\tthis.worldFlipY = skeletonFlipY != this.flipY;\n\t\t}\n\t\tvar radians = this.worldRotation * spine.degRad;\n\t\tvar cos = Math.cos(radians);\n\t\tvar sin = Math.sin(radians);\n\t\tif (this.worldFlipX) {\n\t\t\tthis.m00 = -cos * this.worldScaleX;\n\t\t\tthis.m01 = sin * this.worldScaleY;\n\t\t} else {\n\t\t\tthis.m00 = cos * this.worldScaleX;\n\t\t\tthis.m01 = -sin * this.worldScaleY;\n\t\t}\n\t\tif (this.worldFlipY != spine.Bone.yDown) {\n\t\t\tthis.m10 = -sin * this.worldScaleX;\n\t\t\tthis.m11 = -cos * this.worldScaleY;\n\t\t} else {\n\t\t\tthis.m10 = sin * this.worldScaleX;\n\t\t\tthis.m11 = cos * this.worldScaleY;\n\t\t}\n\t},\n\tsetToSetupPose: function () {\n\t\tvar data = this.data;\n\t\tthis.x = data.x;\n\t\tthis.y = data.y;\n\t\tthis.rotation = data.rotation;\n\t\tthis.rotationIK = this.rotation;\n\t\tthis.scaleX = data.scaleX;\n\t\tthis.scaleY = data.scaleY;\n\t\tthis.flipX = data.flipX;\n\t\tthis.flipY = data.flipY;\n\t},\n\tworldToLocal: function (world) {\n\t\tvar dx = world[0] - this.worldX, dy = world[1] - this.worldY;\n\t\tvar m00 = this.m00, m10 = this.m10, m01 = this.m01, m11 = this.m11;\n\t\tif (this.worldFlipX != (this.worldFlipY != spine.Bone.yDown)) {\n\t\t\tm00 = -m00;\n\t\t\tm11 = -m11;\n\t\t}\n\t\tvar invDet = 1 / (m00 * m11 - m01 * m10);\n\t\tworld[0] = dx * m00 * invDet - dy * m01 * invDet;\n\t\tworld[1] = dy * m11 * invDet - dx * m10 * invDet;\n\t},\n\tlocalToWorld: function (local) {\n\t\tvar localX = local[0], localY = local[1];\n\t\tlocal[0] = localX * this.m00 + localY * this.m01 + this.worldX;\n\t\tlocal[1] = localX * this.m10 + localY * this.m11 + this.worldY;\n\t}\n};\n\nspine.Slot = function (slotData, bone) {\n\tthis.data = slotData;\n\tthis.bone = bone;\n\tthis.setToSetupPose();\n};\nspine.Slot.prototype = {\n\tr: 1, g: 1, b: 1, a: 1,\n\t_attachmentTime: 0,\n\tattachment: null,\n\tattachmentVertices: [],\n\tsetAttachment: function (attachment) {\n\t\tthis.attachment = attachment;\n\t\tthis._attachmentTime = this.bone.skeleton.time;\n\t\tthis.attachmentVertices.length = 0;\n\t},\n\tsetAttachmentTime: function (time) {\n\t\tthis._attachmentTime = this.bone.skeleton.time - time;\n\t},\n\tgetAttachmentTime: function () {\n\t\treturn this.bone.skeleton.time - this._attachmentTime;\n\t},\n\tsetToSetupPose: function () {\n\t\tvar data = this.data;\n\t\tthis.r = data.r;\n\t\tthis.g = data.g;\n\t\tthis.b = data.b;\n\t\tthis.a = data.a;\n\n\t\tvar slotDatas = this.bone.skeleton.data.slots;\n\t\tfor (var i = 0, n = slotDatas.length; i < n; i++) {\n\t\t\tif (slotDatas[i] == data) {\n\t\t\t\tthis.setAttachment(!data.attachmentName ? null : this.bone.skeleton.getAttachmentBySlotIndex(i, data.attachmentName));\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.IkConstraint = function (data, skeleton) {\n\tthis.data = data;\n\tthis.mix = data.mix;\n\tthis.bendDirection = data.bendDirection;\n\n\tthis.bones = [];\n\tfor (var i = 0, n = data.bones.length; i < n; i++)\n\t\tthis.bones.push(skeleton.findBone(data.bones[i].name));\n\tthis.target = skeleton.findBone(data.target.name);\n};\nspine.IkConstraint.prototype = {\n\tapply: function () {\n\t\tvar target = this.target;\n\t\tvar bones = this.bones;\n\t\tswitch (bones.length) {\n\t\tcase 1:\n\t\t\tspine.IkConstraint.apply1(bones[0], target.worldX, target.worldY, this.mix);\n\t\t\tbreak;\n\t\tcase 2:\n\t\t\tspine.IkConstraint.apply2(bones[0], bones[1], target.worldX, target.worldY, this.bendDirection, this.mix);\n\t\t\tbreak;\n\t\t}\n\t}\n};\n/** Adjusts the bone rotation so the tip is as close to the target position as possible. The target is specified in the world\n * coordinate system. */\nspine.IkConstraint.apply1 = function (bone, targetX, targetY, alpha) {\n\tvar parentRotation = (!bone.data.inheritRotation || !bone.parent) ? 0 : bone.parent.worldRotation;\n\tvar rotation = bone.rotation;\n\tvar rotationIK = Math.atan2(targetY - bone.worldY, targetX - bone.worldX) * spine.radDeg - parentRotation;\n\tbone.rotationIK = rotation + (rotationIK - rotation) * alpha;\n};\n/** Adjusts the parent and child bone rotations so the tip of the child is as close to the target position as possible. The\n * target is specified in the world coordinate system.\n * @param child Any descendant bone of the parent. */\nspine.IkConstraint.apply2 = function (parent, child, targetX, targetY, bendDirection, alpha) {\n\tvar childRotation = child.rotation, parentRotation = parent.rotation;\n\tif (!alpha) {\n\t\tchild.rotationIK = childRotation;\n\t\tparent.rotationIK = parentRotation;\n\t\treturn;\n\t}\n\tvar positionX, positionY, tempPosition = spine.temp;\n\tvar parentParent = parent.parent;\n\tif (parentParent) {\n\t\ttempPosition[0] = targetX;\n\t\ttempPosition[1] = targetY;\n\t\tparentParent.worldToLocal(tempPosition);\n\t\ttargetX = (tempPosition[0] - parent.x) * parentParent.worldScaleX;\n\t\ttargetY = (tempPosition[1] - parent.y) * parentParent.worldScaleY;\n\t} else {\n\t\ttargetX -= parent.x;\n\t\ttargetY -= parent.y;\n\t}\n\tif (child.parent == parent) {\n\t\tpositionX = child.x;\n\t\tpositionY = child.y;\n\t} else {\n\t\ttempPosition[0] = child.x;\n\t\ttempPosition[1] = child.y;\n\t\tchild.parent.localToWorld(tempPosition);\n\t\tparent.worldToLocal(tempPosition);\n\t\tpositionX = tempPosition[0];\n\t\tpositionY = tempPosition[1];\n\t}\n\tvar childX = positionX * parent.worldScaleX, childY = positionY * parent.worldScaleY;\n\tvar offset = Math.atan2(childY, childX);\n\tvar len1 = Math.sqrt(childX * childX + childY * childY), len2 = child.data.length * child.worldScaleX;\n\t// Based on code by Ryan Juckett with permission: Copyright (c) 2008-2009 Ryan Juckett, http://www.ryanjuckett.com/\n\tvar cosDenom = 2 * len1 * len2;\n\tif (cosDenom < 0.0001) {\n\t\tchild.rotationIK = childRotation + (Math.atan2(targetY, targetX) * spine.radDeg - parentRotation - childRotation) * alpha;\n\t\treturn;\n\t}\n\tvar cos = (targetX * targetX + targetY * targetY - len1 * len1 - len2 * len2) / cosDenom;\n\tif (cos < -1)\n\t\tcos = -1;\n\telse if (cos > 1)\n\t\tcos = 1;\n\tvar childAngle = Math.acos(cos) * bendDirection;\n\tvar adjacent = len1 + len2 * cos, opposite = len2 * Math.sin(childAngle);\n\tvar parentAngle = Math.atan2(targetY * adjacent - targetX * opposite, targetX * adjacent + targetY * opposite);\n\tvar rotation = (parentAngle - offset) * spine.radDeg - parentRotation;\n\tif (rotation > 180)\n\t\trotation -= 360;\n\telse if (rotation < -180) //\n\t\trotation += 360;\n\tparent.rotationIK = parentRotation + rotation * alpha;\n\trotation = (childAngle + offset) * spine.radDeg - childRotation;\n\tif (rotation > 180)\n\t\trotation -= 360;\n\telse if (rotation < -180) //\n\t\trotation += 360;\n\tchild.rotationIK = childRotation + (rotation + parent.worldRotation - child.parent.worldRotation) * alpha;\n};\n\nspine.Skin = function (name) {\n\tthis.name = name;\n\tthis.attachments = {};\n};\nspine.Skin.prototype = {\n\taddAttachment: function (slotIndex, name, attachment) {\n\t\tthis.attachments[slotIndex + \":\" + name] = attachment;\n\t},\n\tgetAttachment: function (slotIndex, name) {\n\t\treturn this.attachments[slotIndex + \":\" + name];\n\t},\n\t_attachAll: function (skeleton, oldSkin) {\n\t\tfor (var key in oldSkin.attachments) {\n\t\t\tvar colon = key.indexOf(\":\");\n\t\t\tvar slotIndex = parseInt(key.substring(0, colon));\n\t\t\tvar name = key.substring(colon + 1);\n\t\t\tvar slot = skeleton.slots[slotIndex];\n\t\t\tif (slot.attachment && slot.attachment.name == name) {\n\t\t\t\tvar attachment = this.getAttachment(slotIndex, name);\n\t\t\t\tif (attachment) slot.setAttachment(attachment);\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.Animation = function (name, timelines, duration) {\n\tthis.name = name;\n\tthis.timelines = timelines;\n\tthis.duration = duration;\n};\nspine.Animation.prototype = {\n\tapply: function (skeleton, lastTime, time, loop, events) {\n\t\tif (loop && this.duration != 0) {\n\t\t\ttime %= this.duration;\n\t\t\tlastTime %= this.duration;\n\t\t}\n\t\tvar timelines = this.timelines;\n\t\tfor (var i = 0, n = timelines.length; i < n; i++)\n\t\t\ttimelines[i].apply(skeleton, lastTime, time, events, 1);\n\t},\n\tmix: function (skeleton, lastTime, time, loop, events, alpha) {\n\t\tif (loop && this.duration != 0) {\n\t\t\ttime %= this.duration;\n\t\t\tlastTime %= this.duration;\n\t\t}\n\t\tvar timelines = this.timelines;\n\t\tfor (var i = 0, n = timelines.length; i < n; i++)\n\t\t\ttimelines[i].apply(skeleton, lastTime, time, events, alpha);\n\t}\n};\nspine.Animation.binarySearch = function (values, target, step) {\n\tvar low = 0;\n\tvar high = Math.floor(values.length / step) - 2;\n\tif (!high) return step;\n\tvar current = high >>> 1;\n\twhile (true) {\n\t\tif (values[(current + 1) * step] <= target)\n\t\t\tlow = current + 1;\n\t\telse\n\t\t\thigh = current;\n\t\tif (low == high) return (low + 1) * step;\n\t\tcurrent = (low + high) >>> 1;\n\t}\n};\nspine.Animation.binarySearch1 = function (values, target) {\n\tvar low = 0;\n\tvar high = values.length - 2;\n\tif (!high) return 1;\n\tvar current = high >>> 1;\n\twhile (true) {\n\t\tif (values[current + 1] <= target)\n\t\t\tlow = current + 1;\n\t\telse\n\t\t\thigh = current;\n\t\tif (low == high) return low + 1;\n\t\tcurrent = (low + high) >>> 1;\n\t}\n};\nspine.Animation.linearSearch = function (values, target, step) {\n\tfor (var i = 0, last = values.length - step; i <= last; i += step)\n\t\tif (values[i] > target) return i;\n\treturn -1;\n};\n\nspine.Curves = function (frameCount) {\n\tthis.curves = []; // type, x, y, ...\n\t//this.curves.length = (frameCount - 1) * 19/*BEZIER_SIZE*/;\n};\nspine.Curves.prototype = {\n\tsetLinear: function (frameIndex) {\n\t\tthis.curves[frameIndex * 19/*BEZIER_SIZE*/] = 0/*LINEAR*/;\n\t},\n\tsetStepped: function (frameIndex) {\n\t\tthis.curves[frameIndex * 19/*BEZIER_SIZE*/] = 1/*STEPPED*/;\n\t},\n\t/** Sets the control handle positions for an interpolation bezier curve used to transition from this keyframe to the next.\n\t * cx1 and cx2 are from 0 to 1, representing the percent of time between the two keyframes. cy1 and cy2 are the percent of\n\t * the difference between the keyframe's values. */\n\tsetCurve: function (frameIndex, cx1, cy1, cx2, cy2) {\n\t\tvar subdiv1 = 1 / 10/*BEZIER_SEGMENTS*/, subdiv2 = subdiv1 * subdiv1, subdiv3 = subdiv2 * subdiv1;\n\t\tvar pre1 = 3 * subdiv1, pre2 = 3 * subdiv2, pre4 = 6 * subdiv2, pre5 = 6 * subdiv3;\n\t\tvar tmp1x = -cx1 * 2 + cx2, tmp1y = -cy1 * 2 + cy2, tmp2x = (cx1 - cx2) * 3 + 1, tmp2y = (cy1 - cy2) * 3 + 1;\n\t\tvar dfx = cx1 * pre1 + tmp1x * pre2 + tmp2x * subdiv3, dfy = cy1 * pre1 + tmp1y * pre2 + tmp2y * subdiv3;\n\t\tvar ddfx = tmp1x * pre4 + tmp2x * pre5, ddfy = tmp1y * pre4 + tmp2y * pre5;\n\t\tvar dddfx = tmp2x * pre5, dddfy = tmp2y * pre5;\n\n\t\tvar i = frameIndex * 19/*BEZIER_SIZE*/;\n\t\tvar curves = this.curves;\n\t\tcurves[i++] = 2/*BEZIER*/;\n\n\t\tvar x = dfx, y = dfy;\n\t\tfor (var n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) {\n\t\t\tcurves[i] = x;\n\t\t\tcurves[i + 1] = y;\n\t\t\tdfx += ddfx;\n\t\t\tdfy += ddfy;\n\t\t\tddfx += dddfx;\n\t\t\tddfy += dddfy;\n\t\t\tx += dfx;\n\t\t\ty += dfy;\n\t\t}\n\t},\n\tgetCurvePercent: function (frameIndex, percent) {\n\t\tpercent = percent < 0 ? 0 : (percent > 1 ? 1 : percent);\n\t\tvar curves = this.curves;\n\t\tvar i = frameIndex * 19/*BEZIER_SIZE*/;\n\t\tvar type = curves[i];\n\t\tif (type === 0/*LINEAR*/) return percent;\n\t\tif (type == 1/*STEPPED*/) return 0;\n\t\ti++;\n\t\tvar x = 0;\n\t\tfor (var start = i, n = i + 19/*BEZIER_SIZE*/ - 1; i < n; i += 2) {\n\t\t\tx = curves[i];\n\t\t\tif (x >= percent) {\n\t\t\t\tvar prevX, prevY;\n\t\t\t\tif (i == start) {\n\t\t\t\t\tprevX = 0;\n\t\t\t\t\tprevY = 0;\n\t\t\t\t} else {\n\t\t\t\t\tprevX = curves[i - 2];\n\t\t\t\t\tprevY = curves[i - 1];\n\t\t\t\t}\n\t\t\t\treturn prevY + (curves[i + 1] - prevY) * (percent - prevX) / (x - prevX);\n\t\t\t}\n\t\t}\n\t\tvar y = curves[i - 1];\n\t\treturn y + (1 - y) * (percent - x) / (1 - x); // Last point is 1,1.\n\t}\n};\n\nspine.RotateTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, angle, ...\n\tthis.frames.length = frameCount * 2;\n};\nspine.RotateTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 2;\n\t},\n\tsetFrame: function (frameIndex, time, angle) {\n\t\tframeIndex *= 2;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = angle;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar bone = skeleton.bones[this.boneIndex];\n\n\t\tif (time >= frames[frames.length - 2]) { // Time is after last frame.\n\t\t\tvar amount = bone.data.rotation + frames[frames.length - 1] - bone.rotation;\n\t\t\twhile (amount > 180)\n\t\t\t\tamount -= 360;\n\t\t\twhile (amount < -180)\n\t\t\t\tamount += 360;\n\t\t\tbone.rotation += amount * alpha;\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 2);\n\t\tvar prevFrameValue = frames[frameIndex - 1];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex - 2/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 2 - 1, percent);\n\n\t\tvar amount = frames[frameIndex + 1/*FRAME_VALUE*/] - prevFrameValue;\n\t\twhile (amount > 180)\n\t\t\tamount -= 360;\n\t\twhile (amount < -180)\n\t\t\tamount += 360;\n\t\tamount = bone.data.rotation + (prevFrameValue + amount * percent) - bone.rotation;\n\t\twhile (amount > 180)\n\t\t\tamount -= 360;\n\t\twhile (amount < -180)\n\t\t\tamount += 360;\n\t\tbone.rotation += amount * alpha;\n\t}\n};\n\nspine.TranslateTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, x, y, ...\n\tthis.frames.length = frameCount * 3;\n};\nspine.TranslateTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 3;\n\t},\n\tsetFrame: function (frameIndex, time, x, y) {\n\t\tframeIndex *= 3;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = x;\n\t\tthis.frames[frameIndex + 2] = y;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar bone = skeleton.bones[this.boneIndex];\n\n\t\tif (time >= frames[frames.length - 3]) { // Time is after last frame.\n\t\t\tbone.x += (bone.data.x + frames[frames.length - 2] - bone.x) * alpha;\n\t\t\tbone.y += (bone.data.y + frames[frames.length - 1] - bone.y) * alpha;\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 3);\n\t\tvar prevFrameX = frames[frameIndex - 2];\n\t\tvar prevFrameY = frames[frameIndex - 1];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent);\n\n\t\tbone.x += (bone.data.x + prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent - bone.x) * alpha;\n\t\tbone.y += (bone.data.y + prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent - bone.y) * alpha;\n\t}\n};\n\nspine.ScaleTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, x, y, ...\n\tthis.frames.length = frameCount * 3;\n};\nspine.ScaleTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 3;\n\t},\n\tsetFrame: function (frameIndex, time, x, y) {\n\t\tframeIndex *= 3;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = x;\n\t\tthis.frames[frameIndex + 2] = y;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar bone = skeleton.bones[this.boneIndex];\n\n\t\tif (time >= frames[frames.length - 3]) { // Time is after last frame.\n\t\t\tbone.scaleX += (bone.data.scaleX * frames[frames.length - 2] - bone.scaleX) * alpha;\n\t\t\tbone.scaleY += (bone.data.scaleY * frames[frames.length - 1] - bone.scaleY) * alpha;\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 3);\n\t\tvar prevFrameX = frames[frameIndex - 2];\n\t\tvar prevFrameY = frames[frameIndex - 1];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent);\n\n\t\tbone.scaleX += (bone.data.scaleX * (prevFrameX + (frames[frameIndex + 1/*FRAME_X*/] - prevFrameX) * percent) - bone.scaleX) * alpha;\n\t\tbone.scaleY += (bone.data.scaleY * (prevFrameY + (frames[frameIndex + 2/*FRAME_Y*/] - prevFrameY) * percent) - bone.scaleY) * alpha;\n\t}\n};\n\nspine.ColorTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, r, g, b, a, ...\n\tthis.frames.length = frameCount * 5;\n};\nspine.ColorTimeline.prototype = {\n\tslotIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 5;\n\t},\n\tsetFrame: function (frameIndex, time, r, g, b, a) {\n\t\tframeIndex *= 5;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = r;\n\t\tthis.frames[frameIndex + 2] = g;\n\t\tthis.frames[frameIndex + 3] = b;\n\t\tthis.frames[frameIndex + 4] = a;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar r, g, b, a;\n\t\tif (time >= frames[frames.length - 5]) {\n\t\t\t// Time is after last frame.\n\t\t\tvar i = frames.length - 1;\n\t\t\tr = frames[i - 3];\n\t\t\tg = frames[i - 2];\n\t\t\tb = frames[i - 1];\n\t\t\ta = frames[i];\n\t\t} else {\n\t\t\t// Interpolate between the previous frame and the current frame.\n\t\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 5);\n\t\t\tvar prevFrameR = frames[frameIndex - 4];\n\t\t\tvar prevFrameG = frames[frameIndex - 3];\n\t\t\tvar prevFrameB = frames[frameIndex - 2];\n\t\t\tvar prevFrameA = frames[frameIndex - 1];\n\t\t\tvar frameTime = frames[frameIndex];\n\t\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex - 5/*PREV_FRAME_TIME*/] - frameTime);\n\t\t\tpercent = this.curves.getCurvePercent(frameIndex / 5 - 1, percent);\n\n\t\t\tr = prevFrameR + (frames[frameIndex + 1/*FRAME_R*/] - prevFrameR) * percent;\n\t\t\tg = prevFrameG + (frames[frameIndex + 2/*FRAME_G*/] - prevFrameG) * percent;\n\t\t\tb = prevFrameB + (frames[frameIndex + 3/*FRAME_B*/] - prevFrameB) * percent;\n\t\t\ta = prevFrameA + (frames[frameIndex + 4/*FRAME_A*/] - prevFrameA) * percent;\n\t\t}\n\t\tvar slot = skeleton.slots[this.slotIndex];\n\t\tif (alpha < 1) {\n\t\t\tslot.r += (r - slot.r) * alpha;\n\t\t\tslot.g += (g - slot.g) * alpha;\n\t\t\tslot.b += (b - slot.b) * alpha;\n\t\t\tslot.a += (a - slot.a) * alpha;\n\t\t} else {\n\t\t\tslot.r = r;\n\t\t\tslot.g = g;\n\t\t\tslot.b = b;\n\t\t\tslot.a = a;\n\t\t}\n\t}\n};\n\nspine.AttachmentTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, ...\n\tthis.frames.length = frameCount;\n\tthis.attachmentNames = [];\n\tthis.attachmentNames.length = frameCount;\n};\nspine.AttachmentTimeline.prototype = {\n\tslotIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, attachmentName) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.attachmentNames[frameIndex] = attachmentName;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) {\n\t\t\tif (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0);\n\t\t\treturn;\n\t\t} else if (lastTime > time) //\n\t\t\tlastTime = -1;\n\n\t\tvar frameIndex = time >= frames[frames.length - 1] ? frames.length - 1 : spine.Animation.binarySearch1(frames, time) - 1;\n\t\tif (frames[frameIndex] < lastTime) return;\n\n\t\tvar attachmentName = this.attachmentNames[frameIndex];\n\t\tskeleton.slots[this.slotIndex].setAttachment(\n\t\t\t!attachmentName ? null : skeleton.getAttachmentBySlotIndex(this.slotIndex, attachmentName));\n\t}\n};\n\nspine.EventTimeline = function (frameCount) {\n\tthis.frames = []; // time, ...\n\tthis.frames.length = frameCount;\n\tthis.events = [];\n\tthis.events.length = frameCount;\n};\nspine.EventTimeline.prototype = {\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, event) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.events[frameIndex] = event;\n\t},\n\t/** Fires events for frames > lastTime and <= time. */\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tif (!firedEvents) return;\n\n\t\tvar frames = this.frames;\n\t\tvar frameCount = frames.length;\n\n\t\tif (lastTime > time) { // Fire events after last time for looped animations.\n\t\t\tthis.apply(skeleton, lastTime, Number.MAX_VALUE, firedEvents, alpha);\n\t\t\tlastTime = -1;\n\t\t} else if (lastTime >= frames[frameCount - 1]) // Last time is after last frame.\n\t\t\treturn;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar frameIndex;\n\t\tif (lastTime < frames[0])\n\t\t\tframeIndex = 0;\n\t\telse {\n\t\t\tframeIndex = spine.Animation.binarySearch1(frames, lastTime);\n\t\t\tvar frame = frames[frameIndex];\n\t\t\twhile (frameIndex > 0) { // Fire multiple events with the same frame.\n\t\t\t\tif (frames[frameIndex - 1] != frame) break;\n\t\t\t\tframeIndex--;\n\t\t\t}\n\t\t}\n\t\tvar events = this.events;\n\t\tfor (; frameIndex < frameCount && time >= frames[frameIndex]; frameIndex++)\n\t\t\tfiredEvents.push(events[frameIndex]);\n\t}\n};\n\nspine.DrawOrderTimeline = function (frameCount) {\n\tthis.frames = []; // time, ...\n\tthis.frames.length = frameCount;\n\tthis.drawOrders = [];\n\tthis.drawOrders.length = frameCount;\n};\nspine.DrawOrderTimeline.prototype = {\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, drawOrder) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.drawOrders[frameIndex] = drawOrder;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar frameIndex;\n\t\tif (time >= frames[frames.length - 1]) // Time is after last frame.\n\t\t\tframeIndex = frames.length - 1;\n\t\telse\n\t\t\tframeIndex = spine.Animation.binarySearch1(frames, time) - 1;\n\n\t\tvar drawOrder = skeleton.drawOrder;\n\t\tvar slots = skeleton.slots;\n\t\tvar drawOrderToSetupIndex = this.drawOrders[frameIndex];\n\t\tif (!drawOrderToSetupIndex) {\n\t\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\t\tdrawOrder[i] = slots[i];\n\t\t} else {\n\t\t\tfor (var i = 0, n = drawOrderToSetupIndex.length; i < n; i++)\n\t\t\t\tdrawOrder[i] = skeleton.slots[drawOrderToSetupIndex[i]];\n\t\t}\n\n\t}\n};\n\nspine.FfdTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = [];\n\tthis.frames.length = frameCount;\n\tthis.frameVertices = [];\n\tthis.frameVertices.length = frameCount;\n};\nspine.FfdTimeline.prototype = {\n\tslotIndex: 0,\n\tattachment: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length;\n\t},\n\tsetFrame: function (frameIndex, time, vertices) {\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frameVertices[frameIndex] = vertices;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar slot = skeleton.slots[this.slotIndex];\n\t\tif (slot.attachment != this.attachment) return;\n\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar frameVertices = this.frameVertices;\n\t\tvar vertexCount = frameVertices[0].length;\n\n\t\tvar vertices = slot.attachmentVertices;\n\t\tif (vertices.length != vertexCount) alpha = 1;\n\t\tvertices.length = vertexCount;\n\n\t\tif (time >= frames[frames.length - 1]) { // Time is after last frame.\n\t\t\tvar lastVertices = frameVertices[frames.length - 1];\n\t\t\tif (alpha < 1) {\n\t\t\t\tfor (var i = 0; i < vertexCount; i++)\n\t\t\t\t\tvertices[i] += (lastVertices[i] - vertices[i]) * alpha;\n\t\t\t} else {\n\t\t\t\tfor (var i = 0; i < vertexCount; i++)\n\t\t\t\t\tvertices[i] = lastVertices[i];\n\t\t\t}\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch1(frames, time);\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex - 1] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex - 1, percent < 0 ? 0 : (percent > 1 ? 1 : percent));\n\n\t\tvar prevVertices = frameVertices[frameIndex - 1];\n\t\tvar nextVertices = frameVertices[frameIndex];\n\n\t\tif (alpha < 1) {\n\t\t\tfor (var i = 0; i < vertexCount; i++) {\n\t\t\t\tvar prev = prevVertices[i];\n\t\t\t\tvertices[i] += (prev + (nextVertices[i] - prev) * percent - vertices[i]) * alpha;\n\t\t\t}\n\t\t} else {\n\t\t\tfor (var i = 0; i < vertexCount; i++) {\n\t\t\t\tvar prev = prevVertices[i];\n\t\t\t\tvertices[i] = prev + (nextVertices[i] - prev) * percent;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.IkConstraintTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, mix, bendDirection, ...\n\tthis.frames.length = frameCount * 3;\n};\nspine.IkConstraintTimeline.prototype = {\n\tikConstraintIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 3;\n\t},\n\tsetFrame: function (frameIndex, time, mix, bendDirection) {\n\t\tframeIndex *= 3;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = mix;\n\t\tthis.frames[frameIndex + 2] = bendDirection;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) return; // Time is before first frame.\n\n\t\tvar ikConstraint = skeleton.ikConstraints[this.ikConstraintIndex];\n\n\t\tif (time >= frames[frames.length - 3]) { // Time is after last frame.\n\t\t\tikConstraint.mix += (frames[frames.length - 2] - ikConstraint.mix) * alpha;\n\t\t\tikConstraint.bendDirection = frames[frames.length - 1];\n\t\t\treturn;\n\t\t}\n\n\t\t// Interpolate between the previous frame and the current frame.\n\t\tvar frameIndex = spine.Animation.binarySearch(frames, time, 3);\n\t\tvar prevFrameMix = frames[frameIndex + -2/*PREV_FRAME_MIX*/];\n\t\tvar frameTime = frames[frameIndex];\n\t\tvar percent = 1 - (time - frameTime) / (frames[frameIndex + -3/*PREV_FRAME_TIME*/] - frameTime);\n\t\tpercent = this.curves.getCurvePercent(frameIndex / 3 - 1, percent);\n\n\t\tvar mix = prevFrameMix + (frames[frameIndex + 1/*FRAME_MIX*/] - prevFrameMix) * percent;\n\t\tikConstraint.mix += (mix - ikConstraint.mix) * alpha;\n\t\tikConstraint.bendDirection = frames[frameIndex + -1/*PREV_FRAME_BEND_DIRECTION*/];\n\t}\n};\n\nspine.FlipXTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, flip, ...\n\tthis.frames.length = frameCount * 2;\n};\nspine.FlipXTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 2;\n\t},\n\tsetFrame: function (frameIndex, time, flip) {\n\t\tframeIndex *= 2;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = flip ? 1 : 0;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) {\n\t\t\tif (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0);\n\t\t\treturn;\n\t\t} else if (lastTime > time) //\n\t\t\tlastTime = -1;\n\t\tvar frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2;\n\t\tif (frames[frameIndex] < lastTime) return;\n\t\tskeleton.bones[boneIndex].flipX = frames[frameIndex + 1] != 0;\n\t}\n};\n\nspine.FlipYTimeline = function (frameCount) {\n\tthis.curves = new spine.Curves(frameCount);\n\tthis.frames = []; // time, flip, ...\n\tthis.frames.length = frameCount * 2;\n};\nspine.FlipYTimeline.prototype = {\n\tboneIndex: 0,\n\tgetFrameCount: function () {\n\t\treturn this.frames.length / 2;\n\t},\n\tsetFrame: function (frameIndex, time, flip) {\n\t\tframeIndex *= 2;\n\t\tthis.frames[frameIndex] = time;\n\t\tthis.frames[frameIndex + 1] = flip ? 1 : 0;\n\t},\n\tapply: function (skeleton, lastTime, time, firedEvents, alpha) {\n\t\tvar frames = this.frames;\n\t\tif (time < frames[0]) {\n\t\t\tif (lastTime > time) this.apply(skeleton, lastTime, Number.MAX_VALUE, null, 0);\n\t\t\treturn;\n\t\t} else if (lastTime > time) //\n\t\t\tlastTime = -1;\n\t\tvar frameIndex = (time >= frames[frames.length - 2] ? frames.length : spine.Animation.binarySearch(frames, time, 2)) - 2;\n\t\tif (frames[frameIndex] < lastTime) return;\n\t\tskeleton.bones[boneIndex].flipY = frames[frameIndex + 1] != 0;\n\t}\n};\n\nspine.SkeletonData = function () {\n\tthis.bones = [];\n\tthis.slots = [];\n\tthis.skins = [];\n\tthis.events = [];\n\tthis.animations = [];\n\tthis.ikConstraints = [];\n};\nspine.SkeletonData.prototype = {\n\tname: null,\n\tdefaultSkin: null,\n\twidth: 0, height: 0,\n\tversion: null, hash: null,\n\t/** @return May be null. */\n\tfindBone: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].name == boneName) return bones[i];\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindBoneIndex: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].name == boneName) return i;\n\t\treturn -1;\n\t},\n\t/** @return May be null. */\n\tfindSlot: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tif (slots[i].name == slotName) return slot[i];\n\t\t}\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindSlotIndex: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\tif (slots[i].name == slotName) return i;\n\t\treturn -1;\n\t},\n\t/** @return May be null. */\n\tfindSkin: function (skinName) {\n\t\tvar skins = this.skins;\n\t\tfor (var i = 0, n = skins.length; i < n; i++)\n\t\t\tif (skins[i].name == skinName) return skins[i];\n\t\treturn null;\n\t},\n\t/** @return May be null. */\n\tfindEvent: function (eventName) {\n\t\tvar events = this.events;\n\t\tfor (var i = 0, n = events.length; i < n; i++)\n\t\t\tif (events[i].name == eventName) return events[i];\n\t\treturn null;\n\t},\n\t/** @return May be null. */\n\tfindAnimation: function (animationName) {\n\t\tvar animations = this.animations;\n\t\tfor (var i = 0, n = animations.length; i < n; i++)\n\t\t\tif (animations[i].name == animationName) return animations[i];\n\t\treturn null;\n\t},\n\t/** @return May be null. */\n\tfindIkConstraint: function (ikConstraintName) {\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tfor (var i = 0, n = ikConstraints.length; i < n; i++)\n\t\t\tif (ikConstraints[i].name == ikConstraintName) return ikConstraints[i];\n\t\treturn null;\n\t}\n};\n\nspine.Skeleton = function (skeletonData) {\n\tthis.data = skeletonData;\n\n\tthis.bones = [];\n\tfor (var i = 0, n = skeletonData.bones.length; i < n; i++) {\n\t\tvar boneData = skeletonData.bones[i];\n\t\tvar parent = !boneData.parent ? null : this.bones[skeletonData.bones.indexOf(boneData.parent)];\n\t\tthis.bones.push(new spine.Bone(boneData, this, parent));\n\t}\n\n\tthis.slots = [];\n\tthis.drawOrder = [];\n\tfor (var i = 0, n = skeletonData.slots.length; i < n; i++) {\n\t\tvar slotData = skeletonData.slots[i];\n\t\tvar bone = this.bones[skeletonData.bones.indexOf(slotData.boneData)];\n\t\tvar slot = new spine.Slot(slotData, bone);\n\t\tthis.slots.push(slot);\n\t\tthis.drawOrder.push(slot);\n\t}\n\n\tthis.ikConstraints = [];\n\tfor (var i = 0, n = skeletonData.ikConstraints.length; i < n; i++)\n\t\tthis.ikConstraints.push(new spine.IkConstraint(skeletonData.ikConstraints[i], this));\n\n\tthis.boneCache = [];\n\tthis.updateCache();\n};\nspine.Skeleton.prototype = {\n\tx: 0, y: 0,\n\tskin: null,\n\tr: 1, g: 1, b: 1, a: 1,\n\ttime: 0,\n\tflipX: false, flipY: false,\n\t/** Caches information about bones and IK constraints. Must be called if bones or IK constraints are added or removed. */\n\tupdateCache: function () {\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tvar ikConstraintsCount = ikConstraints.length;\n\n\t\tvar arrayCount = ikConstraintsCount + 1;\n\t\tvar boneCache = this.boneCache;\n\t\tif (boneCache.length > arrayCount) boneCache.length = arrayCount;\n\t\tfor (var i = 0, n = boneCache.length; i < n; i++)\n\t\t\tboneCache[i].length = 0;\n\t\twhile (boneCache.length < arrayCount)\n\t\t\tboneCache[boneCache.length] = [];\n\n\t\tvar nonIkBones = boneCache[0];\n\t\tvar bones = this.bones;\n\n\t\touter:\n\t\tfor (var i = 0, n = bones.length; i < n; i++) {\n\t\t\tvar bone = bones[i];\n\t\t\tvar current = bone;\n\t\t\tdo {\n\t\t\t\tfor (var ii = 0; ii < ikConstraintsCount; ii++) {\n\t\t\t\t\tvar ikConstraint = ikConstraints[ii];\n\t\t\t\t\tvar parent = ikConstraint.bones[0];\n\t\t\t\t\tvar child= ikConstraint.bones[ikConstraint.bones.length - 1];\n\t\t\t\t\twhile (true) {\n\t\t\t\t\t\tif (current == child) {\n\t\t\t\t\t\t\tboneCache[ii].push(bone);\n\t\t\t\t\t\t\tboneCache[ii + 1].push(bone);\n\t\t\t\t\t\t\tcontinue outer;\n\t\t\t\t\t\t}\n\t\t\t\t\t\tif (child == parent) break;\n\t\t\t\t\t\tchild = child.parent;\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t\tcurrent = current.parent;\n\t\t\t} while (current);\n\t\t\tnonIkBones[nonIkBones.length] = bone;\n\t\t}\n\t},\n\t/** Updates the world transform for each bone. */\n\tupdateWorldTransform: function () {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++) {\n\t\t\tvar bone = bones[i];\n\t\t\tbone.rotationIK = bone.rotation;\n\t\t}\n\t\tvar i = 0, last = this.boneCache.length - 1;\n\t\twhile (true) {\n\t\t\tvar cacheBones = this.boneCache[i];\n\t\t\tfor (var ii = 0, nn = cacheBones.length; ii < nn; ii++)\n\t\t\t\tcacheBones[ii].updateWorldTransform();\n\t\t\tif (i == last) break;\n\t\t\tthis.ikConstraints[i].apply();\n\t\t\ti++;\n\t\t}\n\t},\n\t/** Sets the bones and slots to their setup pose values. */\n\tsetToSetupPose: function () {\n\t\tthis.setBonesToSetupPose();\n\t\tthis.setSlotsToSetupPose();\n\t},\n\tsetBonesToSetupPose: function () {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tbones[i].setToSetupPose();\n\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tfor (var i = 0, n = ikConstraints.length; i < n; i++) {\n\t\t\tvar ikConstraint = ikConstraints[i];\n\t\t\tikConstraint.bendDirection = ikConstraint.data.bendDirection;\n\t\t\tikConstraint.mix = ikConstraint.data.mix;\n\t\t}\n\t},\n\tsetSlotsToSetupPose: function () {\n\t\tvar slots = this.slots;\n\t\tvar drawOrder = this.drawOrder;\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tdrawOrder[i] = slots[i];\n\t\t\tslots[i].setToSetupPose(i);\n\t\t}\n\t},\n\t/** @return May return null. */\n\tgetRootBone: function () {\n\t\treturn this.bones.length ? this.bones[0] : null;\n\t},\n\t/** @return May be null. */\n\tfindBone: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].data.name == boneName) return bones[i];\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindBoneIndex: function (boneName) {\n\t\tvar bones = this.bones;\n\t\tfor (var i = 0, n = bones.length; i < n; i++)\n\t\t\tif (bones[i].data.name == boneName) return i;\n\t\treturn -1;\n\t},\n\t/** @return May be null. */\n\tfindSlot: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\tif (slots[i].data.name == slotName) return slots[i];\n\t\treturn null;\n\t},\n\t/** @return -1 if the bone was not found. */\n\tfindSlotIndex: function (slotName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++)\n\t\t\tif (slots[i].data.name == slotName) return i;\n\t\treturn -1;\n\t},\n\tsetSkinByName: function (skinName) {\n\t\tvar skin = this.data.findSkin(skinName);\n\t\tif (!skin) throw \"Skin not found: \" + skinName;\n\t\tthis.setSkin(skin);\n\t},\n\t/** Sets the skin used to look up attachments before looking in the {@link SkeletonData#getDefaultSkin() default skin}.\n\t * Attachments from the new skin are attached if the corresponding attachment from the old skin was attached. If there was\n\t * no old skin, each slot's setup mode attachment is attached from the new skin.\n\t * @param newSkin May be null. */\n\tsetSkin: function (newSkin) {\n\t\tif (newSkin) {\n\t\t\tif (this.skin)\n\t\t\t\tnewSkin._attachAll(this, this.skin);\n\t\t\telse {\n\t\t\t\tvar slots = this.slots;\n\t\t\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\t\t\tvar slot = slots[i];\n\t\t\t\t\tvar name = slot.data.attachmentName;\n\t\t\t\t\tif (name) {\n\t\t\t\t\t\tvar attachment = newSkin.getAttachment(i, name);\n\t\t\t\t\t\tif (attachment) slot.setAttachment(attachment);\n\t\t\t\t\t}\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\t\tthis.skin = newSkin;\n\t},\n\t/** @return May be null. */\n\tgetAttachmentBySlotName: function (slotName, attachmentName) {\n\t\treturn this.getAttachmentBySlotIndex(this.data.findSlotIndex(slotName), attachmentName);\n\t},\n\t/** @return May be null. */\n\tgetAttachmentBySlotIndex: function (slotIndex, attachmentName) {\n\t\tif (this.skin) {\n\t\t\tvar attachment = this.skin.getAttachment(slotIndex, attachmentName);\n\t\t\tif (attachment) return attachment;\n\t\t}\n\t\tif (this.data.defaultSkin) return this.data.defaultSkin.getAttachment(slotIndex, attachmentName);\n\t\treturn null;\n\t},\n\t/** @param attachmentName May be null. */\n\tsetAttachment: function (slotName, attachmentName) {\n\t\tvar slots = this.slots;\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tvar slot = slots[i];\n\t\t\tif (slot.data.name == slotName) {\n\t\t\t\tvar attachment = null;\n\t\t\t\tif (attachmentName) {\n\t\t\t\t\tattachment = this.getAttachmentBySlotIndex(i, attachmentName);\n\t\t\t\t\tif (!attachment) throw \"Attachment not found: \" + attachmentName + \", for slot: \" + slotName;\n\t\t\t\t}\n\t\t\t\tslot.setAttachment(attachment);\n\t\t\t\treturn;\n\t\t\t}\n\t\t}\n\t\tthrow \"Slot not found: \" + slotName;\n\t},\n\t/** @return May be null. */\n\tfindIkConstraint: function (ikConstraintName) {\n\t\tvar ikConstraints = this.ikConstraints;\n\t\tfor (var i = 0, n = ikConstraints.length; i < n; i++)\n\t\t\tif (ikConstraints[i].data.name == ikConstraintName) return ikConstraints[i];\n\t\treturn null;\n\t},\n\tupdate: function (delta) {\n\t\tthis.time += delta;\n\t}\n};\n\nspine.EventData = function (name) {\n\tthis.name = name;\n};\nspine.EventData.prototype = {\n\tintValue: 0,\n\tfloatValue: 0,\n\tstringValue: null\n};\n\nspine.Event = function (data) {\n\tthis.data = data;\n};\nspine.Event.prototype = {\n\tintValue: 0,\n\tfloatValue: 0,\n\tstringValue: null\n};\n\nspine.AttachmentType = {\n\tregion: 0,\n\tboundingbox: 1,\n\tmesh: 2,\n\tskinnedmesh: 3\n};\n\nspine.RegionAttachment = function (name) {\n\tthis.name = name;\n\tthis.offset = [];\n\tthis.offset.length = 8;\n\tthis.uvs = [];\n\tthis.uvs.length = 8;\n};\nspine.RegionAttachment.prototype = {\n\ttype: spine.AttachmentType.region,\n\tx: 0, y: 0,\n\trotation: 0,\n\tscaleX: 1, scaleY: 1,\n\twidth: 0, height: 0,\n\tr: 1, g: 1, b: 1, a: 1,\n\tpath: null,\n\trendererObject: null,\n\tregionOffsetX: 0, regionOffsetY: 0,\n\tregionWidth: 0, regionHeight: 0,\n\tregionOriginalWidth: 0, regionOriginalHeight: 0,\n\tsetUVs: function (u, v, u2, v2, rotate) {\n\t\tvar uvs = this.uvs;\n\t\tif (rotate) {\n\t\t\tuvs[2/*X2*/] = u;\n\t\t\tuvs[3/*Y2*/] = v2;\n\t\t\tuvs[4/*X3*/] = u;\n\t\t\tuvs[5/*Y3*/] = v;\n\t\t\tuvs[6/*X4*/] = u2;\n\t\t\tuvs[7/*Y4*/] = v;\n\t\t\tuvs[0/*X1*/] = u2;\n\t\t\tuvs[1/*Y1*/] = v2;\n\t\t} else {\n\t\t\tuvs[0/*X1*/] = u;\n\t\t\tuvs[1/*Y1*/] = v2;\n\t\t\tuvs[2/*X2*/] = u;\n\t\t\tuvs[3/*Y2*/] = v;\n\t\t\tuvs[4/*X3*/] = u2;\n\t\t\tuvs[5/*Y3*/] = v;\n\t\t\tuvs[6/*X4*/] = u2;\n\t\t\tuvs[7/*Y4*/] = v2;\n\t\t}\n\t},\n\tupdateOffset: function () {\n\t\tvar regionScaleX = this.width / this.regionOriginalWidth * this.scaleX;\n\t\tvar regionScaleY = this.height / this.regionOriginalHeight * this.scaleY;\n\t\tvar localX = -this.width / 2 * this.scaleX + this.regionOffsetX * regionScaleX;\n\t\tvar localY = -this.height / 2 * this.scaleY + this.regionOffsetY * regionScaleY;\n\t\tvar localX2 = localX + this.regionWidth * regionScaleX;\n\t\tvar localY2 = localY + this.regionHeight * regionScaleY;\n\t\tvar radians = this.rotation * spine.degRad;\n\t\tvar cos = Math.cos(radians);\n\t\tvar sin = Math.sin(radians);\n\t\tvar localXCos = localX * cos + this.x;\n\t\tvar localXSin = localX * sin;\n\t\tvar localYCos = localY * cos + this.y;\n\t\tvar localYSin = localY * sin;\n\t\tvar localX2Cos = localX2 * cos + this.x;\n\t\tvar localX2Sin = localX2 * sin;\n\t\tvar localY2Cos = localY2 * cos + this.y;\n\t\tvar localY2Sin = localY2 * sin;\n\t\tvar offset = this.offset;\n\t\toffset[0/*X1*/] = localXCos - localYSin;\n\t\toffset[1/*Y1*/] = localYCos + localXSin;\n\t\toffset[2/*X2*/] = localXCos - localY2Sin;\n\t\toffset[3/*Y2*/] = localY2Cos + localXSin;\n\t\toffset[4/*X3*/] = localX2Cos - localY2Sin;\n\t\toffset[5/*Y3*/] = localY2Cos + localX2Sin;\n\t\toffset[6/*X4*/] = localX2Cos - localYSin;\n\t\toffset[7/*Y4*/] = localYCos + localX2Sin;\n\t},\n\tcomputeVertices: function (x, y, bone, vertices) {\n\t\tx += bone.worldX;\n\t\ty += bone.worldY;\n\t\tvar m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11;\n\t\tvar offset = this.offset;\n\t\tvertices[0/*X1*/] = offset[0/*X1*/] * m00 + offset[1/*Y1*/] * m01 + x;\n\t\tvertices[1/*Y1*/] = offset[0/*X1*/] * m10 + offset[1/*Y1*/] * m11 + y;\n\t\tvertices[2/*X2*/] = offset[2/*X2*/] * m00 + offset[3/*Y2*/] * m01 + x;\n\t\tvertices[3/*Y2*/] = offset[2/*X2*/] * m10 + offset[3/*Y2*/] * m11 + y;\n\t\tvertices[4/*X3*/] = offset[4/*X3*/] * m00 + offset[5/*X3*/] * m01 + x;\n\t\tvertices[5/*X3*/] = offset[4/*X3*/] * m10 + offset[5/*X3*/] * m11 + y;\n\t\tvertices[6/*X4*/] = offset[6/*X4*/] * m00 + offset[7/*Y4*/] * m01 + x;\n\t\tvertices[7/*Y4*/] = offset[6/*X4*/] * m10 + offset[7/*Y4*/] * m11 + y;\n\t}\n};\n\nspine.MeshAttachment = function (name) {\n\tthis.name = name;\n};\nspine.MeshAttachment.prototype = {\n\ttype: spine.AttachmentType.mesh,\n\tvertices: null,\n\tuvs: null,\n\tregionUVs: null,\n\ttriangles: null,\n\thullLength: 0,\n\tr: 1, g: 1, b: 1, a: 1,\n\tpath: null,\n\trendererObject: null,\n\tregionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false,\n\tregionOffsetX: 0, regionOffsetY: 0,\n\tregionWidth: 0, regionHeight: 0,\n\tregionOriginalWidth: 0, regionOriginalHeight: 0,\n\tedges: null,\n\twidth: 0, height: 0,\n\tupdateUVs: function () {\n\t\tvar width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV;\n\t\tvar n = this.regionUVs.length;\n\t\tif (!this.uvs || this.uvs.length != n) {\n            this.uvs = new spine.Float32Array(n);\n\t\t}\n\t\tif (this.regionRotate) {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width;\n                this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height;\n\t\t\t}\n\t\t} else {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i] * width;\n                this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height;\n\t\t\t}\n\t\t}\n\t},\n\tcomputeWorldVertices: function (x, y, slot, worldVertices) {\n\t\tvar bone = slot.bone;\n\t\tx += bone.worldX;\n\t\ty += bone.worldY;\n\t\tvar m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11;\n\t\tvar vertices = this.vertices;\n\t\tvar verticesCount = vertices.length;\n\t\tif (slot.attachmentVertices.length == verticesCount) vertices = slot.attachmentVertices;\n\t\tfor (var i = 0; i < verticesCount; i += 2) {\n\t\t\tvar vx = vertices[i];\n\t\t\tvar vy = vertices[i + 1];\n\t\t\tworldVertices[i] = vx * m00 + vy * m01 + x;\n\t\t\tworldVertices[i + 1] = vx * m10 + vy * m11 + y;\n\t\t}\n\t}\n};\n\nspine.SkinnedMeshAttachment = function (name) {\n\tthis.name = name;\n};\nspine.SkinnedMeshAttachment.prototype = {\n\ttype: spine.AttachmentType.skinnedmesh,\n\tbones: null,\n\tweights: null,\n\tuvs: null,\n\tregionUVs: null,\n\ttriangles: null,\n\thullLength: 0,\n\tr: 1, g: 1, b: 1, a: 1,\n\tpath: null,\n\trendererObject: null,\n\tregionU: 0, regionV: 0, regionU2: 0, regionV2: 0, regionRotate: false,\n\tregionOffsetX: 0, regionOffsetY: 0,\n\tregionWidth: 0, regionHeight: 0,\n\tregionOriginalWidth: 0, regionOriginalHeight: 0,\n\tedges: null,\n\twidth: 0, height: 0,\n\tupdateUVs: function (u, v, u2, v2, rotate) {\n\t\tvar width = this.regionU2 - this.regionU, height = this.regionV2 - this.regionV;\n\t\tvar n = this.regionUVs.length;\n\t\tif (!this.uvs || this.uvs.length != n) {\n            this.uvs = new spine.Float32Array(n);\n\t\t}\n\t\tif (this.regionRotate) {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i + 1] * width;\n                this.uvs[i + 1] = this.regionV + height - this.regionUVs[i] * height;\n\t\t\t}\n\t\t} else {\n\t\t\tfor (var i = 0; i < n; i += 2) {\n                this.uvs[i] = this.regionU + this.regionUVs[i] * width;\n                this.uvs[i + 1] = this.regionV + this.regionUVs[i + 1] * height;\n\t\t\t}\n\t\t}\n\t},\n\tcomputeWorldVertices: function (x, y, slot, worldVertices) {\n\t\tvar skeletonBones = slot.bone.skeleton.bones;\n\t\tvar weights = this.weights;\n\t\tvar bones = this.bones;\n\n\t\tvar w = 0, v = 0, b = 0, f = 0, n = bones.length, nn;\n\t\tvar wx, wy, bone, vx, vy, weight;\n\t\tif (!slot.attachmentVertices.length) {\n\t\t\tfor (; v < n; w += 2) {\n\t\t\t\twx = 0;\n\t\t\t\twy = 0;\n\t\t\t\tnn = bones[v++] + v;\n\t\t\t\tfor (; v < nn; v++, b += 3) {\n\t\t\t\t\tbone = skeletonBones[bones[v]];\n\t\t\t\t\tvx = weights[b];\n\t\t\t\t\tvy = weights[b + 1];\n\t\t\t\t\tweight = weights[b + 2];\n\t\t\t\t\twx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight;\n\t\t\t\t\twy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight;\n\t\t\t\t}\n\t\t\t\tworldVertices[w] = wx + x;\n\t\t\t\tworldVertices[w + 1] = wy + y;\n\t\t\t}\n\t\t} else {\n\t\t\tvar ffd = slot.attachmentVertices;\n\t\t\tfor (; v < n; w += 2) {\n\t\t\t\twx = 0;\n\t\t\t\twy = 0;\n\t\t\t\tnn = bones[v++] + v;\n\t\t\t\tfor (; v < nn; v++, b += 3, f += 2) {\n\t\t\t\t\tbone = skeletonBones[bones[v]];\n\t\t\t\t\tvx = weights[b] + ffd[f];\n\t\t\t\t\tvy = weights[b + 1] + ffd[f + 1];\n\t\t\t\t\tweight = weights[b + 2];\n\t\t\t\t\twx += (vx * bone.m00 + vy * bone.m01 + bone.worldX) * weight;\n\t\t\t\t\twy += (vx * bone.m10 + vy * bone.m11 + bone.worldY) * weight;\n\t\t\t\t}\n\t\t\t\tworldVertices[w] = wx + x;\n\t\t\t\tworldVertices[w + 1] = wy + y;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.BoundingBoxAttachment = function (name) {\n\tthis.name = name;\n\tthis.vertices = [];\n};\nspine.BoundingBoxAttachment.prototype = {\n\ttype: spine.AttachmentType.boundingbox,\n\tcomputeWorldVertices: function (x, y, bone, worldVertices) {\n\t\tx += bone.worldX;\n\t\ty += bone.worldY;\n\t\tvar m00 = bone.m00, m01 = bone.m01, m10 = bone.m10, m11 = bone.m11;\n\t\tvar vertices = this.vertices;\n\t\tfor (var i = 0, n = vertices.length; i < n; i += 2) {\n\t\t\tvar px = vertices[i];\n\t\t\tvar py = vertices[i + 1];\n\t\t\tworldVertices[i] = px * m00 + py * m01 + x;\n\t\t\tworldVertices[i + 1] = px * m10 + py * m11 + y;\n\t\t}\n\t}\n};\n\nspine.AnimationStateData = function (skeletonData) {\n\tthis.skeletonData = skeletonData;\n\tthis.animationToMixTime = {};\n};\nspine.AnimationStateData.prototype = {\n\tdefaultMix: 0,\n\tsetMixByName: function (fromName, toName, duration) {\n\t\tvar from = this.skeletonData.findAnimation(fromName);\n\t\tif (!from) throw \"Animation not found: \" + fromName;\n\t\tvar to = this.skeletonData.findAnimation(toName);\n\t\tif (!to) throw \"Animation not found: \" + toName;\n\t\tthis.setMix(from, to, duration);\n\t},\n\tsetMix: function (from, to, duration) {\n\t\tthis.animationToMixTime[from.name + \":\" + to.name] = duration;\n\t},\n\tgetMix: function (from, to) {\n\t\tvar key = from.name + \":\" + to.name;\n\t\treturn this.animationToMixTime.hasOwnProperty(key) ? this.animationToMixTime[key] : this.defaultMix;\n\t}\n};\n\nspine.TrackEntry = function () {};\nspine.TrackEntry.prototype = {\n\tnext: null, previous: null,\n\tanimation: null,\n\tloop: false,\n\tdelay: 0, time: 0, lastTime: -1, endTime: 0,\n\ttimeScale: 1,\n\tmixTime: 0, mixDuration: 0, mix: 1,\n\tonStart: null, onEnd: null, onComplete: null, onEvent: null\n};\n\nspine.AnimationState = function (stateData) {\n\tthis.data = stateData;\n\tthis.tracks = [];\n\tthis.events = [];\n};\nspine.AnimationState.prototype = {\n\tonStart: null,\n\tonEnd: null,\n\tonComplete: null,\n\tonEvent: null,\n\ttimeScale: 1,\n\tupdate: function (delta) {\n\t\tdelta *= this.timeScale;\n\t\tfor (var i = 0; i < this.tracks.length; i++) {\n\t\t\tvar current = this.tracks[i];\n\t\t\tif (!current) continue;\n\n\t\t\tcurrent.time += delta * current.timeScale;\n\t\t\tif (current.previous) {\n\t\t\t\tvar previousDelta = delta * current.previous.timeScale;\n\t\t\t\tcurrent.previous.time += previousDelta;\n\t\t\t\tcurrent.mixTime += previousDelta;\n\t\t\t}\n\n\t\t\tvar next = current.next;\n\t\t\tif (next) {\n\t\t\t\tnext.time = current.lastTime - next.delay;\n\t\t\t\tif (next.time >= 0) this.setCurrent(i, next);\n\t\t\t} else {\n\t\t\t\t// End non-looping animation when it reaches its end time and there is no next entry.\n\t\t\t\tif (!current.loop && current.lastTime >= current.endTime) this.clearTrack(i);\n\t\t\t}\n\t\t}\n\t},\n\tapply: function (skeleton) {\n\t\tfor (var i = 0; i < this.tracks.length; i++) {\n\t\t\tvar current = this.tracks[i];\n\t\t\tif (!current) continue;\n\n\t\t\tthis.events.length = 0;\n\n\t\t\tvar time = current.time;\n\t\t\tvar lastTime = current.lastTime;\n\t\t\tvar endTime = current.endTime;\n\t\t\tvar loop = current.loop;\n\t\t\tif (!loop && time > endTime) time = endTime;\n\n\t\t\tvar previous = current.previous;\n\t\t\tif (!previous) {\n\t\t\t\tif (current.mix == 1)\n\t\t\t\t\tcurrent.animation.apply(skeleton, current.lastTime, time, loop, this.events);\n\t\t\t\telse\n\t\t\t\t\tcurrent.animation.mix(skeleton, current.lastTime, time, loop, this.events, current.mix);\n\t\t\t} else {\n\t\t\t\tvar previousTime = previous.time;\n\t\t\t\tif (!previous.loop && previousTime > previous.endTime) previousTime = previous.endTime;\n\t\t\t\tprevious.animation.apply(skeleton, previousTime, previousTime, previous.loop, null);\n\n\t\t\t\tvar alpha = current.mixTime / current.mixDuration * current.mix;\n\t\t\t\tif (alpha >= 1) {\n\t\t\t\t\talpha = 1;\n\t\t\t\t\tcurrent.previous = null;\n\t\t\t\t}\n\t\t\t\tcurrent.animation.mix(skeleton, current.lastTime, time, loop, this.events, alpha);\n\t\t\t}\n\n\t\t\tfor (var ii = 0, nn = this.events.length; ii < nn; ii++) {\n\t\t\t\tvar event = this.events[ii];\n\t\t\t\tif (current.onEvent) current.onEvent(i, event);\n\t\t\t\tif (this.onEvent) this.onEvent(i, event);\n\t\t\t}\n\n\t\t\t// Check if completed the animation or a loop iteration.\n\t\t\tif (loop ? (lastTime % endTime > time % endTime) : (lastTime < endTime && time >= endTime)) {\n\t\t\t\tvar count = Math.floor(time / endTime);\n\t\t\t\tif (current.onComplete) current.onComplete(i, count);\n\t\t\t\tif (this.onComplete) this.onComplete(i, count);\n\t\t\t}\n\n\t\t\tcurrent.lastTime = current.time;\n\t\t}\n\t},\n\tclearTracks: function () {\n\t\tfor (var i = 0, n = this.tracks.length; i < n; i++)\n\t\t\tthis.clearTrack(i);\n\t\tthis.tracks.length = 0;\n\t},\n\tclearTrack: function (trackIndex) {\n\t\tif (trackIndex >= this.tracks.length) return;\n\t\tvar current = this.tracks[trackIndex];\n\t\tif (!current) return;\n\n\t\tif (current.onEnd) current.onEnd(trackIndex);\n\t\tif (this.onEnd) this.onEnd(trackIndex);\n\n\t\tthis.tracks[trackIndex] = null;\n\t},\n\t_expandToIndex: function (index) {\n\t\tif (index < this.tracks.length) return this.tracks[index];\n\t\twhile (index >= this.tracks.length)\n\t\t\tthis.tracks.push(null);\n\t\treturn null;\n\t},\n\tsetCurrent: function (index, entry) {\n\t\tvar current = this._expandToIndex(index);\n\t\tif (current) {\n\t\t\tvar previous = current.previous;\n\t\t\tcurrent.previous = null;\n\n\t\t\tif (current.onEnd) current.onEnd(index);\n\t\t\tif (this.onEnd) this.onEnd(index);\n\n\t\t\tentry.mixDuration = this.data.getMix(current.animation, entry.animation);\n\t\t\tif (entry.mixDuration > 0) {\n\t\t\t\tentry.mixTime = 0;\n\t\t\t\t// If a mix is in progress, mix from the closest animation.\n\t\t\t\tif (previous && current.mixTime / current.mixDuration < 0.5)\n\t\t\t\t\tentry.previous = previous;\n\t\t\t\telse\n\t\t\t\t\tentry.previous = current;\n\t\t\t}\n\t\t}\n\n\t\tthis.tracks[index] = entry;\n\n\t\tif (entry.onStart) entry.onStart(index);\n\t\tif (this.onStart) this.onStart(index);\n\t},\n\tsetAnimationByName: function (trackIndex, animationName, loop) {\n\t\tvar animation = this.data.skeletonData.findAnimation(animationName);\n\t\tif (!animation) throw \"Animation not found: \" + animationName;\n\t\treturn this.setAnimation(trackIndex, animation, loop);\n\t},\n\t/** Set the current animation. Any queued animations are cleared. */\n\tsetAnimation: function (trackIndex, animation, loop) {\n\t\tvar entry = new spine.TrackEntry();\n\t\tentry.animation = animation;\n\t\tentry.loop = loop;\n\t\tentry.endTime = animation.duration;\n\t\tthis.setCurrent(trackIndex, entry);\n\t\treturn entry;\n\t},\n\taddAnimationByName: function (trackIndex, animationName, loop, delay) {\n\t\tvar animation = this.data.skeletonData.findAnimation(animationName);\n\t\tif (!animation) throw \"Animation not found: \" + animationName;\n\t\treturn this.addAnimation(trackIndex, animation, loop, delay);\n\t},\n\t/** Adds an animation to be played delay seconds after the current or last queued animation.\n\t * @param delay May be <= 0 to use duration of previous animation minus any mix duration plus the negative delay. */\n\taddAnimation: function (trackIndex, animation, loop, delay) {\n\t\tvar entry = new spine.TrackEntry();\n\t\tentry.animation = animation;\n\t\tentry.loop = loop;\n\t\tentry.endTime = animation.duration;\n\n\t\tvar last = this._expandToIndex(trackIndex);\n\t\tif (last) {\n\t\t\twhile (last.next)\n\t\t\t\tlast = last.next;\n\t\t\tlast.next = entry;\n\t\t} else\n\t\t\tthis.tracks[trackIndex] = entry;\n\n\t\tif (delay <= 0) {\n\t\t\tif (last)\n\t\t\t\tdelay += last.endTime - this.data.getMix(last.animation, animation);\n\t\t\telse\n\t\t\t\tdelay = 0;\n\t\t}\n\t\tentry.delay = delay;\n\n\t\treturn entry;\n\t},\n\t/** May be null. */\n\tgetCurrent: function (trackIndex) {\n\t\tif (trackIndex >= this.tracks.length) return null;\n\t\treturn this.tracks[trackIndex];\n\t}\n};\n\nspine.SkeletonJson = function (attachmentLoader) {\n\tthis.attachmentLoader = attachmentLoader;\n};\nspine.SkeletonJson.prototype = {\n\tscale: 1,\n\treadSkeletonData: function (root, name) {\n\t\tvar skeletonData = new spine.SkeletonData();\n\t\tskeletonData.name = name;\n\n\t\t// Skeleton.\n\t\tvar skeletonMap = root[\"skeleton\"];\n\t\tif (skeletonMap) {\n\t\t\tskeletonData.hash = skeletonMap[\"hash\"];\n\t\t\tskeletonData.version = skeletonMap[\"spine\"];\n\t\t\tskeletonData.width = skeletonMap[\"width\"] || 0;\n\t\t\tskeletonData.height = skeletonMap[\"height\"] || 0;\n\t\t}\n\n\t\t// Bones.\n\t\tvar bones = root[\"bones\"];\n\t\tfor (var i = 0, n = bones.length; i < n; i++) {\n\t\t\tvar boneMap = bones[i];\n\t\t\tvar parent = null;\n\t\t\tif (boneMap[\"parent\"]) {\n\t\t\t\tparent = skeletonData.findBone(boneMap[\"parent\"]);\n\t\t\t\tif (!parent) throw \"Parent bone not found: \" + boneMap[\"parent\"];\n\t\t\t}\n\t\t\tvar boneData = new spine.BoneData(boneMap[\"name\"], parent);\n\t\t\tboneData.length = (boneMap[\"length\"] || 0) * this.scale;\n\t\t\tboneData.x = (boneMap[\"x\"] || 0) * this.scale;\n\t\t\tboneData.y = (boneMap[\"y\"] || 0) * this.scale;\n\t\t\tboneData.rotation = (boneMap[\"rotation\"] || 0);\n\t\t\tboneData.scaleX = boneMap.hasOwnProperty(\"scaleX\") ? boneMap[\"scaleX\"] : 1;\n\t\t\tboneData.scaleY = boneMap.hasOwnProperty(\"scaleY\") ? boneMap[\"scaleY\"] : 1;\n\t\t\tboneData.inheritScale = boneMap.hasOwnProperty(\"inheritScale\") ? boneMap[\"inheritScale\"] : true;\n\t\t\tboneData.inheritRotation = boneMap.hasOwnProperty(\"inheritRotation\") ? boneMap[\"inheritRotation\"] : true;\n\t\t\tskeletonData.bones.push(boneData);\n\t\t}\n\n\t\t// IK constraints.\n\t\tvar ik = root[\"ik\"];\n\t\tif (ik) {\n\t\t\tfor (var i = 0, n = ik.length; i < n; i++) {\n\t\t\t\tvar ikMap = ik[i];\n\t\t\t\tvar ikConstraintData = new spine.IkConstraintData(ikMap[\"name\"]);\n\n\t\t\t\tvar bones = ikMap[\"bones\"];\n\t\t\t\tfor (var ii = 0, nn = bones.length; ii < nn; ii++) {\n\t\t\t\t\tvar bone = skeletonData.findBone(bones[ii]);\n\t\t\t\t\tif (!bone) throw \"IK bone not found: \" + bones[ii];\n\t\t\t\t\tikConstraintData.bones.push(bone);\n\t\t\t\t}\n\n\t\t\t\tikConstraintData.target = skeletonData.findBone(ikMap[\"target\"]);\n\t\t\t\tif (!ikConstraintData.target) throw \"Target bone not found: \" + ikMap[\"target\"];\n\n\t\t\t\tikConstraintData.bendDirection = (!ikMap.hasOwnProperty(\"bendPositive\") || ikMap[\"bendPositive\"]) ? 1 : -1;\n\t\t\t\tikConstraintData.mix = ikMap.hasOwnProperty(\"mix\") ? ikMap[\"mix\"] : 1;\n\n\t\t\t\tskeletonData.ikConstraints.push(ikConstraintData);\n\t\t\t}\n\t\t}\n\n\t\t// Slots.\n\t\tvar slots = root[\"slots\"];\n\t\tfor (var i = 0, n = slots.length; i < n; i++) {\n\t\t\tvar slotMap = slots[i];\n\t\t\tvar boneData = skeletonData.findBone(slotMap[\"bone\"]);\n\t\t\tif (!boneData) throw \"Slot bone not found: \" + slotMap[\"bone\"];\n\t\t\tvar slotData = new spine.SlotData(slotMap[\"name\"], boneData);\n\n\t\t\tvar color = slotMap[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tslotData.r = this.toColor(color, 0);\n\t\t\t\tslotData.g = this.toColor(color, 1);\n\t\t\t\tslotData.b = this.toColor(color, 2);\n\t\t\t\tslotData.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tslotData.attachmentName = slotMap[\"attachment\"];\n\t\t\tslotData.additiveBlending = slotMap[\"additive\"] && slotMap[\"additive\"] == \"true\";\n\n\t\t\tskeletonData.slots.push(slotData);\n\t\t}\n\n\t\t// Skins.\n\t\tvar skins = root[\"skins\"];\n\t\tfor (var skinName in skins) {\n\t\t\tif (!skins.hasOwnProperty(skinName)) continue;\n\t\t\tvar skinMap = skins[skinName];\n\t\t\tvar skin = new spine.Skin(skinName);\n\t\t\tfor (var slotName in skinMap) {\n\t\t\t\tif (!skinMap.hasOwnProperty(slotName)) continue;\n\t\t\t\tvar slotIndex = skeletonData.findSlotIndex(slotName);\n\t\t\t\tvar slotEntry = skinMap[slotName];\n\t\t\t\tfor (var attachmentName in slotEntry) {\n\t\t\t\t\tif (!slotEntry.hasOwnProperty(attachmentName)) continue;\n\t\t\t\t\tvar attachment = this.readAttachment(skin, attachmentName, slotEntry[attachmentName]);\n\t\t\t\t\tif (attachment) skin.addAttachment(slotIndex, attachmentName, attachment);\n\t\t\t\t}\n\t\t\t}\n\t\t\tskeletonData.skins.push(skin);\n\t\t\tif (skin.name == \"default\") skeletonData.defaultSkin = skin;\n\t\t}\n\n\t\t// Events.\n\t\tvar events = root[\"events\"];\n\t\tfor (var eventName in events) {\n\t\t\tif (!events.hasOwnProperty(eventName)) continue;\n\t\t\tvar eventMap = events[eventName];\n\t\t\tvar eventData = new spine.EventData(eventName);\n\t\t\teventData.intValue = eventMap[\"int\"] || 0;\n\t\t\teventData.floatValue = eventMap[\"float\"] || 0;\n\t\t\teventData.stringValue = eventMap[\"string\"] || null;\n\t\t\tskeletonData.events.push(eventData);\n\t\t}\n\n\t\t// Animations.\n\t\tvar animations = root[\"animations\"];\n\t\tfor (var animationName in animations) {\n\t\t\tif (!animations.hasOwnProperty(animationName)) continue;\n\t\t\tthis.readAnimation(animationName, animations[animationName], skeletonData);\n\t\t}\n\n\t\treturn skeletonData;\n\t},\n\treadAttachment: function (skin, name, map) {\n\t\tname = map[\"name\"] || name;\n\n\t\tvar type = spine.AttachmentType[map[\"type\"] || \"region\"];\n\t\tvar path = map[\"path\"] || name;\n\n\t\tvar scale = this.scale;\n\t\tif (type == spine.AttachmentType.region) {\n\t\t\tvar region = this.attachmentLoader.newRegionAttachment(skin, name, path);\n\t\t\tif (!region) return null;\n\t\t\tregion.path = path;\n\t\t\tregion.x = (map[\"x\"] || 0) * scale;\n\t\t\tregion.y = (map[\"y\"] || 0) * scale;\n\t\t\tregion.scaleX = map.hasOwnProperty(\"scaleX\") ? map[\"scaleX\"] : 1;\n\t\t\tregion.scaleY = map.hasOwnProperty(\"scaleY\") ? map[\"scaleY\"] : 1;\n\t\t\tregion.rotation = map[\"rotation\"] || 0;\n\t\t\tregion.width = (map[\"width\"] || 0) * scale;\n\t\t\tregion.height = (map[\"height\"] || 0) * scale;\n\n\t\t\tvar color = map[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tregion.r = this.toColor(color, 0);\n\t\t\t\tregion.g = this.toColor(color, 1);\n\t\t\t\tregion.b = this.toColor(color, 2);\n\t\t\t\tregion.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tregion.updateOffset();\n\t\t\treturn region;\n\t\t} else if (type == spine.AttachmentType.mesh) {\n\t\t\tvar mesh = this.attachmentLoader.newMeshAttachment(skin, name, path);\n\t\t\tif (!mesh) return null;\n\t\t\tmesh.path = path;\n\t\t\tmesh.vertices = this.getFloatArray(map, \"vertices\", scale);\n\t\t\tmesh.triangles = this.getIntArray(map, \"triangles\");\n\t\t\tmesh.regionUVs = this.getFloatArray(map, \"uvs\", 1);\n\t\t\tmesh.updateUVs();\n\n\t\t\tcolor = map[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tmesh.r = this.toColor(color, 0);\n\t\t\t\tmesh.g = this.toColor(color, 1);\n\t\t\t\tmesh.b = this.toColor(color, 2);\n\t\t\t\tmesh.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tmesh.hullLength = (map[\"hull\"] || 0) * 2;\n\t\t\tif (map[\"edges\"]) mesh.edges = this.getIntArray(map, \"edges\");\n\t\t\tmesh.width = (map[\"width\"] || 0) * scale;\n\t\t\tmesh.height = (map[\"height\"] || 0) * scale;\n\t\t\treturn mesh;\n\t\t} else if (type == spine.AttachmentType.skinnedmesh) {\n\t\t\tvar mesh = this.attachmentLoader.newSkinnedMeshAttachment(skin, name, path);\n\t\t\tif (!mesh) return null;\n\t\t\tmesh.path = path;\n\n\t\t\tvar uvs = this.getFloatArray(map, \"uvs\", 1);\n\t\t\tvar vertices = this.getFloatArray(map, \"vertices\", 1);\n\t\t\tvar weights = [];\n\t\t\tvar bones = [];\n\t\t\tfor (var i = 0, n = vertices.length; i < n; ) {\n\t\t\t\tvar boneCount = vertices[i++] | 0;\n\t\t\t\tbones[bones.length] = boneCount;\n\t\t\t\tfor (var nn = i + boneCount * 4; i < nn; ) {\n\t\t\t\t\tbones[bones.length] = vertices[i];\n\t\t\t\t\tweights[weights.length] = vertices[i + 1] * scale;\n\t\t\t\t\tweights[weights.length] = vertices[i + 2] * scale;\n\t\t\t\t\tweights[weights.length] = vertices[i + 3];\n\t\t\t\t\ti += 4;\n\t\t\t\t}\n\t\t\t}\n\t\t\tmesh.bones = bones;\n\t\t\tmesh.weights = weights;\n\t\t\tmesh.triangles = this.getIntArray(map, \"triangles\");\n\t\t\tmesh.regionUVs = uvs;\n\t\t\tmesh.updateUVs();\n\n\t\t\tcolor = map[\"color\"];\n\t\t\tif (color) {\n\t\t\t\tmesh.r = this.toColor(color, 0);\n\t\t\t\tmesh.g = this.toColor(color, 1);\n\t\t\t\tmesh.b = this.toColor(color, 2);\n\t\t\t\tmesh.a = this.toColor(color, 3);\n\t\t\t}\n\n\t\t\tmesh.hullLength = (map[\"hull\"] || 0) * 2;\n\t\t\tif (map[\"edges\"]) mesh.edges = this.getIntArray(map, \"edges\");\n\t\t\tmesh.width = (map[\"width\"] || 0) * scale;\n\t\t\tmesh.height = (map[\"height\"] || 0) * scale;\n\t\t\treturn mesh;\n\t\t} else if (type == spine.AttachmentType.boundingbox) {\n\t\t\tvar attachment = this.attachmentLoader.newBoundingBoxAttachment(skin, name);\n\t\t\tvar vertices = map[\"vertices\"];\n\t\t\tfor (var i = 0, n = vertices.length; i < n; i++)\n\t\t\t\tattachment.vertices.push(vertices[i] * scale);\n\t\t\treturn attachment;\n\t\t}\n\t\tthrow \"Unknown attachment type: \" + type;\n\t},\n\treadAnimation: function (name, map, skeletonData) {\n\t\tvar timelines = [];\n\t\tvar duration = 0;\n\n\t\tvar slots = map[\"slots\"];\n\t\tfor (var slotName in slots) {\n\t\t\tif (!slots.hasOwnProperty(slotName)) continue;\n\t\t\tvar slotMap = slots[slotName];\n\t\t\tvar slotIndex = skeletonData.findSlotIndex(slotName);\n\n\t\t\tfor (var timelineName in slotMap) {\n\t\t\t\tif (!slotMap.hasOwnProperty(timelineName)) continue;\n\t\t\t\tvar values = slotMap[timelineName];\n\t\t\t\tif (timelineName == \"color\") {\n\t\t\t\t\tvar timeline = new spine.ColorTimeline(values.length);\n\t\t\t\t\ttimeline.slotIndex = slotIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\tvar color = valueMap[\"color\"];\n\t\t\t\t\t\tvar r = this.toColor(color, 0);\n\t\t\t\t\t\tvar g = this.toColor(color, 1);\n\t\t\t\t\t\tvar b = this.toColor(color, 2);\n\t\t\t\t\t\tvar a = this.toColor(color, 3);\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], r, g, b, a);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 5 - 5]);\n\n\t\t\t\t} else if (timelineName == \"attachment\") {\n\t\t\t\t\tvar timeline = new spine.AttachmentTimeline(values.length);\n\t\t\t\t\ttimeline.slotIndex = slotIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\ttimeline.setFrame(frameIndex++, valueMap[\"time\"], valueMap[\"name\"]);\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]);\n\n\t\t\t\t} else\n\t\t\t\t\tthrow \"Invalid timeline type for a slot: \" + timelineName + \" (\" + slotName + \")\";\n\t\t\t}\n\t\t}\n\n\t\tvar bones = map[\"bones\"];\n\t\tfor (var boneName in bones) {\n\t\t\tif (!bones.hasOwnProperty(boneName)) continue;\n\t\t\tvar boneIndex = skeletonData.findBoneIndex(boneName);\n\t\t\tif (boneIndex == -1) throw \"Bone not found: \" + boneName;\n\t\t\tvar boneMap = bones[boneName];\n\n\t\t\tfor (var timelineName in boneMap) {\n\t\t\t\tif (!boneMap.hasOwnProperty(timelineName)) continue;\n\t\t\t\tvar values = boneMap[timelineName];\n\t\t\t\tif (timelineName == \"rotate\") {\n\t\t\t\t\tvar timeline = new spine.RotateTimeline(values.length);\n\t\t\t\t\ttimeline.boneIndex = boneIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], valueMap[\"angle\"]);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]);\n\n\t\t\t\t} else if (timelineName == \"translate\" || timelineName == \"scale\") {\n\t\t\t\t\tvar timeline;\n\t\t\t\t\tvar timelineScale = 1;\n\t\t\t\t\tif (timelineName == \"scale\")\n\t\t\t\t\t\ttimeline = new spine.ScaleTimeline(values.length);\n\t\t\t\t\telse {\n\t\t\t\t\t\ttimeline = new spine.TranslateTimeline(values.length);\n\t\t\t\t\t\ttimelineScale = this.scale;\n\t\t\t\t\t}\n\t\t\t\t\ttimeline.boneIndex = boneIndex;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\tvar x = (valueMap[\"x\"] || 0) * timelineScale;\n\t\t\t\t\t\tvar y = (valueMap[\"y\"] || 0) * timelineScale;\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], x, y);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 3 - 3]);\n\n\t\t\t\t} else if (timelineName == \"flipX\" || timelineName == \"flipY\") {\n\t\t\t\t\tvar x = timelineName == \"flipX\";\n\t\t\t\t\tvar timeline = x ? new spine.FlipXTimeline(values.length) : new spine.FlipYTimeline(values.length);\n\t\t\t\t\ttimeline.boneIndex = boneIndex;\n\n\t\t\t\t\tvar field = x ? \"x\" : \"y\";\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], valueMap[field] || false);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines.push(timeline);\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() * 2 - 2]);\n\t\t\t\t} else\n\t\t\t\t\tthrow \"Invalid timeline type for a bone: \" + timelineName + \" (\" + boneName + \")\";\n\t\t\t}\n\t\t}\n\n\t\tvar ikMap = map[\"ik\"];\n\t\tfor (var ikConstraintName in ikMap) {\n\t\t\tif (!ikMap.hasOwnProperty(ikConstraintName)) continue;\n\t\t\tvar ikConstraint = skeletonData.findIkConstraint(ikConstraintName);\n\t\t\tvar values = ikMap[ikConstraintName];\n\t\t\tvar timeline = new spine.IkConstraintTimeline(values.length);\n\t\t\ttimeline.ikConstraintIndex = skeletonData.ikConstraints.indexOf(ikConstraint);\n\t\t\tvar frameIndex = 0;\n\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\tvar valueMap = values[i];\n\t\t\t\tvar mix = valueMap.hasOwnProperty(\"mix\") ? valueMap[\"mix\"] : 1;\n\t\t\t\tvar bendDirection = (!valueMap.hasOwnProperty(\"bendPositive\") || valueMap[\"bendPositive\"]) ? 1 : -1;\n\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], mix, bendDirection);\n\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\tframeIndex++;\n\t\t\t}\n\t\t\ttimelines.push(timeline);\n\t\t\tduration = Math.max(duration, timeline.frames[timeline.frameCount * 3 - 3]);\n\t\t}\n\n\t\tvar ffd = map[\"ffd\"];\n\t\tfor (var skinName in ffd) {\n\t\t\tvar skin = skeletonData.findSkin(skinName);\n\t\t\tvar slotMap = ffd[skinName];\n\t\t\tfor (slotName in slotMap) {\n\t\t\t\tvar slotIndex = skeletonData.findSlotIndex(slotName);\n\t\t\t\tvar meshMap = slotMap[slotName];\n\t\t\t\tfor (var meshName in meshMap) {\n\t\t\t\t\tvar values = meshMap[meshName];\n\t\t\t\t\tvar timeline = new spine.FfdTimeline(values.length);\n\t\t\t\t\tvar attachment = skin.getAttachment(slotIndex, meshName);\n\t\t\t\t\tif (!attachment) throw \"FFD attachment not found: \" + meshName;\n\t\t\t\t\ttimeline.slotIndex = slotIndex;\n\t\t\t\t\ttimeline.attachment = attachment;\n\n\t\t\t\t\tvar isMesh = attachment.type == spine.AttachmentType.mesh;\n\t\t\t\t\tvar vertexCount;\n\t\t\t\t\tif (isMesh)\n\t\t\t\t\t\tvertexCount = attachment.vertices.length;\n\t\t\t\t\telse\n\t\t\t\t\t\tvertexCount = attachment.weights.length / 3 * 2;\n\n\t\t\t\t\tvar frameIndex = 0;\n\t\t\t\t\tfor (var i = 0, n = values.length; i < n; i++) {\n\t\t\t\t\t\tvar valueMap = values[i];\n\t\t\t\t\t\tvar vertices;\n\t\t\t\t\t\tif (!valueMap[\"vertices\"]) {\n\t\t\t\t\t\t\tif (isMesh)\n\t\t\t\t\t\t\t\tvertices = attachment.vertices;\n\t\t\t\t\t\t\telse {\n\t\t\t\t\t\t\t\tvertices = [];\n\t\t\t\t\t\t\t\tvertices.length = vertexCount;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\tvar verticesValue = valueMap[\"vertices\"];\n\t\t\t\t\t\t\tvar vertices = [];\n\t\t\t\t\t\t\tvertices.length = vertexCount;\n\t\t\t\t\t\t\tvar start = valueMap[\"offset\"] || 0;\n\t\t\t\t\t\t\tvar nn = verticesValue.length;\n\t\t\t\t\t\t\tif (this.scale == 1) {\n\t\t\t\t\t\t\t\tfor (var ii = 0; ii < nn; ii++)\n\t\t\t\t\t\t\t\t\tvertices[ii + start] = verticesValue[ii];\n\t\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t\tfor (var ii = 0; ii < nn; ii++)\n\t\t\t\t\t\t\t\t\tvertices[ii + start] = verticesValue[ii] * this.scale;\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t\tif (isMesh) {\n\t\t\t\t\t\t\t\tvar meshVertices = attachment.vertices;\n\t\t\t\t\t\t\t\tfor (var ii = 0, nn = vertices.length; ii < nn; ii++)\n\t\t\t\t\t\t\t\t\tvertices[ii] += meshVertices[ii];\n\t\t\t\t\t\t\t}\n\t\t\t\t\t\t}\n\n\t\t\t\t\t\ttimeline.setFrame(frameIndex, valueMap[\"time\"], vertices);\n\t\t\t\t\t\tthis.readCurve(timeline, frameIndex, valueMap);\n\t\t\t\t\t\tframeIndex++;\n\t\t\t\t\t}\n\t\t\t\t\ttimelines[timelines.length] = timeline;\n\t\t\t\t\tduration = Math.max(duration, timeline.frames[timeline.frameCount - 1]);\n\t\t\t\t}\n\t\t\t}\n\t\t}\n\n\t\tvar drawOrderValues = map[\"drawOrder\"];\n\t\tif (!drawOrderValues) drawOrderValues = map[\"draworder\"];\n\t\tif (drawOrderValues) {\n\t\t\tvar timeline = new spine.DrawOrderTimeline(drawOrderValues.length);\n\t\t\tvar slotCount = skeletonData.slots.length;\n\t\t\tvar frameIndex = 0;\n\t\t\tfor (var i = 0, n = drawOrderValues.length; i < n; i++) {\n\t\t\t\tvar drawOrderMap = drawOrderValues[i];\n\t\t\t\tvar drawOrder = null;\n\t\t\t\tif (drawOrderMap[\"offsets\"]) {\n\t\t\t\t\tdrawOrder = [];\n\t\t\t\t\tdrawOrder.length = slotCount;\n\t\t\t\t\tfor (var ii = slotCount - 1; ii >= 0; ii--)\n\t\t\t\t\t\tdrawOrder[ii] = -1;\n\t\t\t\t\tvar offsets = drawOrderMap[\"offsets\"];\n\t\t\t\t\tvar unchanged = [];\n\t\t\t\t\tunchanged.length = slotCount - offsets.length;\n\t\t\t\t\tvar originalIndex = 0, unchangedIndex = 0;\n\t\t\t\t\tfor (var ii = 0, nn = offsets.length; ii < nn; ii++) {\n\t\t\t\t\t\tvar offsetMap = offsets[ii];\n\t\t\t\t\t\tvar slotIndex = skeletonData.findSlotIndex(offsetMap[\"slot\"]);\n\t\t\t\t\t\tif (slotIndex == -1) throw \"Slot not found: \" + offsetMap[\"slot\"];\n\t\t\t\t\t\t// Collect unchanged items.\n\t\t\t\t\t\twhile (originalIndex != slotIndex)\n\t\t\t\t\t\t\tunchanged[unchangedIndex++] = originalIndex++;\n\t\t\t\t\t\t// Set changed items.\n\t\t\t\t\t\tdrawOrder[originalIndex + offsetMap[\"offset\"]] = originalIndex++;\n\t\t\t\t\t}\n\t\t\t\t\t// Collect remaining unchanged items.\n\t\t\t\t\twhile (originalIndex < slotCount)\n\t\t\t\t\t\tunchanged[unchangedIndex++] = originalIndex++;\n\t\t\t\t\t// Fill in unchanged items.\n\t\t\t\t\tfor (var ii = slotCount - 1; ii >= 0; ii--)\n\t\t\t\t\t\tif (drawOrder[ii] == -1) drawOrder[ii] = unchanged[--unchangedIndex];\n\t\t\t\t}\n\t\t\t\ttimeline.setFrame(frameIndex++, drawOrderMap[\"time\"], drawOrder);\n\t\t\t}\n\t\t\ttimelines.push(timeline);\n\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]);\n\t\t}\n\n\t\tvar events = map[\"events\"];\n\t\tif (events) {\n\t\t\tvar timeline = new spine.EventTimeline(events.length);\n\t\t\tvar frameIndex = 0;\n\t\t\tfor (var i = 0, n = events.length; i < n; i++) {\n\t\t\t\tvar eventMap = events[i];\n\t\t\t\tvar eventData = skeletonData.findEvent(eventMap[\"name\"]);\n\t\t\t\tif (!eventData) throw \"Event not found: \" + eventMap[\"name\"];\n\t\t\t\tvar event = new spine.Event(eventData);\n\t\t\t\tevent.intValue = eventMap.hasOwnProperty(\"int\") ? eventMap[\"int\"] : eventData.intValue;\n\t\t\t\tevent.floatValue = eventMap.hasOwnProperty(\"float\") ? eventMap[\"float\"] : eventData.floatValue;\n\t\t\t\tevent.stringValue = eventMap.hasOwnProperty(\"string\") ? eventMap[\"string\"] : eventData.stringValue;\n\t\t\t\ttimeline.setFrame(frameIndex++, eventMap[\"time\"], event);\n\t\t\t}\n\t\t\ttimelines.push(timeline);\n\t\t\tduration = Math.max(duration, timeline.frames[timeline.getFrameCount() - 1]);\n\t\t}\n\n\t\tskeletonData.animations.push(new spine.Animation(name, timelines, duration));\n\t},\n\treadCurve: function (timeline, frameIndex, valueMap) {\n\t\tvar curve = valueMap[\"curve\"];\n\t\tif (!curve)\n\t\t\ttimeline.curves.setLinear(frameIndex);\n\t\telse if (curve == \"stepped\")\n\t\t\ttimeline.curves.setStepped(frameIndex);\n\t\telse if (curve instanceof Array)\n\t\t\ttimeline.curves.setCurve(frameIndex, curve[0], curve[1], curve[2], curve[3]);\n\t},\n\ttoColor: function (hexString, colorIndex) {\n\t\tif (hexString.length != 8) throw \"Color hexidecimal length must be 8, recieved: \" + hexString;\n\t\treturn parseInt(hexString.substring(colorIndex * 2, (colorIndex * 2) + 2), 16) / 255;\n\t},\n\tgetFloatArray: function (map, name, scale) {\n\t\tvar list = map[name];\n\t\tvar values = new spine.Float32Array(list.length);\n\t\tvar i = 0, n = list.length;\n\t\tif (scale == 1) {\n\t\t\tfor (; i < n; i++)\n\t\t\t\tvalues[i] = list[i];\n\t\t} else {\n\t\t\tfor (; i < n; i++)\n\t\t\t\tvalues[i] = list[i] * scale;\n\t\t}\n\t\treturn values;\n\t},\n\tgetIntArray: function (map, name) {\n\t\tvar list = map[name];\n\t\tvar values = new spine.Uint16Array(list.length);\n\t\tfor (var i = 0, n = list.length; i < n; i++)\n\t\t\tvalues[i] = list[i] | 0;\n\t\treturn values;\n\t}\n};\n\nspine.Atlas = function (atlasText, textureLoader) {\n\tthis.textureLoader = textureLoader;\n\tthis.pages = [];\n\tthis.regions = [];\n\n\tvar reader = new spine.AtlasReader(atlasText);\n\tvar tuple = [];\n\ttuple.length = 4;\n\tvar page = null;\n\twhile (true) {\n\t\tvar line = reader.readLine();\n\t\tif (line === null) break;\n\t\tline = reader.trim(line);\n\t\tif (!line.length)\n\t\t\tpage = null;\n\t\telse if (!page) {\n\t\t\tpage = new spine.AtlasPage();\n\t\t\tpage.name = line;\n\n\t\t\tif (reader.readTuple(tuple) == 2) { // size is only optional for an atlas packed with an old TexturePacker.\n\t\t\t\tpage.width = parseInt(tuple[0]);\n\t\t\t\tpage.height = parseInt(tuple[1]);\n\t\t\t\treader.readTuple(tuple);\n\t\t\t}\n\t\t\tpage.format = spine.Atlas.Format[tuple[0]];\n\n\t\t\treader.readTuple(tuple);\n\t\t\tpage.minFilter = spine.Atlas.TextureFilter[tuple[0]];\n\t\t\tpage.magFilter = spine.Atlas.TextureFilter[tuple[1]];\n\n\t\t\tvar direction = reader.readValue();\n\t\t\tpage.uWrap = spine.Atlas.TextureWrap.clampToEdge;\n\t\t\tpage.vWrap = spine.Atlas.TextureWrap.clampToEdge;\n\t\t\tif (direction == \"x\")\n\t\t\t\tpage.uWrap = spine.Atlas.TextureWrap.repeat;\n\t\t\telse if (direction == \"y\")\n\t\t\t\tpage.vWrap = spine.Atlas.TextureWrap.repeat;\n\t\t\telse if (direction == \"xy\")\n\t\t\t\tpage.uWrap = page.vWrap = spine.Atlas.TextureWrap.repeat;\n\n\t\t\ttextureLoader.load(page, line, this);\n\n\t\t\tthis.pages.push(page);\n\n\t\t} else {\n\t\t\tvar region = new spine.AtlasRegion();\n\t\t\tregion.name = line;\n\t\t\tregion.page = page;\n\n\t\t\tregion.rotate = reader.readValue() == \"true\";\n\n\t\t\treader.readTuple(tuple);\n\t\t\tvar x = parseInt(tuple[0]);\n\t\t\tvar y = parseInt(tuple[1]);\n\n\t\t\treader.readTuple(tuple);\n\t\t\tvar width = parseInt(tuple[0]);\n\t\t\tvar height = parseInt(tuple[1]);\n\n\t\t\tregion.u = x / page.width;\n\t\t\tregion.v = y / page.height;\n\t\t\tif (region.rotate) {\n\t\t\t\tregion.u2 = (x + height) / page.width;\n\t\t\t\tregion.v2 = (y + width) / page.height;\n\t\t\t} else {\n\t\t\t\tregion.u2 = (x + width) / page.width;\n\t\t\t\tregion.v2 = (y + height) / page.height;\n\t\t\t}\n\t\t\tregion.x = x;\n\t\t\tregion.y = y;\n\t\t\tregion.width = Math.abs(width);\n\t\t\tregion.height = Math.abs(height);\n\n\t\t\tif (reader.readTuple(tuple) == 4) { // split is optional\n\t\t\t\tregion.splits = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])];\n\n\t\t\t\tif (reader.readTuple(tuple) == 4) { // pad is optional, but only present with splits\n\t\t\t\t\tregion.pads = [parseInt(tuple[0]), parseInt(tuple[1]), parseInt(tuple[2]), parseInt(tuple[3])];\n\n\t\t\t\t\treader.readTuple(tuple);\n\t\t\t\t}\n\t\t\t}\n\n\t\t\tregion.originalWidth = parseInt(tuple[0]);\n\t\t\tregion.originalHeight = parseInt(tuple[1]);\n\n\t\t\treader.readTuple(tuple);\n\t\t\tregion.offsetX = parseInt(tuple[0]);\n\t\t\tregion.offsetY = parseInt(tuple[1]);\n\n\t\t\tregion.index = parseInt(reader.readValue());\n\n\t\t\tthis.regions.push(region);\n\t\t}\n\t}\n};\nspine.Atlas.prototype = {\n\tfindRegion: function (name) {\n\t\tvar regions = this.regions;\n\t\tfor (var i = 0, n = regions.length; i < n; i++)\n\t\t\tif (regions[i].name == name) return regions[i];\n\t\treturn null;\n\t},\n\tdispose: function () {\n\t\tvar pages = this.pages;\n\t\tfor (var i = 0, n = pages.length; i < n; i++)\n\t\t\tthis.textureLoader.unload(pages[i].rendererObject);\n\t},\n\tupdateUVs: function (page) {\n\t\tvar regions = this.regions;\n\t\tfor (var i = 0, n = regions.length; i < n; i++) {\n\t\t\tvar region = regions[i];\n\t\t\tif (region.page != page) continue;\n\t\t\tregion.u = region.x / page.width;\n\t\t\tregion.v = region.y / page.height;\n\t\t\tif (region.rotate) {\n\t\t\t\tregion.u2 = (region.x + region.height) / page.width;\n\t\t\t\tregion.v2 = (region.y + region.width) / page.height;\n\t\t\t} else {\n\t\t\t\tregion.u2 = (region.x + region.width) / page.width;\n\t\t\t\tregion.v2 = (region.y + region.height) / page.height;\n\t\t\t}\n\t\t}\n\t}\n};\n\nspine.Atlas.Format = {\n\talpha: 0,\n\tintensity: 1,\n\tluminanceAlpha: 2,\n\trgb565: 3,\n\trgba4444: 4,\n\trgb888: 5,\n\trgba8888: 6\n};\n\nspine.Atlas.TextureFilter = {\n\tnearest: 0,\n\tlinear: 1,\n\tmipMap: 2,\n\tmipMapNearestNearest: 3,\n\tmipMapLinearNearest: 4,\n\tmipMapNearestLinear: 5,\n\tmipMapLinearLinear: 6\n};\n\nspine.Atlas.TextureWrap = {\n\tmirroredRepeat: 0,\n\tclampToEdge: 1,\n\trepeat: 2\n};\n\nspine.AtlasPage = function () {};\nspine.AtlasPage.prototype = {\n\tname: null,\n\tformat: null,\n\tminFilter: null,\n\tmagFilter: null,\n\tuWrap: null,\n\tvWrap: null,\n\trendererObject: null,\n\twidth: 0,\n\theight: 0\n};\n\nspine.AtlasRegion = function () {};\nspine.AtlasRegion.prototype = {\n\tpage: null,\n\tname: null,\n\tx: 0, y: 0,\n\twidth: 0, height: 0,\n\tu: 0, v: 0, u2: 0, v2: 0,\n\toffsetX: 0, offsetY: 0,\n\toriginalWidth: 0, originalHeight: 0,\n\tindex: 0,\n\trotate: false,\n\tsplits: null,\n\tpads: null\n};\n\nspine.AtlasReader = function (text) {\n\tthis.lines = text.split(/\\r\\n|\\r|\\n/);\n};\nspine.AtlasReader.prototype = {\n\tindex: 0,\n\ttrim: function (value) {\n\t\treturn value.replace(/^\\s+|\\s+$/g, \"\");\n\t},\n\treadLine: function () {\n\t\tif (this.index >= this.lines.length) return null;\n\t\treturn this.lines[this.index++];\n\t},\n\treadValue: function () {\n\t\tvar line = this.readLine();\n\t\tvar colon = line.indexOf(\":\");\n\t\tif (colon == -1) throw \"Invalid line: \" + line;\n\t\treturn this.trim(line.substring(colon + 1));\n\t},\n\t/** Returns the number of tuple values read (1, 2 or 4). */\n\treadTuple: function (tuple) {\n\t\tvar line = this.readLine();\n\t\tvar colon = line.indexOf(\":\");\n\t\tif (colon == -1) throw \"Invalid line: \" + line;\n\t\tvar i = 0, lastMatch = colon + 1;\n\t\tfor (; i < 3; i++) {\n\t\t\tvar comma = line.indexOf(\",\", lastMatch);\n\t\t\tif (comma == -1) break;\n\t\t\ttuple[i] = this.trim(line.substr(lastMatch, comma - lastMatch));\n\t\t\tlastMatch = comma + 1;\n\t\t}\n\t\ttuple[i] = this.trim(line.substring(lastMatch));\n\t\treturn i + 1;\n\t}\n};\n\nspine.AtlasAttachmentLoader = function (atlas) {\n\tthis.atlas = atlas;\n};\nspine.AtlasAttachmentLoader.prototype = {\n\tnewRegionAttachment: function (skin, name, path) {\n\t\tvar region = this.atlas.findRegion(path);\n\t\tif (!region) throw \"Region not found in atlas: \" + path + \" (region attachment: \" + name + \")\";\n\t\tvar attachment = new spine.RegionAttachment(name);\n\t\tattachment.rendererObject = region;\n\t\tattachment.setUVs(region.u, region.v, region.u2, region.v2, region.rotate);\n\t\tattachment.regionOffsetX = region.offsetX;\n\t\tattachment.regionOffsetY = region.offsetY;\n\t\tattachment.regionWidth = region.width;\n\t\tattachment.regionHeight = region.height;\n\t\tattachment.regionOriginalWidth = region.originalWidth;\n\t\tattachment.regionOriginalHeight = region.originalHeight;\n\t\treturn attachment;\n\t},\n\tnewMeshAttachment: function (skin, name, path) {\n\t\tvar region = this.atlas.findRegion(path);\n\t\tif (!region) throw \"Region not found in atlas: \" + path + \" (mesh attachment: \" + name + \")\";\n\t\tvar attachment = new spine.MeshAttachment(name);\n\t\tattachment.rendererObject = region;\n\t\tattachment.regionU = region.u;\n\t\tattachment.regionV = region.v;\n\t\tattachment.regionU2 = region.u2;\n\t\tattachment.regionV2 = region.v2;\n\t\tattachment.regionRotate = region.rotate;\n\t\tattachment.regionOffsetX = region.offsetX;\n\t\tattachment.regionOffsetY = region.offsetY;\n\t\tattachment.regionWidth = region.width;\n\t\tattachment.regionHeight = region.height;\n\t\tattachment.regionOriginalWidth = region.originalWidth;\n\t\tattachment.regionOriginalHeight = region.originalHeight;\n\t\treturn attachment;\n\t},\n\tnewSkinnedMeshAttachment: function (skin, name, path) {\n\t\tvar region = this.atlas.findRegion(path);\n\t\tif (!region) throw \"Region not found in atlas: \" + path + \" (skinned mesh attachment: \" + name + \")\";\n\t\tvar attachment = new spine.SkinnedMeshAttachment(name);\n\t\tattachment.rendererObject = region;\n\t\tattachment.regionU = region.u;\n\t\tattachment.regionV = region.v;\n\t\tattachment.regionU2 = region.u2;\n\t\tattachment.regionV2 = region.v2;\n\t\tattachment.regionRotate = region.rotate;\n\t\tattachment.regionOffsetX = region.offsetX;\n\t\tattachment.regionOffsetY = region.offsetY;\n\t\tattachment.regionWidth = region.width;\n\t\tattachment.regionHeight = region.height;\n\t\tattachment.regionOriginalWidth = region.originalWidth;\n\t\tattachment.regionOriginalHeight = region.originalHeight;\n\t\treturn attachment;\n\t},\n\tnewBoundingBoxAttachment: function (skin, name) {\n\t\treturn new spine.BoundingBoxAttachment(name);\n\t}\n};\n\nspine.SkeletonBounds = function () {\n\tthis.polygonPool = [];\n\tthis.polygons = [];\n\tthis.boundingBoxes = [];\n};\nspine.SkeletonBounds.prototype = {\n\tminX: 0, minY: 0, maxX: 0, maxY: 0,\n\tupdate: function (skeleton, updateAabb) {\n\t\tvar slots = skeleton.slots;\n\t\tvar slotCount = slots.length;\n\t\tvar x = skeleton.x, y = skeleton.y;\n\t\tvar boundingBoxes = this.boundingBoxes;\n\t\tvar polygonPool = this.polygonPool;\n\t\tvar polygons = this.polygons;\n\n\t\tboundingBoxes.length = 0;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++)\n\t\t\tpolygonPool.push(polygons[i]);\n\t\tpolygons.length = 0;\n\n\t\tfor (var i = 0; i < slotCount; i++) {\n\t\t\tvar slot = slots[i];\n\t\t\tvar boundingBox = slot.attachment;\n\t\t\tif (boundingBox.type != spine.AttachmentType.boundingbox) continue;\n\t\t\tboundingBoxes.push(boundingBox);\n\n\t\t\tvar poolCount = polygonPool.length, polygon;\n\t\t\tif (poolCount > 0) {\n\t\t\t\tpolygon = polygonPool[poolCount - 1];\n\t\t\t\tpolygonPool.splice(poolCount - 1, 1);\n\t\t\t} else\n\t\t\t\tpolygon = [];\n\t\t\tpolygons.push(polygon);\n\n\t\t\tpolygon.length = boundingBox.vertices.length;\n\t\t\tboundingBox.computeWorldVertices(x, y, slot.bone, polygon);\n\t\t}\n\n\t\tif (updateAabb) this.aabbCompute();\n\t},\n\taabbCompute: function () {\n\t\tvar polygons = this.polygons;\n\t\tvar minX = Number.MAX_VALUE, minY = Number.MAX_VALUE, maxX = Number.MIN_VALUE, maxY = Number.MIN_VALUE;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++) {\n\t\t\tvar vertices = polygons[i];\n\t\t\tfor (var ii = 0, nn = vertices.length; ii < nn; ii += 2) {\n\t\t\t\tvar x = vertices[ii];\n\t\t\t\tvar y = vertices[ii + 1];\n\t\t\t\tminX = Math.min(minX, x);\n\t\t\t\tminY = Math.min(minY, y);\n\t\t\t\tmaxX = Math.max(maxX, x);\n\t\t\t\tmaxY = Math.max(maxY, y);\n\t\t\t}\n\t\t}\n\t\tthis.minX = minX;\n\t\tthis.minY = minY;\n\t\tthis.maxX = maxX;\n\t\tthis.maxY = maxY;\n\t},\n\t/** Returns true if the axis aligned bounding box contains the point. */\n\taabbContainsPoint: function (x, y) {\n\t\treturn x >= this.minX && x <= this.maxX && y >= this.minY && y <= this.maxY;\n\t},\n\t/** Returns true if the axis aligned bounding box intersects the line segment. */\n\taabbIntersectsSegment: function (x1, y1, x2, y2) {\n\t\tvar minX = this.minX, minY = this.minY, maxX = this.maxX, maxY = this.maxY;\n\t\tif ((x1 <= minX && x2 <= minX) || (y1 <= minY && y2 <= minY) || (x1 >= maxX && x2 >= maxX) || (y1 >= maxY && y2 >= maxY))\n\t\t\treturn false;\n\t\tvar m = (y2 - y1) / (x2 - x1);\n\t\tvar y = m * (minX - x1) + y1;\n\t\tif (y > minY && y < maxY) return true;\n\t\ty = m * (maxX - x1) + y1;\n\t\tif (y > minY && y < maxY) return true;\n\t\tvar x = (minY - y1) / m + x1;\n\t\tif (x > minX && x < maxX) return true;\n\t\tx = (maxY - y1) / m + x1;\n\t\tif (x > minX && x < maxX) return true;\n\t\treturn false;\n\t},\n\t/** Returns true if the axis aligned bounding box intersects the axis aligned bounding box of the specified bounds. */\n\taabbIntersectsSkeleton: function (bounds) {\n\t\treturn this.minX < bounds.maxX && this.maxX > bounds.minX && this.minY < bounds.maxY && this.maxY > bounds.minY;\n\t},\n\t/** Returns the first bounding box attachment that contains the point, or null. When doing many checks, it is usually more\n\t * efficient to only call this method if {@link #aabbContainsPoint(float, float)} returns true. */\n\tcontainsPoint: function (x, y) {\n\t\tvar polygons = this.polygons;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++)\n\t\t\tif (this.polygonContainsPoint(polygons[i], x, y)) return this.boundingBoxes[i];\n\t\treturn null;\n\t},\n\t/** Returns the first bounding box attachment that contains the line segment, or null. When doing many checks, it is usually\n\t * more efficient to only call this method if {@link #aabbIntersectsSegment(float, float, float, float)} returns true. */\n\tintersectsSegment: function (x1, y1, x2, y2) {\n\t\tvar polygons = this.polygons;\n\t\tfor (var i = 0, n = polygons.length; i < n; i++)\n\t\t\tif (polygons[i].intersectsSegment(x1, y1, x2, y2)) return this.boundingBoxes[i];\n\t\treturn null;\n\t},\n\t/** Returns true if the polygon contains the point. */\n\tpolygonContainsPoint: function (polygon, x, y) {\n\t\tvar nn = polygon.length;\n\t\tvar prevIndex = nn - 2;\n\t\tvar inside = false;\n\t\tfor (var ii = 0; ii < nn; ii += 2) {\n\t\t\tvar vertexY = polygon[ii + 1];\n\t\t\tvar prevY = polygon[prevIndex + 1];\n\t\t\tif ((vertexY < y && prevY >= y) || (prevY < y && vertexY >= y)) {\n\t\t\t\tvar vertexX = polygon[ii];\n\t\t\t\tif (vertexX + (y - vertexY) / (prevY - vertexY) * (polygon[prevIndex] - vertexX) < x) inside = !inside;\n\t\t\t}\n\t\t\tprevIndex = ii;\n\t\t}\n\t\treturn inside;\n\t},\n\t/** Returns true if the polygon contains the line segment. */\n\tpolygonIntersectsSegment: function (polygon, x1, y1, x2, y2) {\n\t\tvar nn = polygon.length;\n\t\tvar width12 = x1 - x2, height12 = y1 - y2;\n\t\tvar det1 = x1 * y2 - y1 * x2;\n\t\tvar x3 = polygon[nn - 2], y3 = polygon[nn - 1];\n\t\tfor (var ii = 0; ii < nn; ii += 2) {\n\t\t\tvar x4 = polygon[ii], y4 = polygon[ii + 1];\n\t\t\tvar det2 = x3 * y4 - y3 * x4;\n\t\t\tvar width34 = x3 - x4, height34 = y3 - y4;\n\t\t\tvar det3 = width12 * height34 - height12 * width34;\n\t\t\tvar x = (det1 * width34 - width12 * det2) / det3;\n\t\t\tif (((x >= x3 && x <= x4) || (x >= x4 && x <= x3)) && ((x >= x1 && x <= x2) || (x >= x2 && x <= x1))) {\n\t\t\t\tvar y = (det1 * height34 - height12 * det2) / det3;\n\t\t\t\tif (((y >= y3 && y <= y4) || (y >= y4 && y <= y3)) && ((y >= y1 && y <= y2) || (y >= y2 && y <= y1))) return true;\n\t\t\t}\n\t\t\tx3 = x4;\n\t\t\ty3 = y4;\n\t\t}\n\t\treturn false;\n\t},\n\tgetPolygon: function (attachment) {\n\t\tvar index = this.boundingBoxes.indexOf(attachment);\n\t\treturn index == -1 ? null : this.polygons[index];\n\t},\n\tgetWidth: function () {\n\t\treturn this.maxX - this.minX;\n\t},\n\tgetHeight: function () {\n\t\treturn this.maxY - this.minY;\n\t}\n};\n","/**\n * @file        Main export of the PIXI spine library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    Spine:              require('./Spine')\n};\n","var core = require('../core');\n\n/**\n * A BitmapText object will create a line or multiple lines of text using bitmap font. To\n * split a line you can use '\\n', '\\r' or '\\r\\n' in your string. You can generate the fnt files using:\n *\n * http://www.angelcode.com/products/bmfont/ for windows or\n * http://www.bmglyph.com/ for mac.\n *\n * @class\n * @extends DisplayObjectContainer\n * @namespace PIXI\n * @param text {string} The copy that you would like the text to display\n * @param style {object} The style parameters\n * @param style.font {string} The size (optional) and bitmap font id (required) eq 'Arial' or '20px Arial' (must have loaded previously)\n * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text\n */\nfunction BitmapText(text, style) {\n    core.DisplayObjectContainer.call(this);\n\n    /**\n     * The width of the overall text, different from fontSize,\n     * which is defined in the style object\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.textWidth = 0;\n\n    /**\n     * The height of the overall text, different from fontSize,\n     * which is defined in the style object\n     *\n     * @member {number}\n     * @readOnly\n     */\n    this.textHeight = 0;\n\n    /**\n     * Private tracker for the letter sprite pool.\n     *\n     * @member {Sprite[]}\n     * @private\n     */\n    this._pool = [];\n\n    /**\n     * Private tracker for the current style.\n     *\n     * @member {object}\n     * @private\n     */\n    this._style = {\n        tint: style.tint,\n        align: style.align,\n        fontName: null,\n        fontSize: 0\n    };\n    this.font = style.font; // run font setter\n\n    /**\n     * Private tracker for the current text.\n     *\n     * @member {string}\n     * @private\n     */\n    this._text = text;\n\n    /**\n     * The dirty state of this object.\n     *\n     * @member {boolean}\n     */\n    this.dirty = false;\n\n    this.updateText();\n}\n\n// constructor\nBitmapText.prototype = Object.create(core.DisplayObjectContainer.prototype);\nBitmapText.prototype.constructor = BitmapText;\nmodule.exports = BitmapText;\n\nObject.defineProperties(BitmapText.prototype, {\n    /**\n     * The tint of the BitmapText object\n     *\n     * @member {number}\n     * @memberof BitmapText#\n     */\n    tint: {\n        get: function () {\n            return this._style.tint;\n        },\n        set: function (value) {\n            this._style.tint = (typeof value === 'number' && value >= 0) ? value : 0xFFFFFF;\n\n            this.dirty = true;\n        }\n    },\n\n    /**\n     * The tint of the BitmapText object\n     *\n     * @member {string}\n     * @default 'left'\n     * @memberof BitmapText#\n     */\n    align: {\n        get: function () {\n            return this._style.align;\n        },\n        set: function (value) {\n            this._style.align = value;\n\n            this.dirty = true;\n        }\n    },\n\n    /**\n     * The tint of the BitmapText object\n     *\n     * @member {Font}\n     * @memberof BitmapText#\n     */\n    font: {\n        get: function () {\n            return this._style.font;\n        },\n        set: function (value) {\n            value = value.split(' ');\n\n            // TODO - This should be object-based not string based like it has been.\n            this._style.fontName = value[value.length - 1];\n            this._style.fontSize = value.length >= 2 ? parseInt(value[value.length - 2], 10) : BitmapText.fonts[this.fontName].size;\n\n            this.dirty = true;\n        }\n    },\n\n    /**\n     * The text of the BitmapText object\n     *\n     * @member {string}\n     * @memberof BitmapText#\n     */\n    text: {\n        get: function () {\n            return this._text;\n        },\n        set: function (value) {\n            this._text = value;\n\n            this.dirty = true;\n        }\n    }\n});\n\n/**\n * Renders text and updates it when needed\n *\n * @private\n */\nBitmapText.prototype.updateText = function () {\n    var data = BitmapText.fonts[this.fontName];\n    var pos = new core.math.Point();\n    var prevCharCode = null;\n    var chars = [];\n    var maxLineWidth = 0;\n    var lineWidths = [];\n    var line = 0;\n    var scale = this.fontSize / data.size;\n\n    for (var i = 0; i < this.text.length; i++) {\n        var charCode = this.text.charCodeAt(i);\n\n        if (/(?:\\r\\n|\\r|\\n)/.test(this.text.charAt(i))) {\n            lineWidths.push(pos.x);\n            maxLineWidth = Math.max(maxLineWidth, pos.x);\n            line++;\n\n            pos.x = 0;\n            pos.y += data.lineHeight;\n            prevCharCode = null;\n            continue;\n        }\n\n        var charData = data.chars[charCode];\n\n        if (!charData) {\n            continue;\n        }\n\n        if (prevCharCode && charData.kerning[prevCharCode]) {\n            pos.x += charData.kerning[prevCharCode];\n        }\n\n        chars.push({texture:charData.texture, line: line, charCode: charCode, position: new core.math.Point(pos.x + charData.xOffset, pos.y + charData.yOffset)});\n        pos.x += charData.xAdvance;\n\n        prevCharCode = charCode;\n    }\n\n    lineWidths.push(pos.x);\n    maxLineWidth = Math.max(maxLineWidth, pos.x);\n\n    var lineAlignOffsets = [];\n\n    for (i = 0; i <= line; i++) {\n        var alignOffset = 0;\n\n        if (this.style.align === 'right') {\n            alignOffset = maxLineWidth - lineWidths[i];\n        }\n        else if (this.style.align === 'center') {\n            alignOffset = (maxLineWidth - lineWidths[i]) / 2;\n        }\n\n        lineAlignOffsets.push(alignOffset);\n    }\n\n    var lenChildren = this.children.length;\n    var lenChars = chars.length;\n    var tint = this.tint;\n\n    for (i = 0; i < lenChars; i++) {\n        var c = i < lenChildren ? this.children[i] : this._pool.pop(); // get old child if have. if not - take from pool.\n\n        if (c) {\n            c.setTexture(chars[i].texture); // check if got one before.\n        }\n        else {\n            c = new core.Sprite(chars[i].texture); // if no create new one.\n        }\n\n        c.position.x = (chars[i].position.x + lineAlignOffsets[chars[i].line]) * scale;\n        c.position.y = chars[i].position.y * scale;\n        c.scale.x = c.scale.y = scale;\n        c.tint = tint;\n\n        if (!c.parent) {\n            this.addChild(c);\n        }\n    }\n\n    // remove unnecessary children.\n    // and put their into the pool.\n    while(this.children.length > lenChars) {\n        var child = this.getChildAt(this.children.length - 1);\n        this._pool.push(child);\n        this.removeChild(child);\n    }\n\n    this.textWidth = maxLineWidth * scale;\n    this.textHeight = (pos.y + data.lineHeight) * scale;\n};\n\n/**\n * Updates the transform of this object\n *\n * @private\n */\nBitmapText.prototype.updateTransform = function () {\n    if (this.dirty) {\n        this.updateText();\n        this.dirty = false;\n    }\n\n    this.displayObjectContainerUpdateTransform();\n};\n\nBitmapText.fonts = {};\n","var core = require('../core');\n\n/**\n * A Text Object will create a line or multiple lines of text. To split a line you can use '\\n' in your text string,\n * or add a wordWrap property set to true and and wordWrapWidth property with a value in the style object.\n *\n * @class\n * @extends Sprite\n * @namespace PIXI\n * @param text {string} The copy that you would like the text to display\n * @param [style] {object} The style parameters\n * @param [style.font] {string} default 'bold 20px Arial' The style and size of the font\n * @param [style.fill='black'] {String|Number} A canvas fillstyle that will be used on the text e.g 'red', '#00FF00'\n * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text\n * @param [style.stroke] {String|Number} A canvas fillstyle that will be used on the text stroke e.g 'blue', '#FCFF00'\n * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke)\n * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used\n * @param [style.wordWrapWidth=100] {number} The width at which text will wrap, it needs wordWrap to be set to true\n * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text\n * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00'\n * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow\n * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow\n */\nfunction Text(text, style) {\n    /**\n     * The canvas element that everything is drawn to\n     *\n     * @member {HTMLCanvasElement}\n     */\n    this.canvas = document.createElement('canvas');\n\n    /**\n     * The canvas 2d context that everything is drawn with\n     * @member {HTMLCanvasElement}\n     */\n    this.context = this.canvas.getContext('2d');\n\n    /**\n     * The resolution of the canvas.\n     * @member {number}\n     */\n    this.resolution = 1;\n\n    core.Sprite.call(this, core.Texture.fromCanvas(this.canvas));\n\n    this.setText(text);\n    this.setStyle(style);\n}\n\n// constructor\nText.prototype = Object.create(core.Sprite.prototype);\nText.prototype.constructor = Text;\nmodule.exports = Text;\n\nObject.defineProperties(Text.prototype, {\n    /**\n     * The width of the Text, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof Text#\n     */\n    width: {\n        get: function () {\n            if (this.dirty) {\n                this.updateText();\n                this.dirty = false;\n            }\n\n            return this.scale.x * this.texture.frame.width;\n        },\n        set: function (value) {\n            this.scale.x = value / this.texture.frame.width;\n            this._width = value;\n        }\n    },\n\n    /**\n     * The height of the Text, setting this will actually modify the scale to achieve the value set\n     *\n     * @member {number}\n     * @memberof Text#\n     */\n    height: {\n        get: function () {\n            if (this.dirty) {\n                this.updateText();\n                this.dirty = false;\n            }\n\n            return  this.scale.y * this.texture.frame.height;\n        },\n        set: function (value) {\n            this.scale.y = value / this.texture.frame.height;\n            this._height = value;\n        }\n    }\n});\n\n/**\n * Set the style of the text\n *\n * @param [style] {object} The style parameters\n * @param [style.font='bold 20pt Arial'] {string} The style and size of the font\n * @param [style.fill='black'] {object} A canvas fillstyle that will be used on the text eg 'red', '#00FF00'\n * @param [style.align='left'] {string} Alignment for multiline text ('left', 'center' or 'right'), does not affect single line text\n * @param [style.stroke='black'] {string} A canvas fillstyle that will be used on the text stroke eg 'blue', '#FCFF00'\n * @param [style.strokeThickness=0] {number} A number that represents the thickness of the stroke. Default is 0 (no stroke)\n * @param [style.wordWrap=false] {boolean} Indicates if word wrap should be used\n * @param [style.wordWrapWidth=100] {number} The width at which text will wrap\n * @param [style.dropShadow=false] {boolean} Set a drop shadow for the text\n * @param [style.dropShadowColor='#000000'] {string} A fill style to be used on the dropshadow e.g 'red', '#00FF00'\n * @param [style.dropShadowAngle=Math.PI/4] {number} Set a angle of the drop shadow\n * @param [style.dropShadowDistance=5] {number} Set a distance of the drop shadow\n */\nText.prototype.setStyle = function (style) {\n    style = style || {};\n    style.font = style.font || 'bold 20pt Arial';\n    style.fill = style.fill || 'black';\n    style.align = style.align || 'left';\n    style.stroke = style.stroke || 'black'; //provide a default, see: https://github.com/GoodBoyDigital/pixi.js/issues/136\n    style.strokeThickness = style.strokeThickness || 0;\n    style.wordWrap = style.wordWrap || false;\n    style.wordWrapWidth = style.wordWrapWidth || 100;\n\n    style.dropShadow = style.dropShadow || false;\n    style.dropShadowAngle = style.dropShadowAngle || Math.PI / 6;\n    style.dropShadowDistance = style.dropShadowDistance || 4;\n    style.dropShadowColor = style.dropShadowColor || 'black';\n\n    this.style = style;\n    this.dirty = true;\n};\n\n/**\n * Set the copy for the text object. To split a line you can use '\\n'.\n *\n * @param text {string} The copy that you would like the text to display\n */\nText.prototype.setText = function (text) {\n    this.text = text.toString() || ' ';\n    this.dirty = true;\n};\n\n/**\n * Renders text and updates it when needed\n *\n * @private\n */\nText.prototype.updateText = function () {\n    this.texture.baseTexture.resolution = this.resolution;\n\n    this.context.font = this.style.font;\n\n    var outputText = this.text;\n\n    // word wrap\n    // preserve original text\n    if (this.style.wordWrap) {\n        outputText = this.wordWrap(this.text);\n    }\n\n    //split text into lines\n    var lines = outputText.split(/(?:\\r\\n|\\r|\\n)/);\n\n    //calculate text width\n    var lineWidths = [];\n    var maxLineWidth = 0;\n    var fontProperties = this.determineFontProperties(this.style.font);\n    for (var i = 0; i < lines.length; i++) {\n        var lineWidth = this.context.measureText(lines[i]).width;\n        lineWidths[i] = lineWidth;\n        maxLineWidth = Math.max(maxLineWidth, lineWidth);\n    }\n\n    var width = maxLineWidth + this.style.strokeThickness;\n    if (this.style.dropShadow) {\n        width += this.style.dropShadowDistance;\n    }\n\n    this.canvas.width = ( width + this.context.lineWidth ) * this.resolution;\n\n    //calculate text height\n    var lineHeight = fontProperties.fontSize + this.style.strokeThickness;\n\n    var height = lineHeight * lines.length;\n    if (this.style.dropShadow) {\n        height += this.style.dropShadowDistance;\n    }\n\n    this.canvas.height = height * this.resolution;\n\n    this.context.scale( this.resolution, this.resolution);\n\n    if (navigator.isCocoonJS) {\n        this.context.clearRect(0, 0, this.canvas.width, this.canvas.height);\n    }\n\n    // used for debugging..\n    //this.context.fillStyle =\"#FF0000\"\n    //this.context.fillRect(0, 0, this.canvas.width,this.canvas.height);\n\n    this.context.font = this.style.font;\n    this.context.strokeStyle = this.style.stroke;\n    this.context.lineWidth = this.style.strokeThickness;\n    this.context.textBaseline = 'alphabetic';\n    //this.context.lineJoin = 'round';\n\n    var linePositionX;\n    var linePositionY;\n\n    if (this.style.dropShadow) {\n        this.context.fillStyle = this.style.dropShadowColor;\n\n        var xShadowOffset = Math.cos(this.style.dropShadowAngle) * this.style.dropShadowDistance;\n        var yShadowOffset = Math.sin(this.style.dropShadowAngle) * this.style.dropShadowDistance;\n\n        for (i = 0; i < lines.length; i++) {\n            linePositionX = this.style.strokeThickness / 2;\n            linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent;\n\n            if (this.style.align === 'right') {\n                linePositionX += maxLineWidth - lineWidths[i];\n            }\n            else if (this.style.align === 'center') {\n                linePositionX += (maxLineWidth - lineWidths[i]) / 2;\n            }\n\n            if (this.style.fill) {\n                this.context.fillText(lines[i], linePositionX + xShadowOffset, linePositionY + yShadowOffset);\n            }\n\n          //  if (dropShadow)\n        }\n    }\n\n    //set canvas text styles\n    this.context.fillStyle = this.style.fill;\n\n    //draw lines line by line\n    for (i = 0; i < lines.length; i++) {\n        linePositionX = this.style.strokeThickness / 2;\n        linePositionY = (this.style.strokeThickness / 2 + i * lineHeight) + fontProperties.ascent;\n\n        if (this.style.align === 'right') {\n            linePositionX += maxLineWidth - lineWidths[i];\n        }\n        else if (this.style.align === 'center') {\n            linePositionX += (maxLineWidth - lineWidths[i]) / 2;\n        }\n\n        if (this.style.stroke && this.style.strokeThickness) {\n            this.context.strokeText(lines[i], linePositionX, linePositionY);\n        }\n\n        if (this.style.fill) {\n            this.context.fillText(lines[i], linePositionX, linePositionY);\n        }\n\n      //  if (dropShadow)\n    }\n\n    this.updateTexture();\n};\n\n/**\n * Updates texture size based on canvas size\n *\n * @private\n */\nText.prototype.updateTexture = function () {\n    this.texture.baseTexture.width = this.canvas.width;\n    this.texture.baseTexture.height = this.canvas.height;\n    this.texture.crop.width = this.texture.frame.width = this.canvas.width;\n    this.texture.crop.height = this.texture.frame.height = this.canvas.height;\n\n    this._width = this.canvas.width;\n    this._height = this.canvas.height;\n\n    // update the dirty base textures\n    this.texture.baseTexture.dirty();\n};\n\n/**\n * Renders the object using the WebGL renderer\n*\n * @param renderer {WebGLRenderer}\n */\nText.prototype.renderWebGL = function (renderer) {\n    if (this.dirty) {\n        this.resolution = renderer.resolution;\n\n        this.updateText();\n        this.dirty = false;\n    }\n\n    core.Sprite.prototype.renderWebGL.call(this, renderer);\n};\n\n/**\n * Renders the object using the Canvas renderer\n*\n * @param renderer {CanvasRenderer}\n */\nText.prototype.renderCanvas = function (renderer) {\n    if (this.dirty) {\n        this.resolution = renderer.resolution;\n\n        this.updateText();\n        this.dirty = false;\n    }\n\n    core.Sprite.prototype.renderCanvas.call(this, renderer);\n};\n\n/**\n * Calculates the ascent, descent and fontSize of a given fontStyle\n*\n * @param fontStyle {object}\n * @private\n */\nText.prototype.determineFontProperties = function (fontStyle) {\n    var properties = Text.fontPropertiesCache[fontStyle];\n\n    if (!properties) {\n        properties = {};\n\n        var canvas = Text.fontPropertiesCanvas;\n        var context = Text.fontPropertiesContext;\n\n        context.font = fontStyle;\n\n        var width = Math.ceil(context.measureText('|Mq').width);\n        var baseline = Math.ceil(context.measureText('M').width);\n        var height = 2 * baseline;\n\n        baseline = baseline * 1.4 | 0;\n\n        canvas.width = width;\n        canvas.height = height;\n\n        context.fillStyle = '#f00';\n        context.fillRect(0, 0, width, height);\n\n        context.font = fontStyle;\n\n        context.textBaseline = 'alphabetic';\n        context.fillStyle = '#000';\n        context.fillText('|MÉq', 0, baseline);\n\n        var imagedata = context.getImageData(0, 0, width, height).data;\n        var pixels = imagedata.length;\n        var line = width * 4;\n\n        var i, j;\n\n        var idx = 0;\n        var stop = false;\n\n        // ascent. scan from top to bottom until we find a non red pixel\n        for (i = 0; i < baseline; i++) {\n            for (j = 0; j < line; j += 4) {\n                if (imagedata[idx + j] !== 255) {\n                    stop = true;\n                    break;\n                }\n            }\n            if (!stop) {\n                idx += line;\n            }\n            else {\n                break;\n            }\n        }\n\n        properties.ascent = baseline - i;\n\n        idx = pixels - line;\n        stop = false;\n\n        // descent. scan from bottom to top until we find a non red pixel\n        for (i = height; i > baseline; i--) {\n            for (j = 0; j < line; j += 4) {\n                if (imagedata[idx + j] !== 255) {\n                    stop = true;\n                    break;\n                }\n            }\n            if (!stop) {\n                idx -= line;\n            }\n            else {\n                break;\n            }\n        }\n\n        properties.descent = i - baseline;\n        //TODO might need a tweak. kind of a temp fix!\n        properties.descent += 6;\n        properties.fontSize = properties.ascent + properties.descent;\n\n        Text.fontPropertiesCache[fontStyle] = properties;\n    }\n\n    return properties;\n};\n\n/**\n * Applies newlines to a string to have it optimally fit into the horizontal\n * bounds set by the Text object's wordWrapWidth property.\n *\n * @param text {string}\n * @private\n */\nText.prototype.wordWrap = function (text) {\n    // Greedy wrapping algorithm that will wrap words as the line grows longer\n    // than its horizontal bounds.\n    var result = '';\n    var lines = text.split('\\n');\n    for (var i = 0; i < lines.length; i++) {\n        var spaceLeft = this.style.wordWrapWidth;\n        var words = lines[i].split(' ');\n        for (var j = 0; j < words.length; j++) {\n            var wordWidth = this.context.measureText(words[j]).width;\n            var wordWidthWithSpace = wordWidth + this.context.measureText(' ').width;\n            if (j === 0 || wordWidthWithSpace > spaceLeft) {\n                // Skip printing the newline if it's the first word of the line that is\n                // greater than the word wrap width.\n                if (j > 0) {\n                    result += '\\n';\n                }\n                result += words[j];\n                spaceLeft = this.style.wordWrapWidth - wordWidth;\n            }\n            else {\n                spaceLeft -= wordWidthWithSpace;\n                result += ' ' + words[j];\n            }\n        }\n\n        if (i < lines.length-1) {\n            result += '\\n';\n        }\n    }\n    return result;\n};\n\n/**\n * Returns the bounds of the Text as a rectangle. The bounds calculation takes the worldTransform into account.\n *\n * @param matrix {Matrix} the transformation matrix of the Text\n * @return {Rectangle} the framing rectangle\n */\nText.prototype.getBounds = function (matrix) {\n    if (this.dirty) {\n        this.updateText();\n        this.dirty = false;\n    }\n\n    return core.Sprite.prototype.getBounds.call(this, matrix);\n};\n\n/**\n * Destroys this text object.\n *\n * @param destroyBaseTexture {boolean} whether to destroy the base texture as well\n */\nText.prototype.destroy = function (destroyBaseTexture) {\n    // make sure to reset the the context and canvas.. dont want this hanging around in memory!\n    this.context = null;\n    this.canvas = null;\n\n    this.texture.destroy(destroyBaseTexture === undefined ? true : destroyBaseTexture);\n};\n\nText.fontPropertiesCache = {};\nText.fontPropertiesCanvas = document.createElement('canvas');\nText.fontPropertiesContext = Text.fontPropertiesCanvas.getContext('2d');\n","/**\n * @file        Main export of the PIXI text library\n * @author      Mat Groves <mat@goodboydigital.com>\n * @copyright   2013-2015 GoodBoyDigital\n * @license     {@link https://github.com/GoodBoyDigital/pixi.js/blob/master/LICENSE|MIT License}\n */\n\n/**\n * @namespace PIXI\n */\nmodule.exports = {\n    Text:       require('./Text'),\n    BitmapText: require('./BitmapText')\n};\n"]} diff --git a/dist/pixi.min.js b/dist/pixi.min.js deleted file mode 100644 index 9f2f3aa..0000000 --- a/dist/pixi.min.js +++ /dev/null @@ -1,8 +0,0 @@ -!function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t();else if("function"==typeof define&&define.amd)define([],t);else{var e;"undefined"!=typeof window?e=window:"undefined"!=typeof global?e=global:"undefined"!=typeof self&&(e=self),e.PIXI=t()}}(function(){return function t(e,i,r){function n(o,a){if(!i[o]){if(!e[o]){var h="function"==typeof require&&require;if(!a&&h)return h(o,!0);if(s)return s(o,!0);var l=new Error("Cannot find module '"+o+"'");throw l.code="MODULE_NOT_FOUND",l}var u=i[o]={exports:{}};e[o][0].call(u.exports,function(t){var i=e[o][1][t];return n(i?i:t)},u,u.exports,t,e,i,r)}return i[o].exports}for(var s="function"==typeof require&&require,o=0;o","Richard Davey "],main:"./src/index.js",homepage:"http://goodboydigital.com/",bugs:"https://github.com/GoodBoyDigital/pixi.js/issues",license:"MIT",repository:{type:"git",url:"https://github.com/GoodBoyDigital/pixi.js.git"},scripts:{test:"gulp test",docs:"./node_modules/.bin/jsdoc -c ./gulp/util/jsdoc.conf.json"},devDependencies:{browserify:"^8.0.2",chai:"^1.10.0",del:"^1.1.0",gulp:"^3.8.10","gulp-jshint":"^1.9.0","gulp-plumber":"^0.6.6","gulp-rename":"^1.2.0","gulp-uglify":"^1.0.2","gulp-util":"^3.0.1","ink-docstrap":"^0.4.12",jsdoc:"^3.3.0-alpha13","jshint-summary":"^0.4.0",karma:"^0.12.28","karma-firefox-launcher":"^0.1.0","karma-mocha":"^0.1.10","karma-spec-reporter":"^0.0.16",mocha:"^2.1.0","require-dir":"^0.1.0","run-sequence":"^1.0.2","vinyl-buffer":"^1.0.0","vinyl-source-stream":"^1.0.0",watchify:"^2.2.1"},dependencies:{"webgl-enabled":"^1.0.2"}}},{}],4:[function(t,e){e.exports={WEBGL_RENDERER:1,CANVAS_RENDERER:2,VERSION:t("../../package.json").version,blendModes:{NORMAL:0,ADD:1,MULTIPLY:2,SCREEN:3,OVERLAY:4,DARKEN:5,LIGHTEN:6,COLOR_DODGE:7,COLOR_BURN:8,HARD_LIGHT:9,SOFT_LIGHT:10,DIFFERENCE:11,EXCLUSION:12,HUE:13,SATURATION:14,COLOR:15,LUMINOSITY:16},scaleModes:{DEFAULT:0,LINEAR:0,NEAREST:1},RETINA_PREFIX:"@2x",defaultRenderOptions:{view:null,resolution:1,antialias:!1,autoResize:!1,transparent:!1,backgroundColor:0,clearBeforeRender:!0,preserveDrawingBuffer:!1},SHAPES:{POLY:0,RECT:1,CIRC:2,ELIP:3,RREC:4}}},{"../../package.json":3}],5:[function(t,e){function i(){this.position=new r.Point,this.scale=new r.Point(1,1),this.pivot=new r.Point(0,0),this.rotation=0,this.alpha=1,this.visible=!0,this.renderable=!1,this.parent=null,this.worldAlpha=1,this.worldTransform=new r.Matrix,this.filterArea=null,this._sr=0,this._cr=1,this._bounds=new r.Rectangle(0,0,1,1),this._currentBounds=null,this._mask=null,this._cacheIsDirty=!1}var r=t("../math");i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{x:{get:function(){return this.position.x},set:function(t){this.position.x=t}},y:{get:function(){return this.position.y},set:function(t){this.position.y=t}},worldVisible:{get:function(){var t=this;do{if(!t.visible)return!1;t=t.parent}while(t);return!0}},mask:{get:function(){return this._mask},set:function(t){this._mask&&(this._mask.isMask=!1),this._mask=t,this._mask&&(this._mask.isMask=!0)}},filters:{get:function(){return this._filters},set:function(t){if(t){for(var e=[],i=0;i=0&&e<=this.children.length)return t.parent&&t.parent.removeChild(t),t.parent=this,this.children.splice(e,0,t),t;throw new Error(t+"addChildAt: The index "+e+" supplied is out of bounds "+this.children.length)}},i.prototype.swapChildren=function(t,e){if(t!==e){var i=this.getChildIndex(t),r=this.getChildIndex(e);if(0>i||0>r)throw new Error("swapChildren: Both the supplied DisplayObjects must be a child of the caller.");this.children[i]=e,this.children[r]=t}},i.prototype.getChildIndex=function(t){var e=this.children.indexOf(t);if(-1===e)throw new Error("The supplied DisplayObject must be a child of the caller");return e},i.prototype.setChildIndex=function(t,e){if(0>e||e>=this.children.length)throw new Error("The supplied index is out of bounds");var i=this.getChildIndex(t);this.children.splice(i,1),this.children.splice(e,0,t)},i.prototype.getChildAt=function(t){if(0>t||t>=this.children.length)throw new Error("getChildAt: Supplied index "+t+" does not exist in the child list, or the supplied DisplayObject must be a child of the caller");return this.children[t]},i.prototype.removeChild=function(t){var e=this.children.indexOf(t);if(-1!==e)return this.removeChildAt(e)},i.prototype.removeChildAt=function(t){var e=this.getChildAt(t);return e.parent=null,this.children.splice(t,1),e},i.prototype.removeChildren=function(t,e){var i=t||0,r="number"==typeof e?e:this.children.length,n=r-i;if(n>0&&r>=n){for(var s=this.children.splice(i,n),o=0;ot;++t)this.children[t].updateTransform()},i.prototype.displayObjectContainerUpdateTransform=i.prototype.updateTransform,i.prototype.getBounds=function(){if(0===this.children.length)return r.Rectangle.EMPTY;for(var t,e,i,n=1/0,s=1/0,o=-1/0,a=-1/0,h=!1,l=0,u=this.children.length;u>l;++l){var c=this.children[l];c.visible&&(h=!0,t=this.children[l].getBounds(),n=ne?o:e,a=a>i?a:i)}return h?(this._bounds.x=n,this._bounds.y=s,this._bounds.width=o-n,this._bounds.height=a-s,this._bounds):r.Rectangle.EMPTY},i.prototype.getLocalBounds=function(){var t=this.worldTransform;this.worldTransform=r.Matrix.IDENTITY;for(var e=0,i=this.children.length;i>e;++e)this.children[e].updateTransform();return this.worldTransform=t,this.getBounds()},i.prototype.renderWebGL=function(t){if(this.visible&&!(this.alpha<=0)){if(this._cacheAsBitmap)return void this._renderCachedSprite(t);var e,i;if(this._mask||this._filters){for(this._filters&&(t.spriteBatch.flush(),t.filterManager.pushFilter(this._filterBlock)),this._mask&&(t.spriteBatch.stop(),t.maskManager.pushMask(this.mask,t),t.spriteBatch.start()),this.texture&&t.spriteBatch.render(this),e=0,i=this.children.length;i>e;e++)this.children[e].renderWebGL(t);t.spriteBatch.stop(),this._mask&&t.maskManager.popMask(this._mask,t),this._filters&&t.filterManager.popFilter(),t.spriteBatch.start()}else for(this.texture&&t.spriteBatch.render(this),e=0,i=this.children.length;i>e;++e)this.children[e].renderWebGL(t)}},i.prototype.renderCanvas=function(t){if(this.visible&&!(this.alpha<=0)){if(this._cacheAsBitmap)return void this._renderCachedSprite(t);this._mask&&t.maskManager.pushMask(this._mask,t);for(var e=0,i=this.children.length;i>e;++e)this.children[e].renderCanvas(t);this._mask&&t.maskManager.popMask(t)}},i.prototype._renderCachedSprite=function(t){this._cachedSprite.worldAlpha=this.worldAlpha,t.gl?this._cachedSprite.renderWebGL(t):this._cachedSprite.renderCanvas(t)},i.prototype._generateCachedSprite=function(){var t=this.getLocalBounds();if(this._cachedSprite)this._cachedSprite.texture.resize(0|t.width,0|t.height);else{var e=new s(renderer,0|t.width,0|t.height);this._cachedSprite=new Sprite(e),this._cachedSprite.worldTransform=this.worldTransform}var i=this._filters;this._filters=null,this._cachedSprite.filters=i,o.tx=-t.x,o.ty=-t.y,this._cachedSprite.texture.render(this,o,!0),this._cachedSprite.anchor.x=-(t.x/t.width),this._cachedSprite.anchor.y=-(t.y/t.height),this._filters=i},i.prototype._destroyCachedSprite=function(){this._cachedSprite&&(this._cachedSprite.destroy(!0,!0),this._cachedSprite=null)}},{"../math":12,"../textures/RenderTexture":43,"./DisplayObject":5}],7:[function(t,e){function i(t){s.call(this),this.anchor=new r.Point,this._texture=null,this._width=0,this._height=0,this.tint=16777215,this.blendMode=h.blendModes.NORMAL,this.shader=null,this.renderable=!0,this.texture=t||n.EMPTY}var r=t("../math"),n=t("../textures/Texture"),s=t("./DisplayObjectContainer"),o=t("../renderers/canvas/utils/CanvasTinter"),a=t("../utils"),h=t("../const");i.prototype.destroy=function(t,e){s.prototype.destroy.call(this),this.anchor=null,t&&this._texture.destroy(e),this._texture=null,this.shader=null},i.prototype=Object.create(s.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{width:{get:function(){return this.scale.x*this.texture.frame.width},set:function(t){this.scale.x=t/this.texture.frame.width,this._width=t}},height:{get:function(){return this.scale.y*this.texture.frame.height},set:function(t){this.scale.y=t/this.texture.frame.height,this._height=t}},texture:{get:function(){return this._texture},set:function(t){this._texture!==t&&(this._texture=t,this.cachedTint=16777215,t&&(t.baseTexture.hasLoaded?this._onTextureUpdate():t.once("update",this._onTextureUpdate.bind(this))))}}}),i.prototype._onTextureUpdate=function(){this._width&&(this.scale.x=this._width/this.texture.frame.width),this._height&&(this.scale.y=this._height/this.texture.frame.height)},i.prototype.getBounds=function(t){var e,i,r,n,s=this.texture.frame.width,o=this.texture.frame.height,a=s*(1-this.anchor.x),h=s*-this.anchor.x,l=o*(1-this.anchor.y),u=o*-this.anchor.y,c=t||this.worldTransform,d=c.a,f=c.b,p=c.c,v=c.d,m=c.tx,g=c.ty;if(0===f&&0===p)0>d&&(d*=-1),0>v&&(v*=-1),e=d*h+m,i=d*a+m,r=v*u+g,n=v*l+g;else{var x=d*h+p*u+m,y=v*u+f*h+g,b=d*a+p*u+m,T=v*u+f*a+g,w=d*a+p*l+m,S=v*l+f*a+g,A=d*h+p*l+m,_=v*l+f*h+g;e=x,e=e>b?b:e,e=e>w?w:e,e=e>A?A:e,r=y,r=r>T?T:r,r=r>S?S:r,r=r>_?_:r,i=x,i=b>i?b:i,i=w>i?w:i,i=A>i?A:i,n=y,n=T>n?T:n,n=S>n?S:n,n=_>n?_:n}var C=this._bounds;return C.x=e,C.width=i-e,C.y=r,C.height=n-r,this._currentBounds=C,C},i.prototype.renderCanvas=function(t){if(!(!this.visible||this.alpha<=0||this.texture.crop.width<=0||this.texture.crop.height<=0)){if(this.blendMode!==t.currentBlendMode&&(t.currentBlendMode=this.blendMode,t.context.globalCompositeOperation=t.blendModes[t.currentBlendMode]),this._mask&&t.maskManager.pushMask(this._mask,t),this.texture.valid){var e=this.texture.baseTexture.resolution/t.resolution;t.context.globalAlpha=this.worldAlpha,t.smoothProperty&&t.scaleMode!==this.texture.baseTexture.scaleMode&&(t.scaleMode=this.texture.baseTexture.scaleMode,t.context[t.smoothProperty]=t.scaleMode===h.scaleModes.LINEAR);var i=(this.texture.trim?this.texture.trim.x:0)-this.anchor.x*this.texture.trim.width,r=(this.texture.trim?this.texture.trim.y:0)-this.anchor.y*this.texture.trim.height;t.roundPixels?(t.context.setTransform(this.worldTransform.a,this.worldTransform.b,this.worldTransform.c,this.worldTransform.d,this.worldTransform.tx*t.resolution|0,this.worldTransform.ty*t.resolution|0),i=0|i,r=0|r):t.context.setTransform(this.worldTransform.a,this.worldTransform.b,this.worldTransform.c,this.worldTransform.d,this.worldTransform.tx*t.resolution,this.worldTransform.ty*t.resolution),16777215!==this.tint?(this.cachedTint!==this.tint&&(this.cachedTint=this.tint,this.tintedTexture=o.getTintedTexture(this,this.tint)),t.context.drawImage(this.tintedTexture,0,0,this.texture.crop.width,this.texture.crop.height,i/e,r/e,this.texture.crop.width/e,this.texture.crop.height/e)):t.context.drawImage(this.texture.baseTexture.source,this.texture.crop.x,this.texture.crop.y,this.texture.crop.width,this.texture.crop.height,i/e,r/e,this.texture.crop.width/e,this.texture.crop.height/e)}for(var n=0,s=this.children.length;s>n;n++)this.children[n].renderCanvas(t);this._mask&&t.maskManager.popMask(t)}},i.fromFrame=function(t){var e=a.TextureCache[t];if(!e)throw new Error('The frameId "'+t+'" does not exist in the texture cache'+this);return new i(e)},i.fromImage=function(t,e,r){return new i(n.fromImage(t,e,r))}},{"../const":4,"../math":12,"../renderers/canvas/utils/CanvasTinter":24,"../textures/Texture":44,"../utils":50,"./DisplayObjectContainer":6}],8:[function(t,e){function i(){r.call(this)}{var r=t("./DisplayObjectContainer");t("../renderers/webgl/utils/WebGLFastSpriteBatch")}i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.updateTransform=function(){this.displayObjectUpdateTransform()},i.prototype.renderWebGL=function(t){!this.visible||this.alpha<=0||!this.children.length||(t.spriteBatch.stop(),t.shaderManager.setShader(t.shaderManager.fastShader),t.fastSpriteBatch.begin(this),t.fastSpriteBatch.render(this),t.spriteBatch.start())},i.prototype.renderCanvas=function(t){if(this.visible&&!(this.alpha<=0)&&this.children.length){var e=t.context,i=this.worldTransform,r=!0;e.globalAlpha=this.worldAlpha,this.displayObjectUpdateTransform();for(var n=0;n=i+r},i.prototype.getBounds=function(){return new r(this.x-this.radius,this.y-this.radius,2*this.radius,2*this.radius)}},{"../../const":4,"./Rectangle":16}],14:[function(t,e){function i(t,e,i,r){this.x=t||0,this.y=e||0,this.width=i||0,this.height=r||0,this.type=n.SHAPES.ELIP}var r=t("./Rectangle"),n=t("../../const");i.prototype.constructor=i,e.exports=i,i.prototype.clone=function(){return new i(this.x,this.y,this.width,this.height)},i.prototype.contains=function(t,e){if(this.width<=0||this.height<=0)return!1;var i=(t-this.x)/this.width,r=(e-this.y)/this.height;return i*=i,r*=r,1>=i+r},i.prototype.getBounds=function(){return new r(this.x-this.width,this.y-this.height,this.width,this.height)}},{"../../const":4,"./Rectangle":16}],15:[function(t,e){function i(t){if(t instanceof Array||(t=Array.prototype.slice.call(arguments)),t[0]instanceof r){for(var e=[],i=0,s=t.length;s>i;i++)e.push(t[i].x,t[i].y);t=e}this.closed=!0,this.points=t,this.type=n.SHAPES.POLY}var r=t("../Point"),n=t("../../const");i.prototype.constructor=i,e.exports=i,i.prototype.clone=function(){return new i(this.points.slice())},i.prototype.contains=function(t,e){for(var i=!1,r=this.points.length/2,n=0,s=r-1;r>n;s=n++){var o=this.points[2*n],a=this.points[2*n+1],h=this.points[2*s],l=this.points[2*s+1],u=a>e!=l>e&&(h-o)*(e-a)/(l-a)+o>t;u&&(i=!i)}return i}},{"../../const":4,"../Point":11}],16:[function(t,e){function i(t,e,i,n){this.x=t||0,this.y=e||0,this.width=i||0,this.height=n||0,this.type=r.SHAPES.RECT}var r=t("../../const");i.prototype.constructor=i,e.exports=i,i.EMPTY=new i(0,0,0,0),i.prototype.clone=function(){return new i(this.x,this.y,this.width,this.height)},i.prototype.contains=function(t,e){return this.width<=0||this.height<=0?!1:t>=this.x&&t<=this.x+this.width&&e>=this.y&&e<=this.y+this.height?!0:!1}},{"../../const":4}],17:[function(t,e){function i(t,e,i,n,s){this.x=t||0,this.y=e||0,this.width=i||0,this.height=n||0,this.radius=s||20,this.type=r.SHAPES.RREC}var r=t("../../const");i.prototype.constructor=i,e.exports=i,i.prototype.clone=function(){return new i(this.x,this.y,this.width,this.height,this.radius)},i.prototype.contains=function(t,e){return this.width<=0||this.height<=0?!1:t>=this.x&&t<=this.x+this.width&&e>=this.y&&e<=this.y+this.height?!0:!1}},{"../../const":4}],18:[function(t,e){function i(){r.call(this),this.renderable=!0,this.fillAlpha=1,this.lineWidth=0,this.lineColor=0,this.graphicsData=[],this.tint=16777215,this.blendMode=c.blendModes.NORMAL,this.currentPath=null,this._webGL={},this.isMask=!1,this.boundsPadding=0,this._localBounds=new u.Rectangle(0,0,1,1),this.dirty=!0,this.glDirty=!1,this.cachedSpriteDirty=!1}var r=t("../display/DisplayObjectContainer"),n=t("../display/Sprite"),s=t("../textures/Texture"),o=t("../renderers/canvas/utils/CanvasBuffer"),a=t("../renderers/canvas/utils/CanvasGraphics"),h=t("../renderers/webgl/utils/WebGLGraphics"),l=t("./GraphicsData"),u=t("../math"),c=t("../const");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{cacheAsBitmap:{get:function(){return this._cacheAsBitmap},set:function(t){this._cacheAsBitmap=t,this._cacheAsBitmap?this._generateCachedSprite():(this.destroyCachedSprite(),this.dirty=!0)}}}),l.prototype.clone=function(){var t=new i;t.renderable=this.renderable,t.fillAlpha=this.fillAlpha,t.lineWidth=this.lineWidth,t.lineColor=this.lineColor,t.tint=this.tint,t.blendMode=this.blendMode,t.isMask=this.isMask,t.boundsPadding=this.boundsPadding,t.dirty=this.dirty,t.glDirty=this.glDirty,t.cachedSpriteDirty=this.cachedSpriteDirty;for(var e=0;e=c;++c)u=c/o,n=h+(t-h)*u,s=l+(e-l)*u,a.push(n+(t+(i-t)*u-n)*u,s+(e+(r-e)*u-s)*u);return this.dirty=!0,this},i.prototype.bezierCurveTo=function(t,e,i,r,n,s){this.currentPath?0===this.currentPath.shape.points.length&&(this.currentPath.shape.points=[0,0]):this.moveTo(0,0);for(var o,a,h,l,u,c=20,d=this.currentPath.shape.points,f=d[d.length-2],p=d[d.length-1],v=0,m=1;c>=m;++m)v=m/c,o=1-v,a=o*o,h=a*o,l=v*v,u=l*v,d.push(h*f+3*a*v*t+3*o*l*i+u*n,h*p+3*a*v*e+3*o*l*r+u*s);return this.dirty=!0,this},i.prototype.arcTo=function(t,e,i,r,n){this.currentPath?0===this.currentPath.shape.points.length&&this.currentPath.shape.points.push(t,e):this.moveTo(t,e);var s=this.currentPath.shape.points,o=s[s.length-2],a=s[s.length-1],h=a-e,l=o-t,u=r-e,c=i-t,d=Math.abs(h*c-l*u);if(1e-8>d||0===n)(s[s.length-2]!==t||s[s.length-1]!==e)&&s.push(t,e);else{var f=h*h+l*l,p=u*u+c*c,v=h*u+l*c,m=n*Math.sqrt(f)/d,g=n*Math.sqrt(p)/d,x=m*v/f,y=g*v/p,b=m*c+g*l,T=m*u+g*h,w=l*(g+x),S=h*(g+x),A=c*(m+y),_=u*(m+y),C=Math.atan2(S-T,w-b),E=Math.atan2(_-T,A-b);this.arc(b+t,T+e,n,C,E,l*u>c*h)}return this.dirty=!0,this},i.prototype.arc=function(t,e,i,r,n,s){var o,a=t+Math.cos(r)*i,h=e+Math.sin(r)*i;if(this.currentPath?(o=this.currentPath.shape.points,0===o.length?o.push(a,h):(o[o.length-2]!==a||o[o.length-1]!==h)&&o.push(a,h)):(this.moveTo(a,h),o=this.currentPath.shape.points),r===n)return this;!s&&r>=n?n+=2*Math.PI:s&&n>=r&&(r+=2*Math.PI);var l=s?-1*(r-n):n-r,u=Math.abs(l)/(2*Math.PI)*40;if(0===l)return this;for(var c=l/(2*u),d=2*c,f=Math.cos(c),p=Math.sin(c),v=u-1,m=v%1/v,g=0;v>=g;++g){var x=g+m*g,y=c+r+d*x,b=Math.cos(y),T=-Math.sin(y);o.push((f*b+p*T)*i+t,(f*-T+p*b)*i+e)}return this.dirty=!0,this},i.prototype.beginFill=function(t,e){return this.filling=!0,this.fillColor=t||0,this.fillAlpha=void 0===e?1:e,this.currentPath&&this.currentPath.shape.points.length<=2&&(this.currentPath.fill=this.filling,this.currentPath.fillColor=this.fillColor,this.currentPath.fillAlpha=this.fillAlpha),this},i.prototype.endFill=function(){return this.filling=!1,this.fillColor=null,this.fillAlpha=1,this},i.prototype.drawRect=function(t,e,i,r){return this.drawShape(new u.Rectangle(t,e,i,r)),this},i.prototype.drawRoundedRect=function(t,e,i,r,n){return this.drawShape(new u.RoundedRectangle(t,e,i,r,n)),this},i.prototype.drawCircle=function(t,e,i){return this.drawShape(new u.Circle(t,e,i)),this},i.prototype.drawEllipse=function(t,e,i,r){return this.drawShape(new u.Ellipse(t,e,i,r)),this},i.prototype.drawPolygon=function(t){return t instanceof Array||(t=Array.prototype.slice.call(arguments)),this.drawShape(new u.Polygon(t)),this},i.prototype.clear=function(){return this.lineWidth=0,this.filling=!1,this.dirty=!0,this.clearDirty=!0,this.graphicsData=[],this},i.prototype.generateTexture=function(t,e){t=t||1;var i=this.getBounds(),r=new o(i.width*t,i.height*t),n=s.fromCanvas(r.canvas,e);return n.baseTexture.resolution=t,r.context.scale(t,t),r.context.translate(-i.x,-i.y),a.renderGraphics(this,r.context),n},i.prototype.renderWebGL=function(t){if(this.visible&&!(this.alpha<=0)&&this.isMask!==!0){if(this._cacheAsBitmap)return(this.dirty||this.cachedSpriteDirty)&&(this._generateCachedSprite(),this.updateCachedSpriteTexture(),this.cachedSpriteDirty=!1,this.dirty=!1),this._cachedSprite.worldAlpha=this.worldAlpha,void n.prototype.renderWebGL.call(this._cachedSprite,t);if(t.spriteBatch.stop(),t.blendModeManager.setBlendMode(this.blendMode),this._mask&&t.maskManager.pushMask(this._mask,t),this._filters&&t.filterManager.pushFilter(this._filterBlock),this.blendMode!==t.spriteBatch.currentBlendMode){t.spriteBatch.currentBlendMode=this.blendMode;var e=t.blendModes[t.spriteBatch.currentBlendMode];t.spriteBatch.gl.blendFunc(e[0],e[1])}if(this.glDirty&&(this.dirty=!0,this.glDirty=!1),h.renderGraphics(this,t),this.children.length){t.spriteBatch.start();for(var i=0,r=this.children.length;r>i;++i)this.children[i].renderWebGL(t);t.spriteBatch.stop()}this._filters&&t.filterManager.popFilter(),this._mask&&t.maskManager.popMask(this.mask,t),t.drawCount++,t.spriteBatch.start()}},i.prototype.renderCanvas=function(t){if(this.visible&&!(this.alpha<=0)&&this.isMask!==!0){if(this._cacheAsBitmap)return(this.dirty||this.cachedSpriteDirty)&&(this._generateCachedSprite(),this.updateCachedSpriteTexture(),this.cachedSpriteDirty=!1,this.dirty=!1),this._cachedSprite.alpha=this.alpha,void n.prototype.renderCanvas.call(this._cachedSprite,t);var e=t.context,i=this.worldTransform;this.blendMode!==t.currentBlendMode&&(t.currentBlendMode=this.blendMode,e.globalCompositeOperation=t.blendModes[t.currentBlendMode]),this._mask&&t.maskManager.pushMask(this._mask,t);var r=t.resolution;e.setTransform(i.a*r,i.b*r,i.c*r,i.d*r,i.tx*r,i.ty*r),a.renderGraphics(this,e);for(var s=0,o=this.children.length;o>s;++s)this.children[s].renderCanvas(t);this._mask&&t.maskManager.popMask(t)}},i.prototype.getBounds=function(t){if(this.isMask)return u.Rectangle.EMPTY;this.dirty&&(this.updateLocalBounds(),this.glDirty=!0,this.cachedSpriteDirty=!0,this.dirty=!1);var e=this._localBounds,i=e.x,r=e.width+e.x,n=e.y,s=e.height+e.y,o=t||this.worldTransform,a=o.a,h=o.b,l=o.c,c=o.d,d=o.tx,f=o.ty,p=a*r+l*s+d,v=c*s+h*r+f,m=a*i+l*s+d,g=c*s+h*i+f,x=a*i+l*n+d,y=c*n+h*i+f,b=a*r+l*n+d,T=c*n+h*r+f,w=p,S=v,A=p,_=v;return A=A>m?m:A,A=A>x?x:A,A=A>b?b:A,_=_>g?g:_,_=_>y?y:_,_=_>T?T:_,w=m>w?m:w,w=x>w?x:w,w=b>w?b:w,S=g>S?g:S,S=y>S?y:S,S=T>S?T:S,this._bounds.x=A,this._bounds.width=w-A,this._bounds.y=_,this._bounds.height=S-_,this._bounds},i.prototype.updateLocalBounds=function(){var t=1/0,e=-1/0,i=1/0,r=-1/0;if(this.graphicsData.length)for(var n,s,o,a,h,l,u=0;uo?o:t,e=o+h>e?o+h:e,i=i>a?a:i,r=a+l>r?a+l:r;else if(f===c.SHAPES.CIRC)o=n.x,a=n.y,h=n.radius+p/2,l=n.radius+p/2,t=t>o-h?o-h:t,e=o+h>e?o+h:e,i=i>a-l?a-l:i,r=a+l>r?a+l:r;else if(f===c.SHAPES.ELIP)o=n.x,a=n.y,h=n.width+p/2,l=n.height+p/2,t=t>o-h?o-h:t,e=o+h>e?o+h:e,i=i>a-l?a-l:i,r=a+l>r?a+l:r;else{s=n.points;for(var v=0;vo-p?o-p:t,e=o+p>e?o+p:e,i=i>a-p?a-p:i,r=a+p>r?a+p:r}}else t=0,e=0,i=0,r=0;var m=this.boundsPadding;this._localBounds.x=t-m,this._localBounds.width=e-t+2*m,this._localBounds.y=i-m,this._localBounds.height=r-i+2*m -},i.prototype._generateCachedSprite=function(){var t=this.getLocalBounds();if(this._cachedSprite)this._cachedSprite.buffer.resize(t.width,t.height);else{var e=new o(t.width,t.height),i=s.fromCanvas(e.canvas);this._cachedSprite=new n(i),this._cachedSprite.buffer=e,this._cachedSprite.worldTransform=this.worldTransform}this._cachedSprite.anchor.x=-(t.x/t.width),this._cachedSprite.anchor.y=-(t.y/t.height),this._cachedSprite.buffer.context.translate(-t.x,-t.y),this.worldAlpha=1,a.renderGraphics(this,this._cachedSprite.buffer.context),this._cachedSprite.alpha=this.alpha},i.prototype.updateCachedSpriteTexture=function(){var t=this._cachedSprite,e=t.texture,i=t.buffer.canvas;e.baseTexture.width=i.width,e.baseTexture.height=i.height,e.crop.width=e.frame.width=i.width,e.crop.height=e.frame.height=i.height,t._width=i.width,t._height=i.height,e.baseTexture.dirty()},i.prototype.destroyCachedSprite=function(){this._cachedSprite.texture.destroy(!0),this._cachedSprite=null},i.prototype.drawShape=function(t){this.currentPath&&this.currentPath.shape.points.length<=2&&this.graphicsData.pop(),this.currentPath=null;var e=new l(this.lineWidth,this.lineColor,this.lineAlpha,this.fillColor,this.fillAlpha,this.filling,t);return this.graphicsData.push(e),e.type===c.SHAPES.POLY&&(e.shape.closed=this.filling,this.currentPath=e),this.dirty=!0,e}},{"../const":4,"../display/DisplayObjectContainer":6,"../display/Sprite":7,"../math":12,"../renderers/canvas/utils/CanvasBuffer":21,"../renderers/canvas/utils/CanvasGraphics":22,"../renderers/webgl/utils/WebGLGraphics":39,"../textures/Texture":44,"./GraphicsData":19}],19:[function(t,e){function i(t,e,i,r,n,s,o){this.lineWidth=t,this.lineColor=e,this.lineAlpha=i,this._lineTint=e,this.fillColor=r,this.fillAlpha=n,this._fillTint=r,this.fill=s,this.shape=o,this.type=o.type}i.prototype.constructor=i,e.exports=i,i.prototype.clone=function(){return new i(this.lineWidth,this.lineColor,this.lineAlpha,this.fillColor,this.fillAlpha,this.fill,this.shape)}},{}],20:[function(t,e){function i(t,e,i){if(n.sayHello("Canvas"),i)for(var o in s.defaultRenderOptions)"undefined"==typeof i[o]&&(i[o]=s.defaultRenderOptions[o]);else i=s.defaultRenderOptions;this.type=s.CANVAS_RENDERER,this.resolution=i.resolution,this.clearBeforeRender=i.clearBeforeRender,this._backgroundColor=0,this._backgroundColorString="#000000",this.backgroundColor=i.backgroundColor||this._backgroundColor,this.transparent=i.transparent,this.autoResize=i.autoResize||!1,this.width=t||800,this.height=e||600,this.width*=this.resolution,this.height*=this.resolution,this.view=i.view||document.createElement("canvas"),this.context=this.view.getContext("2d",{alpha:this.transparent}),this.refresh=!0,this.view.width=this.width*this.resolution,this.view.height=this.height*this.resolution,this.count=0,this.maskManager=new r,this.roundPixels=!1,this.scaleMode=null,this.smoothProperty=null,this.context.imageSmoothingEnabled?this.smoothProperty="imageSmoothingEnabled":this.context.webkitImageSmoothingEnabled?this.smoothProperty="webkitImageSmoothingEnabled":this.context.mozImageSmoothingEnabled?this.smoothProperty="mozImageSmoothingEnabled":this.context.oImageSmoothingEnabled?this.smoothProperty="oImageSmoothingEnabled":this.context.msImageSmoothingEnabled&&(this.smoothProperty="msImageSmoothingEnabled"),this.currentBlendMode=s.blendModes.NORMAL,this.blendModes=null,this._mapBlendModes(),this.resize(t,e)}var r=t("./utils/CanvasMaskManager"),n=t("../../utils"),s=t("../../const");i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{backgroundColor:{get:function(){return this._backgroundColor},set:function(t){this._backgroundColor=t,this._backgroundColorString=n.hex2string(t)}}}),i.prototype.render=function(t){t.updateTransform(),this.context.setTransform(1,0,0,1,0,0),this.context.globalAlpha=1,this.currentBlendMode=s.blendModes.NORMAL,this.context.globalCompositeOperation=this.blendModes[s.blendModes.NORMAL],navigator.isCocoonJS&&this.view.screencanvas&&(this.context.fillStyle="black",this.context.clear()),this.clearBeforeRender&&(this.transparent?this.context.clearRect(0,0,this.width,this.height):(this.context.fillStyle=this._backgroundColorString,this.context.fillRect(0,0,this.width,this.height))),this.renderDisplayObject(t)},i.prototype.destroy=function(t){t&&this.view.parent&&this.view.parent.removeChild(this.view),this.view=null,this.context=null,this.maskManager=null},i.prototype.resize=function(t,e){this.width=t*this.resolution,this.height=e*this.resolution,this.view.width=this.width,this.view.height=this.height,this.autoResize&&(this.view.style.width=this.width/this.resolution+"px",this.view.style.height=this.height/this.resolution+"px")},i.prototype.renderDisplayObject=function(t){t.renderCanvas(this)},i.prototype._mapBlendModes=function(){this.blendModes||(this.blendModes={},n.canUseNewCanvasBlendModes()?(this.blendModes[s.blendModes.NORMAL]="source-over",this.blendModes[s.blendModes.ADD]="lighter",this.blendModes[s.blendModes.MULTIPLY]="multiply",this.blendModes[s.blendModes.SCREEN]="screen",this.blendModes[s.blendModes.OVERLAY]="overlay",this.blendModes[s.blendModes.DARKEN]="darken",this.blendModes[s.blendModes.LIGHTEN]="lighten",this.blendModes[s.blendModes.COLOR_DODGE]="color-dodge",this.blendModes[s.blendModes.COLOR_BURN]="color-burn",this.blendModes[s.blendModes.HARD_LIGHT]="hard-light",this.blendModes[s.blendModes.SOFT_LIGHT]="soft-light",this.blendModes[s.blendModes.DIFFERENCE]="difference",this.blendModes[s.blendModes.EXCLUSION]="exclusion",this.blendModes[s.blendModes.HUE]="hue",this.blendModes[s.blendModes.SATURATION]="saturation",this.blendModes[s.blendModes.COLOR]="color",this.blendModes[s.blendModes.LUMINOSITY]="luminosity"):(this.blendModes[s.blendModes.NORMAL]="source-over",this.blendModes[s.blendModes.ADD]="lighter",this.blendModes[s.blendModes.MULTIPLY]="source-over",this.blendModes[s.blendModes.SCREEN]="source-over",this.blendModes[s.blendModes.OVERLAY]="source-over",this.blendModes[s.blendModes.DARKEN]="source-over",this.blendModes[s.blendModes.LIGHTEN]="source-over",this.blendModes[s.blendModes.COLOR_DODGE]="source-over",this.blendModes[s.blendModes.COLOR_BURN]="source-over",this.blendModes[s.blendModes.HARD_LIGHT]="source-over",this.blendModes[s.blendModes.SOFT_LIGHT]="source-over",this.blendModes[s.blendModes.DIFFERENCE]="source-over",this.blendModes[s.blendModes.EXCLUSION]="source-over",this.blendModes[s.blendModes.HUE]="source-over",this.blendModes[s.blendModes.SATURATION]="source-over",this.blendModes[s.blendModes.COLOR]="source-over",this.blendModes[s.blendModes.LUMINOSITY]="source-over"))}},{"../../const":4,"../../utils":50,"./utils/CanvasMaskManager":23}],21:[function(t,e){function i(t,e){this.canvas=document.createElement("canvas"),this.context=this.canvas.getContext("2d"),this.canvas.width=t,this.canvas.height=e}i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{width:{get:function(){return this.canvas.width},set:function(t){this.canvas.width=t}},height:{get:function(){return this.canvas.height},set:function(t){this.canvas.height=t}}}),i.prototype.clear=function(){this.context.setTransform(1,0,0,1,0,0),this.context.clearRect(0,0,this.canvas.width,this.canvas.height)},i.prototype.resize=function(t,e){this.canvas.width=t,this.canvas.height=e}},{}],22:[function(t,e){var i=t("../../../const"),r=e.exports={};r.renderGraphics=function(t,e){var r=t.worldAlpha;t.dirty&&(this.updateGraphicsTint(t),t.dirty=!1);for(var n=0;nE?E:C,e.beginPath(),e.moveTo(w,S+C),e.lineTo(w,S+_-C),e.quadraticCurveTo(w,S+_,w+C,S+_),e.lineTo(w+A-C,S+_),e.quadraticCurveTo(w+A,S+_,w+A,S+_-C),e.lineTo(w+A,S+C),e.quadraticCurveTo(w+A,S,w+A-C,S),e.lineTo(w+C,S),e.quadraticCurveTo(w,S,w,S+C),e.closePath(),(s.fillColor||0===s.fillColor)&&(e.globalAlpha=s.fillAlpha*r,e.fillStyle="#"+("00000"+(0|a).toString(16)).substr(-6),e.fill()),s.lineWidth&&(e.globalAlpha=s.lineAlpha*r,e.strokeStyle="#"+("00000"+(0|h).toString(16)).substr(-6),e.stroke())}}},r.renderGraphicsMask=function(t,e){var r=t.graphicsData.length;if(0!==r){e.beginPath();for(var n=0;r>n;n++){var s=t.graphicsData[n],o=s.shape;if(s.type===i.SHAPES.POLY){var a=o.points;e.moveTo(a[0],a[1]);for(var h=1;h_?_:A,e.moveTo(b,T+A),e.lineTo(b,T+S-A),e.quadraticCurveTo(b,T+S,b+A,T+S),e.lineTo(b+w-A,T+S),e.quadraticCurveTo(b+w,T+S,b+w,T+S-A),e.lineTo(b+w,T+A),e.quadraticCurveTo(b+w,T,b+w-A,T),e.lineTo(b+A,T),e.quadraticCurveTo(b,T,b,T+A),e.closePath()}}}},r.updateGraphicsTint=function(t){if(16777215!==t.tint)for(var e=(t.tint>>16&255)/255,i=(t.tint>>8&255)/255,r=(255&t.tint)/255,n=0;n>16&255)/255*e*255<<16)+((o>>8&255)/255*i*255<<8)+(255&o)/255*r*255,s._lineTint=((a>>16&255)/255*e*255<<16)+((a>>8&255)/255*i*255<<8)+(255&a)/255*r*255}}},{"../../../const":4}],23:[function(t,e){function i(){}var r=t("./CanvasGraphics");i.prototype.constructor=i,e.exports=i,i.prototype.pushMask=function(t,e){e.context.save();var i=t.alpha,n=t.worldTransform,s=e.resolution;e.context.setTransform(n.a*s,n.b*s,n.c*s,n.d*s,n.tx*s,n.ty*s),r.renderGraphicsMask(t,e.context),e.context.clip(),t.worldAlpha=i},i.prototype.popMask=function(t){t.context.restore()}},{"./CanvasGraphics":22}],24:[function(t,e){var i=t("../../../utils"),r=e.exports={};r.getTintedTexture=function(t,e){var i=t.texture;e=r.roundColor(e);var n="#"+("00000"+(0|e).toString(16)).substr(-6);if(i.tintCache=i.tintCache||{},i.tintCache[n])return i.tintCache[n];var s=r.canvas||document.createElement("canvas");if(r.tintMethod(i,e,s),r.convertTintToImage){var o=new Image;o.src=s.toDataURL(),i.tintCache[n]=o}else i.tintCache[n]=s,r.canvas=null;return s},r.tintWithMultiply=function(t,e,i){var r=i.getContext("2d"),n=t.crop;i.width=n.width,i.height=n.height,r.fillStyle="#"+("00000"+(0|e).toString(16)).substr(-6),r.fillRect(0,0,n.width,n.height),r.globalCompositeOperation="multiply",r.drawImage(t.baseTexture.source,n.x,n.y,n.width,n.height,0,0,n.width,n.height),r.globalCompositeOperation="destination-atop",r.drawImage(t.baseTexture.source,n.x,n.y,n.width,n.height,0,0,n.width,n.height)},r.tintWithOverlay=function(t,e,i){var r=i.getContext("2d"),n=t.crop;i.width=n.width,i.height=n.height,r.globalCompositeOperation="copy",r.fillStyle="#"+("00000"+(0|e).toString(16)).substr(-6),r.fillRect(0,0,n.width,n.height),r.globalCompositeOperation="destination-atop",r.drawImage(t.baseTexture.source,n.x,n.y,n.width,n.height,0,0,n.width,n.height)},r.tintWithPerPixel=function(t,e,r){var n=r.getContext("2d"),s=t.crop;r.width=s.width,r.height=s.height,n.globalCompositeOperation="copy",n.drawImage(t.baseTexture.source,s.x,s.y,s.width,s.height,0,0,s.width,s.height);for(var o=i.hex2rgb(e),a=o[0],h=o[1],l=o[2],u=n.getImageData(0,0,s.width,s.height),c=u.data,d=0;dthis.width&&(a.width=this.width-l),u+a.height>this.height&&(a.height=this.height-u),a.width<0&&(a.width=0),a.height<0&&(a.height=0),e.bindFramebuffer(e.FRAMEBUFFER,o.frameBuffer),e.viewport(0,0,a.width,a.height),i.x=a.width/2,i.y=-a.height/2,r.x=-a.x,r.y=-a.y,e.colorMask(!0,!0,!0,!0),e.clearColor(0,0,0,0),e.clear(e.COLOR_BUFFER_BIT),t._glFilterTexture=o},i.prototype.popFilter=function(){var t=this.renderer.gl,e=this.filterStack.pop(),i=e._filterArea,r=e._glFilterTexture,s=this.renderer.projection,o=this.renderer.offset;if(e.filterPasses.length>1){t.viewport(0,0,i.width,i.height),t.bindBuffer(t.ARRAY_BUFFER,this.vertexBuffer),this.vertexArray[0]=0,this.vertexArray[1]=i.height,this.vertexArray[2]=i.width,this.vertexArray[3]=i.height,this.vertexArray[4]=0,this.vertexArray[5]=0,this.vertexArray[6]=i.width,this.vertexArray[7]=0,t.bufferSubData(t.ARRAY_BUFFER,0,this.vertexArray),t.bindBuffer(t.ARRAY_BUFFER,this.uvBuffer),this.uvArray[2]=i.width/this.width,this.uvArray[5]=i.height/this.height,this.uvArray[6]=i.width/this.width,this.uvArray[7]=i.height/this.height,t.bufferSubData(t.ARRAY_BUFFER,0,this.uvArray);var a=r,h=this.texturePool.pop();h||(h=new n(this.renderer.gl,this.width,this.height)),h.resize(this.width,this.height),t.bindFramebuffer(t.FRAMEBUFFER,h.frameBuffer),t.clear(t.COLOR_BUFFER_BIT),t.disable(t.BLEND);for(var l=0;li;++i)s._array[2*i]=a[i].x,s._array[2*i+1]=a[i].y;t.uniform2fv(o,s._array);break;case"v3v":for(s._array||(s._array=new Float32Array(3*a.length)),i=0,n=a.length;n>i;++i)s._array[3*i]=a[i].x,s._array[3*i+1]=a[i].y,s._array[3*i+2]=a[i].z;t.uniform3fv(o,s._array);break;case"v4v":for(s._array||(s._array=new Float32Array(4*a.length)),i=0,n=a.length;n>i;++i)s._array[4*i]=a[i].x,s._array[4*i+1]=a[i].y,s._array[4*i+2]=a[i].z,s._array[4*i+3]=a[i].w;t.uniform4fv(o,s._array);break;case"t":case"sampler2D":if(!s.value||!s.value.baseTexture||!s.value.baseTexture.hasLoaded)break;t.activeTexture(t["TEXTURE"+this.textureCount]),t.bindTexture(t.TEXTURE_2D,s.value.baseTexture._glTextures[t.id]),t.uniform1i(s._location,this.textureCount),this.textureCount++,s._init||(this.initSampler2D(s),s._init=!0);break;default:window.console.warn("Pixi.js Shader Warning: Unknown uniform type: "+s.type)}}},i.prototype.initSampler2D=function(t){var e=this.gl;if(t.textureData){var i=t.textureData,r=i.magFilter?i.magFilter:e.LINEAR,n=i.minFilter?i.minFilter:e.LINEAR,s=i.wrapS?i.wrapS:e.CLAMP_TO_EDGE,o=i.wrapT?i.wrapT:e.CLAMP_TO_EDGE,a=i.luminance?e.LUMINANCE:e.RGBA;if(i.repeat&&(s=e.REPEAT,o=e.REPEAT),e.pixelStorei(e.UNPACK_FLIP_Y_WEBGL,!!i.flipY),i.width){var h=i.width?i.width:512,l=i.height?i.height:2,u=i.border?i.border:0;e.texImage2D(e.TEXTURE_2D,0,a,h,l,u,a,e.UNSIGNED_BYTE,null)}else e.texImage2D(e.TEXTURE_2D,0,a,e.RGBA,e.UNSIGNED_BYTE,t.value.baseTexture.source);e.texParameteri(e.TEXTURE_2D,e.TEXTURE_MAG_FILTER,r),e.texParameteri(e.TEXTURE_2D,e.TEXTURE_MIN_FILTER,n),e.texParameteri(e.TEXTURE_2D,e.TEXTURE_WRAP_S,s),e.texParameteri(e.TEXTURE_2D,e.TEXTURE_WRAP_T,o)}},i.prototype.destroy=function(){this.gl.deleteProgram(this.program),this.gl=null,this.uniforms=null,this.attributes=null,this.vertexSrc=null,this.fragmentSrc=null},i.prototype._glCompile=function(t,e){var i=this.gl.createShader(t);return Array.isArray(e),this.gl.shaderSource(i,e),this.gl.compileShader(i),this.gl.getShaderParameter(i,this.gl.COMPILE_STATUS)?i:(window.console.log(this.gl.getShaderInfoLog(i)),null)}},{"../../../utils":50}],36:[function(t,e){function i(t){r.call(this,t,["attribute vec2 aVertexPosition;","attribute vec2 aTextureCoord;","uniform mat3 translationMatrix;","uniform vec2 projectionVector;","uniform vec2 offsetVector;","varying vec2 vTextureCoord;","void main(void) {"," vec3 v = translationMatrix * vec3(aVertexPosition , 1.0);"," v -= offsetVector.xyx;"," gl_Position = vec4( v.x / projectionVector.x -1.0, v.y / -projectionVector.y + 1.0 , 0.0, 1.0);"," vTextureCoord = aTextureCoord;","}"].join("\n"),["precision mediump float;","uniform float alpha;","uniform sampler2D uSampler;","varying vec2 vTextureCoord;","void main(void) {"," gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * alpha;","}"].join("\n"),{alpha:{type:"1f",value:0},translationMatrix:{type:"mat3",value:new Float32Array(9)}})}var r=t("./Shader");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i},{"./Shader":35}],37:[function(t,e){function i(t,e,i,n){this.gl=t,this.frameBuffer=t.createFramebuffer(),this.texture=t.createTexture(),n=n||r.scaleModes.DEFAULT,t.bindTexture(t.TEXTURE_2D,this.texture),t.texParameteri(t.TEXTURE_2D,t.TEXTURE_MAG_FILTER,n===r.scaleModes.LINEAR?t.LINEAR:t.NEAREST),t.texParameteri(t.TEXTURE_2D,t.TEXTURE_MIN_FILTER,n===r.scaleModes.LINEAR?t.LINEAR:t.NEAREST),t.texParameteri(t.TEXTURE_2D,t.TEXTURE_WRAP_S,t.CLAMP_TO_EDGE),t.texParameteri(t.TEXTURE_2D,t.TEXTURE_WRAP_T,t.CLAMP_TO_EDGE),t.bindFramebuffer(t.FRAMEBUFFER,this.frameBuffer),t.bindFramebuffer(t.FRAMEBUFFER,this.frameBuffer),t.framebufferTexture2D(t.FRAMEBUFFER,t.COLOR_ATTACHMENT0,t.TEXTURE_2D,this.texture,0),this.renderBuffer=t.createRenderbuffer(),t.bindRenderbuffer(t.RENDERBUFFER,this.renderBuffer),t.framebufferRenderbuffer(t.FRAMEBUFFER,t.DEPTH_STENCIL_ATTACHMENT,t.RENDERBUFFER,this.renderBuffer),t.bindRenderbuffer(t.RENDERBUFFER,null),this.resize(e,i)}var r=t("../../../const");i.prototype.constructor=i,e.exports=i,i.prototype.clear=function(){var t=this.gl;t.clearColor(0,0,0,0),t.clear(t.COLOR_BUFFER_BIT)},i.prototype.resize=function(t,e){if(this.width!==t||this.height!==e){this.width=t,this.height=e;var i=this.gl;i.bindTexture(i.TEXTURE_2D,this.texture),i.texImage2D(i.TEXTURE_2D,0,i.RGBA,t,e,0,i.RGBA,i.UNSIGNED_BYTE,null),i.bindRenderbuffer(i.RENDERBUFFER,this.renderBuffer),i.renderbufferStorage(i.RENDERBUFFER,i.DEPTH_STENCIL,t,e),i.bindRenderbuffer(i.RENDERBUFFER,null)}},i.prototype.destroy=function(){var t=this.gl;t.deleteFramebuffer(this.frameBuffer),t.deleteTexture(this.texture),this.frameBuffer=null,this.texture=null}},{"../../../const":4}],38:[function(t,e){function i(t){this.renderer=t,this.vertSize=10,this.maxSize=6e3,this.size=this.maxSize;var e=4*this.size*this.vertSize,i=6*this.maxSize;this.vertices=new Float32Array(e),this.indices=new Uint16Array(i),this.vertexBuffer=null,this.indexBuffer=null,this.lastIndexCount=0;for(var r=0,n=0;i>r;r+=6,n+=4)this.indices[r+0]=n+0,this.indices[r+1]=n+1,this.indices[r+2]=n+2,this.indices[r+3]=n+0,this.indices[r+4]=n+2,this.indices[r+5]=n+3;this.drawing=!1,this.currentBatchSize=0,this.currentBaseTexture=null,this.currentBlendMode=0,this.shader=null,this.matrix=null;var s=this;this.renderer.on("context",function(){s.setupContext()})}i.prototype.constructor=i,e.exports=i,i.prototype.setupContext=function(){var t=this.renderer.gl;this.vertexBuffer=t.createBuffer(),this.indexBuffer=t.createBuffer(),t.bindBuffer(t.ELEMENT_ARRAY_BUFFER,this.indexBuffer),t.bufferData(t.ELEMENT_ARRAY_BUFFER,this.indices,t.STATIC_DRAW),t.bindBuffer(t.ARRAY_BUFFER,this.vertexBuffer),t.bufferData(t.ARRAY_BUFFER,this.vertices,t.DYNAMIC_DRAW)},i.prototype.begin=function(t){this.shader=this.renderer.shaderManager.fastShader,this.matrix=t.worldTransform.toArray(!0),this.start()},i.prototype.end=function(){this.flush()},i.prototype.render=function(t){var e=t.children,i=e[0];if(i.texture._uvs){this.currentBaseTexture=i.texture.baseTexture,i.blendMode!==this.renderer.blendModeManager.currentBlendMode&&(this.flush(),this.renderer.blendModeManager.setBlendMode(i.blendMode));for(var r=0,n=e.length;n>r;r++)this.renderSprite(e[r]);this.flush()}},i.prototype.renderSprite=function(t){if(t.visible&&(t.texture.baseTexture===this.currentBaseTexture||(this.flush(),this.currentBaseTexture=t.texture.baseTexture,t.texture._uvs))){var e,i,r,n,s,o,a,h,l=this.vertices;if(e=t.texture._uvs,i=t.texture.frame.width,r=t.texture.frame.height,t.texture.trim){var u=t.texture.trim;s=u.x-t.anchor.x*u.width,n=s+t.texture.crop.width,a=u.y-t.anchor.y*u.height,o=a+t.texture.crop.height}else n=t.texture.frame.width*(1-t.anchor.x),s=t.texture.frame.width*-t.anchor.x,o=t.texture.frame.height*(1-t.anchor.y),a=t.texture.frame.height*-t.anchor.y;h=4*this.currentBatchSize*this.vertSize,l[h++]=s,l[h++]=a,l[h++]=t.position.x,l[h++]=t.position.y,l[h++]=t.scale.x,l[h++]=t.scale.y,l[h++]=t.rotation,l[h++]=e.x0,l[h++]=e.y1,l[h++]=t.alpha,l[h++]=n,l[h++]=a,l[h++]=t.position.x,l[h++]=t.position.y,l[h++]=t.scale.x,l[h++]=t.scale.y,l[h++]=t.rotation,l[h++]=e.x1,l[h++]=e.y1,l[h++]=t.alpha,l[h++]=n,l[h++]=o,l[h++]=t.position.x,l[h++]=t.position.y,l[h++]=t.scale.x,l[h++]=t.scale.y,l[h++]=t.rotation,l[h++]=e.x2,l[h++]=e.y2,l[h++]=t.alpha,l[h++]=s,l[h++]=o,l[h++]=t.position.x,l[h++]=t.position.y,l[h++]=t.scale.x,l[h++]=t.scale.y,l[h++]=t.rotation,l[h++]=e.x3,l[h++]=e.y3,l[h++]=t.alpha,this.currentBatchSize++,this.currentBatchSize>=this.size&&this.flush()}},i.prototype.flush=function(){if(0!==this.currentBatchSize){var t=this.renderer.gl;if(this.currentBaseTexture._glTextures[t.id]?t.bindTexture(t.TEXTURE_2D,this.currentBaseTexture._glTextures[t.id]):this.renderer.updateTexture(this.currentBaseTexture,t),this.currentBatchSize>.5*this.size)t.bufferSubData(t.ARRAY_BUFFER,0,this.vertices);else{var e=this.vertices.subarray(0,4*this.currentBatchSize*this.vertSize);t.bufferSubData(t.ARRAY_BUFFER,0,e)}t.drawElements(t.TRIANGLES,6*this.currentBatchSize,t.UNSIGNED_SHORT,0),this.currentBatchSize=0,this.renderer.drawCount++}},i.prototype.stop=function(){this.flush()},i.prototype.start=function(){var t=this.renderer.gl;t.activeTexture(t.TEXTURE0),t.bindBuffer(t.ARRAY_BUFFER,this.vertexBuffer),t.bindBuffer(t.ELEMENT_ARRAY_BUFFER,this.indexBuffer);var e=this.renderer.projection;t.uniform2f(this.shader.projectionVector,e.x,e.y),t.uniformMatrix3fv(this.shader.uMatrix,!1,this.matrix);var i=4*this.vertSize;t.vertexAttribPointer(this.shader.attributes.aVertexPosition,2,t.FLOAT,!1,i,0),t.vertexAttribPointer(this.shader.attributes.aPositionCoord,2,t.FLOAT,!1,i,8),t.vertexAttribPointer(this.shader.attributes.aScale,2,t.FLOAT,!1,i,16),t.vertexAttribPointer(this.shader.attributes.aRotation,1,t.FLOAT,!1,i,24),t.vertexAttribPointer(this.shader.attributes.aTextureCoord,2,t.FLOAT,!1,i,28),t.vertexAttribPointer(this.shader.attributes.aColor,1,t.FLOAT,!1,i,36)}},{}],39:[function(t,e){var i=t("../../../utils"),r=t("../../../math"),n=t("../../../const"),s=t("./WebGLGraphicsData"),o=e.exports={};o.renderGraphics=function(t,e){var r,n=e.gl,s=e.projection,a=e.offset,h=e.shaderManager.primitiveShader;t.dirty&&o.updateGraphics(t,n);for(var l=t._webGL[n.id],u=0;u=6)if(h.points.length<12){a=o.switchMode(i,0);var l=o.buildPoly(h,a);l||(a=o.switchMode(i,1),o.buildComplexPoly(h,a))}else a=o.switchMode(i,1),o.buildComplexPoly(h,a);h.lineWidth>0&&(a=o.switchMode(i,0),o.buildLine(h,a))}else a=o.switchMode(i,0),h.type===n.SHAPES.RECT?o.buildRectangle(h,a):h.type===n.SHAPES.CIRC||h.type===n.SHAPES.ELIP?o.buildCircle(h,a):h.type===n.SHAPES.RREC&&o.buildRoundedRectangle(h,a);i.lastIndex++}for(r=0;r=m;m++)v=m/f,a=o(t,i,v),h=o(e,r,v),l=o(i,n,v),u=o(r,s,v),c=o(a,l,v),d=o(h,u,v),p.push(c,d);return p},o.buildCircle=function(t,e){var r,s,a=t.shape,h=a.x,l=a.y;t.type===n.SHAPES.CIRC?(r=a.radius,s=a.radius):(r=a.width,s=a.height);var u=40,c=2*Math.PI/u,d=0;if(t.fill){var f=i.hex2rgb(t.fillColor),p=t.fillAlpha,v=f[0]*p,m=f[1]*p,g=f[2]*p,x=e.points,y=e.indices,b=x.length/6;for(y.push(b),d=0;u+1>d;d++)x.push(h,l,v,m,g,p),x.push(h+Math.sin(c*d)*r,l+Math.cos(c*d)*s,v,m,g,p),y.push(b++,b++);y.push(b-1)}if(t.lineWidth){var T=t.points;for(t.points=[],d=0;u+1>d;d++)t.points.push(h+Math.sin(c*d)*r,l+Math.cos(c*d)*s);o.buildLine(t,e),t.points=T}},o.buildLine=function(t,e){var n=0,s=t.points;if(0!==s.length){if(t.lineWidth%2)for(n=0;nn;n++)d=s[2*(n-1)],f=s[2*(n-1)+1],p=s[2*n],v=s[2*n+1],m=s[2*(n+1)],g=s[2*(n+1)+1],x=-(f-v),y=d-p,D=Math.sqrt(x*x+y*y),x/=D,y/=D,x*=U,y*=U,b=-(v-g),T=p-m,D=Math.sqrt(b*b+T*T),b/=D,T/=D,b*=U,T*=U,A=-y+f-(-y+v),_=-x+p-(-x+d),C=(-x+d)*(-y+v)-(-x+p)*(-y+f),E=-T+g-(-T+v),R=-b+p-(-b+m),M=(-b+m)*(-T+v)-(-b+p)*(-T+g),F=A*R-E*_,Math.abs(F)<.1?(F+=10.1,P.push(p-x,v-y,Y,X,G,j),P.push(p+x,v+y,Y,X,G,j)):(u=(_*M-R*C)/F,c=(E*C-A*M)/F,B=(u-p)*(u-p)+(c-v)+(c-v),B>19600?(w=x-b,S=y-T,D=Math.sqrt(w*w+S*S),w/=D,S/=D,w*=U,S*=U,P.push(p-w,v-S),P.push(Y,X,G,j),P.push(p+w,v+S),P.push(Y,X,G,j),P.push(p-w,v-S),P.push(Y,X,G,j),I++):(P.push(u,c),P.push(Y,X,G,j),P.push(p-(u-p),v-(c-v)),P.push(Y,X,G,j)));for(d=s[2*(O-2)],f=s[2*(O-2)+1],p=s[2*(O-1)],v=s[2*(O-1)+1],x=-(f-v),y=d-p,D=Math.sqrt(x*x+y*y),x/=D,y/=D,x*=U,y*=U,P.push(p-x,v-y),P.push(Y,X,G,j),P.push(p+x,v+y),P.push(Y,X,G,j),L.push(k),n=0;I>n;n++)L.push(k++);L.push(k-1)}},o.buildComplexPoly=function(t,e){var r=t.points.slice();if(!(r.length<6)){var n=e.indices;e.points=r,e.alpha=t.fillAlpha,e.color=i.hex2rgb(t.fillColor);for(var s,o,a=1/0,h=-1/0,l=1/0,u=-1/0,c=0;cs?s:a,h=s>h?s:h,l=l>o?o:l,u=o>u?o:u;r.push(a,l,h,l,h,u,a,u);var d=r.length/2;for(c=0;d>c;c++)n.push(c)}},o.buildPoly=function(t,e){var r=t.points;if(!(r.length<6)){var n=e.points,s=e.indices,o=r.length/2,a=i.hex2rgb(t.fillColor),h=t.fillAlpha,l=a[0]*h,u=a[1]*h,c=a[2]*h,d=i.PolyK.Triangulate(r);if(!d)return!1;var f=n.length/6,p=0;for(p=0;pp;p++)n.push(r[2*p],r[2*p+1],l,u,c,h);return!0}},o.graphicsDataPool=[]},{"../../../const":4,"../../../math":12,"../../../utils":50,"./WebGLGraphicsData":40}],40:[function(t,e){function i(t){this.gl=t,this.color=[0,0,0],this.points=[],this.indices=[],this.buffer=t.createBuffer(),this.indexBuffer=t.createBuffer(),this.mode=1,this.alpha=1,this.dirty=!0}i.prototype.constructor=i,e.exports=i,i.prototype.reset=function(){this.points.length=0,this.indices.length=0},i.prototype.upload=function(){var t=this.gl;this.glPoints=new Float32Array(this.points),t.bindBuffer(t.ARRAY_BUFFER,this.buffer),t.bufferData(t.ARRAY_BUFFER,this.glPoints,t.STATIC_DRAW),this.glIndicies=new Uint16Array(this.indices),t.bindBuffer(t.ELEMENT_ARRAY_BUFFER,this.indexBuffer),t.bufferData(t.ELEMENT_ARRAY_BUFFER,this.glIndicies,t.STATIC_DRAW),this.dirty=!1}},{}],41:[function(t,e){function i(t){this.renderer=t,this.vertSize=5,this.size=2e3;var e=4*this.size*4*this.vertSize,i=6*this.size;this.vertices=new ArrayBuffer(e),this.positions=new Float32Array(this.vertices),this.colors=new Uint32Array(this.vertices),this.indices=new Uint16Array(i),this.lastIndexCount=0;for(var r=0,n=0;i>r;r+=6,n+=4)this.indices[r+0]=n+0,this.indices[r+1]=n+1,this.indices[r+2]=n+2,this.indices[r+3]=n+0,this.indices[r+4]=n+2,this.indices[r+5]=n+3;this.drawing=!1,this.currentBatchSize=0,this.currentBaseTexture=null,this.dirty=!0,this.textures=[],this.blendModes=[],this.shaders=[],this.sprites=[],this.shader=null;var s=this;this.renderer.on("context",function(){s.setupContext()})}var r=t("../../../textures/TextureUvs"),n=t("../shaders/Shader");i.prototype.constructor=i,e.exports=i,i.prototype.setupContext=function(){var t=this.renderer.gl;this.shader=new n(t),this.vertexBuffer=t.createBuffer(),this.indexBuffer=t.createBuffer(),t.bindBuffer(t.ELEMENT_ARRAY_BUFFER,this.indexBuffer),t.bufferData(t.ELEMENT_ARRAY_BUFFER,this.indices,t.STATIC_DRAW),t.bindBuffer(t.ARRAY_BUFFER,this.vertexBuffer),t.bufferData(t.ARRAY_BUFFER,this.vertices,t.DYNAMIC_DRAW),this.currentBlendMode=99999},i.prototype.begin=function(){this.start()},i.prototype.end=function(){this.flush()},i.prototype.render=function(t){var e=t.texture;this.currentBatchSize>=this.size&&(this.flush(),this.currentBaseTexture=e.baseTexture);var i=e._uvs;if(i){var r,n,s,o,a=t.anchor.x,h=t.anchor.y;if(e.trim){var l=e.trim;n=l.x-a*l.width,r=n+e.crop.width,o=l.y-h*l.height,s=o+e.crop.height}else r=e.frame.width*(1-a),n=e.frame.width*-a,s=e.frame.height*(1-h),o=e.frame.height*-h;var u=4*this.currentBatchSize*this.vertSize,c=e.baseTexture.resolution,d=t.worldTransform,f=d.a/c,p=d.b/c,v=d.c/c,m=d.d/c,g=d.tx,x=d.ty,y=this.colors,b=this.positions;this.renderer.roundPixels?(b[u]=f*n+v*o+g|0,b[u+1]=m*o+p*n+x|0,b[u+5]=f*r+v*o+g|0,b[u+6]=m*o+p*r+x|0,b[u+10]=f*r+v*s+g|0,b[u+11]=m*s+p*r+x|0,b[u+15]=f*n+v*s+g|0,b[u+16]=m*s+p*n+x|0):(b[u]=f*n+v*o+g,b[u+1]=m*o+p*n+x,b[u+5]=f*r+v*o+g,b[u+6]=m*o+p*r+x,b[u+10]=f*r+v*s+g,b[u+11]=m*s+p*r+x,b[u+15]=f*n+v*s+g,b[u+16]=m*s+p*n+x),b[u+2]=i.x0,b[u+3]=i.y0,b[u+7]=i.x1,b[u+8]=i.y1,b[u+12]=i.x2,b[u+13]=i.y2,b[u+17]=i.x3,b[u+18]=i.y3;var T=t.tint;y[u+4]=y[u+9]=y[u+14]=y[u+19]=(T>>16)+(65280&T)+((255&T)<<16)+(255*t.worldAlpha<<24),this.sprites[this.currentBatchSize++]=t}},i.prototype.renderTilingSprite=function(t){var e=t.tilingTexture;this.currentBatchSize>=this.size&&(this.flush(),this.currentBaseTexture=e.baseTexture),t._uvs||(t._uvs=new r);var i=t._uvs;t.tilePosition.x%=e.baseTexture.width*t.tileScaleOffset.x,t.tilePosition.y%=e.baseTexture.height*t.tileScaleOffset.y;var n=t.tilePosition.x/(e.baseTexture.width*t.tileScaleOffset.x),s=t.tilePosition.y/(e.baseTexture.height*t.tileScaleOffset.y),o=t.width/e.baseTexture.width/(t.tileScale.x*t.tileScaleOffset.x),a=t.height/e.baseTexture.height/(t.tileScale.y*t.tileScaleOffset.y);i.x0=0-n,i.y0=0-s,i.x1=1*o-n,i.y1=0-s,i.x2=1*o-n,i.y2=1*a-s,i.x3=0-n,i.y3=1*a-s;var h=t.tint,l=(h>>16)+(65280&h)+((255&h)<<16)+(255*t.alpha<<24),u=this.positions,c=this.colors,d=t.width,f=t.height,p=t.anchor.x,v=t.anchor.y,m=d*(1-p),g=d*-p,x=f*(1-v),y=f*-v,b=4*this.currentBatchSize*this.vertSize,T=e.baseTexture.resolution,w=t.worldTransform,S=w.a/T,A=w.b/T,_=w.c/T,C=w.d/T,E=w.tx,R=w.ty;u[b++]=S*g+_*y+E,u[b++]=C*y+A*g+R,u[b++]=i.x0,u[b++]=i.y0,c[b++]=l,u[b++]=S*m+_*y+E,u[b++]=C*y+A*m+R,u[b++]=i.x1,u[b++]=i.y1,c[b++]=l,u[b++]=S*m+_*x+E,u[b++]=C*x+A*m+R,u[b++]=i.x2,u[b++]=i.y2,c[b++]=l,u[b++]=S*g+_*x+E,u[b++]=C*x+A*g+R,u[b++]=i.x3,u[b++]=i.y3,c[b++]=l,this.sprites[this.currentBatchSize++]=t},i.prototype.flush=function(){if(0!==this.currentBatchSize){var t,e=this.renderer.gl;if(this.dirty){this.dirty=!1,e.activeTexture(e.TEXTURE0),e.bindBuffer(e.ARRAY_BUFFER,this.vertexBuffer),e.bindBuffer(e.ELEMENT_ARRAY_BUFFER,this.indexBuffer);var i=4*this.vertSize;e.vertexAttribPointer(this.shader.attributes.aVertexPosition,2,e.FLOAT,!1,i,0),e.vertexAttribPointer(this.shader.attributes.aTextureCoord,2,e.FLOAT,!1,i,8),e.vertexAttribPointer(this.shader.attributes.aColor,4,e.UNSIGNED_BYTE,!0,i,16)}if(this.currentBatchSize>.5*this.size)e.bufferSubData(e.ARRAY_BUFFER,0,this.vertices);else{var r=this.positions.subarray(0,4*this.currentBatchSize*this.vertSize);e.bufferSubData(e.ARRAY_BUFFER,0,r)}for(var s,o,a,h,l=0,u=0,c=null,d=this.renderer.blendModeManager.currentBlendMode,f=null,p=!1,v=!1,m=0,g=this.currentBatchSize;g>m;m++){if(h=this.sprites[m],s=h.texture.baseTexture,o=h.blendMode,a=h.shader||this.shader,p=d!==o,v=f!==a,(c!==s||p||v)&&(this.renderBatch(c,l,u),u=m,l=0,c=s,p&&(d=o,this.renderer.blendModeManager.setBlendMode(d)),v)){f=a,t=f.shaders?f.shaders[e.id]:f,t||(t=new n(e,null,f.fragmentSrc,f.uniforms),f.shaders[e.id]=t),this.renderer.shaderManager.setShader(t),t.dirty&&t.syncUniforms();var x=this.renderer.projection;e.uniform2f(t.uniforms.projectionVector._location,x.x,x.y);var y=this.renderer.offset;e.uniform2f(t.uniforms.offsetVector._location,y.x,y.y)}l++}this.renderBatch(c,l,u),this.currentBatchSize=0}},i.prototype.renderBatch=function(t,e,i){if(0!==e){var r=this.renderer.gl;t._glTextures[r.id]?r.bindTexture(r.TEXTURE_2D,t._glTextures[r.id]):this.renderer.updateTexture(t),r.drawElements(r.TRIANGLES,6*e,r.UNSIGNED_SHORT,6*i*2),this.renderer.drawCount++}},i.prototype.stop=function(){this.flush(),this.dirty=!0},i.prototype.start=function(){this.dirty=!0},i.prototype.destroy=function(){this.renderer.gl.deleteBuffer(this.vertexBuffer),this.renderer.gl.deleteBuffer(this.indexBuffer),this.vertices=null,this.indices=null,this.vertexBuffer=null,this.indexBuffer=null,this.currentBaseTexture=null,this.renderer=null}},{"../../../textures/TextureUvs":45,"../shaders/Shader":35}],42:[function(t,e){function i(t,e){this.uuid=r.uuid(),this.resolution=1,this.width=100,this.height=100,this.scaleMode=e||n.scaleModes.DEFAULT,this.hasLoaded=!1,this.isLoading=!1,this.source=null,this.premultipliedAlpha=!0,this.imageUrl=null,this._powerOf2=!1,this.mipmap=!1,this._glTextures={},this._needsUpdate=!1,t&&this.loadSource(t)}var r=t("../utils"),n=t("../const");i.prototype.constructor=i,e.exports=i,r.eventTarget.mixin(i.prototype),Object.defineProperties(i.prototype,{needsUpdate:{get:function(){return this._needsUpdate},set:function(t){this._needsUpdate=t,t&&this.emit("update",this)}}}),i.prototype.loadSource=function(t){var e=this.isLoading;if(this.hasLoaded=!1,this.isLoading=!1,e&&this.source&&(this.source.onload=null,this.source.onerror=null),this.source=t,(this.source.complete||this.source.getContext)&&this.source.width&&this.source.height)this._sourceLoaded();else if(!t.getContext){this.isLoading=!0;var i=this;t.onload=function(){t.onload=null,t.onerror=null,i.isLoading&&(i.isLoading=!1,i._sourceLoaded(),i.emit("loaded",i))},t.onerror=function(){t.onload=null,t.onerror=null,i.isLoading&&(i.isLoading=!1,i.emit("error",i))},t.complete&&(this.isLoading=!1,t.onload=null,t.onerror=null,t.width&&t.height?(this._sourceLoaded(),e&&this.emit("loaded",this)):e&&this.emit("error",this))}},i.prototype._sourceLoaded=function(){this.hasLoaded=!0,this.width=this.source.naturalWidth||this.source.width,this.height=this.source.naturalHeight||this.source.height,this.needsUpdate=!0},i.prototype.destroy=function(){this.imageUrl?(delete r.BaseTextureCache[this.imageUrl],delete r.TextureCache[this.imageUrl],this.imageUrl=null,navigator.isCocoonJS||(this.source.src="")):this.source&&this.source._pixiId&&delete r.BaseTextureCache[this.source._pixiId],this.source=null,this.dispose()},i.prototype.dispose=function(){this.emit("dispose",this)},i.prototype.updateSourceImage=function(t){this.source.src=t,this.loadSource(this.source)},i.fromImage=function(t,e,s){var o=r.BaseTextureCache[t];if(void 0===e&&0!==t.indexOf("data:")&&(e=!0),!o){var a=new Image;e&&(a.crossOrigin=""),a.src=t,o=new i(a,s),o.imageUrl=t,r.BaseTextureCache[t]=o,-1!==t.indexOf(n.RETINA_PREFIX+".")&&(o.resolution=2) -}return o},i.fromCanvas=function(t,e){t._pixiId||(t._pixiId="canvas_"+r.uuid());var n=r.BaseTextureCache[t._pixiId];return n||(n=new i(t,e),r.BaseTextureCache[t._pixiId]=n),n}},{"../const":4,"../utils":50}],43:[function(t,e){function i(t,e,i,l,u){if(!t)throw new Error("Unable to create RenderTexture, you must pass a renderer into the constructor.");if(this.width=e||100,this.height=i||100,this.resolution=u||1,this.frame=new a.Rectangle(0,0,this.width*this.resolution,this.height*this.resolution),this.crop=new a.Rectangle(0,0,this.width*this.resolution,this.height*this.resolution),this.render=null,this.baseTexture=new r,this.baseTexture.width=this.width*this.resolution,this.baseTexture.height=this.height*this.resolution,this.baseTexture._glTextures=[],this.baseTexture.resolution=this.resolution,this.baseTexture.scaleMode=l||h.scaleModes.DEFAULT,this.baseTexture.hasLoaded=!0,n.call(this,this.baseTexture,new a.Rectangle(0,0,this.width,this.height)),this.renderer=t,this.renderer.type===h.WEBGL_RENDERER){var c=this.renderer.gl;this.textureBuffer=new s(c,this.width*this.resolution,this.height*this.resolution,this.baseTexture.scaleMode),this.baseTexture._glTextures[c.id]=this.textureBuffer.texture,this.render=this.renderWebGL,this.projection=new a.Point(.5*this.width,.5*-this.height)}else this.render=this.renderCanvas,this.textureBuffer=new o(this.width*this.resolution,this.height*this.resolution),this.baseTexture.source=this.textureBuffer.canvas;this.valid=!0,this._updateUvs()}var r=t("./BaseTexture"),n=t("./Texture"),s=t("../renderers/webgl/utils/FilterTexture"),o=t("../renderers/canvas/utils/CanvasBuffer"),a=t("../math"),h=t("../const");i.prototype=Object.create(n.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.resize=function(t,e,i){(t!==this.width||e!==this.height)&&(this.valid=t>0&&e>0,this.width=this.frame.width=this.crop.width=t,this.height=this.frame.height=this.crop.height=e,i&&(this.baseTexture.width=this.width,this.baseTexture.height=this.height),this.renderer.type===h.WEBGL_RENDERER&&(this.projection.x=this.width/2,this.projection.y=-this.height/2),this.valid&&this.textureBuffer.resize(this.width*this.resolution,this.height*this.resolution))},i.prototype.clear=function(){this.valid&&(this.renderer.type===h.WEBGL_RENDERER&&this.renderer.gl.bindFramebuffer(this.renderer.gl.FRAMEBUFFER,this.textureBuffer.frameBuffer),this.textureBuffer.clear())},i.prototype.renderWebGL=function(t,e,i,r){if(this.valid){"undefined"==typeof r&&(r=!0);var n,s;r&&(n=t.worldAlpha,s=t.worldTransform.toArray());var o=t.worldTransform;o.identity(),o.translate(0,2*this.projection.y),e&&o.append(e),o.scale(1,-1),t.worldAlpha=1;var a,h,l=t.children;for(a=0,h=l.length;h>a;++a)l[a].updateTransform();var u=this.renderer.gl;if(u.viewport(0,0,this.width*this.resolution,this.height*this.resolution),u.bindFramebuffer(u.FRAMEBUFFER,this.textureBuffer.frameBuffer),i&&this.textureBuffer.clear(),this.renderer.spriteBatch.dirty=!0,this.renderer.renderDisplayObject(t,this.projection,this.textureBuffer.frameBuffer),this.renderer.spriteBatch.dirty=!0,r)for(t.worldAlpha=n,t.worldTransform.fromArray(s),a=0,h=l.length;h>a;++a)l[a].updateTransform()}},i.prototype.renderCanvas=function(t,e,i,r){if(this.valid){"undefined"==typeof r&&(r=!0);var n,s;r&&(n=t.worldAlpha,s=t.worldTransform.toArray());var o=t.worldTransform;o.identity(),e&&o.append(e),t.worldAlpha=1;var a,h,l=t.children;for(a=0,h=l.length;h>a;++a)l[a].updateTransform();i&&this.textureBuffer.clear();var u=this.textureBuffer.context,c=this.renderer.resolution;if(this.renderer.resolution=this.resolution,this.renderer.renderDisplayObject(t,u),this.renderer.resolution=c,r)for(t.worldAlpha=n,t.worldTransform.fromArray(s),a=0,h=l.length;h>a;++a)l[a].updateTransform()}},i.prototype.getImage=function(){var t=new Image;return t.src=this.getBase64(),t},i.prototype.getBase64=function(){return this.getCanvas().toDataURL()},i.prototype.getCanvas=function(){if(this.renderer.type===h.WEBGL_RENDERER){var t=this.renderer.gl,e=this.textureBuffer.width,i=this.textureBuffer.height,r=new Uint8Array(4*e*i);t.bindFramebuffer(t.FRAMEBUFFER,this.textureBuffer.frameBuffer),t.readPixels(0,0,e,i,t.RGBA,t.UNSIGNED_BYTE,r),t.bindFramebuffer(t.FRAMEBUFFER,null);var n=new o(e,i),s=n.context.getImageData(0,0,e,i);return s.data.set(r),n.context.putImageData(s,0,0),n.canvas}return this.textureBuffer.canvas}},{"../const":4,"../math":12,"../renderers/canvas/utils/CanvasBuffer":21,"../renderers/webgl/utils/FilterTexture":37,"./BaseTexture":42,"./Texture":44}],44:[function(t,e){function i(t,e,r,n){this.noFrame=!1,e||(this.noFrame=!0,e=new a.Rectangle(0,0,1,1)),t instanceof i&&(t=t.baseTexture),this.baseTexture=t,this._frame=e,this.trim=n,this.valid=!1,this.requiresUpdate=!1,this._uvs=null,this.width=0,this.height=0,this.crop=r||new a.Rectangle(0,0,1,1),t.hasLoaded?(this.noFrame&&(e=new a.Rectangle(0,0,t.width,t.height)),this.frame=e):t.addEventListener("loaded",this.onBaseTextureLoaded.bind(this))}var r=t("./BaseTexture"),n=t("./VideoBaseTexture"),s=t("./TextureUvs"),o=t("../utils/eventTarget"),a=t("../math"),h=t("../utils");i.prototype.constructor=i,e.exports=i,o.mixin(i.prototype),Object.defineProperties(i.prototype,{needsUpdate:{get:function(){return this.baseTexture.needsUpdate},set:function(t){this.baseTexture.needsUpdate=t}},frame:{get:function(){return this._frame},set:function(t){if(this._frame=t,this.noFrame=!1,this.width=t.width,this.height=t.height,this.crop.x=t.x,this.crop.y=t.y,this.crop.width=t.width,this.crop.height=t.height,!this.trim&&(t.x+t.width>this.baseTexture.width||t.y+t.height>this.baseTexture.height))throw new Error("Texture Error: frame does not fit inside the base Texture dimensions "+this);this.valid=t&&t.width&&t.height&&this.baseTexture.source&&this.baseTexture.hasLoaded,this.trim&&(this.width=this.trim.width,this.height=this.trim.height,this._frame.width=this.trim.width,this._frame.height=this.trim.height),this.valid&&this._updateUvs()}}}),i.prototype.onBaseTextureLoaded=function(){var t=this.baseTexture;t.removeEventListener("loaded",this.onLoaded),this.noFrame&&(this.frame=new a.Rectangle(0,0,t.width,t.height)),this.dispatchEvent({type:"update",content:this})},i.prototype.destroy=function(t){t&&this.baseTexture.destroy(),this.valid=!1},i.prototype._updateUvs=function(){this._uvs||(this._uvs=new s);var t=this.crop,e=this.baseTexture.width,i=this.baseTexture.height;this._uvs.x0=t.x/e,this._uvs.y0=t.y/i,this._uvs.x1=(t.x+t.width)/e,this._uvs.y1=t.y/i,this._uvs.x2=(t.x+t.width)/e,this._uvs.y2=(t.y+t.height)/i,this._uvs.x3=t.x/e,this._uvs.y3=(t.y+t.height)/i},i.fromImage=function(t,e,n){var s=h.TextureCache[t];return s||(s=new i(r.fromImage(t,e,n)),h.TextureCache[t]=s),s},i.fromFrame=function(t){var e=h.TextureCache[t];if(!e)throw new Error('The frameId "'+t+'" does not exist in the texture cache ');return e},i.fromCanvas=function(t,e){return new i(r.fromCanvas(t,e))},i.fromVideo=function(t,e){return new i(n.baseTextureFromVideo(t,e))},i.addTextureToCache=function(t,e){h.TextureCache[e]=t},i.removeTextureFromCache=function(t){var e=h.TextureCache[t];return delete h.TextureCache[t],delete h.BaseTextureCache[t],e},i.emptyTexture=new i(new r)},{"../math":12,"../utils":50,"../utils/eventTarget":49,"./BaseTexture":42,"./TextureUvs":45,"./VideoBaseTexture":46}],45:[function(t,e){function i(){this.x0=0,this.y0=0,this.x1=0,this.y1=0,this.x2=0,this.y2=0,this.x3=0,this.y3=0}e.exports=i},{}],46:[function(t,e){function i(t,e){if(!t)throw new Error("No video source element specified.");(t.readyState===t.HAVE_ENOUGH_DATA||t.readyState===t.HAVE_FUTURE_DATA)&&t.width&&t.height&&(t.complete=!0),n.call(this,t,e),this.autoUpdate=!1,this._boundOnUpdate=this._onUpdate.bind(this),this._boundOnCanPlay=this._onCanPlay.bind(this),t.complete||(t.addEventListener("canplay",this._boundOnCanPlay),t.addEventListener("canplaythrough",this._boundOnCanPlay),t.addEventListener("play",this._onPlayStart.bind(this)),t.addEventListener("pause",this._onPlayStop.bind(this))),this.__loaded=!1}function r(t,e){e||(e="video/"+t.substr(t.lastIndexOf(".")+1));var i=document.createElement("source");return i.src=t,i.type=e,i}var n=t("./BaseTexture"),s=t("../utils");i.prototype=Object.create(n.prototype),i.prototype.constructor=i,e.exports=i,i.prototype._onUpdate=function(){this.autoUpdate&&(window.requestAnimationFrame(this._boundOnUpdate),this.needsUpdate=!0)},i.prototype._onPlayStart=function(){this.autoUpdate||(window.requestAnimationFrame(this._boundOnUpdate),this.autoUpdate=!0)},i.prototype._onPlayStop=function(){this.autoUpdate=!1},i.prototype._onCanPlay=function(){this.hasLoaded=!0,this.source&&(this.source.removeEventListener("canplay",this._boundOnCanPlay),this.source.removeEventListener("canplaythrough",this._boundOnCanPlay),this.width=this.source.videoWidth,this.height=this.source.videoHeight,this.source.play(),this.__loaded||(this.__loaded=!0,this.emit("loaded",this)))},i.prototype.destroy=function(){this.source&&this.source._pixiId&&(s.BaseTextureCache[this.source._pixiId]=null,delete s.BaseTextureCache[this.source._pixiId],this.source._pixiId=null,delete this.source._pixiId),n.prototype.destroy.call(this)},i.fromVideo=function(t,e){t._pixiId||(t._pixiId="video_"+s.uuid());var r=s.BaseTextureCache[t._pixiId];return r||(r=new i(t,e),s.BaseTextureCache[t._pixiId]=r),r},i.fromUrl=function(t,e){var n=document.createElement("video");if(Array.isArray(t))for(var s=0;s>1;if(3>r)return[];for(var n=[],s=[],o=0;r>o;o++)s.push(o);o=0;for(var a=r;a>3;){var h=s[(o+0)%a],l=s[(o+1)%a],u=s[(o+2)%a],c=t[2*h],d=t[2*h+1],f=t[2*l],p=t[2*l+1],v=t[2*u],m=t[2*u+1],g=!1;if(i._convex(c,d,f,p,v,m,e)){g=!0;for(var x=0;a>x;x++){var y=s[x];if(y!==h&&y!==l&&y!==u&&i._PointInTriangle(t[2*y],t[2*y+1],c,d,f,p,v,m)){g=!1;break}}}if(g)n.push(h,l,u),s.splice((o+1)%a,1),a--,o=0;else if(o++>3*a){if(!e)return null;for(n=[],s=[],o=0;r>o;o++)s.push(o);o=0,a=r,e=!1}}return n.push(s[0],s[1],s[2]),n},i._PointInTriangle=function(t,e,i,r,n,s,o,a){var h=o-i,l=a-r,u=n-i,c=s-r,d=t-i,f=e-r,p=h*h+l*l,v=h*u+l*c,m=h*d+l*f,g=u*u+c*c,x=u*d+c*f,y=1/(p*g-v*v),b=(g*m-v*x)*y,T=(p*x-v*m)*y;return b>=0&&T>=0&&1>b+T},i._convex=function(t,e,i,r,n,s,o){return(e-r)*(n-i)+(i-t)*(s-r)>=0===o}},{}],49:[function(t,e){function i(t){t.listeners=function(t){return this._listeners=this._listeners||{},this._listeners[t]?this._listeners[t].slice():[]},t.emit=t.dispatchEvent=function(t,e){if(this._listeners=this._listeners||{},this._listeners[t]){"object"==typeof t&&(e=t,t=t.type),e&&e.__isEventObject===!0||(e=new r(this,t,e));var i,n=this._listeners[t].slice(0),s=n.length,o=n[0];for(i=0;s>i;o=n[++i])if(o.call(this,e),e.stoppedImmediate)return this;return e.stopped?this:(this.parent&&this.parent.emit&&this.parent.emit.call(this.parent,t,e),this)}},t.on=t.addEventListener=function(t,e){return this._listeners=this._listeners||{},(this._listeners[t]=this._listeners[t]||[]).push(e),this},t.once=function(t,e){function i(){e.apply(r.off(t,i),arguments)}this._listeners=this._listeners||{};var r=this;return i._originalHandler=e,this.on(t,i)},t.off=t.removeEventListener=function(t,e){if(this._listeners=this._listeners||{},!this._listeners[t])return this;for(var i=this._listeners[t],r=e?i.length:0;r-->0;)(i[r]===e||i[r]._originalHandler===e)&&i.splice(r,1);return 0===i.length&&delete this._listeners[t],this},t.removeAllListeners=function(t){return this._listeners=this._listeners||{},this._listeners[t]?(delete this._listeners[t],this):this}}var r=t("./EventData");e.exports={mixin:function(t){i(t)}}},{"./EventData":47}],50:[function(t,e){var i=t("../const"),r=e.exports={_uid:0,_saidHello:!1,PolyK:t("./PolyK"),EventData:t("./EventData"),eventTarget:t("./eventTarget"),uuid:function(){return++r._uid},hex2rgb:function(t,e){return e=e||[],e[0]=(t>>16&255)/255,e[1]=(t>>8&255)/255,e[2]=(255&t)/255,e},hex2string:function(t){return t=t.toString(16),t="000000".substr(0,6-t.length)+t,"#"+t},rgb2hex:function(t){return(255*t[0]<<16)+(255*t[1]<<8)+255*t[2]},canUseNewCanvasBlendModes:function(){if("undefined"==typeof document)return!1;var t=document.createElement("canvas"),e=t.getContext("2d");return t.width=1,t.height=1,e.fillStyle="#000",e.fillRect(0,0,1,1),e.globalCompositeOperation="multiply",e.fillStyle="#fff",e.fillRect(0,0,1,1),0===e.getImageData(0,0,1,1).data[0]},getNextPowerOfTwo:function(t){if(t>0&&0===(t&t-1))return t;for(var e=1;t>e;)e<<=1;return e},isPowerOfTwo:function(t,e){return t>0&&0===(t&t-1)&&e>0&&0===(e&e-1)},sayHello:function(t){if(!r._saidHello){if(navigator.userAgent.toLowerCase().indexOf("chrome")>-1){var e=["%c %c %c Pixi.js "+i.VERSION+" - "+t+" %c %c http://www.pixijs.com/ %c %c ♥%c♥%c♥ ","background: #ff66a5","background: #ff66a5","color: #ff66a5; background: #030307;","background: #ff66a5","background: #ffc3dc","background: #ff66a5","color: #ff2424; background: #fff","color: #ff2424; background: #fff","color: #ff2424; background: #fff"];console.log.apply(console,e)}else window.console&&console.log("Pixi.js "+i.VERSION+" - "+t+" - http://www.pixijs.com/");r._saidHello=!0}},AjaxRequest:function(){var t=["Msxml2.XMLHTTP.6.0","Msxml2.XMLHTTP.3.0","Microsoft.XMLHTTP"];if(!window.ActiveXObject)return window.XMLHttpRequest?new window.XMLHttpRequest:!1;for(var e=0;et?this.loop?(this.currentFrame+=this.textures.length,this.texture=this.textures[this.currentFrame]):(this.gotoAndStop(0),this.onComplete&&this.onComplete()):this.loop||t=this.textures.length&&(this.gotoAndStop(this.textures.length-1),this.onComplete&&this.onComplete())}},e.fromFrames=function(t){for(var r=[],n=0;nh;h++)n=t[h],s=4*h,o=h/(a-1),h%2?(e[s]=o,e[s+1]=0,e[s+2]=o,e[s+3]=1):(e[s]=o,e[s+1]=0,e[s+2]=o,e[s+3]=1),s=2*h,r[s]=1,r[s+1]=1,s=2*h,i[s]=s,i[s+1]=s+1}},i.prototype.updateTransform=function(){var t=this.points;if(!(t.length<1)){for(var e,i,r,n,s,o,a=t[0],h=0,l=0,u=this.vertices,c=t.length,d=0;c>d;d++)i=t[d],r=4*d,e=d1&&(n=1),s=Math.sqrt(h*h+l*l),o=this.texture.height/2,h/=s,l/=s,h*=o,l*=o,u[r]=i.x+h,u[r+1]=i.y+l,u[r+2]=i.x-h,u[r+3]=i.y-l,a=i;this.displayObjectContainerUpdateTransform()}},i.prototype.setTexture=function(t){this.texture=t}},{"./Strip":53}],53:[function(t,e){function i(t){r.DisplayObjectContainer.call(this),this.texture=t,this.uvs=new Float32Array([0,1,1,1,1,0,0,1]),this.vertices=new Float32Array([0,0,100,0,100,100,0,100]),this.colors=new Float32Array([1,1,1,1]),this.indices=new Uint16Array([0,1,2,3]),this.dirty=!0,this.blendMode=r.CONST.blendModes.NORMAL,this.canvasPadding=0,this.drawMode=i.DrawModes.TRIANGLE_STRIP}var r=t("../core");i.prototype=Object.create(r.DisplayObjectContainer.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.renderWebGL=function(t){!this.visible||this.alpha<=0||(t.spriteBatch.stop(),this._vertexBuffer||this._initWebGL(t),t.shaderManager.setShader(t.shaderManager.stripShader),this._renderStrip(t),t.spriteBatch.start())},i.prototype._initWebGL=function(t){var e=t.gl;this._vertexBuffer=e.createBuffer(),this._indexBuffer=e.createBuffer(),this._uvBuffer=e.createBuffer(),this._colorBuffer=e.createBuffer(),e.bindBuffer(e.ARRAY_BUFFER,this._vertexBuffer),e.bufferData(e.ARRAY_BUFFER,this.vertices,e.DYNAMIC_DRAW),e.bindBuffer(e.ARRAY_BUFFER,this._uvBuffer),e.bufferData(e.ARRAY_BUFFER,this.uvs,e.STATIC_DRAW),e.bindBuffer(e.ARRAY_BUFFER,this._colorBuffer),e.bufferData(e.ARRAY_BUFFER,this.colors,e.STATIC_DRAW),e.bindBuffer(e.ELEMENT_ARRAY_BUFFER,this._indexBuffer),e.bufferData(e.ELEMENT_ARRAY_BUFFER,this.indices,e.STATIC_DRAW)},i.prototype._renderStrip=function(t){var e=t.gl,r=t.projection,n=t.offset,s=t.shaderManager.stripShader,o=this.drawMode===i.DrawModes.TRIANGLE_STRIP?e.TRIANGLE_STRIP:e.TRIANGLES;t.blendModeManager.setBlendMode(this.blendMode),e.uniformMatrix3fv(s.translationMatrix,!1,this.worldTransform.toArray(!0)),e.uniform2f(s.projectionVector,r.x,-r.y),e.uniform2f(s.offsetVector,-n.x,-n.y),e.uniform1f(s.alpha,this.worldAlpha),this.dirty?(this.dirty=!1,e.bindBuffer(e.ARRAY_BUFFER,this._vertexBuffer),e.bufferData(e.ARRAY_BUFFER,this.vertices,e.STATIC_DRAW),e.vertexAttribPointer(s.aVertexPosition,2,e.FLOAT,!1,0,0),e.bindBuffer(e.ARRAY_BUFFER,this._uvBuffer),e.bufferData(e.ARRAY_BUFFER,this.uvs,e.STATIC_DRAW),e.vertexAttribPointer(s.aTextureCoord,2,e.FLOAT,!1,0,0),e.activeTexture(e.TEXTURE0),this.texture.baseTexture._dirty[e.id]?t.updateTexture(this.texture.baseTexture):e.bindTexture(e.TEXTURE_2D,this.texture.baseTexture._glTextures[e.id]),e.bindBuffer(e.ELEMENT_ARRAY_BUFFER,this._indexBuffer),e.bufferData(e.ELEMENT_ARRAY_BUFFER,this.indices,e.STATIC_DRAW)):(e.bindBuffer(e.ARRAY_BUFFER,this._vertexBuffer),e.bufferSubData(e.ARRAY_BUFFER,0,this.vertices),e.vertexAttribPointer(s.aVertexPosition,2,e.FLOAT,!1,0,0),e.bindBuffer(e.ARRAY_BUFFER,this._uvBuffer),e.vertexAttribPointer(s.aTextureCoord,2,e.FLOAT,!1,0,0),e.activeTexture(e.TEXTURE0),this.texture.baseTexture._dirty[e.id]?t.updateTexture(this.texture.baseTexture):e.bindTexture(e.TEXTURE_2D,this.texture.baseTexture._glTextures[e.id]),e.bindBuffer(e.ELEMENT_ARRAY_BUFFER,this._indexBuffer)),e.drawElements(o,this.indices.length,e.UNSIGNED_SHORT,0)},i.prototype.renderCanvas=function(t){var e=t.context,r=this.worldTransform;t.roundPixels?e.setTransform(r.a,r.b,r.c,r.d,0|r.tx,0|r.ty):e.setTransform(r.a,r.b,r.c,r.d,r.tx,r.ty),this.drawMode===i.DrawModes.TRIANGLE_STRIP?this._renderCanvasTriangleStrip(e):this._renderCanvasTriangles(e)},i.prototype._renderCanvasTriangleStrip=function(t){for(var e=this.vertices,i=this.uvs,r=e.length/2,n=0;r-2>n;n++){var s=2*n;this._renderCanvasDrawTriangle(t,e,i,s,s+2,s+4)}},i.prototype._renderCanvasTriangles=function(t){for(var e=this.vertices,i=this.uvs,r=this.indices,n=r.length,s=0;n>s;s+=3){var o=2*r[s],a=2*r[s+1],h=2*r[s+2];this._renderCanvasDrawTriangle(t,e,i,o,a,h)}},i.prototype._renderCanvasDrawTriangle=function(t,e,i,r,n,s){var o=this.texture.baseTexture.source,a=this.texture.width,h=this.texture.height,l=e[r],u=e[n],c=e[s],d=e[r+1],f=e[n+1],p=e[s+1],v=i[r]*a,m=i[n]*a,g=i[s]*a,x=i[r+1]*h,y=i[n+1]*h,b=i[s+1]*h;if(this.canvasPadding>0){var T=this.canvasPadding/this.worldTransform.a,w=this.canvasPadding/this.worldTransform.d,S=(l+u+c)/3,A=(d+f+p)/3,_=l-S,C=d-A,E=Math.sqrt(_*_+C*C);l=S+_/E*(E+T),d=A+C/E*(E+w),_=u-S,C=f-A,E=Math.sqrt(_*_+C*C),u=S+_/E*(E+T),f=A+C/E*(E+w),_=c-S,C=p-A,E=Math.sqrt(_*_+C*C),c=S+_/E*(E+T),p=A+C/E*(E+w)}t.save(),t.beginPath(),t.moveTo(l,d),t.lineTo(u,f),t.lineTo(c,p),t.closePath(),t.clip();var R=v*y+x*g+m*b-y*g-x*m-v*b,M=l*y+x*c+u*b-y*c-x*u-l*b,F=v*u+l*g+m*c-u*g-l*m-v*c,B=v*y*c+x*u*g+l*m*b-l*y*g-x*m*c-v*u*b,D=d*y+x*p+f*b-y*p-x*f-d*b,P=v*f+d*g+m*p-f*g-d*m-v*p,L=v*y*p+x*f*g+d*m*b-d*y*g-x*m*p-v*f*b;t.transform(M/R,D/R,F/R,P/R,B/R,L/R),t.drawImage(o,0,0),t.restore()},i.prototype.renderStripFlat=function(t){var e=this.context,i=t.vertices,r=i.length/2;e.beginPath();for(var n=1;r-2>n;n++){var s=2*n,o=i[s],a=i[s+2],h=i[s+4],l=i[s+1],u=i[s+3],c=i[s+5];e.moveTo(o,l),e.lineTo(a,u),e.lineTo(h,c)}e.fillStyle="#FF0000",e.fill(),e.closePath()},i.prototype.onTextureUpdate=function(){this.updateFrame=!0},i.prototype.getBounds=function(t){for(var e=t||this.worldTransform,i=e.a,n=e.b,s=e.c,o=e.d,a=e.tx,h=e.ty,l=-1/0,u=-1/0,c=1/0,d=1/0,f=this.vertices,p=0,v=f.length;v>p;p+=2){var m=f[p],g=f[p+1],x=i*m+s*g+a,y=o*g+n*m+h;c=c>x?x:c,d=d>y?y:d,l=x>l?x:l,u=y>u?y:u}if(c===-1/0||1/0===u)return r.math.Rectangle.EMPTY;var b=this._bounds;return b.x=c,b.width=l-c,b.y=d,b.height=u-d,this._currentBounds=b,b},i.DrawModes={TRIANGLE_STRIP:0,TRIANGLES:1}},{"../core":9}],54:[function(t,e){function i(t,e,i){r.Sprite.call(this,t),this._width=e||100,this._height=i||100,this.tileScale=new r.math.Point(1,1),this.tileScaleOffset=new r.math.Point(1,1),this.tilePosition=new r.math.Point(0,0),this.renderable=!0,this.tint=16777215,this.blendMode=r.CONST.blendModes.NORMAL}var r=t("../core");i.prototype=Object.create(r.Sprite.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{width:{get:function(){return this._width},set:function(t){this._width=t}},height:{get:function(){return this._height},set:function(t){this._height=t}},texture:{get:function(){return this._texture},set:function(t){this._texture!==t&&(this._texture=t,this.refreshTexture=!0,this.cachedTint=16777215)}}}),i.prototype.renderWebGL=function(t){if(this.visible&&!(this.alpha<=0)){var e,i;for(this._mask&&(t.spriteBatch.stop(),t.maskManager.pushMask(this.mask,t),t.spriteBatch.start()),this._filters&&(t.spriteBatch.flush(),t.filterManager.pushFilter(this._filterBlock)),!this.tilingTexture||this.refreshTexture?(this.generateTilingTexture(!0),this.tilingTexture&&this.tilingTexture.needsUpdate&&(t.updateTexture(this.tilingTexture.baseTexture),this.tilingTexture.needsUpdate=!1)):t.spriteBatch.renderTilingSprite(this),e=0,i=this.children.length;i>e;e++)this.children[e].renderWebGL(t);t.spriteBatch.stop(),this._filters&&t.filterManager.popFilter(),this._mask&&t.maskManager.popMask(this._mask,t),t.spriteBatch.start()}},i.prototype.renderCanvas=function(t){if(this.visible&&!(this.alpha<=0)){var e=t.context;this._mask&&t.maskManager.pushMask(this._mask,e),e.globalAlpha=this.worldAlpha;var i,r,n=this.worldTransform,s=t.resolution;if(e.setTransform(n.a*s,n.b*s,n.c*s,n.d*s,n.tx*s,n.ty*s),!this.__tilePattern||this.refreshTexture){if(this.generateTilingTexture(!1),!this.tilingTexture)return;this.__tilePattern=e.createPattern(this.tilingTexture.baseTexture.source,"repeat")}this.blendMode!==t.currentBlendMode&&(t.currentBlendMode=this.blendMode,e.globalCompositeOperation=t.blendModes[t.currentBlendMode]);var o=this.tilePosition,a=this.tileScale;for(o.x%=this.tilingTexture.baseTexture.width,o.y%=this.tilingTexture.baseTexture.height,e.scale(a.x,a.y),e.translate(o.x+this.anchor.x*-this._width,o.y+this.anchor.y*-this._height),e.fillStyle=this.__tilePattern,e.fillRect(-o.x,-o.y,this._width/a.x,this._height/a.y),e.translate(-o.x+this.anchor.x*this._width,-o.y+this.anchor.y*this._height),e.scale(1/a.x,1/a.y),this._mask&&t.maskManager.popMask(t.context),i=0,r=this.children.length;r>i;++i)this.children[i].renderCanvas(t)}},i.prototype.getBounds=function(){var t,e,i,r,n=this._width,s=this._height,o=n*(1-this.anchor.x),a=n*-this.anchor.x,h=s*(1-this.anchor.y),l=s*-this.anchor.y,u=this.worldTransform,c=u.a,d=u.b,f=u.c,p=u.d,v=u.tx,m=u.ty,g=c*a+f*l+v,x=p*l+d*a+m,y=c*o+f*l+v,b=p*l+d*o+m,T=c*o+f*h+v,w=p*h+d*o+m,S=c*a+f*h+v,A=p*h+d*a+m;t=g,t=t>y?y:t,t=t>T?T:t,t=t>S?S:t,i=x,i=i>b?b:i,i=i>w?w:i,i=i>A?A:i,e=g,e=y>e?y:e,e=T>e?T:e,e=S>e?S:e,r=x,r=b>r?b:r,r=w>r?w:r,r=A>r?A:r;var _=this._bounds;return _.x=t,_.width=e-t,_.y=i,_.height=r-i,this._currentBounds=_,_},i.prototype.onTextureUpdate=function(){},i.prototype.generateTilingTexture=function(t){if(this.texture.baseTexture.hasLoaded){var e,i,n=this.originalTexture||this.texture,s=n.frame,o=s.width!==n.baseTexture.width||s.height!==n.baseTexture.height,a=!1;if(t?(e=r.utils.getNextPowerOfTwo(s.width),i=r.utils.getNextPowerOfTwo(s.height),(s.width!==e||s.height!==i||n.baseTexture.width!==e||n.baseTexture.height||i)&&(a=!0)):o&&(e=s.width,i=s.height,a=!0),a){var h;this.tilingTexture&&this.tilingTexture.isTiling?(h=this.tilingTexture.canvasBuffer,h.resize(e,i),this.tilingTexture.baseTexture.width=e,this.tilingTexture.baseTexture.height=i,this.tilingTexture.needsUpdate=!0):(h=new r.CanvasBuffer(e,i),this.tilingTexture=r.Texture.fromCanvas(h.canvas),this.tilingTexture.canvasBuffer=h,this.tilingTexture.isTiling=!0),h.context.drawImage(n.baseTexture.source,n.crop.x,n.crop.y,n.crop.width,n.crop.height,0,0,e,i),this.tileScaleOffset.x=s.width/e,this.tileScaleOffset.y=s.height/i}else this.tilingTexture&&this.tilingTexture.isTiling&&this.tilingTexture.destroy(!0),this.tileScaleOffset.x=1,this.tileScaleOffset.y=1,this.tilingTexture=n;this.refreshTexture=!1,this.originalTexture=this.texture,this.texture=this.tilingTexture,this.tilingTexture.baseTexture._powerOf2=!0}}},{"../core":9}],55:[function(t,e){e.exports={MovieClip:t("./MovieClip"),Rope:t("./Rope"),Strip:t("./Strip"),TilingSprite:t("./TilingSprite")}},{"./MovieClip":51,"./Rope":52,"./Strip":53,"./TilingSprite":54}],56:[function(t,e){function i(t,e){this.passes=[this],this.shaders=[],this.padding=0,this.uniforms=e||{},this.fragmentSrc="string"==typeof t?t.split(""):t||[]}i.prototype.constructor=i,e.exports=i,i.prototype.syncUniforms=function(){for(var t=0,e=this.shaders.length;e>t;++t)this.shaders[t].dirty=!0}},{}],57:[function(t,e){function i(t){r.call(this),t.baseTexture._powerOf2=!0,this.uniforms={mask:{type:"sampler2D",value:t},mapDimensions:{type:"2f",value:{x:1,y:5112}},dimensions:{type:"4fv",value:[0,0,0,0]},offset:{type:"2f",value:{x:0,y:0}}},t.baseTexture.hasLoaded?(this.uniforms.mask.value.x=t.width,this.uniforms.mask.value.y=t.height):(this.boundLoadedFunction=this.onTextureLoaded.bind(this),t.baseTexture.on("loaded",this.boundLoadedFunction)),this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform sampler2D uSampler;","uniform sampler2D mask;","uniform vec2 mapDimensions;","uniform vec4 dimensions;","uniform vec2 offset;","void main() {"," vec2 mapCords = vTextureCoord.xy;"," mapCords += (dimensions.zw + offset)/ dimensions.xy ;"," mapCords.y *= -1.0;"," mapCords.y += 1.0;"," mapCords *= dimensions.xy / mapDimensions;"," vec4 original = texture2D(uSampler, vTextureCoord);"," float maskAlpha = texture2D(mask, mapCords).r;"," original *= maskAlpha;"," gl_FragColor = original;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.onTextureLoaded=function(){this.uniforms.mapDimensions.value.x=this.uniforms.mask.value.width,this.uniforms.mapDimensions.value.y=this.uniforms.mask.value.height,this.uniforms.mask.value.baseTexture.off("loaded",this.boundLoadedFunction)},Object.defineProperties(i.prototype,{map:{get:function(){return this.uniforms.mask.value},set:function(t){this.uniforms.mask.value=t}},offset:{get:function(){return this.uniforms.offset.value},set:function(t){this.uniforms.offset.value=t}}})},{"./AbstractFilter":56}],58:[function(t,e){function i(){r.call(this),this.uniforms={dimensions:{type:"4fv",value:new Float32Array([1e4,100,10,10])},pixelSize:{type:"1f",value:8}},this.fragmentSrc=["precision mediump float;","uniform vec4 dimensions;","uniform float pixelSize;","uniform sampler2D uSampler;","float character(float n, vec2 p)","{"," p = floor(p*vec2(4.0, -4.0) + 2.5);"," if (clamp(p.x, 0.0, 4.0) == p.x && clamp(p.y, 0.0, 4.0) == p.y)"," {"," if (int(mod(n/exp2(p.x + 5.0*p.y), 2.0)) == 1) return 1.0;"," }"," return 0.0;","}","void main()","{"," vec2 uv = gl_FragCoord.xy;"," vec3 col = texture2D(uSampler, floor( uv / pixelSize ) * pixelSize / dimensions.xy).rgb;"," #ifdef HAS_GREENSCREEN"," float gray = (col.r + col.b)/2.0;"," #else"," float gray = (col.r + col.g + col.b)/3.0;"," #endif"," float n = 65536.0; // ."," if (gray > 0.2) n = 65600.0; // :"," if (gray > 0.3) n = 332772.0; // *"," if (gray > 0.4) n = 15255086.0; // o"," if (gray > 0.5) n = 23385164.0; // &"," if (gray > 0.6) n = 15252014.0; // 8"," if (gray > 0.7) n = 13199452.0; // @"," if (gray > 0.8) n = 11512810.0; // #"," vec2 p = mod( uv / ( pixelSize * 0.5 ), 2.0) - vec2(1.0);"," col = col * character(n, p);"," gl_FragColor = vec4(col, 1.0);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{size:{get:function(){return this.uniforms.pixelSize.value},set:function(t){this.uniforms.pixelSize.value=t}}})},{"./AbstractFilter":56}],59:[function(t,e){function i(){r.call(this),this.blurXFilter=new n,this.blurYFilter=new s,this.passes=[this.blurXFilter,this.blurYFilter]}var r=t("./AbstractFilter"),n=t("./BlurXFilter"),s=t("./BlurYFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{blur:{get:function(){return this.blurXFilter.blur},set:function(t){this.blurXFilter.blur=this.blurYFilter.blur=t}},blurX:{get:function(){return this.blurXFilter.blur},set:function(t){this.blurXFilter.blur=t}},blurY:{get:function(){return this.blurYFilter.blur},set:function(t){this.blurYFilter.blur=t}}})},{"./AbstractFilter":56,"./BlurXFilter":60,"./BlurYFilter":61}],60:[function(t,e){function i(){r.call(this),this.uniforms={blur:{type:"1f",value:1/512}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float blur;","uniform sampler2D uSampler;","void main(void) {"," vec4 sum = vec4(0.0);"," sum += texture2D(uSampler, vec2(vTextureCoord.x - 4.0*blur, vTextureCoord.y)) * 0.05;"," sum += texture2D(uSampler, vec2(vTextureCoord.x - 3.0*blur, vTextureCoord.y)) * 0.09;"," sum += texture2D(uSampler, vec2(vTextureCoord.x - 2.0*blur, vTextureCoord.y)) * 0.12;"," sum += texture2D(uSampler, vec2(vTextureCoord.x - blur, vTextureCoord.y)) * 0.15;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;"," sum += texture2D(uSampler, vec2(vTextureCoord.x + blur, vTextureCoord.y)) * 0.15;"," sum += texture2D(uSampler, vec2(vTextureCoord.x + 2.0*blur, vTextureCoord.y)) * 0.12;"," sum += texture2D(uSampler, vec2(vTextureCoord.x + 3.0*blur, vTextureCoord.y)) * 0.09;"," sum += texture2D(uSampler, vec2(vTextureCoord.x + 4.0*blur, vTextureCoord.y)) * 0.05;"," gl_FragColor = sum;","}"]}var r=t("./AbstractFilter"),n=1/7e3;i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{blur:{get:function(){return this.uniforms.blur.value/n -},set:function(t){this.uniforms.blur.value=n*t}}})},{"./AbstractFilter":56}],61:[function(t,e){function i(){r.call(this),this.uniforms={blur:{type:"1f",value:1/512}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float blur;","uniform sampler2D uSampler;","void main(void) {"," vec4 sum = vec4(0.0);"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 4.0*blur)) * 0.05;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 3.0*blur)) * 0.09;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - 2.0*blur)) * 0.12;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y - blur)) * 0.15;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y)) * 0.16;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + blur)) * 0.15;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 2.0*blur)) * 0.12;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 3.0*blur)) * 0.09;"," sum += texture2D(uSampler, vec2(vTextureCoord.x, vTextureCoord.y + 4.0*blur)) * 0.05;"," gl_FragColor = sum;","}"]}var r=t("./AbstractFilter"),n=1/7e3;i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{blur:{get:function(){return this.uniforms.blur.value/n},set:function(t){this.uniforms.blur.value=n*t}}})},{"./AbstractFilter":56}],62:[function(t,e){function i(){r.call(this),this.uniforms={matrix:{type:"mat4",value:[1,0,0,0,0,1,0,0,0,0,1,0,0,0,0,1]}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float invert;","uniform mat4 matrix;","uniform sampler2D uSampler;","void main(void) {"," gl_FragColor = texture2D(uSampler, vTextureCoord) * matrix;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{matrix:{get:function(){return this.uniforms.matrix.value},set:function(t){this.uniforms.matrix.value=t}}})},{"./AbstractFilter":56}],63:[function(t,e){function i(){r.call(this),this.uniforms={step:{type:"1f",value:5}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform sampler2D uSampler;","uniform float step;","void main(void) {"," vec4 color = texture2D(uSampler, vTextureCoord);"," color = floor(color * step) / step;"," gl_FragColor = color;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{step:{get:function(){return this.uniforms.step.value},set:function(t){this.uniforms.step.value=t}}})},{"./AbstractFilter":56}],64:[function(t,e){function i(t,e,i){r.call(this),this.uniforms={matrix:{type:"1fv",value:new Float32Array(t)},texelSizeX:{type:"1f",value:1/e},texelSizeY:{type:"1f",value:1/i}},this.fragmentSrc=["precision mediump float;","varying mediump vec2 vTextureCoord;","uniform sampler2D texture;","uniform float texelSizeX;","uniform float texelSizeY;","uniform float matrix[9];","vec2 px = vec2(texelSizeX, texelSizeY);","void main(void) {"," vec4 c11 = texture2D(texture, vTextureCoord - px);"," vec4 c12 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y - px.y));"," vec4 c13 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y - px.y));"," vec4 c21 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y) );"," vec4 c22 = texture2D(texture, vTextureCoord);"," vec4 c23 = texture2D(texture, vec2(vTextureCoord.x + px.x, vTextureCoord.y) );"," vec4 c31 = texture2D(texture, vec2(vTextureCoord.x - px.x, vTextureCoord.y + px.y) );"," vec4 c32 = texture2D(texture, vec2(vTextureCoord.x, vTextureCoord.y + px.y) );"," vec4 c33 = texture2D(texture, vTextureCoord + px );"," gl_FragColor = "," c11 * matrix[0] + c12 * matrix[1] + c13 * matrix[2] +"," c21 * matrix[3] + c22 * matrix[4] + c23 * matrix[5] +"," c31 * matrix[6] + c32 * matrix[7] + c33 * matrix[8];"," gl_FragColor.a = c22.a;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{matrix:{get:function(){return this.uniforms.matrix.value},set:function(t){this.uniforms.matrix.value=new Float32Array(t)}},width:{get:function(){return this.uniforms.texelSizeX.value},set:function(t){this.uniforms.texelSizeX.value=1/t}},height:{get:function(){return this.uniforms.texelSizeY.value},set:function(t){this.uniforms.texelSizeY.value=1/t}}})},{"./AbstractFilter":56}],65:[function(t,e){function i(){r.call(this),this.uniforms={blur:{type:"1f",value:1/512}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float blur;","uniform sampler2D uSampler;","void main(void) {"," float lum = length(texture2D(uSampler, vTextureCoord.xy).rgb);"," gl_FragColor = vec4(1.0, 1.0, 1.0, 1.0);"," if (lum < 1.00) {"," if (mod(gl_FragCoord.x + gl_FragCoord.y, 10.0) == 0.0) {"," gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);"," }"," }"," if (lum < 0.75) {"," if (mod(gl_FragCoord.x - gl_FragCoord.y, 10.0) == 0.0) {"," gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);"," }"," }"," if (lum < 0.50) {"," if (mod(gl_FragCoord.x + gl_FragCoord.y - 5.0, 10.0) == 0.0) {"," gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);"," }"," }"," if (lum < 0.3) {"," if (mod(gl_FragCoord.x - gl_FragCoord.y - 5.0, 10.0) == 0.0) {"," gl_FragColor = vec4(0.0, 0.0, 0.0, 1.0);"," }"," }","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{blur:{get:function(){return this.uniforms.blur.value/(1/7e3)},set:function(t){this.uniforms.blur.value=1/7e3*t}}})},{"./AbstractFilter":56}],66:[function(t,e){function i(t){r.call(this),t.baseTexture._powerOf2=!0,this.uniforms={displacementMap:{type:"sampler2D",value:t},scale:{type:"2f",value:{x:30,y:30}},offset:{type:"2f",value:{x:0,y:0}},mapDimensions:{type:"2f",value:{x:1,y:5112}},dimensions:{type:"4fv",value:[0,0,0,0]}},t.baseTexture.hasLoaded?this.onTextureLoaded():(this.boundLoadedFunction=this.onTextureLoaded.bind(this),t.baseTexture.on("loaded",this.boundLoadedFunction)),this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform sampler2D displacementMap;","uniform sampler2D uSampler;","uniform vec2 scale;","uniform vec2 offset;","uniform vec4 dimensions;","uniform vec2 mapDimensions;","void main(void) {"," vec2 mapCords = vTextureCoord.xy;"," mapCords += (dimensions.zw + offset)/ dimensions.xy ;"," mapCords.y *= -1.0;"," mapCords.y += 1.0;"," vec2 matSample = texture2D(displacementMap, mapCords).xy;"," matSample -= 0.5;"," matSample *= scale;"," matSample /= mapDimensions;"," gl_FragColor = texture2D(uSampler, vec2(vTextureCoord.x + matSample.x, vTextureCoord.y + matSample.y));"," gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb, 1.0);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.onTextureLoaded=function(){this.uniforms.mapDimensions.value.x=this.uniforms.displacementMap.value.width,this.uniforms.mapDimensions.value.y=this.uniforms.displacementMap.value.height,this.uniforms.displacementMap.value.baseTexture.off("loaded",this.boundLoadedFunction)},Object.defineProperties(i.prototype,{map:{get:function(){return this.uniforms.displacementMap.value},set:function(t){this.uniforms.displacementMap.value=t}},scale:{get:function(){return this.uniforms.scale.value},set:function(t){this.uniforms.scale.value=t}},offset:{get:function(){return this.uniforms.offset.value},set:function(t){this.uniforms.offset.value=t}}})},{"./AbstractFilter":56}],67:[function(t,e){function i(){r.call(this),this.uniforms={scale:{type:"1f",value:1},angle:{type:"1f",value:5},dimensions:{type:"4fv",value:[0,0,0,0]}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform vec4 dimensions;","uniform sampler2D uSampler;","uniform float angle;","uniform float scale;","float pattern() {"," float s = sin(angle), c = cos(angle);"," vec2 tex = vTextureCoord * dimensions.xy;"," vec2 point = vec2("," c * tex.x - s * tex.y,"," s * tex.x + c * tex.y"," ) * scale;"," return (sin(point.x) * sin(point.y)) * 4.0;","}","void main() {"," vec4 color = texture2D(uSampler, vTextureCoord);"," float average = (color.r + color.g + color.b) / 3.0;"," gl_FragColor = vec4(vec3(average * 10.0 - 5.0 + pattern()), color.a);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{scale:{get:function(){return this.uniforms.scale.value},set:function(t){this.uniforms.scale.value=t}},angle:{get:function(){return this.uniforms.angle.value},set:function(t){this.uniforms.angle.value=t}}})},{"./AbstractFilter":56}],68:[function(t,e){function i(){this.visible=!0,this.renderable=!0}i.prototype.constructor=i,e.exports=i},{}],69:[function(t,e){function i(){r.call(this),this.uniforms={gray:{type:"1f",value:1}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform sampler2D uSampler;","uniform float gray;","void main(void) {"," gl_FragColor = texture2D(uSampler, vTextureCoord);"," gl_FragColor.rgb = mix(gl_FragColor.rgb, vec3(0.2126*gl_FragColor.r + 0.7152*gl_FragColor.g + 0.0722*gl_FragColor.b), gray);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{gray:{get:function(){return this.uniforms.gray.value},set:function(t){this.uniforms.gray.value=t}}})},{"./AbstractFilter":56}],70:[function(t,e){function i(){r.call(this),this.uniforms={invert:{type:"1f",value:1}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float invert;","uniform sampler2D uSampler;","void main(void) {"," gl_FragColor = texture2D(uSampler, vTextureCoord);"," gl_FragColor.rgb = mix( (vec3(1)-gl_FragColor.rgb) * gl_FragColor.a, gl_FragColor.rgb, 1.0 - invert);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{invert:{get:function(){return this.uniforms.invert.value},set:function(t){this.uniforms.invert.value=t}}})},{"./AbstractFilter":56}],71:[function(t,e){function i(){r.call(this),this.uniforms={noise:{type:"1f",value:.5}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float noise;","uniform sampler2D uSampler;","float rand(vec2 co) {"," return fract(sin(dot(co.xy ,vec2(12.9898,78.233))) * 43758.5453);","}","void main() {"," vec4 color = texture2D(uSampler, vTextureCoord);"," float diff = (rand(vTextureCoord) - 0.5) * noise;"," color.r += diff;"," color.g += diff;"," color.b += diff;"," gl_FragColor = color;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{noise:{get:function(){return this.uniforms.noise.value},set:function(t){this.dirty=!0,this.uniforms.noise.value=t}}})},{"./AbstractFilter":56}],72:[function(t,e){function i(t){r.call(this),t.baseTexture._powerOf2=!0,this.uniforms={displacementMap:{type:"sampler2D",value:t},scale:{type:"2f",value:{x:15,y:15}},offset:{type:"2f",value:{x:0,y:0}},mapDimensions:{type:"2f",value:{x:1,y:1}},dimensions:{type:"4f",value:[0,0,0,0]},LightPos:{type:"3f",value:[0,1,0]}},t.baseTexture.hasLoaded?this.onTextureLoaded():(this.boundLoadedFunction=this.onTextureLoaded.bind(this),t.baseTexture.on("loaded",this.boundLoadedFunction)),this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying float vColor;","uniform sampler2D displacementMap;","uniform sampler2D uSampler;","uniform vec4 dimensions;","const vec2 Resolution = vec2(1.0,1.0);","uniform vec3 LightPos;","const vec4 LightColor = vec4(1.0, 1.0, 1.0, 1.0);","const vec4 AmbientColor = vec4(1.0, 1.0, 1.0, 0.5);","const vec3 Falloff = vec3(0.0, 1.0, 0.2);","uniform vec3 LightDir;","uniform vec2 mapDimensions;","void main(void) {"," vec2 mapCords = vTextureCoord.xy;"," vec4 color = texture2D(uSampler, vTextureCoord.st);"," vec3 nColor = texture2D(displacementMap, vTextureCoord.st).rgb;"," mapCords *= vec2(dimensions.x/512.0, dimensions.y/512.0);"," mapCords.y *= -1.0;"," mapCords.y += 1.0;"," vec4 DiffuseColor = texture2D(uSampler, vTextureCoord);"," vec3 NormalMap = texture2D(displacementMap, mapCords).rgb;"," vec3 LightDir = vec3(LightPos.xy - (mapCords.xy), LightPos.z);"," float D = length(LightDir);"," vec3 N = normalize(NormalMap * 2.0 - 1.0);"," vec3 L = normalize(LightDir);"," vec3 Diffuse = (LightColor.rgb * LightColor.a) * max(dot(N, L), 0.0);"," vec3 Ambient = AmbientColor.rgb * AmbientColor.a;"," float Attenuation = 1.0 / ( Falloff.x + (Falloff.y*D) + (Falloff.z*D*D) );"," vec3 Intensity = Ambient + Diffuse * Attenuation;"," vec3 FinalColor = DiffuseColor.rgb * Intensity;"," gl_FragColor = vColor * vec4(FinalColor, DiffuseColor.a);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.onTextureLoaded=function(){this.uniforms.mapDimensions.value.x=this.uniforms.displacementMap.value.width,this.uniforms.mapDimensions.value.y=this.uniforms.displacementMap.value.height,this.uniforms.displacementMap.value.baseTexture.off("loaded",this.boundLoadedFunction)},Object.defineProperties(i.prototype,{map:{get:function(){return this.uniforms.displacementMap.value},set:function(t){this.uniforms.displacementMap.value=t}},scale:{get:function(){return this.uniforms.scale.value},set:function(t){this.uniforms.scale.value=t}},offset:{get:function(){return this.uniforms.offset.value},set:function(t){this.uniforms.offset.value=t}}})},{"./AbstractFilter":56}],73:[function(t,e){function i(){r.call(this),this.uniforms={invert:{type:"1f",value:0},dimensions:{type:"4fv",value:new Float32Array([1e4,100,10,10])},pixelSize:{type:"2f",value:{x:10,y:10}}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform vec2 testDim;","uniform vec4 dimensions;","uniform vec2 pixelSize;","uniform sampler2D uSampler;","void main(void) {"," vec2 coord = vTextureCoord;"," vec2 size = dimensions.xy/pixelSize;"," vec2 color = floor( ( vTextureCoord * size ) ) / size + pixelSize/dimensions.xy * 0.5;"," gl_FragColor = texture2D(uSampler, color);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{size:{get:function(){return this.uniforms.pixelSize.value},set:function(t){this.uniforms.pixelSize.value=t}}})},{"./AbstractFilter":56}],74:[function(t,e){function i(){r.call(this),this.passes=[this],this.uniforms={red:{type:"2f",value:{x:20,y:20}},green:{type:"2f",value:{x:-20,y:20}},blue:{type:"2f",value:{x:20,y:-20}},dimensions:{type:"4fv",value:[0,0,0,0]}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform vec2 red;","uniform vec2 green;","uniform vec2 blue;","uniform vec4 dimensions;","uniform sampler2D uSampler;","void main(void) {"," gl_FragColor.r = texture2D(uSampler, vTextureCoord + red/dimensions.xy).r;"," gl_FragColor.g = texture2D(uSampler, vTextureCoord + green/dimensions.xy).g;"," gl_FragColor.b = texture2D(uSampler, vTextureCoord + blue/dimensions.xy).b;"," gl_FragColor.a = texture2D(uSampler, vTextureCoord).a;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{red:{get:function(){return this.uniforms.red.value},set:function(t){this.uniforms.red.value=t}},green:{get:function(){return this.uniforms.green.value},set:function(t){this.uniforms.green.value=t}},blue:{get:function(){return this.uniforms.blue.value},set:function(t){this.uniforms.blue.value=t}}})},{"./AbstractFilter":56}],75:[function(t,e){function i(){r.call(this),this.uniforms={sepia:{type:"1f",value:1}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float sepia;","uniform sampler2D uSampler;","const mat3 sepiaMatrix = mat3(0.3588, 0.7044, 0.1368, 0.2990, 0.5870, 0.1140, 0.2392, 0.4696, 0.0912);","void main(void) {"," gl_FragColor = texture2D(uSampler, vTextureCoord);"," gl_FragColor.rgb = mix( gl_FragColor.rgb, gl_FragColor.rgb * sepiaMatrix, sepia);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{sepia:{get:function(){return this.uniforms.sepia.value},set:function(t){this.uniforms.sepia.value=t}}})},{"./AbstractFilter":56}],76:[function(t,e){function i(){r.call(this),this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","uniform sampler2D uSampler;","const vec2 delta = vec2(1.0/10.0, 0.0);","float random(vec3 scale, float seed) {"," return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);","}","void main(void) {"," vec4 color = vec4(0.0);"," float total = 0.0;"," float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);"," for (float t = -30.0; t <= 30.0; t++) {"," float percent = (t + offset - 0.5) / 30.0;"," float weight = 1.0 - abs(percent);"," vec4 sample = texture2D(uSampler, vTextureCoord + delta * percent);"," sample.rgb *= sample.a;"," color += sample * weight;"," total += weight;"," }"," gl_FragColor = color / total;"," gl_FragColor.rgb /= gl_FragColor.a + 0.00001;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i},{"./AbstractFilter":56}],77:[function(t,e){function i(){r.call(this),this.tiltShiftXFilter=new n,this.tiltShiftYFilter=new s,this.tiltShiftXFilter.updateDelta(),this.tiltShiftXFilter.updateDelta(),this.passes=[this.tiltShiftXFilter,this.tiltShiftYFilter]}var r=t("./AbstractFilter"),n=t("./TiltShiftXFilter"),s=t("./TiltShiftYFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{blur:{get:function(){return this.tiltShiftXFilter.blur},set:function(t){this.tiltShiftXFilter.blur=this.tiltShiftYFilter.blur=t}},gradientBlur:{get:function(){return this.tiltShiftXFilter.gradientBlur},set:function(t){this.tiltShiftXFilter.gradientBlur=this.tiltShiftYFilter.gradientBlur=t}},start:{get:function(){return this.tiltShiftXFilter.start},set:function(t){this.tiltShiftXFilter.start=this.tiltShiftYFilter.start=t}},end:{get:function(){return this.tiltShiftXFilter.end},set:function(t){this.tiltShiftXFilter.end=this.tiltShiftYFilter.end=t}}})},{"./AbstractFilter":56,"./TiltShiftXFilter":78,"./TiltShiftYFilter":79}],78:[function(t,e){function i(){r.call(this),this.uniforms={blur:{type:"1f",value:100},gradientBlur:{type:"1f",value:600},start:{type:"2f",value:{x:0,y:window.screenHeight/2}},end:{type:"2f",value:{x:600,y:window.screenHeight/2}},delta:{type:"2f",value:{x:30,y:30}},texSize:{type:"2f",value:{x:window.screenWidth,y:window.screenHeight}}},this.updateDelta(),this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","uniform sampler2D uSampler;","uniform float blur;","uniform float gradientBlur;","uniform vec2 start;","uniform vec2 end;","uniform vec2 delta;","uniform vec2 texSize;","float random(vec3 scale, float seed) {"," return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);","}","void main(void) {"," vec4 color = vec4(0.0);"," float total = 0.0;"," float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);"," vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));"," float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;"," for (float t = -30.0; t <= 30.0; t++) {"," float percent = (t + offset - 0.5) / 30.0;"," float weight = 1.0 - abs(percent);"," vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);"," sample.rgb *= sample.a;"," color += sample * weight;"," total += weight;"," }"," gl_FragColor = color / total;"," gl_FragColor.rgb /= gl_FragColor.a + 0.00001;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.updateDelta=function(){var t=this.uniforms.end.value.x-this.uniforms.start.value.x,e=this.uniforms.end.value.y-this.uniforms.start.value.y,i=Math.sqrt(t*t+e*e);this.uniforms.delta.value.x=t/i,this.uniforms.delta.value.y=e/i},Object.defineProperties(i.prototype,{blur:{get:function(){return this.uniforms.blur.value},set:function(t){this.uniforms.blur.value=t}},gradientBlur:{get:function(){return this.uniforms.gradientBlur.value},set:function(t){this.uniforms.gradientBlur.value=t}},start:{get:function(){return this.uniforms.start.value},set:function(t){this.uniforms.start.value=t,this.updateDelta()}},end:{get:function(){return this.uniforms.end.value},set:function(t){this.uniforms.end.value=t,this.updateDelta()}}})},{"./AbstractFilter":56}],79:[function(t,e){function i(){r.call(this),this.uniforms={blur:{type:"1f",value:100},gradientBlur:{type:"1f",value:600},start:{type:"2f",value:{x:0,y:window.screenHeight/2}},end:{type:"2f",value:{x:600,y:window.screenHeight/2}},delta:{type:"2f",value:{x:30,y:30}},texSize:{type:"2f",value:{x:window.screenWidth,y:window.screenHeight}}},this.updateDelta(),this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","uniform sampler2D uSampler;","uniform float blur;","uniform float gradientBlur;","uniform vec2 start;","uniform vec2 end;","uniform vec2 delta;","uniform vec2 texSize;","float random(vec3 scale, float seed) {"," return fract(sin(dot(gl_FragCoord.xyz + seed, scale)) * 43758.5453 + seed);","}","void main(void) {"," vec4 color = vec4(0.0);"," float total = 0.0;"," float offset = random(vec3(12.9898, 78.233, 151.7182), 0.0);"," vec2 normal = normalize(vec2(start.y - end.y, end.x - start.x));"," float radius = smoothstep(0.0, 1.0, abs(dot(vTextureCoord * texSize - start, normal)) / gradientBlur) * blur;"," for (float t = -30.0; t <= 30.0; t++) {"," float percent = (t + offset - 0.5) / 30.0;"," float weight = 1.0 - abs(percent);"," vec4 sample = texture2D(uSampler, vTextureCoord + delta / texSize * percent * radius);"," sample.rgb *= sample.a;"," color += sample * weight;"," total += weight;"," }"," gl_FragColor = color / total;"," gl_FragColor.rgb /= gl_FragColor.a + 0.00001;","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,i.prototype.updateDelta=function(){var t=this.uniforms.end.value.x-this.uniforms.start.value.x,e=this.uniforms.end.value.y-this.uniforms.start.value.y,i=Math.sqrt(t*t+e*e);this.uniforms.delta.value.x=-e/i,this.uniforms.delta.value.y=t/i},Object.defineProperties(i.prototype,{blur:{get:function(){return this.uniforms.blur.value},set:function(t){this.uniforms.blur.value=t}},gradientBlur:{get:function(){return this.uniforms.gradientBlur.value},set:function(t){this.uniforms.gradientBlur.value=t}},start:{get:function(){return this.uniforms.start.value},set:function(t){this.uniforms.start.value=t,this.updateDelta()}},end:{get:function(){return this.uniforms.end.value},set:function(t){this.uniforms.end.value=t,this.updateDelta()}}})},{"./AbstractFilter":56}],80:[function(t,e){function i(){r.call(this),this.uniforms={radius:{type:"1f",value:.5},angle:{type:"1f",value:5},offset:{type:"2f",value:{x:.5,y:.5}}},this.fragmentSrc=["precision mediump float;","varying vec2 vTextureCoord;","varying vec4 vColor;","uniform float radius;","uniform float angle;","uniform vec2 offset;","uniform sampler2D uSampler;","void main(void) {"," vec2 coord = vTextureCoord - offset;"," float distance = length(coord);"," if (distance < radius) {"," float ratio = (radius - distance) / radius;"," float angleMod = ratio * ratio * angle;"," float s = sin(angleMod);"," float c = cos(angleMod);"," coord = vec2(coord.x * c - coord.y * s, coord.x * s + coord.y * c);"," }"," gl_FragColor = texture2D(uSampler, coord+offset);","}"]}var r=t("./AbstractFilter");i.prototype=Object.create(r.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{offset:{get:function(){return this.uniforms.offset.value},set:function(t){this.uniforms.offset.value=t}},radius:{get:function(){return this.uniforms.radius.value},set:function(t){this.uniforms.radius.value=t}},angle:{get:function(){return this.uniforms.angle.value},set:function(t){this.uniforms.angle.value=t}}})},{"./AbstractFilter":56}],81:[function(t,e){e.exports={AbstractFilter:t("./AbstractFilter"),AlphaMaskFilter:t("./AlphaMaskFilter"),AsciiFilter:t("./AsciiFilter"),BlurFilter:t("./BlurFilter"),BlurXFilter:t("./BlurXFilter"),BlurYFilter:t("./BlurYFilter"),ColorMatrixFilter:t("./ColorMatrixFilter"),ColorStepFilter:t("./ColorStepFilter"),ConvolutionFilter:t("./ConvolutionFilter"),CrossHatchFilter:t("./CrossHatchFilter"),DisplacementFilter:t("./DisplacementFilter"),DotScreenFilter:t("./DotScreenFilter"),FilterBlock:t("./FilterBlock"),GrayFilter:t("./GrayFilter"),InvertFilter:t("./InvertFilter"),NoiseFilter:t("./NoiseFilter"),NormalMapFilter:t("./NormalMapFilter"),PixelateFilter:t("./PixelateFilter"),RGBSplitFilter:t("./RGBSplitFilter"),SepiaFilter:t("./SepiaFilter"),SmartBlurFilter:t("./SmartBlurFilter"),TiltShiftFilter:t("./TiltShiftFilter"),TiltShiftXFilter:t("./TiltShiftXFilter"),TiltShiftYFilter:t("./TiltShiftYFilter"),TwistFilter:t("./TwistFilter")}},{"./AbstractFilter":56,"./AlphaMaskFilter":57,"./AsciiFilter":58,"./BlurFilter":59,"./BlurXFilter":60,"./BlurYFilter":61,"./ColorMatrixFilter":62,"./ColorStepFilter":63,"./ConvolutionFilter":64,"./CrossHatchFilter":65,"./DisplacementFilter":66,"./DotScreenFilter":67,"./FilterBlock":68,"./GrayFilter":69,"./InvertFilter":70,"./NoiseFilter":71,"./NormalMapFilter":72,"./PixelateFilter":73,"./RGBSplitFilter":74,"./SepiaFilter":75,"./SmartBlurFilter":76,"./TiltShiftFilter":77,"./TiltShiftXFilter":78,"./TiltShiftYFilter":79,"./TwistFilter":80}],82:[function(t,e){function i(){this.global=new r.math.Point,this.target=null,this.originalEvent=null}var r=t("../core");i.prototype.constructor=i,e.exports=i,i.prototype.getLocalPosition=function(t,e){var i=t.worldTransform,n=this.global,s=i.a,o=i.c,a=i.tx,h=i.b,l=i.d,u=i.ty,c=1/(s*l+o*-h);return e=e||new r.math.Point,e.x=l*c*n.x+-o*c*n.y+(u*o-a*l)*c,e.y=s*c*n.y+-h*c*n.x+(-u*s+a*h)*c,e}},{"../core":9}],83:[function(t,e){function i(t){this.stage=t,this.mouse=new n,this.touches={},this.tempPoint=new r.math.Point,this.mouseoverEnabled=!0,this.pool=[],this.interactiveItems=[],this.interactionDOMElement=null,this.eventsAdded=!1,this.onMouseMove=this.onMouseMove.bind(this),this.onMouseDown=this.onMouseDown.bind(this),this.onMouseOut=this.onMouseOut.bind(this),this.onMouseUp=this.onMouseUp.bind(this),this.onTouchStart=this.onTouchStart.bind(this),this.onTouchEnd=this.onTouchEnd.bind(this),this.onTouchMove=this.onTouchMove.bind(this),this.last=0,this.currentCursorStyle="inherit",this.mouseOut=!1,this.resolution=1,this._tempPoint=new r.math.Point}var r=t("../core"),n=t("./InteractionData"),s=30,o=!0;i.prototype.constructor=i,e.exports=i,i.prototype.collectInteractiveSprite=function(t,e){for(var i=t.children,r=i.length,n=r-1;n>=0;n--){var s=i[n];s._interactive?(e.interactiveChildren=!0,this.interactiveItems.push(s),s.children.length>0&&this.collectInteractiveSprite(s,s)):(s.__iParent=null,s.children.length>0&&this.collectInteractiveSprite(s,e))}},i.prototype.setTargetElement=function(t,e){this.removeEvents(),this.interactionDOMElement=t,this.resolution=e||1,this.addEvents()},i.prototype.addEvents=function(){this.interactionDOMElement&&(window.navigator.msPointerEnabled&&(this.interactionDOMElement.style["-ms-content-zooming"]="none",this.interactionDOMElement.style["-ms-touch-action"]="none"),this.interactionDOMElement.addEventListener("mousemove",this.onMouseMove,!0),this.interactionDOMElement.addEventListener("mousedown",this.onMouseDown,!0),this.interactionDOMElement.addEventListener("mouseout",this.onMouseOut,!0),this.interactionDOMElement.addEventListener("touchstart",this.onTouchStart,!0),this.interactionDOMElement.addEventListener("touchend",this.onTouchEnd,!0),this.interactionDOMElement.addEventListener("touchmove",this.onTouchMove,!0),window.addEventListener("mouseup",this.onMouseUp,!0),this.eventsAdded=!0)},i.prototype.removeEvents=function(){this.interactionDOMElement&&(window.navigator.msPointerEnabled&&(this.interactionDOMElement.style["-ms-content-zooming"]="",this.interactionDOMElement.style["-ms-touch-action"]=""),this.interactionDOMElement.removeEventListener("mousemove",this.onMouseMove,!0),this.interactionDOMElement.removeEventListener("mousedown",this.onMouseDown,!0),this.interactionDOMElement.removeEventListener("mouseout",this.onMouseOut,!0),this.interactionDOMElement.removeEventListener("touchstart",this.onTouchStart,!0),this.interactionDOMElement.removeEventListener("touchend",this.onTouchEnd,!0),this.interactionDOMElement.removeEventListener("touchmove",this.onTouchMove,!0),this.interactionDOMElement=null,window.removeEventListener("mouseup",this.onMouseUp,!0),this.eventsAdded=!1)},i.prototype.update=function(){if(this.interactionDOMElement){var t=Date.now(),e=t-this.last;if(e=e*s/1e3,!(1>e)){this.last=t;var i=0;this.dirty&&this.rebuildInteractiveGraph();var r=this.interactiveItems.length,n="inherit",o=!1;for(i=0;r>i;i++){var a=this.interactiveItems[i];a.__hit=this.hitTest(a,this.mouse),this.mouse.target=a,a.__hit&&!o?(a.buttonMode&&(n=a.defaultCursor),a.interactiveChildren||(o=!0),a.__isOver||(a.mouseover&&a.mouseover(this.mouse),a.__isOver=!0)):a.__isOver&&(a.mouseout&&a.mouseout(this.mouse),a.__isOver=!1)}this.currentCursorStyle!==n&&(this.currentCursorStyle=n,this.interactionDOMElement.style.cursor=n)}}},i.prototype.rebuildInteractiveGraph=function(){this.dirty=!1;for(var t=this.interactiveItems.length,e=0;t>e;e++)this.interactiveItems[e].interactiveChildren=!1;this.interactiveItems.length=0,this.stage.interactive&&this.interactiveItems.push(this.stage),this.collectInteractiveSprite(this.stage,this.stage)},i.prototype.onMouseMove=function(t){this.dirty&&this.rebuildInteractiveGraph(),this.mouse.originalEvent=t;var e=this.interactionDOMElement.getBoundingClientRect();this.mouse.global.x=(t.clientX-e.left)*(this.interactionDOMElement.width/e.width)/this.resolution,this.mouse.global.y=(t.clientY-e.top)*(this.interactionDOMElement.height/e.height)/this.resolution;for(var i=this.interactiveItems.length,r=0;i>r;r++){var n=this.interactiveItems[r];n.mousemove&&n.mousemove(this.mouse)}},i.prototype.onMouseDown=function(t){this.dirty&&this.rebuildInteractiveGraph(),this.mouse.originalEvent=t,o&&this.mouse.originalEvent.preventDefault();for(var e=this.interactiveItems.length,i=this.mouse.originalEvent,r=2===i.button||3===i.which,n=r?"rightdown":"mousedown",s=r?"rightclick":"click",a=r?"__rightIsDown":"__mouseIsDown",h=r?"__isRightDown":"__isDown",l=0;e>l;l++){var u=this.interactiveItems[l];if((u[n]||u[s])&&(u[a]=!0,u.__hit=this.hitTest(u,this.mouse),u.__hit&&(u[n]&&u[n](this.mouse),u[h]=!0,!u.interactiveChildren)))break}},i.prototype.onMouseOut=function(t){this.dirty&&this.rebuildInteractiveGraph(),this.mouse.originalEvent=t;var e=this.interactiveItems.length; -this.interactionDOMElement.style.cursor="inherit";for(var i=0;e>i;i++){var r=this.interactiveItems[i];r.__isOver&&(this.mouse.target=r,r.mouseout&&r.mouseout(this.mouse),r.__isOver=!1)}this.mouseOut=!0,this.mouse.global.x=-1e4,this.mouse.global.y=-1e4},i.prototype.onMouseUp=function(t){this.dirty&&this.rebuildInteractiveGraph(),this.mouse.originalEvent=t;for(var e=this.interactiveItems.length,i=!1,r=this.mouse.originalEvent,n=2===r.button||3===r.which,s=n?"rightup":"mouseup",o=n?"rightclick":"click",a=n?"rightupoutside":"mouseupoutside",h=n?"__isRightDown":"__isDown",l=0;e>l;l++){var u=this.interactiveItems[l];(u[o]||u[s]||u[a])&&(u.__hit=this.hitTest(u,this.mouse),u.__hit&&!i?(u[s]&&u[s](this.mouse),u[h]&&u[o]&&u[o](this.mouse),u.interactiveChildren||(i=!0)):u[h]&&u[a]&&u[a](this.mouse),u[h]=!1)}},i.prototype.hitTest=function(t,e){var i=e.global;if(!t.worldVisible)return!1;t.worldTransform.applyInverse(i,this._tempPoint);var n,s=this._tempPoint.x,o=this._tempPoint.y;if(e.target=t,t.hitArea&&t.hitArea.contains)return t.hitArea.contains(s,o);if(t instanceof r.Sprite){var a,h=t.texture.frame.width,l=t.texture.frame.height,u=-h*t.anchor.x;if(s>u&&u+h>s&&(a=-l*t.anchor.y,o>a&&a+l>o))return!0}else if(t instanceof r.Graphics){var c=t.graphicsData;for(n=0;nn;n++){var p=t.children[n],v=this.hitTest(p,e);if(v)return e.target=t,!0}return!1},i.prototype.onTouchMove=function(t){this.dirty&&this.rebuildInteractiveGraph();var e,i=this.interactionDOMElement.getBoundingClientRect(),r=t.changedTouches,n=0;for(n=0;nl;l++){var u=this.interactiveItems[l];if((u.touchstart||u.tap)&&(u.__hit=this.hitTest(u,a),u.__hit&&(u.touchstart&&u.touchstart(a),u.__isDown=!0,u.__touchData=u.__touchData||{},u.__touchData[s.identifier]=a,!u.interactiveChildren)))break}}},i.prototype.onTouchEnd=function(t){this.dirty&&this.rebuildInteractiveGraph();for(var e=this.interactionDOMElement.getBoundingClientRect(),i=t.changedTouches,r=0;rh;h++){var l=this.interactiveItems[h];l.__touchData&&l.__touchData[n.identifier]&&(l.__hit=this.hitTest(l,l.__touchData[n.identifier]),s.originalEvent=t,(l.touchend||l.tap)&&(l.__hit&&!o?(l.touchend&&l.touchend(s),l.__isDown&&l.tap&&l.tap(s),l.interactiveChildren||(o=!0)):l.__isDown&&l.touchendoutside&&l.touchendoutside(s),l.__isDown=!1),l.__touchData[n.identifier]=null)}this.pool.push(s),this.touches[n.identifier]=null}}},{"../core":9,"./InteractionData":82}],84:[function(t,e){e.exports={InteractionData:t("./InteractionData"),InteractionManager:t("./InteractionManager")}},{"./InteractionData":82,"./InteractionManager":83}],85:[function(t,e){function i(t,e){this.url=t,this.baseUrl=t.replace(/[^\/]*$/,""),this.crossorigin=e,this.loaded=!1}var r=t("../core"),n=t("./ImageLoader");i.prototype.constructor=i,e.exports=i,r.utils.eventTarget.mixin(i.prototype),i.prototype.load=function(){this.ajaxRequest=new r.utils.AjaxRequest,this.ajaxRequest.onreadystatechange=this.onAtlasLoaded.bind(this),this.ajaxRequest.open("GET",this.url,!0),this.ajaxRequest.overrideMimeType&&this.ajaxRequest.overrideMimeType("application/json"),this.ajaxRequest.send(null)},i.prototype.onAtlasLoaded=function(){if(4===this.ajaxRequest.readyState)if(200===this.ajaxRequest.status||-1===window.location.href.indexOf("http")){this.atlas={meta:{image:[]},frames:[]};var t=this.ajaxRequest.responseText.split(/\r?\n/),e=-3,i=0,s=null,o=!1,a=0,h=0,l=this.onLoaded.bind(this);for(a=0;a0){if(o===a)this.atlas.meta.image.push(t[a]),i=this.atlas.meta.image.length-1,this.atlas.frames.push({}),e=-3;else if(e>0)if(e%7===1)null!=s&&(this.atlas.frames[i][s.name]=s),s={name:t[a],frame:{}};else{var u=t[a].split(" ");if(e%7===3)s.frame.x=Number(u[1].replace(",","")),s.frame.y=Number(u[2]);else if(e%7===4)s.frame.w=Number(u[1].replace(",","")),s.frame.h=Number(u[2]);else if(e%7===5){var c={x:0,y:0,w:Number(u[1].replace(",","")),h:Number(u[2])};c.w>s.frame.w||c.h>s.frame.h?(s.trimmed=!0,s.realSize=c):s.trimmed=!1}}e++}if(null!=s&&(this.atlas.frames[i][s.name]=s),this.atlas.meta.image.length>0){for(this.images=[],h=0;hthis.currentImageId?(this.currentImageId++,this.images[this.currentImageId].load()):(this.loaded=!0,this.emit("loaded",{content:this}))},i.prototype.onError=function(){this.emit("error",{content:this})}},{"../core":9,"./ImageLoader":87}],86:[function(t,e){function i(t,e){this.url=t,this.crossorigin=e,this.baseUrl=t.replace(/[^\/]*$/,""),this.texture=null}var r=t("../core"),n=t("./ImageLoader");i.prototype.constructor=i,e.exports=i,r.utils.eventTarget.mixin(i.prototype),i.prototype.load=function(){this.ajaxRequest=new r.utils.AjaxRequest,this.ajaxRequest.onreadystatechange=this.onXMLLoaded.bind(this),this.ajaxRequest.open("GET",this.url,!0),this.ajaxRequest.overrideMimeType&&this.ajaxRequest.overrideMimeType("application/xml"),this.ajaxRequest.send(null)},i.prototype.onXMLLoaded=function(){if(4===this.ajaxRequest.readyState&&(200===this.ajaxRequest.status||-1===window.location.protocol.indexOf("http"))){var t=this.ajaxRequest.responseXML;if(!t||/MSIE 9/i.test(navigator.userAgent)||navigator.isCocoonJS)if("function"==typeof window.DOMParser){var e=new DOMParser;t=e.parseFromString(this.ajaxRequest.responseText,"text/xml")}else{var i=document.createElement("div");i.innerHTML=this.ajaxRequest.responseText,t=i}var s=this.baseUrl+t.getElementsByTagName("page")[0].getAttribute("file"),o=new n(s,this.crossorigin);this.texture=o.texture.baseTexture;var a={},h=t.getElementsByTagName("info")[0],l=t.getElementsByTagName("common")[0];a.font=h.getAttribute("face"),a.size=parseInt(h.getAttribute("size"),10),a.lineHeight=parseInt(l.getAttribute("lineHeight"),10),a.chars={};for(var u=t.getElementsByTagName("char"),c=0;ca;++a)for(var h=0;n>h;++h,++o){var l=new r.Texture(this.texture.baseTexture,new r.math.Rectangle(h*t,a*e,t,e));this.frames.push(l),i&&(r.utils.TextureCache[i+"-"+o]=l)}this.load()}},{"../core":9}],88:[function(t,e){function i(t,e){this.url=t,this.crossorigin=e,this.baseUrl=t.replace(/[^\/]*$/,""),this.loaded=!1}var r=t("../core"),n=t("../spine/SpineRuntime"),s=t("./ImageLoader"),o=t("./SpineTextureLoader");i.prototype.constructor=i,e.exports=i,r.utils.eventTarget.mixin(i.prototype),i.prototype.load=function(){window.XDomainRequest&&this.crossorigin?(this.ajaxRequest=new window.XDomainRequest,this.ajaxRequest.timeout=3e3,this.ajaxRequest.onerror=this.onError.bind(this),this.ajaxRequest.ontimeout=this.onError.bind(this),this.ajaxRequest.onprogress=function(){},this.ajaxRequest.onload=this.onJSONLoaded.bind(this)):(this.ajaxRequest=window.XMLHttpRequest?new window.XMLHttpRequest:new window.ActiveXObject("Microsoft.XMLHTTP"),this.ajaxRequest.onreadystatechange=this.onReadyStateChanged.bind(this)),this.ajaxRequest.open("GET",this.url,!0),this.ajaxRequest.send()},i.prototype.onReadyStateChanged=function(){4!==this.ajaxRequest.readyState||200!==this.ajaxRequest.status&&-1!==window.location.href.indexOf("http")||this.onJSONLoaded()},i.prototype.onJSONLoaded=function(){if(!this.ajaxRequest.responseText)return void this.onError();if(this.json=JSON.parse(this.ajaxRequest.responseText),this.json.frames){var t=this.baseUrl+this.json.meta.image,e=new s(t,this.crossorigin),a=this.json.frames;this.texture=e.texture.baseTexture,e.addEventListener("loaded",this.onLoaded.bind(this)),e.addEventListener("error",this.onError.bind(this));for(var h in a){var l=a[h].frame;if(l){var u=new r.math.Rectangle(l.x,l.y,l.w,l.h),c=u.clone(),d=null;if(a[h].trimmed){var f=a[h].sourceSize,p=a[h].spriteSourceSize;d=new r.math.Rectangle(p.x,p.y,f.w,f.h)}r.utils.TextureCache[h]=new r.Texture(this.texture,u,c,d)}}e.load()}else if(this.json.bones)if(r.utils.AnimCache[this.url])this.onLoaded();else{var v=this.url.substr(0,this.url.lastIndexOf("."))+".atlas",m=new i(v,this.crossorigin),g=this;m.onJSONLoaded=function(){if(!this.ajaxRequest.responseText)return void this.onError();var t=new o(this.url.substring(0,this.url.lastIndexOf("/"))),e=new n.Atlas(this.ajaxRequest.responseText,t),i=new n.AtlasAttachmentLoader(e),s=new n.SkeletonJson(i),a=s.readSkeletonData(g.json);r.utils.AnimCache[g.url]=a,g.spine=a,g.spineAtlas=e,g.spineAtlasLoader=m,t.loadingCount>0?t.addEventListener("loadedBaseTexture",function(t){t.content.content.loadingCount<=0&&g.onLoaded()}):g.onLoaded()},m.load()}else this.onLoaded()},i.prototype.onLoaded=function(){this.loaded=!0,this.dispatchEvent({type:"loaded",content:this})},i.prototype.onError=function(){this.dispatchEvent({type:"error",content:this})}},{"../core":9,"../spine/SpineRuntime":94,"./ImageLoader":87,"./SpineTextureLoader":90}],89:[function(t,e){function i(t,e){this.url=t,this.crossorigin=e,this.loaded=!1}var r=t("../core"),n=t("./JsonLoader");i.prototype.constructor=i,e.exports=i,r.utils.eventTarget.mixin(i.prototype),i.prototype.load=function(){var t=this,e=new n(this.url,this.crossorigin);e.on("loaded",function(e){t.json=e.data.content.json,t.onLoaded()}),e.load()},i.prototype.onLoaded=function(){this.loaded=!0,this.emit("loaded",{content:this})}},{"../core":9,"./JsonLoader":88}],90:[function(t,e){function i(t,e){this.basePath=t,this.crossorigin=e,this.loadingCount=0}var r=t("../core");i.prototype.constructor=i,e.exports=i,r.utils.eventTarget.mixin(i.prototype),i.prototype.load=function(t,e){if(t.rendererObject=r.BaseTexture.fromImage(this.basePath+"/"+e,this.crossorigin),!t.rendererObject.hasLoaded){var i=this;++i.loadingCount,t.rendererObject.addEventListener("loaded",function(){--i.loadingCount,i.dispatchEvent({type:"loadedBaseTexture",content:i})})}},i.prototype.unload=function(t){t.destroy(!0)}},{"../core":9}],91:[function(t,e){function i(t,e){this.url=t,this.crossorigin=e,this.baseUrl=t.replace(/[^\/]*$/,""),this.texture=null,this.frames={}}var r=t("../core"),n=t("./JsonLoader");i.prototype.constructor=i,e.exports=i,r.utils.eventTarget.mixin(i.prototype),i.prototype.load=function(){var t=this,e=new n(this.url,this.crossorigin);e.on("loaded",function(e){t.json=e.data.content.json,t.onLoaded()}),e.load()},i.prototype.onLoaded=function(){this.emit("loaded",{content:this})}},{"../core":9,"./JsonLoader":88}],92:[function(t,e){e.exports={AtlasLoader:t("./AtlasLoader"),BitmapFontLoader:t("./BitmapFontLoader"),ImageLoader:t("./ImageLoader"),JsonLoader:t("./JsonLoader"),SpineLoader:t("./SpineLoader"),SpriteSheetLoader:t("./SpriteSheetLoader")}},{"./AtlasLoader":85,"./BitmapFontLoader":86,"./ImageLoader":87,"./JsonLoader":88,"./SpineLoader":89,"./SpriteSheetLoader":91}],93:[function(t,e){function i(t){if(r.DisplayObjectContainer.call(this),this.spineData=r.utils.AnimCache[t],!this.spineData)throw new Error("Spine data must be preloaded using SpineLoader or AssetLoader: "+t);this.skeleton=new n.Skeleton(this.spineData),this.skeleton.updateWorldTransform(),this.stateData=new n.AnimationStateData(this.spineData),this.state=new n.AnimationState(this.stateData),this.slotContainers=[];for(var e=0,i=this.skeleton.drawOrder.length;i>e;e++){var s=this.skeleton.drawOrder[e],o=s.attachment,a=new r.DisplayObjectContainer;if(this.slotContainers.push(a),this.addChild(a),o instanceof n.RegionAttachment){var h=o.rendererObject.name,l=this.createSprite(s,o);s.currentSprite=l,s.currentSpriteName=h,a.addChild(l)}else{if(!(o instanceof n.MeshAttachment))continue;var u=this.createMesh(s,o);s.currentMesh=u,s.currentMeshName=o.name,a.addChild(u)}}this.autoUpdate=!0}var r=t("../core"),n=t("./SpineRuntime");n.Bone.yDown=!0,i.prototype=Object.create(r.DisplayObjectContainer.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{autoUpdate:{get:function(){return this.updateTransform===i.prototype.autoUpdateTransform},set:function(t){this.updateTransform=t?i.prototype.autoUpdateTransform:r.DisplayObjectContainer.prototype.updateTransform}}}),i.prototype.update=function(t){this.state.update(t),this.state.apply(this.skeleton),this.skeleton.updateWorldTransform();for(var e=this.skeleton.drawOrder,i=0,s=e.length;s>i;i++){var o=e[i],a=o.attachment,h=this.slotContainers[i];if(a){var l=a.type;if(l===n.AttachmentType.region){if(a.rendererObject&&(!o.currentSpriteName||o.currentSpriteName!==a.rendererObject.name)){var u=a.rendererObject.name;if(void 0!==o.currentSprite&&(o.currentSprite.visible=!1),o.sprites=o.sprites||{},void 0!==o.sprites[u])o.sprites[u].visible=!0;else{var c=this.createSprite(o,a);h.addChild(c)}o.currentSprite=o.sprites[u],o.currentSpriteName=u}var d=o.bone;h.position.x=d.worldX+a.x*d.m00+a.y*d.m01,h.position.y=d.worldY+a.x*d.m10+a.y*d.m11,h.scale.x=d.worldScaleX,h.scale.y=d.worldScaleY,h.rotation=-(o.bone.worldRotation*n.degRad),o.currentSprite.tint=r.utils.rgb2hex([o.r,o.g,o.b])}else{if(l!==n.AttachmentType.skinnedmesh){h.visible=!1;continue}if(!o.currentMeshName||o.currentMeshName!==a.name){var f=a.name;if(void 0!==o.currentMesh&&(o.currentMesh.visible=!1),o.meshes=o.meshes||{},void 0!==o.meshes[f])o.meshes[f].visible=!0;else{var p=this.createMesh(o,a);h.addChild(p)}o.currentMesh=o.meshes[f],o.currentMeshName=f}a.computeWorldVertices(o.bone.skeleton.x,o.bone.skeleton.y,o,o.currentMesh.vertices)}h.visible=!0,h.alpha=o.a}else h.visible=!1}},i.prototype.autoUpdateTransform=function(){this.lastTime=this.lastTime||Date.now();var t=.001*(Date.now()-this.lastTime);this.lastTime=Date.now(),this.update(t),r.DisplayObjectContainer.prototype.updateTransform.call(this)},i.prototype.createSprite=function(t,e){var i=e.rendererObject,s=i.page.rendererObject,o=new r.math.Rectangle(i.x,i.y,i.rotate?i.height:i.width,i.rotate?i.width:i.height),a=new r.Texture(s,o),h=new r.Sprite(a),l=i.rotate?.5*Math.PI:0;return h.scale.set(i.width/i.originalWidth,i.height/i.originalHeight),h.rotation=l-e.rotation*n.degRad,h.anchor.x=h.anchor.y=.5,t.sprites=t.sprites||{},t.sprites[i.name]=h,h},i.prototype.createMesh=function(t,e){var i=e.rendererObject,n=i.page.rendererObject,s=new r.Texture(n),o=new r.Strip(s);return o.drawMode=r.Strip.DrawModes.TRIANGLES,o.canvasPadding=1.5,o.vertices=new Float32Array(e.uvs.length),o.uvs=e.uvs,o.indices=e.triangles,t.meshes=t.meshes||{},t.meshes[e.name]=o,o}},{"../core":9,"./SpineRuntime":94}],94:[function(t,e){var i=e.exports={radDeg:180/Math.PI,degRad:Math.PI/180,temp:[],Float32Array:"undefined"==typeof Float32Array?Array:Float32Array,Uint16Array:"undefined"==typeof Uint16Array?Array:Uint16Array};i.BoneData=function(t,e){this.name=t,this.parent=e},i.BoneData.prototype={length:0,x:0,y:0,rotation:0,scaleX:1,scaleY:1,inheritScale:!0,inheritRotation:!0,flipX:!1,flipY:!1},i.SlotData=function(t,e){this.name=t,this.boneData=e},i.SlotData.prototype={r:1,g:1,b:1,a:1,attachmentName:null,additiveBlending:!1},i.IkConstraintData=function(t){this.name=t,this.bones=[]},i.IkConstraintData.prototype={target:null,bendDirection:1,mix:1},i.Bone=function(t,e,i){this.data=t,this.skeleton=e,this.parent=i,this.setToSetupPose()},i.Bone.yDown=!1,i.Bone.prototype={x:0,y:0,rotation:0,rotationIK:0,scaleX:1,scaleY:1,flipX:!1,flipY:!1,m00:0,m01:0,worldX:0,m10:0,m11:0,worldY:0,worldRotation:0,worldScaleX:1,worldScaleY:1,worldFlipX:!1,worldFlipY:!1,updateWorldTransform:function(){var t=this.parent;if(t)this.worldX=this.x*t.m00+this.y*t.m01+t.worldX,this.worldY=this.x*t.m10+this.y*t.m11+t.worldY,this.data.inheritScale?(this.worldScaleX=t.worldScaleX*this.scaleX,this.worldScaleY=t.worldScaleY*this.scaleY):(this.worldScaleX=this.scaleX,this.worldScaleY=this.scaleY),this.worldRotation=this.data.inheritRotation?t.worldRotation+this.rotationIK:this.rotationIK,this.worldFlipX=t.worldFlipX!=this.flipX,this.worldFlipY=t.worldFlipY!=this.flipY;else{var e=this.skeleton.flipX,r=this.skeleton.flipY;this.worldX=e?-this.x:this.x,this.worldY=r!=i.Bone.yDown?-this.y:this.y,this.worldScaleX=this.scaleX,this.worldScaleY=this.scaleY,this.worldRotation=this.rotationIK,this.worldFlipX=e!=this.flipX,this.worldFlipY=r!=this.flipY}var n=this.worldRotation*i.degRad,s=Math.cos(n),o=Math.sin(n);this.worldFlipX?(this.m00=-s*this.worldScaleX,this.m01=o*this.worldScaleY):(this.m00=s*this.worldScaleX,this.m01=-o*this.worldScaleY),this.worldFlipY!=i.Bone.yDown?(this.m10=-o*this.worldScaleX,this.m11=-s*this.worldScaleY):(this.m10=o*this.worldScaleX,this.m11=s*this.worldScaleY)},setToSetupPose:function(){var t=this.data;this.x=t.x,this.y=t.y,this.rotation=t.rotation,this.rotationIK=this.rotation,this.scaleX=t.scaleX,this.scaleY=t.scaleY,this.flipX=t.flipX,this.flipY=t.flipY},worldToLocal:function(t){var e=t[0]-this.worldX,r=t[1]-this.worldY,n=this.m00,s=this.m10,o=this.m01,a=this.m11;this.worldFlipX!=(this.worldFlipY!=i.Bone.yDown)&&(n=-n,a=-a);var h=1/(n*a-o*s);t[0]=e*n*h-r*o*h,t[1]=r*a*h-e*s*h},localToWorld:function(t){var e=t[0],i=t[1];t[0]=e*this.m00+i*this.m01+this.worldX,t[1]=e*this.m10+i*this.m11+this.worldY}},i.Slot=function(t,e){this.data=t,this.bone=e,this.setToSetupPose()},i.Slot.prototype={r:1,g:1,b:1,a:1,_attachmentTime:0,attachment:null,attachmentVertices:[],setAttachment:function(t){this.attachment=t,this._attachmentTime=this.bone.skeleton.time,this.attachmentVertices.length=0},setAttachmentTime:function(t){this._attachmentTime=this.bone.skeleton.time-t},getAttachmentTime:function(){return this.bone.skeleton.time-this._attachmentTime},setToSetupPose:function(){var t=this.data;this.r=t.r,this.g=t.g,this.b=t.b,this.a=t.a;for(var e=this.bone.skeleton.data.slots,i=0,r=e.length;r>i;i++)if(e[i]==t){this.setAttachment(t.attachmentName?this.bone.skeleton.getAttachmentBySlotIndex(i,t.attachmentName):null);break}}},i.IkConstraint=function(t,e){this.data=t,this.mix=t.mix,this.bendDirection=t.bendDirection,this.bones=[];for(var i=0,r=t.bones.length;r>i;i++)this.bones.push(e.findBone(t.bones[i].name));this.target=e.findBone(t.target.name)},i.IkConstraint.prototype={apply:function(){var t=this.target,e=this.bones;switch(e.length){case 1:i.IkConstraint.apply1(e[0],t.worldX,t.worldY,this.mix);break;case 2:i.IkConstraint.apply2(e[0],e[1],t.worldX,t.worldY,this.bendDirection,this.mix)}}},i.IkConstraint.apply1=function(t,e,r,n){var s=t.data.inheritRotation&&t.parent?t.parent.worldRotation:0,o=t.rotation,a=Math.atan2(r-t.worldY,e-t.worldX)*i.radDeg-s;t.rotationIK=o+(a-o)*n},i.IkConstraint.apply2=function(t,e,r,n,s,o){var a=e.rotation,h=t.rotation;if(!o)return e.rotationIK=a,void(t.rotationIK=h);var l,u,c=i.temp,d=t.parent;d?(c[0]=r,c[1]=n,d.worldToLocal(c),r=(c[0]-t.x)*d.worldScaleX,n=(c[1]-t.y)*d.worldScaleY):(r-=t.x,n-=t.y),e.parent==t?(l=e.x,u=e.y):(c[0]=e.x,c[1]=e.y,e.parent.localToWorld(c),t.worldToLocal(c),l=c[0],u=c[1]);var f=l*t.worldScaleX,p=u*t.worldScaleY,v=Math.atan2(p,f),m=Math.sqrt(f*f+p*p),g=e.data.length*e.worldScaleX,x=2*m*g;if(1e-4>x)return void(e.rotationIK=a+(Math.atan2(n,r)*i.radDeg-h-a)*o);var y=(r*r+n*n-m*m-g*g)/x;-1>y?y=-1:y>1&&(y=1);var b=Math.acos(y)*s,T=m+g*y,w=g*Math.sin(b),S=Math.atan2(n*T-r*w,r*T+n*w),A=(S-v)*i.radDeg-h;A>180?A-=360:-180>A&&(A+=360),t.rotationIK=h+A*o,A=(b+v)*i.radDeg-a,A>180?A-=360:-180>A&&(A+=360),e.rotationIK=a+(A+t.worldRotation-e.parent.worldRotation)*o},i.Skin=function(t){this.name=t,this.attachments={}},i.Skin.prototype={addAttachment:function(t,e,i){this.attachments[t+":"+e]=i},getAttachment:function(t,e){return this.attachments[t+":"+e]},_attachAll:function(t,e){for(var i in e.attachments){var r=i.indexOf(":"),n=parseInt(i.substring(0,r)),s=i.substring(r+1),o=t.slots[n];if(o.attachment&&o.attachment.name==s){var a=this.getAttachment(n,s);a&&o.setAttachment(a)}}}},i.Animation=function(t,e,i){this.name=t,this.timelines=e,this.duration=i},i.Animation.prototype={apply:function(t,e,i,r,n){r&&0!=this.duration&&(i%=this.duration,e%=this.duration);for(var s=this.timelines,o=0,a=s.length;a>o;o++)s[o].apply(t,e,i,n,1)},mix:function(t,e,i,r,n,s){r&&0!=this.duration&&(i%=this.duration,e%=this.duration);for(var o=this.timelines,a=0,h=o.length;h>a;a++)o[a].apply(t,e,i,n,s)}},i.Animation.binarySearch=function(t,e,i){var r=0,n=Math.floor(t.length/i)-2;if(!n)return i;for(var s=n>>>1;;){if(t[(s+1)*i]<=e?r=s+1:n=s,r==n)return(r+1)*i;s=r+n>>>1}},i.Animation.binarySearch1=function(t,e){var i=0,r=t.length-2;if(!r)return 1;for(var n=r>>>1;;){if(t[n+1]<=e?i=n+1:r=n,i==r)return i+1;n=i+r>>>1}},i.Animation.linearSearch=function(t,e,i){for(var r=0,n=t.length-i;n>=r;r+=i)if(t[r]>e)return r;return-1},i.Curves=function(){this.curves=[]},i.Curves.prototype={setLinear:function(t){this.curves[19*t]=0},setStepped:function(t){this.curves[19*t]=1},setCurve:function(t,e,i,r,n){var s=.1,o=s*s,a=o*s,h=3*s,l=3*o,u=6*o,c=6*a,d=2*-e+r,f=2*-i+n,p=3*(e-r)+1,v=3*(i-n)+1,m=e*h+d*l+p*a,g=i*h+f*l+v*a,x=d*u+p*c,y=f*u+v*c,b=p*c,T=v*c,w=19*t,S=this.curves;S[w++]=2;for(var A=m,_=g,C=w+19-1;C>w;w+=2)S[w]=A,S[w+1]=_,m+=x,g+=y,x+=b,y+=T,A+=m,_+=g},getCurvePercent:function(t,e){e=0>e?0:e>1?1:e;var i=this.curves,r=19*t,n=i[r];if(0===n)return e;if(1==n)return 0;r++;for(var s=0,o=r,a=r+19-1;a>r;r+=2)if(s=i[r],s>=e){var h,l;return r==o?(h=0,l=0):(h=i[r-2],l=i[r-1]),l+(i[r+1]-l)*(e-h)/(s-h)}var u=i[r-1];return u+(1-u)*(e-s)/(1-s)}},i.RotateTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=2*t},i.RotateTimeline.prototype={boneIndex:0,getFrameCount:function(){return this.frames.length/2},setFrame:function(t,e,i){t*=2,this.frames[t]=e,this.frames[t+1]=i},apply:function(t,e,r,n,s){var o=this.frames;if(!(r=o[o.length-2]){for(var h=a.data.rotation+o[o.length-1]-a.rotation;h>180;)h-=360;for(;-180>h;)h+=360;return void(a.rotation+=h*s)}var l=i.Animation.binarySearch(o,r,2),u=o[l-1],c=o[l],d=1-(r-c)/(o[l-2]-c);d=this.curves.getCurvePercent(l/2-1,d);for(var h=o[l+1]-u;h>180;)h-=360;for(;-180>h;)h+=360;for(h=a.data.rotation+(u+h*d)-a.rotation;h>180;)h-=360;for(;-180>h;)h+=360;a.rotation+=h*s}}},i.TranslateTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=3*t},i.TranslateTimeline.prototype={boneIndex:0,getFrameCount:function(){return this.frames.length/3},setFrame:function(t,e,i,r){t*=3,this.frames[t]=e,this.frames[t+1]=i,this.frames[t+2]=r},apply:function(t,e,r,n,s){var o=this.frames;if(!(r=o[o.length-3])return a.x+=(a.data.x+o[o.length-2]-a.x)*s,void(a.y+=(a.data.y+o[o.length-1]-a.y)*s);var h=i.Animation.binarySearch(o,r,3),l=o[h-2],u=o[h-1],c=o[h],d=1-(r-c)/(o[h+-3]-c);d=this.curves.getCurvePercent(h/3-1,d),a.x+=(a.data.x+l+(o[h+1]-l)*d-a.x)*s,a.y+=(a.data.y+u+(o[h+2]-u)*d-a.y)*s}}},i.ScaleTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=3*t},i.ScaleTimeline.prototype={boneIndex:0,getFrameCount:function(){return this.frames.length/3},setFrame:function(t,e,i,r){t*=3,this.frames[t]=e,this.frames[t+1]=i,this.frames[t+2]=r},apply:function(t,e,r,n,s){var o=this.frames;if(!(r=o[o.length-3])return a.scaleX+=(a.data.scaleX*o[o.length-2]-a.scaleX)*s,void(a.scaleY+=(a.data.scaleY*o[o.length-1]-a.scaleY)*s);var h=i.Animation.binarySearch(o,r,3),l=o[h-2],u=o[h-1],c=o[h],d=1-(r-c)/(o[h+-3]-c);d=this.curves.getCurvePercent(h/3-1,d),a.scaleX+=(a.data.scaleX*(l+(o[h+1]-l)*d)-a.scaleX)*s,a.scaleY+=(a.data.scaleY*(u+(o[h+2]-u)*d)-a.scaleY)*s}}},i.ColorTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=5*t},i.ColorTimeline.prototype={slotIndex:0,getFrameCount:function(){return this.frames.length/5},setFrame:function(t,e,i,r,n,s){t*=5,this.frames[t]=e,this.frames[t+1]=i,this.frames[t+2]=r,this.frames[t+3]=n,this.frames[t+4]=s},apply:function(t,e,r,n,s){var o=this.frames;if(!(r=o[o.length-5]){var c=o.length-1;a=o[c-3],h=o[c-2],l=o[c-1],u=o[c]}else{var d=i.Animation.binarySearch(o,r,5),f=o[d-4],p=o[d-3],v=o[d-2],m=o[d-1],g=o[d],x=1-(r-g)/(o[d-5]-g);x=this.curves.getCurvePercent(d/5-1,x),a=f+(o[d+1]-f)*x,h=p+(o[d+2]-p)*x,l=v+(o[d+3]-v)*x,u=m+(o[d+4]-m)*x}var y=t.slots[this.slotIndex];1>s?(y.r+=(a-y.r)*s,y.g+=(h-y.g)*s,y.b+=(l-y.b)*s,y.a+=(u-y.a)*s):(y.r=a,y.g=h,y.b=l,y.a=u)}}},i.AttachmentTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=t,this.attachmentNames=[],this.attachmentNames.length=t},i.AttachmentTimeline.prototype={slotIndex:0,getFrameCount:function(){return this.frames.length},setFrame:function(t,e,i){this.frames[t]=e,this.attachmentNames[t]=i},apply:function(t,e,r){var n=this.frames;if(rr&&this.apply(t,e,Number.MAX_VALUE,null,0));e>r&&(e=-1);var s=r>=n[n.length-1]?n.length-1:i.Animation.binarySearch1(n,r)-1;if(!(n[s]r)this.apply(t,e,Number.MAX_VALUE,n,s),e=-1;else if(e>=o[a-1])return;if(!(r0&&o[h-1]==l;)h--}for(var u=this.events;a>h&&r>=o[h];h++)n.push(u[h])}}}},i.DrawOrderTimeline=function(t){this.frames=[],this.frames.length=t,this.drawOrders=[],this.drawOrders.length=t},i.DrawOrderTimeline.prototype={getFrameCount:function(){return this.frames.length},setFrame:function(t,e,i){this.frames[t]=e,this.drawOrders[t]=i},apply:function(t,e,r){var n=this.frames;if(!(r=n[n.length-1]?n.length-1:i.Animation.binarySearch1(n,r)-1;var o=t.drawOrder,a=t.slots,h=this.drawOrders[s];if(h)for(var l=0,u=h.length;u>l;l++)o[l]=t.slots[h[l]];else for(var l=0,u=a.length;u>l;l++)o[l]=a[l]}}},i.FfdTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=t,this.frameVertices=[],this.frameVertices.length=t},i.FfdTimeline.prototype={slotIndex:0,attachment:0,getFrameCount:function(){return this.frames.length},setFrame:function(t,e,i){this.frames[t]=e,this.frameVertices[t]=i},apply:function(t,e,r,n,s){var o=t.slots[this.slotIndex];if(o.attachment==this.attachment){var a=this.frames;if(!(r=a[a.length-1]){var c=h[a.length-1];if(1>s)for(var d=0;l>d;d++)u[d]+=(c[d]-u[d])*s;else for(var d=0;l>d;d++)u[d]=c[d]}else{var f=i.Animation.binarySearch1(a,r),p=a[f],v=1-(r-p)/(a[f-1]-p);v=this.curves.getCurvePercent(f-1,0>v?0:v>1?1:v);var m=h[f-1],g=h[f];if(1>s)for(var d=0;l>d;d++){var x=m[d];u[d]+=(x+(g[d]-x)*v-u[d])*s}else for(var d=0;l>d;d++){var x=m[d];u[d]=x+(g[d]-x)*v}}}}}},i.IkConstraintTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=3*t},i.IkConstraintTimeline.prototype={ikConstraintIndex:0,getFrameCount:function(){return this.frames.length/3},setFrame:function(t,e,i,r){t*=3,this.frames[t]=e,this.frames[t+1]=i,this.frames[t+2]=r},apply:function(t,e,r,n,s){var o=this.frames;if(!(r=o[o.length-3])return a.mix+=(o[o.length-2]-a.mix)*s,void(a.bendDirection=o[o.length-1]);var h=i.Animation.binarySearch(o,r,3),l=o[h+-2],u=o[h],c=1-(r-u)/(o[h+-3]-u);c=this.curves.getCurvePercent(h/3-1,c);var d=l+(o[h+1]-l)*c;a.mix+=(d-a.mix)*s,a.bendDirection=o[h+-1]}}},i.FlipXTimeline=function(t){this.curves=new i.Curves(t),this.frames=[],this.frames.length=2*t},i.FlipXTimeline.prototype={boneIndex:0,getFrameCount:function(){return this.frames.length/2},setFrame:function(t,e,i){t*=2,this.frames[t]=e,this.frames[t+1]=i?1:0},apply:function(t,e,r){var n=this.frames;if(rr&&this.apply(t,e,Number.MAX_VALUE,null,0));e>r&&(e=-1);var s=(r>=n[n.length-2]?n.length:i.Animation.binarySearch(n,r,2))-2;n[s]r&&this.apply(t,e,Number.MAX_VALUE,null,0));e>r&&(e=-1);var s=(r>=n[n.length-2]?n.length:i.Animation.binarySearch(n,r,2))-2;n[s]i;i++)if(e[i].name==t)return e[i];return null},findBoneIndex:function(t){for(var e=this.bones,i=0,r=e.length;r>i;i++)if(e[i].name==t)return i;return-1},findSlot:function(t){for(var e=this.slots,i=0,r=e.length;r>i;i++)if(e[i].name==t)return slot[i];return null},findSlotIndex:function(t){for(var e=this.slots,i=0,r=e.length;r>i;i++)if(e[i].name==t)return i;return-1},findSkin:function(t){for(var e=this.skins,i=0,r=e.length;r>i;i++)if(e[i].name==t)return e[i];return null},findEvent:function(t){for(var e=this.events,i=0,r=e.length;r>i;i++)if(e[i].name==t)return e[i];return null},findAnimation:function(t){for(var e=this.animations,i=0,r=e.length;r>i;i++)if(e[i].name==t)return e[i];return null},findIkConstraint:function(t){for(var e=this.ikConstraints,i=0,r=e.length;r>i;i++)if(e[i].name==t)return e[i];return null}},i.Skeleton=function(t){this.data=t,this.bones=[];for(var e=0,r=t.bones.length;r>e;e++){var n=t.bones[e],s=n.parent?this.bones[t.bones.indexOf(n.parent)]:null;this.bones.push(new i.Bone(n,this,s))}this.slots=[],this.drawOrder=[];for(var e=0,r=t.slots.length;r>e;e++){var o=t.slots[e],a=this.bones[t.bones.indexOf(o.boneData)],h=new i.Slot(o,a);this.slots.push(h),this.drawOrder.push(h)}this.ikConstraints=[];for(var e=0,r=t.ikConstraints.length;r>e;e++)this.ikConstraints.push(new i.IkConstraint(t.ikConstraints[e],this));this.boneCache=[],this.updateCache()},i.Skeleton.prototype={x:0,y:0,skin:null,r:1,g:1,b:1,a:1,time:0,flipX:!1,flipY:!1,updateCache:function(){var t=this.ikConstraints,e=t.length,i=e+1,r=this.boneCache;r.length>i&&(r.length=i);for(var n=0,s=r.length;s>n;n++)r[n].length=0;for(;r.lengthn;n++){var h=a[n],l=h;do{for(var u=0;e>u;u++)for(var c=t[u],d=c.bones[0],f=c.bones[c.bones.length-1];;){if(l==f){r[u].push(h),r[u+1].push(h);continue t}if(f==d)break;f=f.parent}l=l.parent}while(l);o[o.length]=h}},updateWorldTransform:function(){for(var t=this.bones,e=0,i=t.length;i>e;e++){var r=t[e];r.rotationIK=r.rotation}for(var e=0,n=this.boneCache.length-1;;){for(var s=this.boneCache[e],o=0,a=s.length;a>o;o++)s[o].updateWorldTransform();if(e==n)break;this.ikConstraints[e].apply(),e++}},setToSetupPose:function(){this.setBonesToSetupPose(),this.setSlotsToSetupPose()},setBonesToSetupPose:function(){for(var t=this.bones,e=0,i=t.length;i>e;e++)t[e].setToSetupPose();for(var r=this.ikConstraints,e=0,i=r.length;i>e;e++){var n=r[e];n.bendDirection=n.data.bendDirection,n.mix=n.data.mix}},setSlotsToSetupPose:function(){for(var t=this.slots,e=this.drawOrder,i=0,r=t.length;r>i;i++)e[i]=t[i],t[i].setToSetupPose(i)},getRootBone:function(){return this.bones.length?this.bones[0]:null},findBone:function(t){for(var e=this.bones,i=0,r=e.length;r>i;i++)if(e[i].data.name==t)return e[i];return null},findBoneIndex:function(t){for(var e=this.bones,i=0,r=e.length;r>i;i++)if(e[i].data.name==t)return i;return-1},findSlot:function(t){for(var e=this.slots,i=0,r=e.length;r>i;i++)if(e[i].data.name==t)return e[i];return null},findSlotIndex:function(t){for(var e=this.slots,i=0,r=e.length;r>i;i++)if(e[i].data.name==t)return i;return-1},setSkinByName:function(t){var e=this.data.findSkin(t);if(!e)throw"Skin not found: "+t;this.setSkin(e)},setSkin:function(t){if(t)if(this.skin)t._attachAll(this,this.skin);else for(var e=this.slots,i=0,r=e.length;r>i;i++){var n=e[i],s=n.data.attachmentName;if(s){var o=t.getAttachment(i,s);o&&n.setAttachment(o)}}this.skin=t},getAttachmentBySlotName:function(t,e){return this.getAttachmentBySlotIndex(this.data.findSlotIndex(t),e)},getAttachmentBySlotIndex:function(t,e){if(this.skin){var i=this.skin.getAttachment(t,e);if(i)return i}return this.data.defaultSkin?this.data.defaultSkin.getAttachment(t,e):null},setAttachment:function(t,e){for(var i=this.slots,r=0,n=i.length;n>r;r++){var s=i[r];if(s.data.name==t){var o=null;if(e&&(o=this.getAttachmentBySlotIndex(r,e),!o))throw"Attachment not found: "+e+", for slot: "+t;return void s.setAttachment(o)}}throw"Slot not found: "+t},findIkConstraint:function(t){for(var e=this.ikConstraints,i=0,r=e.length;r>i;i++)if(e[i].data.name==t)return e[i];return null},update:function(t){this.time+=t}},i.EventData=function(t){this.name=t},i.EventData.prototype={intValue:0,floatValue:0,stringValue:null},i.Event=function(t){this.data=t},i.Event.prototype={intValue:0,floatValue:0,stringValue:null},i.AttachmentType={region:0,boundingbox:1,mesh:2,skinnedmesh:3},i.RegionAttachment=function(t){this.name=t,this.offset=[],this.offset.length=8,this.uvs=[],this.uvs.length=8},i.RegionAttachment.prototype={type:i.AttachmentType.region,x:0,y:0,rotation:0,scaleX:1,scaleY:1,width:0,height:0,r:1,g:1,b:1,a:1,path:null,rendererObject:null,regionOffsetX:0,regionOffsetY:0,regionWidth:0,regionHeight:0,regionOriginalWidth:0,regionOriginalHeight:0,setUVs:function(t,e,i,r,n){var s=this.uvs;n?(s[2]=t,s[3]=r,s[4]=t,s[5]=e,s[6]=i,s[7]=e,s[0]=i,s[1]=r):(s[0]=t,s[1]=r,s[2]=t,s[3]=e,s[4]=i,s[5]=e,s[6]=i,s[7]=r)},updateOffset:function(){var t=this.width/this.regionOriginalWidth*this.scaleX,e=this.height/this.regionOriginalHeight*this.scaleY,r=-this.width/2*this.scaleX+this.regionOffsetX*t,n=-this.height/2*this.scaleY+this.regionOffsetY*e,s=r+this.regionWidth*t,o=n+this.regionHeight*e,a=this.rotation*i.degRad,h=Math.cos(a),l=Math.sin(a),u=r*h+this.x,c=r*l,d=n*h+this.y,f=n*l,p=s*h+this.x,v=s*l,m=o*h+this.y,g=o*l,x=this.offset;x[0]=u-f,x[1]=d+c,x[2]=u-g,x[3]=m+c,x[4]=p-g,x[5]=m+v,x[6]=p-f,x[7]=d+v},computeVertices:function(t,e,i,r){t+=i.worldX,e+=i.worldY;var n=i.m00,s=i.m01,o=i.m10,a=i.m11,h=this.offset;r[0]=h[0]*n+h[1]*s+t,r[1]=h[0]*o+h[1]*a+e,r[2]=h[2]*n+h[3]*s+t,r[3]=h[2]*o+h[3]*a+e,r[4]=h[4]*n+h[5]*s+t,r[5]=h[4]*o+h[5]*a+e,r[6]=h[6]*n+h[7]*s+t,r[7]=h[6]*o+h[7]*a+e}},i.MeshAttachment=function(t){this.name=t},i.MeshAttachment.prototype={type:i.AttachmentType.mesh,vertices:null,uvs:null,regionUVs:null,triangles:null,hullLength:0,r:1,g:1,b:1,a:1,path:null,rendererObject:null,regionU:0,regionV:0,regionU2:0,regionV2:0,regionRotate:!1,regionOffsetX:0,regionOffsetY:0,regionWidth:0,regionHeight:0,regionOriginalWidth:0,regionOriginalHeight:0,edges:null,width:0,height:0,updateUVs:function(){var t=this.regionU2-this.regionU,e=this.regionV2-this.regionV,r=this.regionUVs.length;if(this.uvs&&this.uvs.length==r||(this.uvs=new i.Float32Array(r)),this.regionRotate)for(var n=0;r>n;n+=2)this.uvs[n]=this.regionU+this.regionUVs[n+1]*t,this.uvs[n+1]=this.regionV+e-this.regionUVs[n]*e;else for(var n=0;r>n;n+=2)this.uvs[n]=this.regionU+this.regionUVs[n]*t,this.uvs[n+1]=this.regionV+this.regionUVs[n+1]*e},computeWorldVertices:function(t,e,i,r){var n=i.bone;t+=n.worldX,e+=n.worldY;var s=n.m00,o=n.m01,a=n.m10,h=n.m11,l=this.vertices,u=l.length;i.attachmentVertices.length==u&&(l=i.attachmentVertices);for(var c=0;u>c;c+=2){var d=l[c],f=l[c+1];r[c]=d*s+f*o+t,r[c+1]=d*a+f*h+e}}},i.SkinnedMeshAttachment=function(t){this.name=t},i.SkinnedMeshAttachment.prototype={type:i.AttachmentType.skinnedmesh,bones:null,weights:null,uvs:null,regionUVs:null,triangles:null,hullLength:0,r:1,g:1,b:1,a:1,path:null,rendererObject:null,regionU:0,regionV:0,regionU2:0,regionV2:0,regionRotate:!1,regionOffsetX:0,regionOffsetY:0,regionWidth:0,regionHeight:0,regionOriginalWidth:0,regionOriginalHeight:0,edges:null,width:0,height:0,updateUVs:function(){var t=this.regionU2-this.regionU,e=this.regionV2-this.regionV,r=this.regionUVs.length;if(this.uvs&&this.uvs.length==r||(this.uvs=new i.Float32Array(r)),this.regionRotate)for(var n=0;r>n;n+=2)this.uvs[n]=this.regionU+this.regionUVs[n+1]*t,this.uvs[n+1]=this.regionV+e-this.regionUVs[n]*e;else for(var n=0;r>n;n+=2)this.uvs[n]=this.regionU+this.regionUVs[n]*t,this.uvs[n+1]=this.regionV+this.regionUVs[n+1]*e},computeWorldVertices:function(t,e,i,r){var n,s,o,a,h,l,u,c=i.bone.skeleton.bones,d=this.weights,f=this.bones,p=0,v=0,m=0,g=0,x=f.length;if(i.attachmentVertices.length)for(var y=i.attachmentVertices;x>v;p+=2){for(s=0,o=0,n=f[v++]+v;n>v;v++,m+=3,g+=2)a=c[f[v]],h=d[m]+y[g],l=d[m+1]+y[g+1],u=d[m+2],s+=(h*a.m00+l*a.m01+a.worldX)*u,o+=(h*a.m10+l*a.m11+a.worldY)*u;r[p]=s+t,r[p+1]=o+e}else for(;x>v;p+=2){for(s=0,o=0,n=f[v++]+v;n>v;v++,m+=3)a=c[f[v]],h=d[m],l=d[m+1],u=d[m+2],s+=(h*a.m00+l*a.m01+a.worldX)*u,o+=(h*a.m10+l*a.m11+a.worldY)*u;r[p]=s+t,r[p+1]=o+e}}},i.BoundingBoxAttachment=function(t){this.name=t,this.vertices=[]},i.BoundingBoxAttachment.prototype={type:i.AttachmentType.boundingbox,computeWorldVertices:function(t,e,i,r){t+=i.worldX,e+=i.worldY;for(var n=i.m00,s=i.m01,o=i.m10,a=i.m11,h=this.vertices,l=0,u=h.length;u>l;l+=2){var c=h[l],d=h[l+1];r[l]=c*n+d*s+t,r[l+1]=c*o+d*a+e}}},i.AnimationStateData=function(t){this.skeletonData=t,this.animationToMixTime={}},i.AnimationStateData.prototype={defaultMix:0,setMixByName:function(t,e,i){var r=this.skeletonData.findAnimation(t);if(!r)throw"Animation not found: "+t;var n=this.skeletonData.findAnimation(e);if(!n)throw"Animation not found: "+e;this.setMix(r,n,i)},setMix:function(t,e,i){this.animationToMixTime[t.name+":"+e.name]=i},getMix:function(t,e){var i=t.name+":"+e.name;return this.animationToMixTime.hasOwnProperty(i)?this.animationToMixTime[i]:this.defaultMix}},i.TrackEntry=function(){},i.TrackEntry.prototype={next:null,previous:null,animation:null,loop:!1,delay:0,time:0,lastTime:-1,endTime:0,timeScale:1,mixTime:0,mixDuration:0,mix:1,onStart:null,onEnd:null,onComplete:null,onEvent:null},i.AnimationState=function(t){this.data=t,this.tracks=[],this.events=[]},i.AnimationState.prototype={onStart:null,onEnd:null,onComplete:null,onEvent:null,timeScale:1,update:function(t){t*=this.timeScale;for(var e=0;e=0&&this.setCurrent(e,n)):!i.loop&&i.lastTime>=i.endTime&&this.clearTrack(e)}}},apply:function(t){for(var e=0;es&&(r=s);var a=i.previous;if(a){var h=a.time;!a.loop&&h>a.endTime&&(h=a.endTime),a.animation.apply(t,h,h,a.loop,null);var l=i.mixTime/i.mixDuration*i.mix;l>=1&&(l=1,i.previous=null),i.animation.mix(t,i.lastTime,r,o,this.events,l)}else 1==i.mix?i.animation.apply(t,i.lastTime,r,o,this.events):i.animation.mix(t,i.lastTime,r,o,this.events,i.mix);for(var u=0,c=this.events.length;c>u;u++){var d=this.events[u];i.onEvent&&i.onEvent(e,d),this.onEvent&&this.onEvent(e,d)}if(o?n%s>r%s:s>n&&r>=s){var f=Math.floor(r/s);i.onComplete&&i.onComplete(e,f),this.onComplete&&this.onComplete(e,f)}i.lastTime=i.time}}},clearTracks:function(){for(var t=0,e=this.tracks.length;e>t;t++)this.clearTrack(t);this.tracks.length=0},clearTrack:function(t){if(!(t>=this.tracks.length)){var e=this.tracks[t];e&&(e.onEnd&&e.onEnd(t),this.onEnd&&this.onEnd(t),this.tracks[t]=null)}},_expandToIndex:function(t){if(t=this.tracks.length;)this.tracks.push(null);return null},setCurrent:function(t,e){var i=this._expandToIndex(t);if(i){var r=i.previous;i.previous=null,i.onEnd&&i.onEnd(t),this.onEnd&&this.onEnd(t),e.mixDuration=this.data.getMix(i.animation,e.animation),e.mixDuration>0&&(e.mixTime=0,e.previous=r&&i.mixTime/i.mixDuration<.5?r:i)}this.tracks[t]=e,e.onStart&&e.onStart(t),this.onStart&&this.onStart(t)},setAnimationByName:function(t,e,i){var r=this.data.skeletonData.findAnimation(e);if(!r)throw"Animation not found: "+e;return this.setAnimation(t,r,i)},setAnimation:function(t,e,r){var n=new i.TrackEntry;return n.animation=e,n.loop=r,n.endTime=e.duration,this.setCurrent(t,n),n},addAnimationByName:function(t,e,i,r){var n=this.data.skeletonData.findAnimation(e);if(!n)throw"Animation not found: "+e;return this.addAnimation(t,n,i,r)},addAnimation:function(t,e,r,n){var s=new i.TrackEntry;s.animation=e,s.loop=r,s.endTime=e.duration;var o=this._expandToIndex(t);if(o){for(;o.next;)o=o.next;o.next=s}else this.tracks[t]=s;return 0>=n&&(o?n+=o.endTime-this.data.getMix(o.animation,e):n=0),s.delay=n,s},getCurrent:function(t){return t>=this.tracks.length?null:this.tracks[t]}},i.SkeletonJson=function(t){this.attachmentLoader=t},i.SkeletonJson.prototype={scale:1,readSkeletonData:function(t,e){var r=new i.SkeletonData;r.name=e;var n=t.skeleton;n&&(r.hash=n.hash,r.version=n.spine,r.width=n.width||0,r.height=n.height||0);for(var s=t.bones,o=0,a=s.length;a>o;o++){var h=s[o],l=null;if(h.parent&&(l=r.findBone(h.parent),!l))throw"Parent bone not found: "+h.parent;var u=new i.BoneData(h.name,l);u.length=(h.length||0)*this.scale,u.x=(h.x||0)*this.scale,u.y=(h.y||0)*this.scale,u.rotation=h.rotation||0,u.scaleX=h.hasOwnProperty("scaleX")?h.scaleX:1,u.scaleY=h.hasOwnProperty("scaleY")?h.scaleY:1,u.inheritScale=h.hasOwnProperty("inheritScale")?h.inheritScale:!0,u.inheritRotation=h.hasOwnProperty("inheritRotation")?h.inheritRotation:!0,r.bones.push(u)}var c=t.ik;if(c)for(var o=0,a=c.length;a>o;o++){for(var d=c[o],f=new i.IkConstraintData(d.name),s=d.bones,p=0,v=s.length;v>p;p++){var m=r.findBone(s[p]);if(!m)throw"IK bone not found: "+s[p];f.bones.push(m)}if(f.target=r.findBone(d.target),!f.target)throw"Target bone not found: "+d.target;f.bendDirection=!d.hasOwnProperty("bendPositive")||d.bendPositive?1:-1,f.mix=d.hasOwnProperty("mix")?d.mix:1,r.ikConstraints.push(f)}for(var g=t.slots,o=0,a=g.length;a>o;o++){var x=g[o],u=r.findBone(x.bone);if(!u)throw"Slot bone not found: "+x.bone;var y=new i.SlotData(x.name,u),b=x.color;b&&(y.r=this.toColor(b,0),y.g=this.toColor(b,1),y.b=this.toColor(b,2),y.a=this.toColor(b,3)),y.attachmentName=x.attachment,y.additiveBlending=x.additive&&"true"==x.additive,r.slots.push(y)}var T=t.skins;for(var w in T)if(T.hasOwnProperty(w)){var S=T[w],A=new i.Skin(w);for(var _ in S)if(S.hasOwnProperty(_)){var C=r.findSlotIndex(_),E=S[_];for(var R in E)if(E.hasOwnProperty(R)){var M=this.readAttachment(A,R,E[R]);M&&A.addAttachment(C,R,M)}}r.skins.push(A),"default"==A.name&&(r.defaultSkin=A)}var F=t.events;for(var B in F)if(F.hasOwnProperty(B)){var D=F[B],P=new i.EventData(B);P.intValue=D["int"]||0,P.floatValue=D["float"]||0,P.stringValue=D.string||null,r.events.push(P)}var L=t.animations;for(var O in L)L.hasOwnProperty(O)&&this.readAnimation(O,L[O],r);return r},readAttachment:function(t,e,r){e=r.name||e;var n=i.AttachmentType[r.type||"region"],s=r.path||e,o=this.scale;if(n==i.AttachmentType.region){var a=this.attachmentLoader.newRegionAttachment(t,e,s);if(!a)return null;a.path=s,a.x=(r.x||0)*o,a.y=(r.y||0)*o,a.scaleX=r.hasOwnProperty("scaleX")?r.scaleX:1,a.scaleY=r.hasOwnProperty("scaleY")?r.scaleY:1,a.rotation=r.rotation||0,a.width=(r.width||0)*o,a.height=(r.height||0)*o;var h=r.color;return h&&(a.r=this.toColor(h,0),a.g=this.toColor(h,1),a.b=this.toColor(h,2),a.a=this.toColor(h,3)),a.updateOffset(),a}if(n==i.AttachmentType.mesh){var l=this.attachmentLoader.newMeshAttachment(t,e,s);return l?(l.path=s,l.vertices=this.getFloatArray(r,"vertices",o),l.triangles=this.getIntArray(r,"triangles"),l.regionUVs=this.getFloatArray(r,"uvs",1),l.updateUVs(),h=r.color,h&&(l.r=this.toColor(h,0),l.g=this.toColor(h,1),l.b=this.toColor(h,2),l.a=this.toColor(h,3)),l.hullLength=2*(r.hull||0),r.edges&&(l.edges=this.getIntArray(r,"edges")),l.width=(r.width||0)*o,l.height=(r.height||0)*o,l):null}if(n==i.AttachmentType.skinnedmesh){var l=this.attachmentLoader.newSkinnedMeshAttachment(t,e,s);if(!l)return null;l.path=s;for(var u=this.getFloatArray(r,"uvs",1),c=this.getFloatArray(r,"vertices",1),d=[],f=[],p=0,v=c.length;v>p;){var m=0|c[p++];f[f.length]=m;for(var g=p+4*m;g>p;)f[f.length]=c[p],d[d.length]=c[p+1]*o,d[d.length]=c[p+2]*o,d[d.length]=c[p+3],p+=4}return l.bones=f,l.weights=d,l.triangles=this.getIntArray(r,"triangles"),l.regionUVs=u,l.updateUVs(),h=r.color,h&&(l.r=this.toColor(h,0),l.g=this.toColor(h,1),l.b=this.toColor(h,2),l.a=this.toColor(h,3)),l.hullLength=2*(r.hull||0),r.edges&&(l.edges=this.getIntArray(r,"edges")),l.width=(r.width||0)*o,l.height=(r.height||0)*o,l}if(n==i.AttachmentType.boundingbox){for(var x=this.attachmentLoader.newBoundingBoxAttachment(t,e),c=r.vertices,p=0,v=c.length;v>p;p++)x.vertices.push(c[p]*o);return x}throw"Unknown attachment type: "+n},readAnimation:function(t,e,r){var n=[],s=0,o=e.slots;for(var a in o)if(o.hasOwnProperty(a)){var h=o[a],l=r.findSlotIndex(a);for(var u in h)if(h.hasOwnProperty(u)){var c=h[u];if("color"==u){var d=new i.ColorTimeline(c.length);d.slotIndex=l;for(var f=0,p=0,v=c.length;v>p;p++){var m=c[p],g=m.color,x=this.toColor(g,0),y=this.toColor(g,1),b=this.toColor(g,2),T=this.toColor(g,3);d.setFrame(f,m.time,x,y,b,T),this.readCurve(d,f,m),f++}n.push(d),s=Math.max(s,d.frames[5*d.getFrameCount()-5])}else{if("attachment"!=u)throw"Invalid timeline type for a slot: "+u+" ("+a+")";var d=new i.AttachmentTimeline(c.length);d.slotIndex=l;for(var f=0,p=0,v=c.length;v>p;p++){var m=c[p];d.setFrame(f++,m.time,m.name)}n.push(d),s=Math.max(s,d.frames[d.getFrameCount()-1])}}}var w=e.bones;for(var S in w)if(w.hasOwnProperty(S)){var A=r.findBoneIndex(S);if(-1==A)throw"Bone not found: "+S;var _=w[S];for(var u in _)if(_.hasOwnProperty(u)){var c=_[u];if("rotate"==u){var d=new i.RotateTimeline(c.length);d.boneIndex=A;for(var f=0,p=0,v=c.length;v>p;p++){var m=c[p];d.setFrame(f,m.time,m.angle),this.readCurve(d,f,m),f++}n.push(d),s=Math.max(s,d.frames[2*d.getFrameCount()-2])}else if("translate"==u||"scale"==u){var d,C=1;"scale"==u?d=new i.ScaleTimeline(c.length):(d=new i.TranslateTimeline(c.length),C=this.scale),d.boneIndex=A;for(var f=0,p=0,v=c.length;v>p;p++){var m=c[p],E=(m.x||0)*C,R=(m.y||0)*C;d.setFrame(f,m.time,E,R),this.readCurve(d,f,m),f++}n.push(d),s=Math.max(s,d.frames[3*d.getFrameCount()-3])}else{if("flipX"!=u&&"flipY"!=u)throw"Invalid timeline type for a bone: "+u+" ("+S+")";var E="flipX"==u,d=E?new i.FlipXTimeline(c.length):new i.FlipYTimeline(c.length);d.boneIndex=A;for(var M=E?"x":"y",f=0,p=0,v=c.length;v>p;p++){var m=c[p];d.setFrame(f,m.time,m[M]||!1),f++}n.push(d),s=Math.max(s,d.frames[2*d.getFrameCount()-2])}}}var F=e.ik;for(var B in F)if(F.hasOwnProperty(B)){var D=r.findIkConstraint(B),c=F[B],d=new i.IkConstraintTimeline(c.length);d.ikConstraintIndex=r.ikConstraints.indexOf(D);for(var f=0,p=0,v=c.length;v>p;p++){var m=c[p],P=m.hasOwnProperty("mix")?m.mix:1,L=!m.hasOwnProperty("bendPositive")||m.bendPositive?1:-1;d.setFrame(f,m.time,P,L),this.readCurve(d,f,m),f++}n.push(d),s=Math.max(s,d.frames[3*d.frameCount-3])}var O=e.ffd;for(var I in O){var k=r.findSkin(I),h=O[I];for(a in h){var l=r.findSlotIndex(a),U=h[a];for(var N in U){var c=U[N],d=new i.FfdTimeline(c.length),j=k.getAttachment(l,N);if(!j)throw"FFD attachment not found: "+N;d.slotIndex=l,d.attachment=j;var Y,X=j.type==i.AttachmentType.mesh;Y=X?j.vertices.length:j.weights.length/3*2;for(var f=0,p=0,v=c.length;v>p;p++){var G,m=c[p];if(m.vertices){var W=m.vertices,G=[];G.length=Y;var V=m.offset||0,z=W.length;if(1==this.scale)for(var H=0;z>H;H++)G[H+V]=W[H];else for(var H=0;z>H;H++)G[H+V]=W[H]*this.scale;if(X)for(var q=j.vertices,H=0,z=G.length;z>H;H++)G[H]+=q[H]}else X?G=j.vertices:(G=[],G.length=Y);d.setFrame(f,m.time,G),this.readCurve(d,f,m),f++}n[n.length]=d,s=Math.max(s,d.frames[d.frameCount-1])}}}var K=e.drawOrder;if(K||(K=e.draworder),K){for(var d=new i.DrawOrderTimeline(K.length),J=r.slots.length,f=0,p=0,v=K.length;v>p;p++){var Q=K[p],$=null;if(Q.offsets){$=[],$.length=J;for(var H=J-1;H>=0;H--)$[H]=-1;var Z=Q.offsets,te=[];te.length=J-Z.length;for(var ee=0,ie=0,H=0,z=Z.length;z>H;H++){var re=Z[H],l=r.findSlotIndex(re.slot);if(-1==l)throw"Slot not found: "+re.slot;for(;ee!=l;)te[ie++]=ee++;$[ee+re.offset]=ee++}for(;J>ee;)te[ie++]=ee++;for(var H=J-1;H>=0;H--)-1==$[H]&&($[H]=te[--ie])}d.setFrame(f++,Q.time,$)}n.push(d),s=Math.max(s,d.frames[d.getFrameCount()-1])}var ne=e.events;if(ne){for(var d=new i.EventTimeline(ne.length),f=0,p=0,v=ne.length;v>p;p++){var se=ne[p],oe=r.findEvent(se.name);if(!oe)throw"Event not found: "+se.name;var ae=new i.Event(oe);ae.intValue=se.hasOwnProperty("int")?se["int"]:oe.intValue,ae.floatValue=se.hasOwnProperty("float")?se["float"]:oe.floatValue,ae.stringValue=se.hasOwnProperty("string")?se.string:oe.stringValue,d.setFrame(f++,se.time,ae)}n.push(d),s=Math.max(s,d.frames[d.getFrameCount()-1])}r.animations.push(new i.Animation(t,n,s))},readCurve:function(t,e,i){var r=i.curve;r?"stepped"==r?t.curves.setStepped(e):r instanceof Array&&t.curves.setCurve(e,r[0],r[1],r[2],r[3]):t.curves.setLinear(e)},toColor:function(t,e){if(8!=t.length)throw"Color hexidecimal length must be 8, recieved: "+t;return parseInt(t.substring(2*e,2*e+2),16)/255},getFloatArray:function(t,e,r){var n=t[e],s=new i.Float32Array(n.length),o=0,a=n.length;if(1==r)for(;a>o;o++)s[o]=n[o];else for(;a>o;o++)s[o]=n[o]*r;return s},getIntArray:function(t,e){for(var r=t[e],n=new i.Uint16Array(r.length),s=0,o=r.length;o>s;s++)n[s]=0|r[s];return n}},i.Atlas=function(t,e){this.textureLoader=e,this.pages=[],this.regions=[];var r=new i.AtlasReader(t),n=[];n.length=4;for(var s=null;;){var o=r.readLine();if(null===o)break;if(o=r.trim(o),o.length)if(s){var a=new i.AtlasRegion;a.name=o,a.page=s,a.rotate="true"==r.readValue(),r.readTuple(n);var h=parseInt(n[0]),l=parseInt(n[1]);r.readTuple(n);var u=parseInt(n[0]),c=parseInt(n[1]);a.u=h/s.width,a.v=l/s.height,a.rotate?(a.u2=(h+c)/s.width,a.v2=(l+u)/s.height):(a.u2=(h+u)/s.width,a.v2=(l+c)/s.height),a.x=h,a.y=l,a.width=Math.abs(u),a.height=Math.abs(c),4==r.readTuple(n)&&(a.splits=[parseInt(n[0]),parseInt(n[1]),parseInt(n[2]),parseInt(n[3])],4==r.readTuple(n)&&(a.pads=[parseInt(n[0]),parseInt(n[1]),parseInt(n[2]),parseInt(n[3])],r.readTuple(n))),a.originalWidth=parseInt(n[0]),a.originalHeight=parseInt(n[1]),r.readTuple(n),a.offsetX=parseInt(n[0]),a.offsetY=parseInt(n[1]),a.index=parseInt(r.readValue()),this.regions.push(a)}else{s=new i.AtlasPage,s.name=o,2==r.readTuple(n)&&(s.width=parseInt(n[0]),s.height=parseInt(n[1]),r.readTuple(n)),s.format=i.Atlas.Format[n[0]],r.readTuple(n),s.minFilter=i.Atlas.TextureFilter[n[0]],s.magFilter=i.Atlas.TextureFilter[n[1]];var d=r.readValue();s.uWrap=i.Atlas.TextureWrap.clampToEdge,s.vWrap=i.Atlas.TextureWrap.clampToEdge,"x"==d?s.uWrap=i.Atlas.TextureWrap.repeat:"y"==d?s.vWrap=i.Atlas.TextureWrap.repeat:"xy"==d&&(s.uWrap=s.vWrap=i.Atlas.TextureWrap.repeat),e.load(s,o,this),this.pages.push(s)}else s=null}},i.Atlas.prototype={findRegion:function(t){for(var e=this.regions,i=0,r=e.length;r>i;i++)if(e[i].name==t)return e[i];return null},dispose:function(){for(var t=this.pages,e=0,i=t.length;i>e;e++)this.textureLoader.unload(t[e].rendererObject)},updateUVs:function(t){for(var e=this.regions,i=0,r=e.length;r>i;i++){var n=e[i];n.page==t&&(n.u=n.x/t.width,n.v=n.y/t.height,n.rotate?(n.u2=(n.x+n.height)/t.width,n.v2=(n.y+n.width)/t.height):(n.u2=(n.x+n.width)/t.width,n.v2=(n.y+n.height)/t.height))}}},i.Atlas.Format={alpha:0,intensity:1,luminanceAlpha:2,rgb565:3,rgba4444:4,rgb888:5,rgba8888:6},i.Atlas.TextureFilter={nearest:0,linear:1,mipMap:2,mipMapNearestNearest:3,mipMapLinearNearest:4,mipMapNearestLinear:5,mipMapLinearLinear:6},i.Atlas.TextureWrap={mirroredRepeat:0,clampToEdge:1,repeat:2},i.AtlasPage=function(){},i.AtlasPage.prototype={name:null,format:null,minFilter:null,magFilter:null,uWrap:null,vWrap:null,rendererObject:null,width:0,height:0},i.AtlasRegion=function(){},i.AtlasRegion.prototype={page:null,name:null,x:0,y:0,width:0,height:0,u:0,v:0,u2:0,v2:0,offsetX:0,offsetY:0,originalWidth:0,originalHeight:0,index:0,rotate:!1,splits:null,pads:null},i.AtlasReader=function(t){this.lines=t.split(/\r\n|\r|\n/)},i.AtlasReader.prototype={index:0,trim:function(t){return t.replace(/^\s+|\s+$/g,"")},readLine:function(){return this.index>=this.lines.length?null:this.lines[this.index++]},readValue:function(){var t=this.readLine(),e=t.indexOf(":");if(-1==e)throw"Invalid line: "+t;return this.trim(t.substring(e+1))},readTuple:function(t){var e=this.readLine(),i=e.indexOf(":");if(-1==i)throw"Invalid line: "+e;for(var r=0,n=i+1;3>r;r++){var s=e.indexOf(",",n);if(-1==s)break;t[r]=this.trim(e.substr(n,s-n)),n=s+1}return t[r]=this.trim(e.substring(n)),r+1}},i.AtlasAttachmentLoader=function(t){this.atlas=t},i.AtlasAttachmentLoader.prototype={newRegionAttachment:function(t,e,r){var n=this.atlas.findRegion(r);if(!n)throw"Region not found in atlas: "+r+" (region attachment: "+e+")";var s=new i.RegionAttachment(e);return s.rendererObject=n,s.setUVs(n.u,n.v,n.u2,n.v2,n.rotate),s.regionOffsetX=n.offsetX,s.regionOffsetY=n.offsetY,s.regionWidth=n.width,s.regionHeight=n.height,s.regionOriginalWidth=n.originalWidth,s.regionOriginalHeight=n.originalHeight,s},newMeshAttachment:function(t,e,r){var n=this.atlas.findRegion(r);if(!n)throw"Region not found in atlas: "+r+" (mesh attachment: "+e+")";var s=new i.MeshAttachment(e);return s.rendererObject=n,s.regionU=n.u,s.regionV=n.v,s.regionU2=n.u2,s.regionV2=n.v2,s.regionRotate=n.rotate,s.regionOffsetX=n.offsetX,s.regionOffsetY=n.offsetY,s.regionWidth=n.width,s.regionHeight=n.height,s.regionOriginalWidth=n.originalWidth,s.regionOriginalHeight=n.originalHeight,s},newSkinnedMeshAttachment:function(t,e,r){var n=this.atlas.findRegion(r);if(!n)throw"Region not found in atlas: "+r+" (skinned mesh attachment: "+e+")";var s=new i.SkinnedMeshAttachment(e);return s.rendererObject=n,s.regionU=n.u,s.regionV=n.v,s.regionU2=n.u2,s.regionV2=n.v2,s.regionRotate=n.rotate,s.regionOffsetX=n.offsetX,s.regionOffsetY=n.offsetY,s.regionWidth=n.width,s.regionHeight=n.height,s.regionOriginalWidth=n.originalWidth,s.regionOriginalHeight=n.originalHeight,s},newBoundingBoxAttachment:function(t,e){return new i.BoundingBoxAttachment(e)}},i.SkeletonBounds=function(){this.polygonPool=[],this.polygons=[],this.boundingBoxes=[]},i.SkeletonBounds.prototype={minX:0,minY:0,maxX:0,maxY:0,update:function(t,e){var r=t.slots,n=r.length,s=t.x,o=t.y,a=this.boundingBoxes,h=this.polygonPool,l=this.polygons;a.length=0;for(var u=0,c=l.length;c>u;u++)h.push(l[u]);l.length=0;for(var u=0;n>u;u++){var d=r[u],f=d.attachment;if(f.type==i.AttachmentType.boundingbox){a.push(f);var p,v=h.length;v>0?(p=h[v-1],h.splice(v-1,1)):p=[],l.push(p),p.length=f.vertices.length,f.computeWorldVertices(s,o,d.bone,p)}}e&&this.aabbCompute()},aabbCompute:function(){for(var t=this.polygons,e=Number.MAX_VALUE,i=Number.MAX_VALUE,r=Number.MIN_VALUE,n=Number.MIN_VALUE,s=0,o=t.length;o>s;s++)for(var a=t[s],h=0,l=a.length;l>h;h+=2){var u=a[h],c=a[h+1];e=Math.min(e,u),i=Math.min(i,c),r=Math.max(r,u),n=Math.max(n,c)}this.minX=e,this.minY=i,this.maxX=r,this.maxY=n},aabbContainsPoint:function(t,e){return t>=this.minX&&t<=this.maxX&&e>=this.minY&&e<=this.maxY},aabbIntersectsSegment:function(t,e,i,r){var n=this.minX,s=this.minY,o=this.maxX,a=this.maxY;if(n>=t&&n>=i||s>=e&&s>=r||t>=o&&i>=o||e>=a&&r>=a)return!1;var h=(r-e)/(i-t),l=h*(n-t)+e;if(l>s&&a>l)return!0;if(l=h*(o-t)+e,l>s&&a>l)return!0;var u=(s-e)/h+t;return u>n&&o>u?!0:(u=(a-e)/h+t,u>n&&o>u?!0:!1)},aabbIntersectsSkeleton:function(t){return this.minXt.minX&&this.minYt.minY},containsPoint:function(t,e){for(var i=this.polygons,r=0,n=i.length;n>r;r++)if(this.polygonContainsPoint(i[r],t,e))return this.boundingBoxes[r];return null},intersectsSegment:function(t,e,i,r){for(var n=this.polygons,s=0,o=n.length;o>s;s++)if(n[s].intersectsSegment(t,e,i,r))return this.boundingBoxes[s];return null},polygonContainsPoint:function(t,e,i){for(var r=t.length,n=r-2,s=!1,o=0;r>o;o+=2){var a=t[o+1],h=t[n+1];if(i>a&&h>=i||i>h&&a>=i){var l=t[o];l+(i-a)/(h-a)*(t[n]-l)c;c+=2){var d=t[c],f=t[c+1],p=l*f-u*d,v=l-d,m=u-f,g=o*m-a*v,x=(h*v-o*p)/g;if((x>=l&&d>=x||x>=d&&l>=x)&&(x>=e&&r>=x||x>=r&&e>=x)){var y=(h*m-a*p)/g;if((y>=u&&f>=y||y>=f&&u>=y)&&(y>=i&&n>=y||y>=n&&i>=y))return!0}l=d,u=f}return!1},getPolygon:function(t){var e=this.boundingBoxes.indexOf(t);return-1==e?null:this.polygons[e]},getWidth:function(){return this.maxX-this.minX},getHeight:function(){return this.maxY-this.minY}}},{}],95:[function(t,e){e.exports={Spine:t("./Spine")}},{"./Spine":93}],96:[function(t,e){function i(t,e){r.DisplayObjectContainer.call(this),this.textWidth=0,this.textHeight=0,this._pool=[],this._style={tint:e.tint,align:e.align,fontName:null,fontSize:0},this.font=e.font,this._text=t,this.dirty=!1,this.updateText()}var r=t("../core");i.prototype=Object.create(r.DisplayObjectContainer.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{tint:{get:function(){return this._style.tint},set:function(t){this._style.tint="number"==typeof t&&t>=0?t:16777215,this.dirty=!0}},align:{get:function(){return this._style.align},set:function(t){this._style.align=t,this.dirty=!0}},font:{get:function(){return this._style.font},set:function(t){t=t.split(" "),this._style.fontName=t[t.length-1],this._style.fontSize=t.length>=2?parseInt(t[t.length-2],10):i.fonts[this.fontName].size,this.dirty=!0}},text:{get:function(){return this._text},set:function(t){this._text=t,this.dirty=!0}}}),i.prototype.updateText=function(){for(var t=i.fonts[this.fontName],e=new r.math.Point,n=null,s=[],o=0,a=[],h=0,l=this.fontSize/t.size,u=0;u=u;u++){var p=0;"right"===this.style.align?p=o-a[u]:"center"===this.style.align&&(p=(o-a[u])/2),f.push(p)}var v=this.children.length,m=s.length,g=this.tint;for(u=0;m>u;u++){var x=v>u?this.children[u]:this._pool.pop();x?x.setTexture(s[u].texture):x=new r.Sprite(s[u].texture),x.position.x=(s[u].position.x+f[s[u].line])*l,x.position.y=s[u].position.y*l,x.scale.x=x.scale.y=l,x.tint=g,x.parent||this.addChild(x)}for(;this.children.length>m;){var y=this.getChildAt(this.children.length-1);this._pool.push(y),this.removeChild(y)}this.textWidth=o*l,this.textHeight=(e.y+t.lineHeight)*l},i.prototype.updateTransform=function(){this.dirty&&(this.updateText(),this.dirty=!1),this.displayObjectContainerUpdateTransform()},i.fonts={}},{"../core":9}],97:[function(t,e){function i(t,e){this.canvas=document.createElement("canvas"),this.context=this.canvas.getContext("2d"),this.resolution=1,r.Sprite.call(this,r.Texture.fromCanvas(this.canvas)),this.setText(t),this.setStyle(e)}var r=t("../core");i.prototype=Object.create(r.Sprite.prototype),i.prototype.constructor=i,e.exports=i,Object.defineProperties(i.prototype,{width:{get:function(){return this.dirty&&(this.updateText(),this.dirty=!1),this.scale.x*this.texture.frame.width},set:function(t){this.scale.x=t/this.texture.frame.width,this._width=t}},height:{get:function(){return this.dirty&&(this.updateText(),this.dirty=!1),this.scale.y*this.texture.frame.height},set:function(t){this.scale.y=t/this.texture.frame.height,this._height=t}}}),i.prototype.setStyle=function(t){t=t||{},t.font=t.font||"bold 20pt Arial",t.fill=t.fill||"black",t.align=t.align||"left",t.stroke=t.stroke||"black",t.strokeThickness=t.strokeThickness||0,t.wordWrap=t.wordWrap||!1,t.wordWrapWidth=t.wordWrapWidth||100,t.dropShadow=t.dropShadow||!1,t.dropShadowAngle=t.dropShadowAngle||Math.PI/6,t.dropShadowDistance=t.dropShadowDistance||4,t.dropShadowColor=t.dropShadowColor||"black",this.style=t,this.dirty=!0},i.prototype.setText=function(t){this.text=t.toString()||" ",this.dirty=!0},i.prototype.updateText=function(){this.texture.baseTexture.resolution=this.resolution,this.context.font=this.style.font;var t=this.text;this.style.wordWrap&&(t=this.wordWrap(this.text));for(var e=t.split(/(?:\r\n|\r|\n)/),i=[],r=0,n=this.determineFontProperties(this.style.font),s=0;sh;h++){for(l=0;d>l;l+=4)if(255!==u[f+l]){p=!0;break}if(p)break;f+=d}for(e.ascent=o-h,f=c-d,p=!1,h=a;h>o;h--){for(l=0;d>l;l+=4)if(255!==u[f+l]){p=!0;break}if(p)break;f-=d}e.descent=h-o,e.descent+=6,e.fontSize=e.ascent+e.descent,i.fontPropertiesCache[t]=e}return e},i.prototype.wordWrap=function(t){for(var e="",i=t.split("\n"),r=0;rn?(o>0&&(e+="\n"),e+=s[o],n=this.style.wordWrapWidth-a):(n-=h,e+=" "+s[o])}r