Skip to content

Commit 8f96497

Browse files
committed
Adds basic TileMap.transform(fn)
`TileMap.getTileAt()` now returns `Tile` object instead of number Moves `TileMap.query` to separate file
1 parent ddd627f commit 8f96497

File tree

6 files changed

+155
-52
lines changed

6 files changed

+155
-52
lines changed

examples/transform.js

+16
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
import labyrinthos from '../lib/labyrinthos.js';
2+
3+
let map = new labyrinthos.TileMap({
4+
width: 10,
5+
height: 10
6+
});
7+
8+
map.fill(1); // fill entire map with 1
9+
10+
// Example transformation: Increment each tile's value by 2
11+
map.transform(function(tile){
12+
tile.value += 2;
13+
return tile;
14+
});
15+
16+
console.log(map.data);

lib/TileMap.js

+9-52
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,10 @@ import generateTiledJSON from './util/generateTiledJSON.js'; // Exports to Tiled
22
import Mersenne from './util/mersenne.js'; // Randomness
33
import noise from './util/noise.js'; // Perlin Noise
44

5+
import getTileAt from './TileMap/getTileAt.js';
6+
import transform from './TileMap/transform.js';
7+
import query from './TileMap/query.js';
8+
59
export default class TileMap {
610
constructor({ x = 0, y = 0, width = 16, height = 16, depth = 1, tileWidth = 16, tileHeight = 16, is3D = false } = {}) {
711
this.x = x;
@@ -16,7 +20,12 @@ export default class TileMap {
1620
this.Noise = new noise();
1721
this.noise = this.Noise.noise;
1822
this.seedNoise = this.Noise.noiseSeed;
23+
this.transform = transform.bind(this);
24+
this.query = query.bind(this);
25+
this.getTileAt = getTileAt.bind(this);
1926
this.data = this.initializeDataArray();
27+
this.tileSet = [];
28+
2029
// ASCII representations for tiles 0-10
2130
// TODO: Is there a better default set of ASCII characters we can use?
2231
this.defaultRogueLike = ['-', '#', '+', '0', '<', '>', '$', '⌂', '@', '&', '?'];
@@ -155,14 +164,6 @@ export default class TileMap {
155164
}
156165
}
157166

158-
getTileAt(x, y, z) {
159-
if (this.is3D) {
160-
return this.data[z][y * this.width + x];
161-
} else {
162-
return this.data[y * this.width + x];
163-
}
164-
}
165-
166167
toJSON() {
167168
return JSON.stringify({
168169
width: this.width,
@@ -177,50 +178,6 @@ export default class TileMap {
177178
return generateTiledJSON(this);
178179
}
179180

180-
query({ x, y, width, height, z, tileName } = {}) {
181-
let results = [];
182-
183-
if (x !== undefined && y !== undefined && width !== undefined && height !== undefined) {
184-
for (let offsetY = 0; offsetY < height; offsetY++) {
185-
for (let offsetX = 0; offsetX < width; offsetX++) {
186-
const queryX = x + offsetX;
187-
const queryY = y + offsetY;
188-
189-
if (queryX >= this.width || queryY >= this.height) {
190-
results.push(undefined); // Add undefined for out-of-bounds indices
191-
} else {
192-
const index = queryY * this.width + queryX; // Calculate the correct index in the 1D array
193-
if (this.is3D) {
194-
if (z !== undefined && this.data[z] && this.data[z][index] !== undefined) {
195-
results.push(this.data[z][index]);
196-
} else {
197-
results.push(undefined);
198-
}
199-
} else {
200-
if (this.data[index] !== undefined) {
201-
results.push(this.data[index]);
202-
} else {
203-
results.push(undefined);
204-
}
205-
}
206-
}
207-
}
208-
}
209-
}
210-
211-
// create a new TileMap instance from the results
212-
let subsection = new TileMap({
213-
width,
214-
height,
215-
is3D: this.is3D
216-
});
217-
218-
subsection.data = results;
219-
220-
return subsection;
221-
};
222-
223-
224181
// query3D is WIP - not fully implemented yet, see: ./test/tilemap-query-test.js
225182
query3D({ x, y, z, width, height, depth, tileName } = {}) {
226183
let results = [];

lib/TileMap/getTileAt.js

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
export default function getTileAt(x, y, z) {
2+
let tileId; // This will hold the actual value from the data array
3+
4+
// Retrieve the tile value based on whether the map is 3D or 2D
5+
if (this.is3D) {
6+
if (z !== undefined && this.data[z] !== undefined) {
7+
tileId = this.data[z][y * this.width + x];
8+
} else {
9+
return undefined; // Return undefined if the z layer does not exist
10+
}
11+
} else {
12+
tileId = this.data[y * this.width + x];
13+
}
14+
15+
// Construct the tile object with the id, value, and coordinates
16+
let tileObject = {
17+
// id: tileId, // Tile ID from the tileSet array
18+
id: tileId, // Actual tile ID from the data array
19+
value: tileId, // Actual value from the data array
20+
x: x,
21+
y: y,
22+
...(this.is3D ? { z: z } : {}), // Include z coordinate if the map is 3D
23+
};
24+
25+
// If the tileSet is defined and contains an entry for this tile ID, merge its properties into the tile object
26+
if (Array.isArray(this.tileSet) && this.tileSet[tileId]) {
27+
tileObject = { ...tileObject, ...this.tileSet[tileId] };
28+
}
29+
return tileObject;
30+
}

lib/TileMap/query.js

+44
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,44 @@
1+
import TileMap from '../TileMap.js'; // self reference required for new TileMap instance
2+
3+
export default function query({ x, y, width, height, z, tile } = {}) {
4+
let results = [];
5+
6+
if (x !== undefined && y !== undefined && width !== undefined && height !== undefined) {
7+
for (let offsetY = 0; offsetY < height; offsetY++) {
8+
for (let offsetX = 0; offsetX < width; offsetX++) {
9+
const queryX = x + offsetX;
10+
const queryY = y + offsetY;
11+
12+
if (queryX >= this.width || queryY >= this.height) {
13+
results.push(undefined); // Add undefined for out-of-bounds indices
14+
} else {
15+
const index = queryY * this.width + queryX; // Calculate the correct index in the 1D array
16+
if (this.is3D) {
17+
if (z !== undefined && this.data[z] && this.data[z][index] !== undefined) {
18+
results.push(this.data[z][index]);
19+
} else {
20+
results.push(undefined);
21+
}
22+
} else {
23+
if (this.data[index] !== undefined) {
24+
results.push(this.data[index]);
25+
} else {
26+
results.push(undefined);
27+
}
28+
}
29+
}
30+
}
31+
}
32+
}
33+
34+
// create a new TileMap instance from the results
35+
let subsection = new TileMap({
36+
width,
37+
height,
38+
is3D: this.is3D
39+
});
40+
41+
subsection.data = results;
42+
43+
return subsection;
44+
};

lib/TileMap/transform.js

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
export default function transform(transformFunction) {
2+
// Check if the map is 3D
3+
if (this.depth > 1) {
4+
for (let z = 0; z < this.depth; z++) {
5+
for (let y = 0; y < this.height; y++) {
6+
for (let x = 0; x < this.width; x++) {
7+
// Use getTileAt to get the tile object
8+
const tileObject = this.getTileAt(x, y, z);
9+
// Apply the transformation function to the tile object
10+
const transformedTile = transformFunction(tileObject, x, y, z);
11+
// Assuming the transformation function returns a tile object with an updated 'value'
12+
// Update the data array with the new value
13+
this.data[z][y * this.width + x] = transformedTile;
14+
}
15+
}
16+
}
17+
} else {
18+
// Handle 2D map transformation
19+
for (let y = 0; y < this.height; y++) {
20+
for (let x = 0; x < this.width; x++) {
21+
// Use getTileAt to get the tile object for 2D maps (z is undefined)
22+
const tileObject = this.getTileAt(x, y);
23+
// Apply the transformation function to the tile object
24+
const transformedTile = transformFunction(tileObject, x, y);
25+
// Update the data array with the new value
26+
this.data[y * this.width + x] = transformedTile.id;
27+
}
28+
}
29+
}
30+
}

test/tilemap-transform-test.js

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import tap from 'tape';
2+
import TileMap from '../lib/TileMap.js';
3+
4+
tap.test('applies a transformation function to all tiles', (t) => {
5+
const map = new TileMap({ width: 2, height: 2, tileSet: [{}, { value: 'tile1' }, { value: 'tile2' }] });
6+
map.fill(1); // Assuming '1' corresponds to a tile in the tileSet
7+
8+
// Transformation function that increments the id of the tile object
9+
const increment = tile => ({ ...tile, id: tile.id + 1 });
10+
11+
map.transform(increment);
12+
13+
let passed = true;
14+
for (let y = 0; y < map.height; y++) {
15+
for (let x = 0; x < map.width; x++) {
16+
const tile = map.getTileAt(x, y);
17+
// Check if the transformation has been applied correctly
18+
if (typeof tile !== 'object' || tile.id !== 2) {
19+
passed = false;
20+
}
21+
}
22+
}
23+
24+
t.ok(passed, 'All tiles should have their id incremented by 1');
25+
t.end();
26+
});

0 commit comments

Comments
 (0)