@jsprismarine/prismarine
Version:
Dedicated Minecraft Bedrock Edition server written in TypeScript
105 lines (100 loc) • 14.5 kB
JavaScript
;
Object.defineProperties(exports, { __esModule: { value: true }, [Symbol.toStringTag]: { value: 'Module' } });
const block_BlockMappings = require('../../block/BlockMappings.cjs.cjs');
class BlockStorage {
blocks;
palette;
constructor({ blocks, palette }) {
this.palette = palette ?? [block_BlockMappings.BlockMappings.getRuntimeId("minecraft:air")];
this.blocks = blocks ?? Array.from({ length: 4096 }).fill(0);
}
static getIndex(bx, by, bz) {
bx = bx & 15;
bz = bz & 15;
by = by & 15;
return (bx << 8) + (bz << 4) | by;
}
getBlock(bx, by, bz) {
const paletteIndex = this.blocks[BlockStorage.getIndex(bx, by, bz)];
const runtimeId = this.palette[paletteIndex];
return block_BlockMappings.BlockMappings.getLegacyId(runtimeId);
}
setBlock(bx, by, bz, runtimeId) {
if (!this.palette.includes(runtimeId)) {
this.palette.push(runtimeId);
}
this.blocks[BlockStorage.getIndex(bx, by, bz)] = this.palette.indexOf(runtimeId);
}
networkSerialize(stream) {
let bitsPerBlock = Math.ceil(Math.log2(this.palette.length));
switch (bitsPerBlock) {
case 0:
bitsPerBlock = 1;
break;
case 1:
case 2:
case 3:
case 4:
case 5:
case 6:
break;
case 7:
case 8:
bitsPerBlock = 8;
break;
default:
bitsPerBlock = 16;
break;
}
stream.writeByte(bitsPerBlock << 1 | 1);
const blocksPerWord = Math.floor(32 / bitsPerBlock);
const wordsPerChunk = Math.ceil(4096 / blocksPerWord);
let position = 0;
for (let w = 0; w < wordsPerChunk; w++) {
let word = 0;
for (let block = 0; block < blocksPerWord; block++) {
const state = this.blocks[position++];
word |= state << bitsPerBlock * block;
}
stream.writeIntLE(word);
}
stream.writeVarInt(this.palette.length);
for (const val of this.palette) {
stream.writeVarInt(val);
}
}
static networkDeserialize(stream) {
const bitsPerBlock = stream.readByte() >> 1;
const blocksPerWord = Math.floor(32 / bitsPerBlock);
const wordsPerChunk = Math.ceil(4096 / blocksPerWord);
const words = new Array(wordsPerChunk);
for (let w = 0; w < wordsPerChunk; w++) {
words[w] = stream.readIntLE();
}
const paletteCount = stream.readVarInt();
const palette = new Array(paletteCount);
for (let i = 0; i < paletteCount; i++) {
palette[i] = stream.readVarInt();
}
let positon = 0;
const storage = new BlockStorage({ palette });
for (let w = 0; w < wordsPerChunk; w++) {
const word = words[w];
for (let block = 0; block < blocksPerWord; block++) {
const state = word >> positon % blocksPerWord * bitsPerBlock & (1 << bitsPerBlock) - 1;
const x = positon >> 8 & 15;
const y = positon & 15;
const z = positon >> 4 & 15;
const translated = palette[state];
storage.setBlock(x, y, z, translated);
positon++;
}
}
return storage;
}
isEmpty() {
return this.palette.length === 1;
}
}
exports.default = BlockStorage;
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"BlockStorage.cjs.cjs","sources":["../../../src/world/chunk/BlockStorage.ts"],"sourcesContent":["import type { LegacyId } from '../../block/BlockMappings';\nimport { BlockMappings } from '../../block/BlockMappings';\n\nimport type BinaryStream from '@jsprismarine/jsbinaryutils';\n\ninterface BlockStorageData {\n    blocks?: number[];\n    palette?: number[];\n}\n\nexport default class BlockStorage {\n    private blocks: number[];\n    private palette: number[];\n\n    public constructor({ blocks, palette }: BlockStorageData) {\n        this.palette = palette ?? [BlockMappings.getRuntimeId('minecraft:air')];\n        this.blocks = blocks ?? Array.from<number>({ length: 4096 }).fill(0);\n    }\n\n    private static getIndex(bx: number, by: number, bz: number): number {\n        bx = bx & 0x0f;\n        bz = bz & 0x0f;\n        by = by & 0x0f;\n        return ((bx << 8) + (bz << 4)) | by;\n    }\n\n    public getBlock(bx: number, by: number, bz: number): LegacyId {\n        const paletteIndex = this.blocks[BlockStorage.getIndex(bx, by, bz)]!;\n        const runtimeId = this.palette[paletteIndex]!;\n        return BlockMappings.getLegacyId(runtimeId);\n    }\n\n    public setBlock(bx: number, by: number, bz: number, runtimeId: number): void {\n        if (!this.palette.includes(runtimeId)) {\n            this.palette.push(runtimeId);\n        }\n        this.blocks[BlockStorage.getIndex(bx, by, bz)] = this.palette.indexOf(runtimeId);\n    }\n\n    public networkSerialize(stream: BinaryStream): void {\n        // https://gist.github.com/Tomcc/a96af509e275b1af483b25c543cfbf37\n        let bitsPerBlock = Math.ceil(Math.log2(this.palette.length));\n\n        switch (bitsPerBlock) {\n            case 0:\n                bitsPerBlock = 1;\n                break;\n            case 1:\n            case 2:\n            case 3:\n            case 4:\n            case 5:\n            case 6:\n                break;\n            case 7:\n            case 8:\n                bitsPerBlock = 8;\n                break;\n            default:\n                bitsPerBlock = 16;\n                break;\n        }\n\n        // 7 bit: storage type, 1 bit (shift to end): network format (always 1)\n        stream.writeByte((bitsPerBlock << 1) | 1);\n        const blocksPerWord = Math.floor(32 / bitsPerBlock);\n        const wordsPerChunk = Math.ceil(4096 / blocksPerWord);\n\n        // Encoding example\n        // https://github.com/NiclasOlofsson/MiNET/blob/4acbccb6dedae066547f8486a2ace1c9d6db0084/src/MiNET/MiNET/Worlds/SubChunk.cs#L294\n        let position = 0;\n        for (let w = 0; w < wordsPerChunk; w++) {\n            let word = 0;\n            for (let block = 0; block < blocksPerWord; block++) {\n                const state = this.blocks[position++]!;\n                word |= state << (bitsPerBlock * block);\n            }\n            stream.writeIntLE(word);\n        }\n\n        // Write palette entries as runtime ids\n        stream.writeVarInt(this.palette.length);\n        for (const val of this.palette) {\n            stream.writeVarInt(val);\n        }\n    }\n\n    public static networkDeserialize(stream: BinaryStream): BlockStorage {\n        const bitsPerBlock = stream.readByte() >> 1;\n        const blocksPerWord = Math.floor(32 / bitsPerBlock);\n        const wordsPerChunk = Math.ceil(4096 / blocksPerWord);\n\n        const words: number[] = new Array(wordsPerChunk);\n        for (let w = 0; w < wordsPerChunk; w++) {\n            words[w] = stream.readIntLE();\n        }\n\n        const paletteCount = stream.readVarInt();\n        const palette: number[] = new Array(paletteCount);\n        for (let i = 0; i < paletteCount; i++) {\n            palette[i] = stream.readVarInt();\n        }\n\n        // Encoding example\n        // https://github.com/kennyvv/Alex/blob/dcca0d697bbb25637a8bcfa93830f8a762c463af/src/Alex/Worlds/Multiplayer/Bedrock/ChunkProcessor.cs#L367\n\n        let positon = 0;\n        const storage = new BlockStorage({ palette });\n        for (let w = 0; w < wordsPerChunk; w++) {\n            const word = words[w]!;\n            for (let block = 0; block < blocksPerWord; block++) {\n                const state = (word >> ((positon % blocksPerWord) * bitsPerBlock)) & ((1 << bitsPerBlock) - 1);\n\n                const x = (positon >> 8) & 0xf;\n                const y = positon & 0xf;\n                const z = (positon >> 4) & 0xf;\n\n                const translated = palette[state]!;\n                storage.setBlock(x, y, z, translated);\n                positon++;\n            }\n        }\n        return storage;\n    }\n\n    public isEmpty(): boolean {\n        return this.palette.length === 1;\n    }\n}\n"],"names":["BlockMappings"],"mappings":"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"}