phaser/Phaser/input/Pointer.ts

618 lines
19 KiB
TypeScript
Raw Normal View History

2013-05-28 20:38:37 +00:00
/// <reference path="../Game.ts" />
/// <reference path="../core/Vec2.ts" />
2013-05-16 01:36:58 +00:00
/**
* Phaser - Pointer
*
* A Pointer object is used by the Touch and MSPoint managers and represents a single finger on the touch screen.
*/
module Phaser {
export class Pointer {
/**
* Constructor
* @param {Phaser.Game} game.
* @return {Phaser.Pointer} This object.
*/
constructor(game: Game, id: number) {
2013-06-02 13:19:12 +00:00
this.game = game;
2013-05-16 01:36:58 +00:00
this.id = id;
this.active = false;
2013-05-28 20:38:37 +00:00
this.position = new Vec2;
this.positionDown = new Vec2;
2013-05-16 01:36:58 +00:00
this.circle = new Circle(0, 0, 44);
2013-05-16 20:34:24 +00:00
if (id == 0)
{
this.isMouse = true;
}
2013-05-16 01:36:58 +00:00
}
/**
* Local private reference to game.
2013-06-02 13:19:12 +00:00
* @property game
2013-05-16 01:36:58 +00:00
* @type {Phaser.Game}
* @private
**/
2013-06-02 13:19:12 +00:00
public game: Game;
2013-05-16 01:36:58 +00:00
/**
2013-05-16 20:34:24 +00:00
* Local private variable to store the status of dispatching a hold event
* @property _holdSent
* @type {Boolean}
* @private
*/
private _holdSent: bool = false;
/**
* Local private variable storing the short-term history of pointer movements
* @property _history
* @type {Array}
* @private
*/
private _history = [];
/**
* Local private variable storing the time at which the next history drop should occur
* @property _lastDrop
* @type {Number}
* @private
*/
private _nextDrop: number = 0;
/**
* The Pointer ID (a number between 1 and 10, 0 is reserved for the mouse pointer specifically)
2013-05-16 01:36:58 +00:00
* @property id
* @type {Number}
*/
public id: number;
/**
* An identification number for each touch point.
* When a touch point becomes active, it is assigned an identifier that is distinct from any other active touch point.
* While the touch point remains active, all events that refer to it are assigned the same identifier.
* @property identifier
* @type {Number}
*/
public identifier: number;
/**
* Is this Pointer active or not? An active Pointer is one that is in contact with the touch screen.
* @property active
* @type {Boolean}
*/
public active: bool;
/**
2013-05-22 23:01:58 +00:00
* A Vector object containing the initial position when the Pointer was engaged with the screen.
* @property positionDown
2013-05-28 20:38:37 +00:00
* @type {Vec2}
2013-05-16 01:36:58 +00:00
**/
2013-05-28 20:38:37 +00:00
public positionDown: Vec2 = null;
2013-05-16 01:36:58 +00:00
/**
2013-05-22 23:01:58 +00:00
* A Vector object containing the current position of the Pointer on the screen.
* @property position
2013-05-28 20:38:37 +00:00
* @type {Vec2}
2013-05-16 01:36:58 +00:00
**/
2013-05-28 20:38:37 +00:00
public position: Vec2 = null;
2013-05-16 01:36:58 +00:00
/**
* A Circle object centered on the x/y screen coordinates of the Pointer.
* Default size of 44px (Apple's recommended "finger tip" size)
* @property circle
* @type {Circle}
**/
public circle: Circle = null;
/**
*
* @property withinGame
* @type {Boolean}
*/
public withinGame: bool = false;
2013-05-16 20:34:24 +00:00
/**
* If this Pointer is a mouse the button property holds the value of which mouse button was pressed down
* @property button
* @type {Number}
*/
public button: number;
2013-05-16 01:36:58 +00:00
/**
* The horizontal coordinate of point relative to the viewport in pixels, excluding any scroll offset
* @property clientX
* @type {Number}
*/
public clientX: number = -1;
/**
* The vertical coordinate of point relative to the viewport in pixels, excluding any scroll offset
* @property clientY
* @type {Number}
*/
public clientY: number = -1;
/**
* The horizontal coordinate of point relative to the viewport in pixels, including any scroll offset
* @property pageX
* @type {Number}
*/
public pageX: number = -1;
/**
* The vertical coordinate of point relative to the viewport in pixels, including any scroll offset
* @property pageY
* @type {Number}
*/
public pageY: number = -1;
/**
* The horizontal coordinate of point relative to the screen in pixels
* @property screenX
* @type {Number}
*/
public screenX: number = -1;
/**
* The vertical coordinate of point relative to the screen in pixels
* @property screenY
* @type {Number}
*/
public screenY: number = -1;
/**
* The horizontal coordinate of point relative to the game element
* @property x
* @type {Number}
*/
public x: number = -1;
/**
* The vertical coordinate of point relative to the game element
* @property y
* @type {Number}
*/
public y: number = -1;
/**
* The Element on which the touch point started when it was first placed on the surface, even if the touch point has since moved outside the interactive area of that element.
* @property target
* @type {Any}
*/
public target;
/**
2013-05-16 20:34:24 +00:00
* If the Pointer is a mouse this is true, otherwise false
* @property isMouse
* @type {Boolean}
**/
public isMouse: bool = false;
/**
* If the Pointer is touching the touchscreen, or the mouse button is held down, isDown is set to true
2013-05-16 01:36:58 +00:00
* @property isDown
* @type {Boolean}
**/
public isDown: bool = false;
/**
2013-05-16 20:34:24 +00:00
* If the Pointer is not touching the touchscreen, or the mouse button is up, isUp is set to true
2013-05-16 01:36:58 +00:00
* @property isUp
* @type {Boolean}
**/
public isUp: bool = true;
/**
* A timestamp representing when the Pointer first touched the touchscreen.
* @property timeDown
* @type {Number}
**/
public timeDown: number = 0;
/**
* A timestamp representing when the Pointer left the touchscreen.
* @property timeUp
* @type {Number}
**/
public timeUp: number = 0;
/**
2013-05-16 20:34:24 +00:00
* A timestamp representing when the Pointer was last tapped or clicked
* @property previousTapTime
2013-05-16 01:36:58 +00:00
* @type {Number}
**/
2013-05-16 20:34:24 +00:00
public previousTapTime: number = 0;
2013-05-16 01:36:58 +00:00
/**
* The total number of times this Pointer has been touched to the touchscreen
* @property totalTouches
* @type {Number}
**/
public totalTouches: number = 0;
/**
2013-05-16 20:34:24 +00:00
* How long the Pointer has been depressed on the touchscreen. If not currently down it returns -1.
2013-05-16 01:36:58 +00:00
* @property duration
* @type {Number}
**/
public get duration(): number {
2013-05-16 20:34:24 +00:00
if (this.isUp)
{
return -1;
}
2013-06-02 13:19:12 +00:00
return this.game.time.now - this.timeDown;
2013-05-16 01:36:58 +00:00
}
2013-06-02 13:19:12 +00:00
/**
* The Game Object this Pointer is currently dragging.
* @property draggedObject
* @type {Any}
**/
public draggedObject;
2013-05-16 01:36:58 +00:00
/**
* Gets the X value of this Pointer in world coordinate space
* @param {Camera} [camera]
*/
2013-06-02 13:19:12 +00:00
public getWorldX(camera?: Camera = this.game.camera) {
2013-05-16 01:36:58 +00:00
return camera.worldView.x + this.x;
}
/**
* Gets the Y value of this Pointer in world coordinate space
* @param {Camera} [camera]
*/
2013-06-02 13:19:12 +00:00
public getWorldY(camera?: Camera = this.game.camera) {
2013-05-16 01:36:58 +00:00
return camera.worldView.y + this.y;
}
/**
* Called when the Pointer is pressed onto the touchscreen
* @method start
* @param {Any} event
*/
public start(event): Pointer {
this.identifier = event.identifier;
this.target = event.target;
2013-05-16 20:34:24 +00:00
if (event.button)
{
this.button = event.button;
}
2013-05-22 23:01:58 +00:00
// Fix to stop rogue browser plugins from blocking the visibility state event
2013-06-02 13:19:12 +00:00
if (this.game.paused == true)
2013-05-22 23:01:58 +00:00
{
2013-06-02 13:19:12 +00:00
this.game.stage.resumeGame();
2013-05-22 23:01:58 +00:00
return this;
}
2013-05-16 20:34:24 +00:00
this._history.length = 0;
2013-05-16 01:36:58 +00:00
this.move(event);
2013-05-22 23:01:58 +00:00
this.positionDown.setTo(this.x, this.y);
2013-05-16 01:36:58 +00:00
this.active = true;
this.withinGame = true;
this.isDown = true;
this.isUp = false;
2013-06-02 13:19:12 +00:00
this.timeDown = this.game.time.now;
2013-05-16 20:34:24 +00:00
this._holdSent = false;
2013-05-16 01:36:58 +00:00
2013-06-02 13:19:12 +00:00
if (this.game.input.multiInputOverride == Input.MOUSE_OVERRIDES_TOUCH || this.game.input.multiInputOverride == Input.MOUSE_TOUCH_COMBINE || (this.game.input.multiInputOverride == Input.TOUCH_OVERRIDES_MOUSE && this.game.input.currentPointers == 0))
2013-05-16 20:34:24 +00:00
{
2013-06-02 13:19:12 +00:00
this.game.input.x = this.x * this.game.input.scaleX;
this.game.input.y = this.y * this.game.input.scaleY;
this.game.input.onDown.dispatch(this);
2013-05-16 20:34:24 +00:00
}
2013-05-16 01:36:58 +00:00
this.totalTouches++;
2013-05-16 20:34:24 +00:00
if (this.isMouse == false)
{
2013-06-02 13:19:12 +00:00
this.game.input.currentPointers++;
2013-05-16 20:34:24 +00:00
}
2013-05-16 01:36:58 +00:00
return this;
}
2013-05-16 20:34:24 +00:00
public update() {
if (this.active)
{
2013-06-02 13:19:12 +00:00
if (this._holdSent == false && this.duration >= this.game.input.holdRate)
2013-05-16 20:34:24 +00:00
{
2013-06-02 13:19:12 +00:00
if (this.game.input.multiInputOverride == Input.MOUSE_OVERRIDES_TOUCH || this.game.input.multiInputOverride == Input.MOUSE_TOUCH_COMBINE || (this.game.input.multiInputOverride == Input.TOUCH_OVERRIDES_MOUSE && this.game.input.currentPointers == 0))
2013-05-16 20:34:24 +00:00
{
2013-06-02 13:19:12 +00:00
this.game.input.onHold.dispatch(this);
2013-05-16 20:34:24 +00:00
}
this._holdSent = true;
}
// Update the droppings history
2013-06-02 13:19:12 +00:00
if (this.game.input.recordPointerHistory && this.game.time.now >= this._nextDrop)
2013-05-16 20:34:24 +00:00
{
2013-06-02 13:19:12 +00:00
this._nextDrop = this.game.time.now + this.game.input.recordRate;
2013-05-22 23:01:58 +00:00
this._history.push({ x: this.position.x, y: this.position.y });
2013-05-16 20:34:24 +00:00
2013-06-02 13:19:12 +00:00
if (this._history.length > this.game.input.recordLimit)
2013-05-16 20:34:24 +00:00
{
this._history.shift();
}
}
2013-06-02 13:19:12 +00:00
// Iterate through the tracked objects
// Build our temporary click stack
var _highestPriority = 0;
for (var i = 0; i < this.game.input.totalTrackedObjects; i++)
{
if (this.game.input.inputObjects[i].input.enabled)
{
if (this.game.input.inputObjects[i].input.update(this) && this.game.input.inputObjects[i].input.priorityID > _highestPriority)
2013-06-02 13:19:12 +00:00
{
_highestPriority = this.game.input.inputObjects[i].input.priorityID;
}
}
}
if (this.isDown)
{
// Now update all objects with the highest priority ID (can be more than 1)
for (var i = 0; i < this.game.input.totalTrackedObjects; i++)
{
if (this.game.input.inputObjects[i].input.priorityID == _highestPriority)
{
this.game.input.inputObjects[i].input._touchedHandler(this);
}
}
}
2013-05-16 20:34:24 +00:00
}
}
2013-05-16 01:36:58 +00:00
/**
* Called when the Pointer is moved on the touchscreen
* @method move
* @param {Any} event
*/
public move(event): Pointer {
2013-05-16 20:34:24 +00:00
if (event.button)
{
this.button = event.button;
}
2013-05-16 01:36:58 +00:00
this.clientX = event.clientX;
this.clientY = event.clientY;
this.pageX = event.pageX;
this.pageY = event.pageY;
this.screenX = event.screenX;
this.screenY = event.screenY;
2013-06-02 13:19:12 +00:00
this.x = this.pageX - this.game.stage.offset.x;
this.y = this.pageY - this.game.stage.offset.y;
2013-05-16 01:36:58 +00:00
2013-05-22 23:01:58 +00:00
this.position.setTo(this.x, this.y);
this.circle.x = this.x;
this.circle.y = this.y;
2013-05-16 01:36:58 +00:00
2013-06-02 13:19:12 +00:00
if (this.game.input.multiInputOverride == Input.MOUSE_OVERRIDES_TOUCH || this.game.input.multiInputOverride == Input.MOUSE_TOUCH_COMBINE || (this.game.input.multiInputOverride == Input.TOUCH_OVERRIDES_MOUSE && this.game.input.currentPointers == 0))
2013-05-16 20:34:24 +00:00
{
2013-06-02 13:19:12 +00:00
this.game.input.x = this.x * this.game.input.scaleX;
this.game.input.y = this.y * this.game.input.scaleY;
this.game.input.position.setTo(this.game.input.x, this.game.input.y);
this.game.input.circle.x = this.game.input.x;
this.game.input.circle.y = this.game.input.y;
2013-05-16 20:34:24 +00:00
}
2013-05-16 01:36:58 +00:00
return this;
}
/**
* Called when the Pointer leaves the target area
* @method leave
* @param {Any} event
*/
public leave(event) {
this.withinGame = false;
this.move(event);
}
/**
* Called when the Pointer leaves the touchscreen
* @method stop
* @param {Any} event
*/
public stop(event): Pointer {
2013-06-02 13:19:12 +00:00
this.timeUp = this.game.time.now;
2013-05-16 20:34:24 +00:00
2013-06-02 13:19:12 +00:00
if (this.game.input.multiInputOverride == Input.MOUSE_OVERRIDES_TOUCH || this.game.input.multiInputOverride == Input.MOUSE_TOUCH_COMBINE || (this.game.input.multiInputOverride == Input.TOUCH_OVERRIDES_MOUSE && this.game.input.currentPointers == 0))
2013-05-16 20:34:24 +00:00
{
2013-06-02 13:19:12 +00:00
this.game.input.onUp.dispatch(this);
2013-05-16 20:34:24 +00:00
// Was it a tap?
2013-06-02 13:19:12 +00:00
if (this.duration >= 0 && this.duration <= this.game.input.tapRate)
2013-05-16 20:34:24 +00:00
{
// Was it a double-tap?
2013-06-02 13:19:12 +00:00
if (this.timeUp - this.previousTapTime < this.game.input.doubleTapRate)
2013-05-16 20:34:24 +00:00
{
// Yes, let's dispatch the signal then with the 2nd parameter set to true
2013-06-02 13:19:12 +00:00
this.game.input.onTap.dispatch(this, true);
}
else
{
// Wasn't a double-tap, so dispatch a single tap signal
2013-06-02 13:19:12 +00:00
this.game.input.onTap.dispatch(this, false);
2013-05-16 20:34:24 +00:00
}
this.previousTapTime = this.timeUp;
}
}
2013-06-02 13:19:12 +00:00
// Mouse is always active
if (this.id > 0)
{
this.active = false;
}
2013-05-16 01:36:58 +00:00
this.withinGame = false;
this.isDown = false;
this.isUp = true;
2013-05-16 20:34:24 +00:00
if (this.isMouse == false)
{
2013-06-02 13:19:12 +00:00
this.game.input.currentPointers--;
2013-05-16 20:34:24 +00:00
}
2013-05-16 01:36:58 +00:00
for (var i = 0; i < this.game.input.totalTrackedObjects; i++)
{
if (this.game.input.inputObjects[i].input.enabled)
{
this.game.input.inputObjects[i].input._releasedHandler(this);
}
}
this.draggedObject = null;
2013-05-16 01:36:58 +00:00
return this;
}
/**
2013-05-16 20:34:24 +00:00
* The Pointer is considered justPressed if the time it was pressed onto the touchscreen or clicked is less than justPressedRate
2013-05-16 01:36:58 +00:00
* @method justPressed
* @param {Number} [duration].
* @return {Boolean}
*/
2013-06-02 13:19:12 +00:00
public justPressed(duration?: number = this.game.input.justPressedRate): bool {
2013-05-16 01:36:58 +00:00
2013-06-02 13:19:12 +00:00
if (this.isDown === true && (this.timeDown + duration) > this.game.time.now)
2013-05-16 01:36:58 +00:00
{
return true;
}
else
{
return false;
}
}
/**
* The Pointer is considered justReleased if the time it left the touchscreen is less than justReleasedRate
* @method justReleased
* @param {Number} [duration].
* @return {Boolean}
*/
2013-06-02 13:19:12 +00:00
public justReleased(duration?: number = this.game.input.justReleasedRate): bool {
2013-05-16 01:36:58 +00:00
2013-06-02 13:19:12 +00:00
if (this.isUp === true && (this.timeUp + duration) > this.game.time.now)
2013-05-16 01:36:58 +00:00
{
return true;
}
else
{
return false;
}
}
2013-05-16 20:34:24 +00:00
/**
* Resets the Pointer properties. Called by Input.reset when you perform a State change.
* @method reset
*/
2013-05-16 01:36:58 +00:00
public reset() {
this.active = false;
this.identifier = null;
this.isDown = false;
this.isUp = true;
this.totalTouches = 0;
2013-05-16 20:34:24 +00:00
this._holdSent = false;
this._history.length = 0;
2013-05-16 01:36:58 +00:00
}
/**
* Renders the Pointer.circle object onto the stage in green if down or red if up.
* @method renderDebug
*/
public renderDebug(hideIfUp: bool = false) {
if (hideIfUp == true && this.isUp == true)
{
return;
}
2013-06-02 13:19:12 +00:00
this.game.stage.context.beginPath();
this.game.stage.context.arc(this.x, this.y, this.circle.radius, 0, Math.PI * 2);
2013-05-16 01:36:58 +00:00
if (this.active)
{
2013-06-02 13:19:12 +00:00
this.game.stage.context.fillStyle = 'rgba(0,255,0,0.5)';
this.game.stage.context.strokeStyle = 'rgb(0,255,0)';
2013-05-16 01:36:58 +00:00
}
else
{
2013-06-02 13:19:12 +00:00
this.game.stage.context.fillStyle = 'rgba(255,0,0,0.5)';
this.game.stage.context.strokeStyle = 'rgb(100,0,0)';
2013-05-16 01:36:58 +00:00
}
2013-06-02 13:19:12 +00:00
this.game.stage.context.fill();
this.game.stage.context.closePath();
2013-05-16 01:36:58 +00:00
// Render the points
2013-06-02 13:19:12 +00:00
this.game.stage.context.beginPath();
this.game.stage.context.moveTo(this.positionDown.x, this.positionDown.y);
this.game.stage.context.lineTo(this.position.x, this.position.y);
this.game.stage.context.lineWidth = 2;
this.game.stage.context.stroke();
this.game.stage.context.closePath();
2013-05-16 01:36:58 +00:00
// Render the text
2013-06-02 13:19:12 +00:00
this.game.stage.context.fillStyle = 'rgb(255,255,255)';
this.game.stage.context.font = 'Arial 16px';
this.game.stage.context.fillText('ID: ' + this.id + " Active: " + this.active, this.x, this.y - 100);
this.game.stage.context.fillText('Screen X: ' + this.x + " Screen Y: " + this.y, this.x, this.y - 80);
this.game.stage.context.fillText('Duration: ' + this.duration + " ms", this.x, this.y - 60);
2013-05-16 01:36:58 +00:00
}
/**
* Returns a string representation of this object.
* @method toString
* @return {String} a string representation of the instance.
**/
public toString(): string {
return "[{Pointer (id=" + this.id + " identifer=" + this.identifier + " active=" + this.active + " duration=" + this.duration + " withinGame=" + this.withinGame + " x=" + this.x + " y=" + this.y + " clientX=" + this.clientX + " clientY=" + this.clientY + " screenX=" + this.screenX + " screenY=" + this.screenY + " pageX=" + this.pageX + " pageY=" + this.pageY + ")}]";
}
}
}