phaser/src/tilemap/TilemapParser.js

379 lines
11 KiB
JavaScript
Raw Normal View History

/**
* @author Richard Davey <rich@photonstorm.com>
* @copyright 2013 Photon Storm Ltd.
* @license {@link https://github.com/photonstorm/phaser/blob/master/license.txt|MIT License}
*/
/**
* Phaser.TilemapParser parses data objects from Phaser.Loader that need more preparation before they can be inserted into a Tilemap.
*
* @class Phaser.TilemapParser
*/
Phaser.TilemapParser = {
/**
* Creates a Tileset object.
* @method Phaser.TilemapParser.tileset
* @param {Phaser.Game} game - Game reference to the currently running game.
* @param {string} key - The Cache key of this tileset.
* @param {number} tileWidth - Width of each single tile in pixels.
* @param {number} tileHeight - Height of each single tile in pixels.
* @param {number} [tileMargin=0] - If the tiles have been drawn with a margin, specify the amount here.
* @param {number} [tileSpacing=0] - If the tiles have been drawn with spacing between them, specify the amount here.
* @param {number} [rows=-1] - How many tiles are placed horizontally in each row? If -1 it will calculate rows by dividing the image width by tileWidth.
* @param {number} [columns=-1] - How many tiles are placed vertically in each column? If -1 it will calculate columns by dividing the image height by tileHeight.
* @param {number} [total=-1] - The maximum number of tiles to extract from the image. If -1 it will extract `rows * columns` worth. You can also set a value lower than the actual number of tiles.
* @return {Phaser.Tileset} Generated Tileset object.
*/
tileset: function (game, key, tileWidth, tileHeight, tileMargin, tileSpacing, rows, columns, total) {
// How big is our image?
var img = game.cache.getTilesetImage(key);
if (img === null)
{
console.warn("Phaser.TilemapParser.tileSet: Invalid image key given");
return null;
}
var width = img.width;
var height = img.height;
if (rows === -1)
{
rows = Math.round(width / tileWidth);
}
if (columns === -1)
{
columns = Math.round(height / tileHeight);
}
if (total === -1)
{
total = rows * columns;
}
// Zero or smaller than tile sizes?
if (width === 0 || height === 0 || width < tileWidth || height < tileHeight || total === 0)
{
console.warn("Phaser.TilemapParser.tileSet: width/height zero or width/height < given tileWidth/tileHeight");
return null;
}
return new Phaser.Tileset(img, key, tileWidth, tileHeight, tileMargin, tileSpacing, rows, columns, total);
2013-10-11 05:30:28 +00:00
},
2013-10-11 05:30:28 +00:00
/**
* Parse tileset data from the cache and creates a Tileset object.
* @method Phaser.TilemapParser.parse
* @param {Phaser.Game} game - Game reference to the currently running game.
2013-12-18 16:56:14 +00:00
* @param {string} key - The key of the tilemap in the Cache.
* @return {object} The parsed map object.
*/
2013-12-18 16:56:14 +00:00
parse: function (game, key) {
var map = game.cache.getTilemapData(key);
2013-10-11 05:30:28 +00:00
2013-12-18 16:56:14 +00:00
if (map)
{
2013-12-18 16:56:14 +00:00
if (map.format === Phaser.Tilemap.CSV)
{
return this.parseCSV(map.data);
}
else if (map.format === Phaser.Tilemap.TILED_JSON)
{
return this.parseTiledJSON(map.data);
}
}
2013-12-18 16:56:14 +00:00
else
{
2013-12-18 16:56:14 +00:00
return { layers: [], objects: [], images: [], tilesets: [] };
}
2013-10-11 05:30:28 +00:00
},
2013-10-11 05:30:28 +00:00
/**
* Parses a CSV file into valid map data.
* @method Phaser.TilemapParser.parseCSV
* @param {string} data - The CSV file data.
* @return {object} Generated map data.
*/
parseCSV: function (data) {
// Trim any rogue whitespace from the data
data = data.trim();
var output = [];
var rows = data.split("\n");
var height = rows.length;
var width = 0;
for (var i = 0; i < rows.length; i++)
{
output[i] = [];
var column = rows[i].split(",");
2013-10-11 05:30:28 +00:00
for (var c = 0; c < column.length; c++)
{
output[i][c] = parseInt(column[c], 10);
}
2013-10-11 05:30:28 +00:00
if (width === 0)
{
width = column.length;
}
}
// Build collision map
return [{ name: 'csv', width: width, height: height, alpha: 1, visible: true, indexes: [], tileMargin: 0, tileSpacing: 0, data: output }];
},
2013-10-11 05:30:28 +00:00
/**
* Parses a Tiled JSON file into valid map data.
* @method Phaser.TilemapParser.parseJSON
2013-12-18 16:56:14 +00:00
* @param {object} json - The JSON map data.
* @return {object} Generated and parsed map data.
*/
parseTiledJSON: function (json) {
2013-10-11 05:30:28 +00:00
if (json.orientation !== 'orthogonal')
{
console.warn('TilemapParser.parseTiledJSON: Only orthogonal map types are supported in this version of Phaser');
return null;
}
// Map data will consist of: layers, objects, images, tilesets, sizes
2013-12-18 04:40:10 +00:00
var map = {};
map.width = json.width;
map.height = json.height;
map.tileWidth = json.tilewidth;
map.tileHeight = json.tileheight;
map.orientation = json.orientation;
map.version = json.version;
map.properties = json.properties;
map.widthInPixels = map.width * map.tileWidth;
map.heightInPixels = map.height * map.tileHeight;
2013-12-18 04:40:10 +00:00
// Tile Layers
var layers = [];
for (var i = 0; i < json.layers.length; i++)
{
2013-12-18 04:40:10 +00:00
if (json.layers[i].type !== 'tilelayer')
{
continue;
}
var layer = {
name: json.layers[i].name,
x: json.layers[i].x,
y: json.layers[i].y,
width: json.layers[i].width,
height: json.layers[i].height,
widthInPixels: json.layers[i].width * json.tilewidth,
heightInPixels: json.layers[i].height * json.tileheight,
alpha: json.layers[i].opacity,
visible: json.layers[i].visible,
properties: {},
indexes: [],
callbacks: []
};
if (json.layers[i].properties)
{
layer.properties = json.layers[i].properties;
}
var x = 0;
var row = [];
var output = [];
// Loop through the data field in the JSON.
// This is an array containing the tile indexes, one after the other. 0 = no tile, everything else = the tile index (starting at 1)
// If the map contains multiple tilesets then the indexes are relative to that which the set starts from.
// Need to set which tileset in the cache = which tileset in the JSON, if you do this manually it means you can use the same map data but a new tileset.
2013-12-18 04:40:10 +00:00
for (var t = 0, len = json.layers[i].data.length; t < len; t++)
{
// index, x, y, width, height
if (json.layers[i].data[t] > 0)
{
row.push(new Phaser.Tile(layer, json.layers[i].data[t], x, output.length, json.tilewidth, json.tileheight));
}
else
{
row.push(null);
}
x++;
if (x === json.layers[i].width)
{
output.push(row);
x = 0;
row = [];
}
2013-12-18 04:40:10 +00:00
}
2013-12-18 04:40:10 +00:00
layer.data = output;
2013-12-18 04:40:10 +00:00
layers.push(layer);
}
map.layers = layers;
// Images
var images = [];
2013-12-18 04:40:10 +00:00
for (var i = 0; i < json.layers.length; i++)
{
if (json.layers[i].type !== 'imagelayer')
{
continue;
}
2013-12-18 04:40:10 +00:00
var image = {
name: json.layers[i].name,
image: json.layers[i].image,
x: json.layers[i].x,
y: json.layers[i].y,
alpha: json.layers[i].opacity,
visible: json.layers[i].visible,
properties: {}
};
if (json.layers[i].properties)
{
image.properties = json.layers[i].properties;
}
2013-12-18 04:40:10 +00:00
images.push(image);
}
map.images = images;
// Objects
var objects = {};
2013-12-18 04:40:10 +00:00
for (var i = 0; i < json.layers.length; i++)
{
if (json.layers[i].type !== 'objectgroup')
{
continue;
}
objects[json.layers[i].name] = [];
for (var v = 0, len = json.layers[i].objects.length; v < len; v++)
2013-12-18 04:40:10 +00:00
{
// For now we'll just support object tiles
if (json.layers[i].objects[v].gid)
{
2013-12-18 04:40:10 +00:00
var object = {
gid: json.layers[i].objects[v].gid,
name: json.layers[i].objects[v].name,
x: json.layers[i].objects[v].x,
y: json.layers[i].objects[v].y,
visible: json.layers[i].objects[v].visible,
properties: json.layers[i].objects[v].properties
};
objects[json.layers[i].name].push(object);
}
2013-12-18 04:40:10 +00:00
}
2013-12-18 04:40:10 +00:00
}
2013-10-11 05:30:28 +00:00
2013-12-18 04:40:10 +00:00
map.objects = objects;
2013-12-18 04:40:10 +00:00
// Tilesets
var tilesets = [];
for (var i = 0; i < json.tilesets.length; i++)
2013-12-18 16:56:14 +00:00
{
// name, firstgid, width, height, margin, spacing, properties
var set = json.tilesets[i];
var newSet = new Phaser.Tileset(set.name, set.firstgid, set.tilewidth, set.tileheight, set.margin, set.spacing, set.properties);
2013-12-18 16:56:14 +00:00
if (set.tileproperties)
2013-12-18 16:56:14 +00:00
{
newSet.tileProperties = set.tileproperties;
2013-12-18 16:56:14 +00:00
}
newSet.rows = (set.imageheight - set.margin) / (set.tileheight + set.spacing);
newSet.columns = (set.imagewidth - set.margin) / (set.tilewidth + set.spacing);
newSet.total = newSet.rows * newSet.columns;
2013-12-18 16:56:14 +00:00
tilesets.push(newSet);
}
2013-12-18 16:56:14 +00:00
map.tilesets = tilesets;
2013-12-18 16:56:14 +00:00
map.tiles = [];
2013-12-18 16:56:14 +00:00
// Finally lets build our super tileset index
for (var i = 0; i < map.tilesets.length; i++)
2013-12-18 04:40:10 +00:00
{
var set = map.tilesets[i];
var x = set.tileMargin;
var y = set.tileMargin;
2013-12-18 04:40:10 +00:00
var count = 0;
var countX = 0;
var countY = 0;
2013-12-18 04:40:10 +00:00
for (var t = set.firstgid; t < set.firstgid + set.total; t++)
{
// Can add extra properties here as needed
map.tiles[t] = [x, y, i];
2013-12-18 16:56:14 +00:00
x += set.tileWidth + set.tileSpacing;
2013-12-18 16:56:14 +00:00
count++;
2013-10-11 05:30:28 +00:00
if (count === set.total)
{
break;
}
2013-12-18 16:56:14 +00:00
countX++;
2013-12-18 16:56:14 +00:00
if (countX === set.columns)
{
x = set.tileMargin;
y += set.tileHeight + set.tileSpacing;
2013-12-18 16:56:14 +00:00
countX = 0;
countY++;
2013-12-18 16:56:14 +00:00
if (countY === set.rows)
{
break;
}
}
}
2013-12-18 16:56:14 +00:00
}
2013-12-18 04:40:10 +00:00
return map;
}
}