2017-11-10 02:50:25 +00:00
|
|
|
var Class = require('../../utils/Class');
|
2017-11-14 21:34:33 +00:00
|
|
|
var Components = require('../components');
|
2017-11-10 02:50:25 +00:00
|
|
|
|
|
|
|
var Tile = new Class({
|
|
|
|
|
2017-11-14 21:34:33 +00:00
|
|
|
Mixins: [
|
|
|
|
Components.Alpha,
|
|
|
|
Components.Flip,
|
|
|
|
Components.Visible
|
|
|
|
],
|
|
|
|
|
2017-11-10 02:50:25 +00:00
|
|
|
initialize:
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* A Tile is a representation of a single tile within the Tilemap. This is a lightweight data
|
|
|
|
* representation, so it's position information is stored without factoring in scroll, layer
|
|
|
|
* scale or layer position.
|
|
|
|
*
|
|
|
|
* @class Tile
|
|
|
|
* @constructor
|
|
|
|
*
|
|
|
|
* @param {LayerData} layer - The LayerData object in the Tilemap that this tile belongs to.
|
|
|
|
* @param {integer} index - The unique index of this tile within the map.
|
|
|
|
* @param {integer} x - The x coordinate of this tile in tile coordinates.
|
|
|
|
* @param {integer} y - The y coordinate of this tile in tile coordinates.
|
|
|
|
* @param {integer} width - Width of the tile in pixels.
|
|
|
|
* @param {integer} height - Height of the tile in pixels.
|
|
|
|
*/
|
2017-11-10 02:50:25 +00:00
|
|
|
function Tile (layer, index, x, y, width, height)
|
|
|
|
{
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The LayerData in the Tilemap data that this tile belongs to.
|
|
|
|
* @property {LayerData} layer
|
|
|
|
*/
|
2017-11-10 02:50:25 +00:00
|
|
|
this.layer = layer;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The index of this tile within the map data corresponding to the tileset, or -1 if this
|
|
|
|
* represents a blank tile.
|
|
|
|
* @property {integer} index
|
|
|
|
*/
|
2017-11-10 02:50:25 +00:00
|
|
|
this.index = index;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The x map coordinate of this tile in tile units.
|
|
|
|
* @property {integer} x
|
|
|
|
*/
|
2017-11-10 02:50:25 +00:00
|
|
|
this.x = x;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The y map coordinate of this tile in tile units.
|
|
|
|
* @property {integer} y
|
|
|
|
*/
|
2017-11-10 02:50:25 +00:00
|
|
|
this.y = y;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The world x coordinate of this tile in pixels. This does not factor in camera scroll,
|
|
|
|
* layer scale or layer position.
|
|
|
|
* @property {number} x
|
|
|
|
*/
|
2017-11-14 21:34:33 +00:00
|
|
|
this.worldX = x * width;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The world y coordinate of this tile in pixels. This does not factor in camera scroll,
|
|
|
|
* layer scale or layer position.
|
|
|
|
* @property {number} y
|
|
|
|
*/
|
2017-11-14 21:34:33 +00:00
|
|
|
this.worldY = y * height;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The width of the tile in pixels.
|
|
|
|
* @property {integer} width
|
|
|
|
*/
|
2017-11-10 02:50:25 +00:00
|
|
|
this.width = width;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The height of the tile in pixels.
|
|
|
|
* @property {integer} height
|
|
|
|
*/
|
2017-11-10 02:50:25 +00:00
|
|
|
this.height = height;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Tile specific properties. These usually come from Tiled.
|
|
|
|
* @property {object} properties
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
this.properties = {};
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The rotation angle of this tile.
|
|
|
|
* @property {number} rotation
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.rotation = 0;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile should collide with any object on the left side.
|
|
|
|
* @property {boolean} collideLeft
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.collideLeft = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile should collide with any object on the right side.
|
|
|
|
* @property {boolean} collideRight
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.collideRight = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile should collide with any object on the top side.
|
|
|
|
* @property {boolean} collideUp
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.collideUp = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile should collide with any object on the bottom side.
|
|
|
|
* @property {boolean} collideDown
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.collideDown = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile's left edge is interesting for collisions.
|
|
|
|
* @property {boolean} faceLeft
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.faceLeft = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile's right edge is interesting for collisions.
|
|
|
|
* @property {boolean} faceRight
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.faceRight = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile's top edge is interesting for collisions.
|
|
|
|
* @property {boolean} faceTop
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.faceTop = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether the tile's bottom edge is interesting for collisions.
|
|
|
|
* @property {boolean} faceBottom
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.faceBottom = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Tile collision callback.
|
|
|
|
* @property {function} collisionCallback
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.collisionCallback = null;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The context in which the collision callback will be called.
|
|
|
|
* @property {object} collisionCallbackContext
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
this.collisionCallbackContext = this;
|
2017-11-10 02:50:25 +00:00
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The tint to apply to this tile. Note: tint is currently a single color value instead of
|
|
|
|
* the 4 corner tint component on other GameObjects.
|
|
|
|
* @property {number} Tint
|
|
|
|
* @default
|
|
|
|
*/
|
|
|
|
this.tint = 0xffffff;
|
2017-11-16 19:09:07 +00:00
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Check if the given x and y world coordinates are within this Tile. This does not factor in
|
|
|
|
* camera scroll, layer scale or layer position.
|
|
|
|
*
|
|
|
|
* @param {number} x - The x coordinate to test.
|
|
|
|
* @param {number} y - The y coordinate to test.
|
|
|
|
* @return {boolean} True if the coordinates are within this Tile, otherwise false.
|
|
|
|
*/
|
|
|
|
containsPoint: function (x, y)
|
|
|
|
{
|
|
|
|
return !(x < this.worldX || y < this.worldY || x > this.right || y > this.bottom);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copies the tile data & properties from the given tile to this tile. This copies everything
|
|
|
|
* except for position and interesting faces.
|
|
|
|
*
|
|
|
|
* @param {Tile} tile - The tile to copy from.
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
copy: function (tile)
|
|
|
|
{
|
|
|
|
this.index = tile.index;
|
|
|
|
this.alpha = tile.alpha;
|
|
|
|
this.properties = tile.properties;
|
|
|
|
this.visible = tile.visible;
|
|
|
|
this.setFlip(tile.flipX, tile.flipY);
|
|
|
|
this.tint = tile.tint;
|
2017-11-18 21:38:46 +00:00
|
|
|
this.rotation = tile.rotation;
|
|
|
|
this.collideUp = tile.collideUp;
|
|
|
|
this.collideDown = tile.collideDown;
|
|
|
|
this.collideLeft = tile.collideLeft;
|
|
|
|
this.collideRight = tile.collideRight;
|
|
|
|
this.collisionCallback = tile.collisionCallback;
|
|
|
|
this.collisionCallbackContext = tile.collisionCallbackContext;
|
2017-11-16 19:09:07 +00:00
|
|
|
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Clean up memory.
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
destroy: function ()
|
|
|
|
{
|
|
|
|
this.collisionCallback = undefined;
|
|
|
|
this.collisionCallbackContext = undefined;
|
|
|
|
this.properties = undefined;
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Check for intersection with this tile. This does not factor in camera scroll, layer scale or
|
|
|
|
* layer position.
|
|
|
|
*
|
|
|
|
* @param {number} x - The x axis in pixels.
|
|
|
|
* @param {number} y - The y axis in pixels.
|
|
|
|
* @param {number} right - The right point.
|
|
|
|
* @param {number} bottom - The bottom point.
|
|
|
|
* @return {boolean}
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
intersects: function (x, y, right, bottom)
|
|
|
|
{
|
2017-11-29 21:37:23 +00:00
|
|
|
return !(
|
|
|
|
right <= this.worldX || bottom <= this.worldY ||
|
|
|
|
x >= this.right || y >= this.bottom
|
|
|
|
);
|
2017-11-18 21:38:46 +00:00
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Checks if the tile is interesting.
|
|
|
|
*
|
|
|
|
* @param {boolean} collides - If true, will consider the tile interesting if it collides on any
|
|
|
|
* side.
|
|
|
|
* @param {boolean} faces - If true, will consider the tile interesting if it has an interesting
|
|
|
|
* face.
|
|
|
|
* @returns {boolean} True if the Tile is interesting, otherwise false.
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
isInteresting: function (collides, faces)
|
|
|
|
{
|
|
|
|
if (collides && faces) { return (this.canCollide || this.hasInterestingFace); }
|
|
|
|
else if (collides) { return this.collides; }
|
|
|
|
else if (faces) { return this.hasInterestingFace; }
|
|
|
|
return false;
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Reset collision status flags.
|
|
|
|
*
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
resetCollision: function ()
|
|
|
|
{
|
|
|
|
this.collideLeft = false;
|
|
|
|
this.collideRight = false;
|
|
|
|
this.collideUp = false;
|
|
|
|
this.collideDown = false;
|
|
|
|
|
|
|
|
this.faceTop = false;
|
|
|
|
this.faceBottom = false;
|
|
|
|
this.faceLeft = false;
|
|
|
|
this.faceRight = false;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
return this;
|
2017-11-18 21:38:46 +00:00
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Sets the collision flags for each side of this tile and updates the interesting faces list.
|
|
|
|
*
|
|
|
|
* @param {boolean} left - Indicating collide with any object on the left.
|
|
|
|
* @param {boolean} right - Indicating collide with any object on the right.
|
|
|
|
* @param {boolean} up - Indicating collide with any object on the top.
|
|
|
|
* @param {boolean} down - Indicating collide with any object on the bottom.
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
setCollision: function (left, right, up, down)
|
|
|
|
{
|
2017-11-22 01:18:34 +00:00
|
|
|
if (right === undefined) { right = left; }
|
|
|
|
if (up === undefined) { up = left; }
|
|
|
|
if (down === undefined) { down = left; }
|
|
|
|
|
2017-11-18 21:38:46 +00:00
|
|
|
this.collideLeft = left;
|
|
|
|
this.collideRight = right;
|
|
|
|
this.collideUp = up;
|
|
|
|
this.collideDown = down;
|
|
|
|
|
|
|
|
this.faceLeft = left;
|
|
|
|
this.faceRight = right;
|
|
|
|
this.faceTop = up;
|
|
|
|
this.faceBottom = down;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
return this;
|
2017-11-18 21:38:46 +00:00
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Set a callback to be called when this tile is hit by an object. The callback must true for
|
|
|
|
* collision processing to take place.
|
|
|
|
*
|
|
|
|
* @param {function} callback - Callback function.
|
|
|
|
* @param {object} context - Callback will be called within this context.
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
setCollisionCallback: function (callback, context)
|
|
|
|
{
|
2017-11-29 21:37:23 +00:00
|
|
|
if (callback === null)
|
|
|
|
{
|
2017-11-29 14:20:24 +00:00
|
|
|
this.collisionCallback = undefined;
|
|
|
|
this.collisionCallbackContext = undefined;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
this.collisionCallback = callback;
|
|
|
|
this.collisionCallbackContext = context;
|
|
|
|
}
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
return this;
|
2017-11-18 21:38:46 +00:00
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* Sets the size of the tile and updates its worldX and worldY.
|
|
|
|
*
|
|
|
|
* @param {integer} tileWidth - The width of the tile in pixels.
|
|
|
|
* @param {integer} tileHeight - The height of the tile in pixels.
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
2017-11-18 21:40:27 +00:00
|
|
|
setSize: function (tileWidth, tileHeight)
|
|
|
|
{
|
|
|
|
this.worldX = this.x * tileWidth;
|
|
|
|
this.worldY = this.y * tileHeight;
|
|
|
|
this.width = tileWidth;
|
|
|
|
this.height = tileHeight;
|
2017-11-29 21:37:23 +00:00
|
|
|
|
|
|
|
return this;
|
2017-11-18 21:40:27 +00:00
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* True if this tile can collide on any of its faces or has a collision callback set.
|
|
|
|
* @property {boolean} canCollide
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
canCollide: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return (this.collideLeft || this.collideRight || this.collideUp || this.collideDown || this.collisionCallback);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* True if this tile can collide on any of its faces.
|
|
|
|
* @property {boolean} canCollide
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
collides: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return (this.collideLeft || this.collideRight || this.collideUp || this.collideDown);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* True if this tile has any interesting faces.
|
|
|
|
* @property {boolean} canCollide
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-18 21:38:46 +00:00
|
|
|
hasInterestingFace: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return (this.faceTop || this.faceBottom || this.faceLeft || this.faceRight);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The world position of the left side of the tile. This does not factor in camera scroll, layer
|
|
|
|
* scale or layer position.
|
|
|
|
* @property {integer} left
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
left: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return this.worldX;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The world position of the right side of the tile. This does not factor in camera scroll,
|
|
|
|
* layer scale or layer position.
|
|
|
|
* @property {integer} right
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
right: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return this.worldX + this.width;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The world position of the top side of the tile. This does not factor in camera scroll,
|
|
|
|
* layer scale or layer position.
|
|
|
|
* @property {integer} top
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
top: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return this.worldY;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The world position of the bottom side of the tile. This does not factor in camera scroll,
|
|
|
|
* layer scale or layer position.
|
|
|
|
* @property {integer} bottom
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
bottom: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return this.worldY + this.height;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The x world position of the center of the tile. This does not factor in camera scroll, layer
|
|
|
|
* scale or layer position.
|
|
|
|
* @property {integer} centerX
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
centerX: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return this.worldX + this.width / 2;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2017-11-29 21:37:23 +00:00
|
|
|
/**
|
|
|
|
* The y world position of the center of the tile. This does not factor in camera scroll, layer
|
|
|
|
* scale or layer position.
|
|
|
|
* @property {integer} centerY
|
|
|
|
* @readonly
|
|
|
|
*/
|
2017-11-16 19:09:07 +00:00
|
|
|
centerY: {
|
|
|
|
get: function ()
|
|
|
|
{
|
|
|
|
return this.worldY + this.height / 2;
|
|
|
|
}
|
2017-11-14 21:34:33 +00:00
|
|
|
}
|
2017-11-16 19:09:07 +00:00
|
|
|
|
2017-11-10 02:50:25 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
module.exports = Tile;
|