mirror of
https://github.com/photonstorm/phaser
synced 2024-12-23 11:33:28 +00:00
357 lines
No EOL
14 KiB
TypeScript
357 lines
No EOL
14 KiB
TypeScript
/// <reference path="../_definitions.ts" />
|
|
|
|
/**
|
|
* Phaser - Components - TransformManager
|
|
*/
|
|
|
|
module Phaser.Components {
|
|
|
|
export class TransformManager {
|
|
|
|
/**
|
|
* Creates a new TransformManager component
|
|
* @param parent The game object using this transform
|
|
*/
|
|
constructor(parent) {
|
|
|
|
this.game = parent.game;
|
|
this.parent = parent;
|
|
|
|
this.local = new Phaser.Mat3;
|
|
|
|
this.scrollFactor = new Phaser.Vec2(1, 1);
|
|
this.origin = new Phaser.Vec2;
|
|
this.scale = new Phaser.Vec2(1, 1);
|
|
this.skew = new Phaser.Vec2;
|
|
|
|
this.center = new Phaser.Point;
|
|
this.upperLeft = new Phaser.Point;
|
|
this.upperRight = new Phaser.Point;
|
|
this.bottomLeft = new Phaser.Point;
|
|
this.bottomRight = new Phaser.Point;
|
|
|
|
this._pos = new Phaser.Point;
|
|
this._scale = new Phaser.Point;
|
|
this._size = new Phaser.Point;
|
|
this._halfSize = new Phaser.Point;
|
|
this._offset = new Phaser.Point;
|
|
this._origin = new Phaser.Point;
|
|
this._sc = new Phaser.Point;
|
|
this._scA = new Phaser.Point;
|
|
|
|
}
|
|
|
|
private _rotation: number;
|
|
|
|
private _dirty: boolean = false;
|
|
|
|
// Cache vars
|
|
private _pos: Phaser.Point;
|
|
private _scale: Phaser.Point;
|
|
private _size: Phaser.Point;
|
|
private _halfSize: Phaser.Point;
|
|
private _offset: Phaser.Point;
|
|
private _origin: Phaser.Point;
|
|
private _sc: Phaser.Point;
|
|
private _scA: Phaser.Point;
|
|
private _angle: number;
|
|
private _distance: number;
|
|
private _prevRotation: number;
|
|
private _flippedX: boolean;
|
|
private _flippedY: boolean;
|
|
|
|
/**
|
|
* Reference to Phaser.Game
|
|
*/
|
|
public game: Phaser.Game;
|
|
|
|
/**
|
|
* Reference to the parent object (Sprite, Group, etc)
|
|
*/
|
|
public parent: Phaser.Sprite;
|
|
|
|
/**
|
|
* Scale of the object. A scale of 1.0 is the original size. 0.5 half size. 2.0 double sized.
|
|
*/
|
|
public scale: Phaser.Vec2;
|
|
|
|
/**
|
|
* Skew the object along the x and y axis. A skew value of 0 is no skew.
|
|
*/
|
|
public skew: Phaser.Vec2;
|
|
|
|
/**
|
|
* The influence of camera movement upon the object, if supported.
|
|
*/
|
|
public scrollFactor: Phaser.Vec2;
|
|
|
|
/**
|
|
* The origin is the point around which scale and rotation takes place and defaults to the top-left of the sprite.
|
|
*/
|
|
public origin: Phaser.Vec2;
|
|
|
|
/**
|
|
* This value is added to the rotation of the object.
|
|
* For example if you had a texture drawn facing straight up then you could set
|
|
* rotationOffset to 90 and it would correspond correctly with Phasers right-handed coordinate system.
|
|
* @type {number}
|
|
*/
|
|
public rotationOffset: number = 0;
|
|
|
|
/**
|
|
* The rotation of the object in degrees. Phaser uses a right-handed coordinate system, where 0 points to the right.
|
|
*/
|
|
public rotation: number = 0;
|
|
|
|
/**
|
|
* The center of the Sprite in world coordinates, after taking scaling and rotation into consideration
|
|
*/
|
|
public center: Phaser.Point;
|
|
|
|
/**
|
|
* The upper-left corner of the Sprite in world coordinates, after taking scaling and rotation into consideration
|
|
*/
|
|
public upperLeft: Phaser.Point;
|
|
|
|
/**
|
|
* The upper-right corner of the Sprite in world coordinates, after taking scaling and rotation into consideration
|
|
*/
|
|
public upperRight: Phaser.Point;
|
|
|
|
/**
|
|
* The bottom-left corner of the Sprite in world coordinates, after taking scaling and rotation into consideration
|
|
*/
|
|
public bottomLeft: Phaser.Point;
|
|
|
|
/**
|
|
* The bottom-right corner of the Sprite in world coordinates, after taking scaling and rotation into consideration
|
|
*/
|
|
public bottomRight: Phaser.Point;
|
|
|
|
/**
|
|
* The local transform matrix
|
|
*/
|
|
public local: Phaser.Mat3;
|
|
|
|
/**
|
|
* The distance from the center of the transform to the rotation origin.
|
|
*/
|
|
public get distance(): number {
|
|
return this._distance;
|
|
}
|
|
|
|
/**
|
|
* The angle between the center of the transform to the rotation origin.
|
|
*/
|
|
public get angleToCenter(): number {
|
|
return this._angle;
|
|
}
|
|
|
|
/**
|
|
* The offset on the X axis of the origin That is the difference between the top left of the Sprite and the origin.x.
|
|
* So if the origin.x is 0 the offsetX will be 0. If the origin.x is 0.5 then offsetX will be sprite width / 2, and so on.
|
|
*/
|
|
public get offsetX(): number {
|
|
return this._offset.x;
|
|
}
|
|
|
|
/**
|
|
* The offset on the Y axis of the origin
|
|
*/
|
|
public get offsetY(): number {
|
|
return this._offset.y;
|
|
}
|
|
|
|
/**
|
|
* Half the width of the parent sprite, taking into consideration scaling
|
|
*/
|
|
public get halfWidth(): number {
|
|
return this._halfSize.x;
|
|
}
|
|
|
|
/**
|
|
* Half the height of the parent sprite, taking into consideration scaling
|
|
*/
|
|
public get halfHeight(): number {
|
|
return this._halfSize.y;
|
|
}
|
|
|
|
/**
|
|
* The equivalent of Math.sin(rotation + rotationOffset)
|
|
*/
|
|
public get sin(): number {
|
|
return this._sc.x;
|
|
}
|
|
|
|
/**
|
|
* The equivalent of Math.cos(rotation + rotationOffset)
|
|
*/
|
|
public get cos(): number {
|
|
return this._sc.y;
|
|
}
|
|
|
|
/**
|
|
* Moves the sprite so its center is located on the given x and y coordinates.
|
|
* Doesn't change the origin of the sprite.
|
|
*/
|
|
public centerOn(x: number, y: number) {
|
|
|
|
this.parent.x = x + (this.parent.x - this.center.x);
|
|
this.parent.y = y + (this.parent.y - this.center.y);
|
|
|
|
//this.setCache();
|
|
|
|
}
|
|
|
|
/**
|
|
* Populates the transform cache. Called by the parent object on creation.
|
|
*/
|
|
public setCache() {
|
|
|
|
this._pos.x = this.parent.x;
|
|
this._pos.y = this.parent.y;
|
|
this._halfSize.x = this.parent.width / 2;
|
|
this._halfSize.y = this.parent.height / 2;
|
|
this._offset.x = this.origin.x * this.parent.width;
|
|
this._offset.y = this.origin.y * this.parent.height;
|
|
this._angle = Math.atan2(this.halfHeight - this._offset.x, this.halfWidth - this._offset.y);
|
|
this._distance = Math.sqrt(((this._offset.x - this._halfSize.x) * (this._offset.x - this._halfSize.x)) + ((this._offset.y - this._halfSize.y) * (this._offset.y - this._halfSize.y)));
|
|
this._size.x = this.parent.width;
|
|
this._size.y = this.parent.height;
|
|
this._origin.x = this.origin.x;
|
|
this._origin.y = this.origin.y;
|
|
this._scA.x = Math.sin((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD + this._angle);
|
|
this._scA.y = Math.cos((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD + this._angle);
|
|
this._prevRotation = this.rotation;
|
|
|
|
if (this.parent.texture && this.parent.texture.renderRotation)
|
|
{
|
|
this._sc.x = Math.sin((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD);
|
|
this._sc.y = Math.cos((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD);
|
|
}
|
|
else
|
|
{
|
|
this._sc.x = 0;
|
|
this._sc.y = 1;
|
|
}
|
|
|
|
this.center.x = this.parent.x + this._distance * this._scA.y;
|
|
this.center.y = this.parent.y + this._distance * this._scA.x;
|
|
|
|
this.upperLeft.setTo(this.center.x - this._halfSize.x * this._sc.y + this._halfSize.y * this._sc.x, this.center.y - this._halfSize.y * this._sc.y - this._halfSize.x * this._sc.x);
|
|
this.upperRight.setTo(this.center.x + this._halfSize.x * this._sc.y + this._halfSize.y * this._sc.x, this.center.y - this._halfSize.y * this._sc.y + this._halfSize.x * this._sc.x);
|
|
this.bottomLeft.setTo(this.center.x - this._halfSize.x * this._sc.y - this._halfSize.y * this._sc.x, this.center.y + this._halfSize.y * this._sc.y - this._halfSize.x * this._sc.x);
|
|
this.bottomRight.setTo(this.center.x + this._halfSize.x * this._sc.y - this._halfSize.y * this._sc.x, this.center.y + this._halfSize.y * this._sc.y + this._halfSize.x * this._sc.x);
|
|
|
|
this._pos.x = this.parent.x;
|
|
this._pos.y = this.parent.y;
|
|
|
|
this._flippedX = this.parent.texture.flippedX;
|
|
this._flippedY = this.parent.texture.flippedY;
|
|
|
|
}
|
|
|
|
/**
|
|
* Updates the local transform matrix and the cache values if anything has changed in the parent.
|
|
*/
|
|
public update() {
|
|
|
|
// Check cache
|
|
this._dirty = false;
|
|
|
|
// 1) Height or Width change (also triggered by a change in scale) or an Origin change
|
|
if (this.parent.width !== this._size.x || this.parent.height !== this._size.y || this.origin.x !== this._origin.x|| this.origin.y !== this._origin.y)
|
|
{
|
|
this._halfSize.x = this.parent.width / 2;
|
|
this._halfSize.y = this.parent.height / 2;
|
|
this._offset.x = this.origin.x * this.parent.width;
|
|
this._offset.y = this.origin.y * this.parent.height;
|
|
this._angle = Math.atan2(this.halfHeight - this._offset.y, this.halfWidth - this._offset.x);
|
|
this._distance = Math.sqrt(((this._offset.x - this._halfSize.x) * (this._offset.x - this._halfSize.x)) + ((this._offset.y - this._halfSize.y) * (this._offset.y - this._halfSize.y)));
|
|
// Store
|
|
this._size.x = this.parent.width;
|
|
this._size.y = this.parent.height;
|
|
this._origin.x = this.origin.x;
|
|
this._origin.y = this.origin.y;
|
|
this._dirty = true;
|
|
}
|
|
|
|
// 2) Rotation change
|
|
if (this.rotation != this._prevRotation || this._dirty)
|
|
{
|
|
this._scA.y = Math.cos((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD + this._angle);
|
|
this._scA.x = Math.sin((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD + this._angle);
|
|
|
|
if (this.parent.texture.renderRotation)
|
|
{
|
|
this._sc.x = Math.sin((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD);
|
|
this._sc.y = Math.cos((this.rotation + this.rotationOffset) * Phaser.GameMath.DEG_TO_RAD);
|
|
}
|
|
else
|
|
{
|
|
this._sc.x = 0;
|
|
this._sc.y = 1;
|
|
}
|
|
|
|
// Store
|
|
this._prevRotation = this.rotation;
|
|
this._dirty = true;
|
|
}
|
|
|
|
// 3) If it has moved (or any of the above) then update the edges and center
|
|
if (this._dirty || this.parent.x != this._pos.x || this.parent.y != this._pos.y)
|
|
{
|
|
this.center.x = this.parent.x + this._distance * this._scA.y;
|
|
this.center.y = this.parent.y + this._distance * this._scA.x;
|
|
|
|
this.upperLeft.setTo(this.center.x - this._halfSize.x * this._sc.y + this._halfSize.y * this._sc.x, this.center.y - this._halfSize.y * this._sc.y - this._halfSize.x * this._sc.x);
|
|
this.upperRight.setTo(this.center.x + this._halfSize.x * this._sc.y + this._halfSize.y * this._sc.x, this.center.y - this._halfSize.y * this._sc.y + this._halfSize.x * this._sc.x);
|
|
this.bottomLeft.setTo(this.center.x - this._halfSize.x * this._sc.y - this._halfSize.y * this._sc.x, this.center.y + this._halfSize.y * this._sc.y - this._halfSize.x * this._sc.x);
|
|
this.bottomRight.setTo(this.center.x + this._halfSize.x * this._sc.y - this._halfSize.y * this._sc.x, this.center.y + this._halfSize.y * this._sc.y + this._halfSize.x * this._sc.x);
|
|
|
|
this._pos.x = this.parent.x;
|
|
this._pos.y = this.parent.y;
|
|
|
|
// Translate
|
|
this.local.data[2] = this.parent.x;
|
|
this.local.data[5] = this.parent.y;
|
|
}
|
|
|
|
// Scale and Skew
|
|
if (this._dirty || this._flippedX != this.parent.texture.flippedX)
|
|
{
|
|
this._flippedX = this.parent.texture.flippedX;
|
|
|
|
if (this._flippedX)
|
|
{
|
|
this.local.data[0] = this._sc.y * -this.scale.x;
|
|
this.local.data[3] = (this._sc.x * -this.scale.x) + this.skew.x;
|
|
}
|
|
else
|
|
{
|
|
this.local.data[0] = this._sc.y * this.scale.x;
|
|
this.local.data[3] = (this._sc.x * this.scale.x) + this.skew.x;
|
|
}
|
|
}
|
|
|
|
if (this._dirty || this._flippedY != this.parent.texture.flippedY)
|
|
{
|
|
this._flippedY = this.parent.texture.flippedY;
|
|
|
|
if (this._flippedY)
|
|
{
|
|
this.local.data[4] = this._sc.y * -this.scale.y;
|
|
this.local.data[1] = -(this._sc.x * -this.scale.y) + this.skew.y;
|
|
}
|
|
else
|
|
{
|
|
this.local.data[4] = this._sc.y * this.scale.y;
|
|
this.local.data[1] = -(this._sc.x * this.scale.y) + this.skew.y;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} |