-
Notifications
You must be signed in to change notification settings - Fork 16
/
Copy pathforest.js
123 lines (105 loc) · 4.15 KB
/
forest.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
const OPEN_GROUND = '.';
const TREES = '|';
const LUMBERYARD = '#';
class Forest {
constructor(raw_forest) {
this.grid = raw_forest.split('\n').map(line => line.split(''));
this.resources = this.calculateTotalResources();
this.time = 0;
}
getAdjacent(x, y) {
let adjacents = [];
if (this.grid[y - 1]) {
if (this.grid[y - 1][x]) adjacents.push(this.grid[y - 1][x]);
if (this.grid[y - 1][x + 1]) adjacents.push(this.grid[y - 1][x + 1]);
if (this.grid[y - 1][x - 1]) adjacents.push(this.grid[y - 1][x - 1]);
}
if (this.grid[y + 1]) {
if (this.grid[y + 1][x]) adjacents.push(this.grid[y + 1][x]);
if (this.grid[y + 1][x + 1]) adjacents.push(this.grid[y + 1][x + 1]);
if (this.grid[y + 1][x - 1]) adjacents.push(this.grid[y + 1][x - 1]);
}
if (this.grid[y]) {
if (this.grid[y][x + 1]) adjacents.push(this.grid[y][x + 1]);
if (this.grid[y][x - 1]) adjacents.push(this.grid[y][x - 1]);
}
return adjacents;
}
tick() {
let new_grid = JSON.parse(JSON.stringify(this.grid));
// Keep running tally of new counts
let new_grid_resources = {
[TREES]: 0,
[LUMBERYARD]: 0,
[OPEN_GROUND]: 0,
};
for (let y = 0; y < this.grid.length; y++) {
for (let x = 0; x < this.grid.length; x++) {
let cell = this.grid[y][x];
let adjacents = this.getAdjacent(x, y);
if (cell === OPEN_GROUND) {
// An open acre will become filled with trees if
// three or more adjacent acres contained trees. Otherwise, nothing happens.
if (adjacents.filter(c => c === TREES).length >= 3) {
new_grid[y][x] = TREES;
new_grid_resources[TREES]++;
} else {
new_grid_resources[OPEN_GROUND]++;
}
} else if (cell === TREES) {
// An acre filled with trees will become a lumberyard if
// three or more adjacent acres were lumberyards. Otherwise, nothing happens.
if (adjacents.filter(c => c === LUMBERYARD).length >= 3) {
new_grid[y][x] = LUMBERYARD;
new_grid_resources[LUMBERYARD]++;
} else {
new_grid_resources[TREES]++;
}
} else {
// An acre containing a lumberyard will remain a lumberyard if it was adjacent
// to at least one other lumberyard and at least one acre containing trees.
// Otherwise, it becomes open.
if (!(adjacents.includes(LUMBERYARD) && adjacents.includes(TREES))) {
new_grid[y][x] = OPEN_GROUND;
new_grid_resources[OPEN_GROUND]++;
} else {
new_grid_resources[LUMBERYARD]++;
}
}
}
}
this.time++;
this.grid = new_grid;
this.resources = new_grid_resources;
}
calculateTotalResources() {
let num_trees = 0;
let num_lumberyards = 0;
let num_open = 0;
for (let y = 0; y < this.grid.length; y++) {
for (let x = 0; x < this.grid.length; x++) {
let cell = this.grid[y][x];
switch (cell) {
case TREES:
num_trees++;
break;
case LUMBERYARD:
num_lumberyards++;
break;
case OPEN_GROUND:
num_open++;
break;
}
}
}
return {
[TREES]: num_trees,
[LUMBERYARD]: num_lumberyards,
[OPEN_GROUND]: num_open,
};
}
getTotalResources() {
return this.resources[TREES] * this.resources[LUMBERYARD];
}
}
module.exports = Forest;