324 lines
9.3 KiB
TypeScript
324 lines
9.3 KiB
TypeScript
import { makeFace } from '../geometry';
|
|
import * as linalg from './linalg';
|
|
import {memoize} from '../memoize';
|
|
|
|
type direction = ('-x' | '+x' | '-y' | '+y' | '-z' | '+z');
|
|
|
|
const BlockType = {
|
|
UNDEFINED: 0,
|
|
AIR: 1,
|
|
DIRT: 2,
|
|
GRASS: 3,
|
|
STONE: 4,
|
|
WATER: 5,
|
|
TREE: 6,
|
|
LEAVES: 7,
|
|
SUN: 8,
|
|
};
|
|
|
|
const CHUNKSIZE = 32;
|
|
|
|
|
|
/** seed: some kind of number uniquely defining the body
|
|
* x, y, z: space coordinates in the body's frame
|
|
*
|
|
* returns: a chunk
|
|
*/
|
|
function makeDirtBlock(seed: number) {
|
|
// XXX: for now, return a premade chunk: a 24x24x24 cube of dirt
|
|
// surrounded by 4 blocks of air all around
|
|
const cs = CHUNKSIZE;
|
|
|
|
if (seed !== 1337) {
|
|
return {};
|
|
}
|
|
// if (Math.abs
|
|
|
|
const blocks = new Array(cs * cs * cs);
|
|
blocks.fill(BlockType.AIR);
|
|
|
|
const dirt = new Array(24).fill(BlockType.DIRT);
|
|
|
|
for (let i = 0; i < 24; i++) {
|
|
for (let j = 4; j < 28; j++) {
|
|
const offset = cs * cs * (i + 4) + cs * j;
|
|
blocks.splice(offset + 4, 24, ...dirt);
|
|
}
|
|
}
|
|
|
|
const half = cs / 2;
|
|
|
|
return {
|
|
position: [-half, -half, -half],
|
|
blocks,
|
|
underground: false,
|
|
seed,
|
|
};
|
|
}
|
|
|
|
function makeSunChunk(seed: number, i: number, j: number, k: number) {
|
|
const cs = CHUNKSIZE;
|
|
const radius = 42;
|
|
if (Math.abs(cs * i) > radius
|
|
|| Math.abs(cs * j) > radius
|
|
|| Math.abs(cs * k) > radius) {
|
|
return undefined;
|
|
}
|
|
const half = cs / 2;
|
|
const blocks = new Array(cs**3);
|
|
blocks.fill(BlockType.SUN);
|
|
|
|
let underground = true;
|
|
|
|
for (let x = 0; x < cs; x++) {
|
|
for (let y = 0; y < cs; y++) {
|
|
for (let z = 0; z < cs; z++) {
|
|
const pos = [
|
|
x + i * cs - half,
|
|
y + j * cs - half,
|
|
z + k * cs - half,
|
|
];
|
|
const idx = (
|
|
z * cs * cs +
|
|
y * cs +
|
|
x
|
|
);
|
|
if (pos[0]**2 + pos[1]**2 + pos[2]**2 > radius**2) {
|
|
blocks[idx] = BlockType.AIR;
|
|
underground = false;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return {
|
|
position: [i * cs - half, j * cs - half, k * cs - half],
|
|
layout: [i, j, k],
|
|
blocks,
|
|
seed,
|
|
underground,
|
|
};
|
|
}
|
|
|
|
function _getChunk(seed: number, chunkX: number, chunkY: number, chunkZ: number) {
|
|
if (seed === 0) {
|
|
return makeSunChunk(seed, chunkX, chunkY, chunkZ);
|
|
}
|
|
if (chunkX === 0 && chunkY === 0 && chunkZ === 0) {
|
|
return makeDirtBlock(seed); // x, y, z unused right now
|
|
}
|
|
return undefined;
|
|
}
|
|
const getChunk = memoize(_getChunk);
|
|
|
|
function faceTexture(type: number, dir: direction) {
|
|
switch (type) {
|
|
case BlockType.GRASS:
|
|
switch (dir) {
|
|
case '+y': return [0, 15];
|
|
case '-y': return [2, 15];
|
|
default: return [1, 15];
|
|
}
|
|
case BlockType.DIRT: return [2, 15];
|
|
case BlockType.STONE: return [3, 15];
|
|
case BlockType.WATER: return [4, 15];
|
|
case BlockType.TREE:
|
|
switch (dir) {
|
|
case '+y':
|
|
case '-y':
|
|
return [5, 15];
|
|
default: return [6, 15];
|
|
}
|
|
case BlockType.LEAVES: return [7, 15];
|
|
case BlockType.SUN: return [0, 4];
|
|
default: return [0, 0];
|
|
}
|
|
}
|
|
|
|
function* makeChunkFaces(chunk, getChunk) {
|
|
const cs = CHUNKSIZE;
|
|
|
|
console.log(`make chunk faces for ${chunk.seed} at ${chunk.position}`);
|
|
|
|
function faceCenter(pos: linalg.Vec3, dir: direction) {
|
|
switch (dir) {
|
|
case '-x': return [pos[0] - 0.5, pos[1], pos[2]];
|
|
case '+x': return [pos[0] + 0.5, pos[1], pos[2]];
|
|
case '-y': return [pos[0], pos[1] - 0.5, pos[2]];
|
|
case '+y': return [pos[0], pos[1] + 0.5, pos[2]];
|
|
case '-z': return [pos[0], pos[1], pos[2] - 0.5];
|
|
case '+z': return [pos[0], pos[1], pos[2] + 0.5];
|
|
}
|
|
}
|
|
|
|
function neighborChunk(dir: direction) {
|
|
const [chunkX, chunkY, chunkZ] = chunk.layout;
|
|
if (chunk.neighbors === undefined) {
|
|
chunk.neighbors = {};
|
|
}
|
|
if (!(dir in chunk.neighbors)) {
|
|
switch (dir) {
|
|
case '-x':
|
|
chunk.neighbors[dir] = getChunk(chunk.seed, chunkX - 1, chunkY, chunkZ);
|
|
break;
|
|
case '+x':
|
|
chunk.neighbors[dir] = getChunk(chunk.seed, chunkX + 1, chunkY, chunkZ);
|
|
break;
|
|
case '-y':
|
|
chunk.neighbors[dir] = getChunk(chunk.seed, chunkX, chunkY - 1, chunkZ);
|
|
break;
|
|
case '+y':
|
|
chunk.neighbors[dir] = getChunk(chunk.seed, chunkX, chunkY + 1, chunkZ);
|
|
break;
|
|
case '-z':
|
|
chunk.neighbors[dir] = getChunk(chunk.seed, chunkX, chunkY, chunkZ - 1);
|
|
break;
|
|
case '+z':
|
|
chunk.neighbors[dir] = getChunk(chunk.seed, chunkX, chunkY, chunkZ + 1);
|
|
break;
|
|
}
|
|
}
|
|
return chunk.neighbors[dir];
|
|
}
|
|
|
|
const neighborIndices = {
|
|
'-x': (x, y, z) => z * cs * cs + y * cs + (cs - 1),
|
|
'+x': (x, y, z) => z * cs * cs + y * cs + 0,
|
|
'-y': (x, y, z) => z * cs * cs + (cs - 1) * cs + x,
|
|
'+y': (x, y, z) => z * cs * cs + 0 * cs + x,
|
|
'-z': (x, y, z) => (cs - 1) * cs * cs + y * cs + x,
|
|
'+z': (x, y, z) => 0 * cs * cs + y * cs + x,
|
|
};
|
|
|
|
function neighborBlock(dir: direction, x, y, z) {
|
|
const neighbor = neighborChunk(dir);
|
|
let block;
|
|
if (neighbor === undefined) {
|
|
block = BlockType.AIR;
|
|
} else {
|
|
block = neighbor.blocks[neighborIndices[dir](x, y, z)];
|
|
}
|
|
return { dir, block };
|
|
}
|
|
|
|
function* neighbors(x, y, z) {
|
|
const idx = (
|
|
z * cs * cs +
|
|
y * cs +
|
|
x
|
|
);
|
|
if (x > 0) {
|
|
yield {
|
|
block: chunk.blocks[idx - 1],
|
|
dir: '-x',
|
|
};
|
|
} else {
|
|
yield neighborBlock('-x', x, y, z);
|
|
}
|
|
if (x < cs - 1) {
|
|
yield {
|
|
block: chunk.blocks[idx + 1],
|
|
dir: '+x',
|
|
};
|
|
} else {
|
|
yield neighborBlock('+x', x, y, z);
|
|
}
|
|
if (y > 0) {
|
|
yield {
|
|
block: chunk.blocks[idx - cs],
|
|
dir: '-y',
|
|
};
|
|
} else {
|
|
yield neighborBlock('-y', x, y, z);
|
|
}
|
|
if (y < cs - 1) {
|
|
yield {
|
|
block: chunk.blocks[idx + cs],
|
|
dir: '+y',
|
|
};
|
|
} else {
|
|
yield neighborBlock('+y', x, y, z);
|
|
}
|
|
if (z > 0) {
|
|
yield {
|
|
block: chunk.blocks[idx - cs * cs],
|
|
dir: '-z',
|
|
};
|
|
} else {
|
|
yield neighborBlock('-z', x, y, z);
|
|
}
|
|
if (z < cs - 1) {
|
|
yield {
|
|
block: chunk.blocks[idx + cs * cs],
|
|
dir: '+z',
|
|
};
|
|
} else {
|
|
yield neighborBlock('+z', x, y, z);
|
|
}
|
|
}
|
|
|
|
for (let x = 0; x < cs; x++) {
|
|
for (let y = 0; y < cs; y++) {
|
|
for (let z = 0; z < cs; z++) {
|
|
const idx = (
|
|
z * cs * cs +
|
|
y * cs +
|
|
x
|
|
);
|
|
const chpos = chunk.position;
|
|
const bkpos = [
|
|
chpos[0] + x,
|
|
chpos[1] + y,
|
|
chpos[2] + z,
|
|
];
|
|
const bt = chunk.blocks[idx];
|
|
if (bt === BlockType.AIR) {
|
|
continue;
|
|
}
|
|
for (const { block, dir } of neighbors(x, y, z)) {
|
|
if (block !== BlockType.AIR) {
|
|
continue;
|
|
}
|
|
yield makeFace(dir, faceTexture(bt, dir), faceCenter(bkpos, dir));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function getBodyChunks(seed: number) {
|
|
const chunks = [];
|
|
const toCheck = [[0, 0, 0]];
|
|
while (toCheck.length > 0) {
|
|
const [chunkX, chunkY, chunkZ] = toCheck.pop();
|
|
const thisChunk = getChunk(seed, chunkX, chunkY, chunkZ);
|
|
if (thisChunk === undefined || chunks.includes(thisChunk)) {
|
|
continue;
|
|
}
|
|
chunks.push(thisChunk);
|
|
toCheck.push([chunkX - 1, chunkY, chunkZ]);
|
|
toCheck.push([chunkX + 1, chunkY, chunkZ]);
|
|
toCheck.push([chunkX, chunkY - 1, chunkZ]);
|
|
toCheck.push([chunkX, chunkY + 1, chunkZ]);
|
|
toCheck.push([chunkX, chunkY, chunkZ - 1]);
|
|
toCheck.push([chunkX, chunkY, chunkZ + 1]);
|
|
}
|
|
return chunks;
|
|
}
|
|
|
|
export function getBodyGeometry(chunks) {
|
|
function lookup(seed: number, chunkX: number, chunkY: number, chunkZ: number) {
|
|
for (const chunk of chunks) {
|
|
const [cx, cy, cz] = chunk.layout;
|
|
if (chunkX == cx && chunkY == cy && chunkZ == cz) {
|
|
return chunk;
|
|
}
|
|
}
|
|
}
|
|
|
|
const faces = chunks
|
|
.filter(chunk => !chunk.underground)
|
|
.map(chunk => [...makeChunkFaces(chunk, memoize(lookup))]);
|
|
|
|
return faces.reduce((a, b) => a.concat(b));
|
|
} |