-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathlight_worker.js
94 lines (82 loc) · 3.21 KB
/
light_worker.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
// vim: sw=2 ts=2 expandtab smartindent ft=javascript
const MAP_SIZE = 16;
const MAX_HEIGHT = 64;
const MAX_LIGHT = 8;
const state = { chunks: {} };
const modulo = (n, d) => ((n % d) + d) % d;
const clamp = (n, min, max) => Math.min(max, Math.max(min, n));
const map_index = (x, y, z) => modulo(x, MAP_SIZE) *MAP_SIZE*MAX_HEIGHT +
clamp(y, 0, MAX_HEIGHT)*MAP_SIZE +
modulo(z, MAP_SIZE);
const map_chunk = (x, y, z) => {
const c_x = Math.floor(x / MAP_SIZE);
const c_z = Math.floor(z / MAP_SIZE);
const chunk_key = c_x + ',' + c_z;
return state.chunks[chunk_key];
}
function *map_chunks_near(pos, range) {
const c_x = Math.floor(pos[0] / MAP_SIZE);
const c_z = Math.floor(pos[2] / MAP_SIZE);
for (let o_x = -range; o_x <= range; o_x++)
for (let o_z = -range; o_z <= range; o_z++)
yield (c_x+o_x) + ',' + (c_z+o_z);
}
const map_light_src = (x, y, z ) => map_chunk(x, y, z).light_src[map_index(x, y, z)];
const map_light = (x, y, z ) => map_chunk(x, y, z).light [map_index(x, y, z)];
const map_light_set = (x, y, z, v) => map_chunk(x, y, z).light [map_index(x, y, z)] = v;
function chunk_light_from_src(chunk, chunk_out) {
for (let __x = 0; __x < MAP_SIZE; __x++)
for (let __y = 0; __y < MAX_HEIGHT; __y++)
for (let __z = 0; __z < MAP_SIZE; __z++) {
const t_x = chunk.x + __x;
const t_y = __y;
const t_z = chunk.z + __z;
let light = Math.max(0, map_light(t_x, t_y, t_z)-1);
if (map_light_src(t_x, t_y, t_z))
light = MAX_LIGHT;
else {
for (const [x, y, z] of [[1, 0, 0], [-1, 0, 0], [0, 0, 1], [0, 0, -1], [0, 1, 0], [0, -1, 0]]) {
if (map_chunk(t_x+x, t_y+y, t_z+z) == undefined) continue;
let nbr_light = map_light(t_x+x, t_y+y, t_z+z);
light = Math.max(light, nbr_light-1);
}
}
chunk_out[map_index(__x, __y, __z)] = light;
}
}
let n_computes = 0;
const chunk_out = new Uint8Array(MAP_SIZE*MAP_SIZE*MAX_HEIGHT);
let invocation_i = 0;
onmessage = async ({ data }) => {
if (data.chunk) {
state.chunks[data.chunk.key] = {
x: data.chunk.x,
z: data.chunk.z,
light_src: data.chunk.light_src,
light: new Uint8Array(MAP_SIZE*MAP_SIZE*MAX_HEIGHT)
};
}
if (data.compute) {
const my_invocation = ++invocation_i;
for (let range = 0; range <= 3; range++) {
for (const chunk_key of map_chunks_near(data.around, range)) {
const chunk = state.chunks[chunk_key];
if (chunk == undefined) return;
chunk.light.fill(0);
}
for (let i = 0; i < MAX_LIGHT; i++) {
for (const chunk_key of map_chunks_near(data.around, range)) {
const chunk = state.chunks[chunk_key];
chunk_out.fill(0);
chunk_light_from_src(chunk, chunk_out);
chunk.light.set(chunk_out);
}
/* yield to event loop so fresh data = early quit */
await new Promise(res => setTimeout(res));
if (invocation_i != my_invocation) return;
}
for (const chunk_key of map_chunks_near(data.around, range))
postMessage({ chunk_key, light: state.chunks[chunk_key].light });
}
}
}