/// /// /// /// /// module Phaser { export class CanvasRenderer implements Phaser.IRenderer { constructor(game: Phaser.Game) { this._game = game; } /** * The essential reference to the main game object */ private _game: Phaser.Game; // local rendering related temp vars to help avoid gc spikes with var creation private _ga: number = 1; private _sx: number = 0; private _sy: number = 0; private _sw: number = 0; private _sh: number = 0; private _dx: number = 0; private _dy: number = 0; private _dw: number = 0; private _dh: number = 0; private _fx: number = 1; private _fy: number = 1; private _sin: number = 0; private _cos: number = 1; private _cameraList; private _camera: Camera; private _groupLength: number; private _count: number; public renderTotal: number; public render() { // Get a list of all the active cameras this._cameraList = this._game.world.getAllCameras(); this._count = 0; // Then iterate through world.group on them all (where not blacklisted, etc) for (var c = 0; c < this._cameraList.length; c++) { this._camera = this._cameraList[c]; this._camera.preRender(); this._game.world.group.render(this._camera); this._camera.postRender(); } this.renderTotal = this._count; } public renderGameObject(object) { if (object.type == Types.SPRITE) { this.renderSprite(this._camera, object); } else if (object.type == Types.SCROLLZONE) { this.renderScrollZone(this._camera, object); } } /** * Check whether this object is visible in a specific camera rectangle. * @param camera {Rectangle} The rectangle you want to check. * @return {boolean} Return true if bounds of this sprite intersects the given rectangle, otherwise return false. */ public inCamera(camera: Camera, sprite: Sprite): bool { // Object fixed in place regardless of the camera scrolling? Then it's always visible if (sprite.scrollFactor.x == 0 && sprite.scrollFactor.y == 0) { return true; } this._dx = sprite.frameBounds.x - camera.worldView.x; this._dy = sprite.frameBounds.y - camera.worldView.y; this._dw = sprite.frameBounds.width * sprite.scale.x; this._dh = sprite.frameBounds.height * sprite.scale.y; return (camera.worldView.right > this._dx) && (camera.worldView.x < this._dx + this._dw) && (camera.worldView.bottom > this._dy) && (camera.worldView.y < this._dy + this._dh); } /** * Render this sprite to specific camera. Called by game loop after update(). * @param camera {Camera} Camera this sprite will be rendered to. * @return {boolean} Return false if not rendered, otherwise return true. */ public renderSprite(camera: Camera, sprite: Sprite): bool { if (sprite.scale.x == 0 || sprite.scale.y == 0 || sprite.texture.alpha < 0.1 || this.inCamera(camera, sprite) == false) { return false; } this._count++; // Reset our temp vars this._ga = -1; this._sx = 0; this._sy = 0; this._sw = sprite.frameBounds.width; this._sh = sprite.frameBounds.height; this._fx = sprite.scale.x; this._fy = sprite.scale.y; this._sin = 0; this._cos = 1; this._dx = (camera.scaledX * sprite.scrollFactor.x) + sprite.frameBounds.x - (camera.worldView.x * sprite.scrollFactor.x); this._dy = (camera.scaledY * sprite.scrollFactor.y) + sprite.frameBounds.y - (camera.worldView.y * sprite.scrollFactor.y); this._dw = sprite.frameBounds.width; this._dh = sprite.frameBounds.height; // Alpha if (sprite.texture.alpha !== 1) { this._ga = sprite.texture.context.globalAlpha; sprite.texture.context.globalAlpha = sprite.texture.alpha; } // Sprite Flip X if (sprite.texture.flippedX) { this._fx = -sprite.scale.x; } // Sprite Flip Y if (sprite.texture.flippedY) { this._fy = -sprite.scale.y; } if (sprite.animations.currentFrame !== null) { this._sx = sprite.animations.currentFrame.x; this._sy = sprite.animations.currentFrame.y; if (sprite.animations.currentFrame.trimmed) { this._dx += sprite.animations.currentFrame.spriteSourceSizeX; this._dy += sprite.animations.currentFrame.spriteSourceSizeY; } } // Rotation and Flipped if (sprite.modified) { if (sprite.texture.renderRotation == true && (sprite.angle !== 0 || sprite.angleOffset !== 0)) { this._sin = Math.sin(sprite.game.math.degreesToRadians(sprite.angleOffset + sprite.angle)); this._cos = Math.cos(sprite.game.math.degreesToRadians(sprite.angleOffset + sprite.angle)); } // setTransform(a, b, c, d, e, f); // a = scale x // b = skew x // c = skew y // d = scale y // e = translate x // f = translate y sprite.texture.context.save(); sprite.texture.context.setTransform(this._cos * this._fx, (this._sin * this._fx) + sprite.skew.x, -(this._sin * this._fy) + sprite.skew.y, this._cos * this._fy, this._dx, this._dy); this._dx = -sprite.origin.x; this._dy = -sprite.origin.y; } else { if (!sprite.origin.equals(0)) { this._dx -= sprite.origin.x; this._dy -= sprite.origin.y; } } this._sx = Math.round(this._sx); this._sy = Math.round(this._sy); this._sw = Math.round(this._sw); this._sh = Math.round(this._sh); this._dx = Math.round(this._dx); this._dy = Math.round(this._dy); this._dw = Math.round(this._dw); this._dh = Math.round(this._dh); if (sprite.texture.loaded) { sprite.texture.context.drawImage( sprite.texture.texture, // Source Image this._sx, // Source X (location within the source image) this._sy, // Source Y this._sw, // Source Width this._sh, // Source Height this._dx, // Destination X (where on the canvas it'll be drawn) this._dy, // Destination Y this._dw, // Destination Width (always same as Source Width unless scaled) this._dh // Destination Height (always same as Source Height unless scaled) ); } else { //sprite.texture.context.fillStyle = this.fillColor; sprite.texture.context.fillStyle = 'rgb(255,255,255)'; sprite.texture.context.fillRect(this._dx, this._dy, this._dw, this._dh); } if (sprite.modified) { sprite.texture.context.restore(); } if (this._ga > -1) { sprite.texture.context.globalAlpha = this._ga; } return true; } public renderScrollZone(camera: Camera, scrollZone: ScrollZone): bool { if (scrollZone.scale.x == 0 || scrollZone.scale.y == 0 || scrollZone.texture.alpha < 0.1 || this.inCamera(camera, scrollZone) == false) { return false; } this._count++; // Reset our temp vars this._ga = -1; this._sx = 0; this._sy = 0; this._sw = scrollZone.frameBounds.width; this._sh = scrollZone.frameBounds.height; this._fx = scrollZone.scale.x; this._fy = scrollZone.scale.y; this._sin = 0; this._cos = 1; this._dx = (camera.scaledX * scrollZone.scrollFactor.x) + scrollZone.frameBounds.x - (camera.worldView.x * scrollZone.scrollFactor.x); this._dy = (camera.scaledY * scrollZone.scrollFactor.y) + scrollZone.frameBounds.y - (camera.worldView.y * scrollZone.scrollFactor.y); this._dw = scrollZone.frameBounds.width; this._dh = scrollZone.frameBounds.height; // Alpha if (scrollZone.texture.alpha !== 1) { this._ga = scrollZone.texture.context.globalAlpha; scrollZone.texture.context.globalAlpha = scrollZone.texture.alpha; } // Sprite Flip X if (scrollZone.texture.flippedX) { this._fx = -scrollZone.scale.x; } // Sprite Flip Y if (scrollZone.texture.flippedY) { this._fy = -scrollZone.scale.y; } // Rotation and Flipped if (scrollZone.modified) { if (scrollZone.texture.renderRotation == true && (scrollZone.angle !== 0 || scrollZone.angleOffset !== 0)) { this._sin = Math.sin(scrollZone.game.math.degreesToRadians(scrollZone.angleOffset + scrollZone.angle)); this._cos = Math.cos(scrollZone.game.math.degreesToRadians(scrollZone.angleOffset + scrollZone.angle)); } // setTransform(a, b, c, d, e, f); // a = scale x // b = skew x // c = skew y // d = scale y // e = translate x // f = translate y scrollZone.texture.context.save(); scrollZone.texture.context.setTransform(this._cos * this._fx, (this._sin * this._fx) + scrollZone.skew.x, -(this._sin * this._fy) + scrollZone.skew.y, this._cos * this._fy, this._dx, this._dy); this._dx = -scrollZone.origin.x; this._dy = -scrollZone.origin.y; } else { if (!scrollZone.origin.equals(0)) { this._dx -= scrollZone.origin.x; this._dy -= scrollZone.origin.y; } } this._sx = Math.round(this._sx); this._sy = Math.round(this._sy); this._sw = Math.round(this._sw); this._sh = Math.round(this._sh); this._dx = Math.round(this._dx); this._dy = Math.round(this._dy); this._dw = Math.round(this._dw); this._dh = Math.round(this._dh); for (var i = 0; i < scrollZone.regions.length; i++) { if (scrollZone.texture.isDynamic) { scrollZone.regions[i].render(scrollZone.texture.context, scrollZone.texture.texture, this._dx, this._dy, this._dw, this._dh); } else { scrollZone.regions[i].render(scrollZone.texture.context, scrollZone.texture.texture, this._dx, this._dy, this._dw, this._dh); } } if (scrollZone.modified) { scrollZone.texture.context.restore(); } if (this._ga > -1) { scrollZone.texture.context.globalAlpha = this._ga; } return true; } } }