UNPKG

@jsprismarine/prismarine

Version:

Dedicated Minecraft Bedrock Edition server written in TypeScript

101 lines (98 loc) 14.3 kB
import { BlockMappings } from '../../block/BlockMappings.es.js'; class BlockStorage { blocks; palette; constructor({ blocks, palette }) { this.palette = palette ?? [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 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; } } export { BlockStorage as default }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"BlockStorage.es.js","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":[],"mappings":"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"}