/// /// /// /// /// /// /// /// /// /** * Phaser - Sprite */ module Phaser { export class Sprite implements IGameObject { /** * Create a new Sprite. * * @param game {Phaser.Game} Current game instance. * @param [x] {number} the initial x position of the sprite. * @param [y] {number} the initial y position of the sprite. * @param [key] {string} Key of the graphic you want to load for this sprite. * @param [bodyType] {number} The physics body type of the object (defaults to BODY_DYNAMIC) */ constructor(game: Game, x?: number = 0, y?: number = 0, key?: string = null, frame? = null, bodyType?: number = Phaser.Types.BODY_DYNAMIC) { this.game = game; this.type = Phaser.Types.SPRITE; this.exists = true; this.active = true; this.visible = true; this.alive = true; this.x = x; this.y = y; this.z = -1; this.group = null; this.animations = new Phaser.Components.AnimationManager(this); this.input = new Phaser.Components.Sprite.Input(this); this.events = new Phaser.Components.Sprite.Events(this); this.texture = new Phaser.Components.Texture(this); this.transform = new Phaser.Components.Transform(this); if (key !== null) { this.texture.loadImage(key, false); } else { this.texture.opaque = true; } if (frame !== null) { if (typeof frame == 'string') { this.frameName = frame; } else { this.frame = frame; } } this.body = new Phaser.Physics.Body(this, bodyType); this.worldView = new Rectangle(x, y, this.width, this.height); this.cameraView = new Rectangle(x, y, this.width, this.height); this.transform.setCache(); // Handy proxies this.scale = this.transform.scale; this.alpha = this.texture.alpha; this.origin = this.transform.origin; } /** * Reference to the main game object */ public game: Game; /** * The type of game object. */ public type: number; /** * The Group this Sprite belongs to. */ public group: Group; /** * Controls if both update and render are called by the core game loop. */ public exists: bool; /** * Controls if update() is automatically called by the core game loop. */ public active: bool; /** * Controls if this Sprite is rendered or skipped during the core game loop. */ public visible: bool; /** * A useful state for many game objects. Kill and revive both flip this switch. */ public alive: bool; /** * Sprite physics body. */ public body: Phaser.Physics.Body; /** * The texture used to render the Sprite. */ public texture: Phaser.Components.Texture; /** * The Sprite transform component. */ public transform: Phaser.Components.Transform; /** * The Input component */ public input: Phaser.Components.Sprite.Input; /** * The Events component */ public events: Phaser.Components.Sprite.Events; /** * This manages animations of the sprite. You can modify animations though it. (see AnimationManager) * @type AnimationManager */ public animations: Phaser.Components.AnimationManager; /** * A Rectangle that defines the size and placement of the Sprite in the game world, * after taking into consideration both scrollFactor and scaling. */ public worldView: Phaser.Rectangle; /** * A Rectangle that maps to the placement of this sprite with respect to a specific Camera. * This value is constantly updated and modified during the internal render pass, it is not meant to be accessed directly. */ public cameraView: Phaser.Rectangle; /** * A boolean representing if the Sprite has been modified in any way via a scale, rotate, flip or skew. */ public modified: bool = false; /** * x value of the object. */ public x: number = 0; /** * y value of the object. */ public y: number = 0; /** * z order value of the object. */ public z: number = 0; /** * Render iteration counter */ public renderOrderID: number = 0; /** * The rotation of the sprite in degrees. Phaser uses a right-handed coordinate system, where 0 points to the right. */ public get rotation(): number { return this.transform.rotation; } /** * Set the rotation of the sprite in degrees. Phaser uses a right-handed coordinate system, where 0 points to the right. * The value is automatically wrapped to be between 0 and 360. */ public set rotation(value: number) { this.transform.rotation = this.game.math.wrap(value, 360, 0); } /** * The scale of the Sprite. A value of 1 is original scale. 0.5 is half size. 2 is double the size. * This is a reference to Sprite.transform.scale */ public scale: Phaser.Vec2; /** * The alpha of the Sprite between 0 and 1, a value of 1 being fully opaque. */ public alpha: number; /** * The origin of the Sprite around which rotation and positioning takes place. * This is a reference to Sprite.transform.origin */ public origin: Phaser.Vec2; /** * Set the animation frame by frame number. */ public set frame(value: number) { this.animations.frame = value; } /** * Get the animation frame number. */ public get frame(): number { return this.animations.frame; } /** * Set the animation frame by frame name. */ public set frameName(value: string) { this.animations.frameName = value; } /** * Get the animation frame name. */ public get frameName(): string { return this.animations.frameName; } public set width(value: number) { this.transform.scale.x = value / this.texture.width; } public get width(): number { return this.texture.width * this.transform.scale.x; } public set height(value: number) { this.transform.scale.y = value / this.texture.height; } public get height(): number { return this.texture.height * this.transform.scale.y; } /** * Pre-update is called right before update() on each object in the game loop. */ public preUpdate() { this.transform.update(); this.worldView.x = (this.x * this.transform.scrollFactor.x) - (this.width * this.transform.origin.x); this.worldView.y = (this.y * this.transform.scrollFactor.y) - (this.height * this.transform.origin.y); //this.worldView.x = this.x * this.transform.scrollFactor.x; //this.worldView.y = this.y * this.transform.scrollFactor.y; this.worldView.width = this.width; this.worldView.height = this.height; if (this.modified == false && (!this.transform.scale.equals(1) || !this.transform.skew.equals(0) || this.transform.rotation != 0 || this.transform.rotationOffset != 0 || this.texture.flippedX || this.texture.flippedY)) { this.modified = true; } } /** * Override this function to update your class's position and appearance. */ public update() { } /** * Automatically called after update() by the game loop. */ public postUpdate() { this.animations.update(); this.body.postUpdate(); /* if (this.worldBounds != null) { if (this.outOfBoundsAction == GameObject.OUT_OF_BOUNDS_KILL) { if (this.x < this.worldBounds.x || this.x > this.worldBounds.right || this.y < this.worldBounds.y || this.y > this.worldBounds.bottom) { this.kill(); } } else { if (this.x < this.worldBounds.x) { this.x = this.worldBounds.x; } else if (this.x > this.worldBounds.right) { this.x = this.worldBounds.right; } if (this.y < this.worldBounds.y) { this.y = this.worldBounds.y; } else if (this.y > this.worldBounds.bottom) { this.y = this.worldBounds.bottom; } } } */ if (this.modified == true && this.transform.scale.equals(1) && this.transform.skew.equals(0) && this.transform.rotation == 0 && this.transform.rotationOffset == 0 && this.texture.flippedX == false && this.texture.flippedY == false) { this.modified = false; } } /** * Clean up memory. */ public destroy() { this.input.destroy(); } /** * Handy for "killing" game objects. * Default behavior is to flag them as nonexistent AND dead. * However, if you want the "corpse" to remain in the game, * like to animate an effect or whatever, you should override this, * setting only alive to false, and leaving exists true. */ public kill(removeFromGroup:bool = false) { this.alive = false; this.exists = false; if (removeFromGroup && this.group) { this.group.remove(this); } this.events.onKilled.dispatch(this); } /** * Handy for bringing game objects "back to life". Just sets alive and exists back to true. * In practice, this is most often called by Object.reset(). */ public revive() { this.alive = true; this.exists = true; this.events.onRevived.dispatch(this); } } }