UNPKG

@jsprismarine/prismarine

Version:

Dedicated Minecraft Bedrock Edition server written in TypeScript

136 lines (135 loc) • 16 kB
import { BlockMappings } from "../../block/BlockMappings.es.js"; import SubChunk from "./SubChunk.es.js"; import BinaryStream from "@jsprismarine/jsbinaryutils"; import { Vector3 } from "@jsprismarine/math"; //#region src/world/chunk/Chunk.ts var MAX_SUBCHUNKS = 16; var Chunk = class Chunk { x; z; hasChanged; subChunks = /* @__PURE__ */ new Map(); static EMPTY_SUBCHUNK = new SubChunk(); constructor(chunkX = 0, chunkZ = 0, _subChunks = /* @__PURE__ */ new Map()) { this.x = chunkX; this.z = chunkZ; this.hasChanged = false; } getX() { return this.x; } getZ() { return this.z; } getHasChanged() { return this.hasChanged; } getHeight() { return this.subChunks.size; } /** * Returns the highest empty sub chunk (so we don't send empty sub chunks). * @returns {number} The highest empty sub chunk. */ getTopEmpty() { let topEmpty = MAX_SUBCHUNKS - 1; while (topEmpty >= 0 && !this.subChunks.has(topEmpty) || this.subChunks.has(topEmpty) && this.subChunks.get(topEmpty).isEmpty()) topEmpty--; return ++topEmpty; } /** * Returns the Chunk slice at the given layer. * @param {number} y - The layer to get. */ getSubChunk(y) { if (y < 0 || y > MAX_SUBCHUNKS) throw new Error(`Invalid subchunk height: ${y}`); return this.subChunks.get(y) ?? null; } getOrCreateSubChunk(y) { const subChunk = new SubChunk(); this.subChunks.set(y, subChunk); return subChunk; } getSubChunks() { return this.subChunks; } getHighestBlockAt(_x, _z) { return 100; } /** * Returns block legacy id (DATA) in the corresponding sub chunk. * Use world to get the actual block instance (this is to keep code clean). * @param {Vector3 | number} x - block x. * @param {number} [y=0] - block y. * @param {number} [z=0] - block z. * @param {number} [layer=0] - block storage layer (0 for blocks, 1 for liquids). */ getBlock(x, y = 0, z = 0, layer = 0) { if (x instanceof Vector3) return this.getBlock(x.getX(), x.getY(), x.getZ(), layer); const subChunk = this.getSubChunk(Math.floor(y / 16)); if (!subChunk) return BlockMappings.getLegacyId(BlockMappings.getRuntimeId("minecraft:air")); return subChunk.getBlock(x, y & 15, z, layer); } /** * Sets a block into the chunk by its runtime Id. * @param {number} x - block x * @param {number} y - block y * @param {number} z - block z * @param {Block} block - block to set * @param {number} [layer=0] - block storage layer (0 for blocks, 1 for liquids) */ setBlock(x, y, z, block, layer = 0) { let subChunk = this.getSubChunk(Math.floor(y / 16)); if (!subChunk) subChunk = this.getOrCreateSubChunk(y >> 4); subChunk.setBlock(x, y & 15, z, BlockMappings.getRuntimeId(block.getName()), layer); this.hasChanged = true; } /** * Helper method used to hash into a single 64 bits integer * both Chunk X and Z coordinates. * @param {number} chunkX - Target Chunk X coordinate. * @param {number} chunkZ - Target Chunk Z coordinate. * @returns {bigint} A 64 bit intger containing a hash of X and Z. */ static packXZ(chunkX, chunkZ) { return (BigInt(chunkX) & 4294967295n) << 32n | BigInt(chunkZ) & 4294967295n; } /** * Helper method used to decode a 64 bit hash containing * both Chunk X and Z coordinates. * @param {bigint} packed - Target Chunk coordinate hash. * @returns {number[]} An array containing decoded Chunk X and Z coordinates. */ static unpackXZ(packed) { return [Number(BigInt.asIntN(32, packed >> 32n)), Number(BigInt.asIntN(32, packed & 4294967295n))]; } networkSerialize() { const stream = new BinaryStream(); for (let y = 0; y < 4; ++y) { stream.writeByte(8); stream.writeByte(0); } for (let y = 0; y < this.getTopEmpty(); ++y) (this.subChunks.get(y) ?? Chunk.EMPTY_SUBCHUNK).networkSerialize(stream); for (let i = 0; i < 24; i++) { stream.writeByte(0); stream.writeUnsignedVarInt(2); } stream.writeByte(0); return stream.getBuffer(); } /** * Deserialize network stream into chunk * useful for client applications and/or our Filesystem impl * @param {BinaryStream} stream - the network stream * @param {number} [x] - the chunk x coordinate * @param {number} [z] - the chunk z coordinate */ static networkDeserialize(stream, x, z) { stream.read(8); const subChunks = /* @__PURE__ */ new Map(); for (let i = 0; i < MAX_SUBCHUNKS; i++) subChunks.set(i, SubChunk.networkDeserialize(stream)); return new Chunk(x, z, subChunks); } }; //#endregion export { Chunk as default }; //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiQ2h1bmsuZXMuanMiLCJuYW1lcyI6W10sInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL3dvcmxkL2NodW5rL0NodW5rLnRzIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB0eXBlIHsgTGVnYWN5SWQgfSBmcm9tICcuLi8uLi9ibG9jay9CbG9ja01hcHBpbmdzJztcbmltcG9ydCB7IEJsb2NrTWFwcGluZ3MgfSBmcm9tICcuLi8uLi9ibG9jay9CbG9ja01hcHBpbmdzJztcblxuaW1wb3J0IEJpbmFyeVN0cmVhbSBmcm9tICdAanNwcmlzbWFyaW5lL2pzYmluYXJ5dXRpbHMnO1xuaW1wb3J0IHsgVmVjdG9yMyB9IGZyb20gJ0Bqc3ByaXNtYXJpbmUvbWF0aCc7XG5pbXBvcnQgdHlwZSB7IEJsb2NrIH0gZnJvbSAnLi4vLi4vYmxvY2svQmxvY2snO1xuaW1wb3J0IFN1YkNodW5rIGZyb20gJy4vU3ViQ2h1bmsnO1xuXG5jb25zdCBNQVhfU1VCQ0hVTktTID0gMTY7XG5cbmV4cG9ydCBkZWZhdWx0IGNsYXNzIENodW5rIHtcbiAgICBwcml2YXRlIHg6IG51bWJlcjtcbiAgICBwcml2YXRlIHo6IG51bWJlcjtcbiAgICBwcml2YXRlIGhhc0NoYW5nZWQ6IGJvb2xlYW47XG5cbiAgICBwcml2YXRlIHN1YkNodW5rczogTWFwPG51bWJlciwgU3ViQ2h1bms+ID0gbmV3IE1hcCgpO1xuICAgIHByaXZhdGUgc3RhdGljIHJlYWRvbmx5IEVNUFRZX1NVQkNIVU5LID0gbmV3IFN1YkNodW5rKCk7XG5cbiAgICBwdWJsaWMgY29uc3RydWN0b3IoY2h1bmtYID0gMCwgY2h1bmtaID0gMCwgX3N1YkNodW5rczogTWFwPG51bWJlciwgU3ViQ2h1bms+ID0gbmV3IE1hcCgpKSB7XG4gICAgICAgIHRoaXMueCA9IGNodW5rWDtcbiAgICAgICAgdGhpcy56ID0gY2h1bmtaO1xuICAgICAgICB0aGlzLmhhc0NoYW5nZWQgPSBmYWxzZTtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0WCgpOiBudW1iZXIge1xuICAgICAgICByZXR1cm4gdGhpcy54O1xuICAgIH1cblxuICAgIHB1YmxpYyBnZXRaKCk6IG51bWJlciB7XG4gICAgICAgIHJldHVybiB0aGlzLno7XG4gICAgfVxuXG4gICAgcHVibGljIGdldEhhc0NoYW5nZWQoKTogYm9vbGVhbiB7XG4gICAgICAgIHJldHVybiB0aGlzLmhhc0NoYW5nZWQ7XG4gICAgfVxuXG4gICAgcHVibGljIGdldEhlaWdodCgpOiBudW1iZXIge1xuICAgICAgICByZXR1cm4gdGhpcy5zdWJDaHVua3Muc2l6ZTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBSZXR1cm5zIHRoZSBoaWdoZXN0IGVtcHR5IHN1YiBjaHVuayAoc28gd2UgZG9uJ3Qgc2VuZCBlbXB0eSBzdWIgY2h1bmtzKS5cbiAgICAgKiBAcmV0dXJucyB7bnVtYmVyfSBUaGUgaGlnaGVzdCBlbXB0eSBzdWIgY2h1bmsuXG4gICAgICovXG4gICAgcHVibGljIGdldFRvcEVtcHR5KCk6IG51bWJlciB7XG4gICAgICAgIGxldCB0b3BFbXB0eSA9IE1BWF9TVUJDSFVOS1MgLSAxO1xuICAgICAgICB3aGlsZSAoXG4gICAgICAgICAgICAodG9wRW1wdHkgPj0gMCAmJiAhdGhpcy5zdWJDaHVua3MuaGFzKHRvcEVtcHR5KSkgfHxcbiAgICAgICAgICAgICh0aGlzLnN1YkNodW5rcy5oYXModG9wRW1wdHkpICYmIHRoaXMuc3ViQ2h1bmtzLmdldCh0b3BFbXB0eSkhLmlzRW1wdHkoKSlcbiAgICAgICAgKSB7XG4gICAgICAgICAgICB0b3BFbXB0eS0tO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiArK3RvcEVtcHR5O1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIFJldHVybnMgdGhlIENodW5rIHNsaWNlIGF0IHRoZSBnaXZlbiBsYXllci5cbiAgICAgKiBAcGFyYW0ge251bWJlcn0geSAtIFRoZSBsYXllciB0byBnZXQuXG4gICAgICovXG4gICAgcHVibGljIGdldFN1YkNodW5rKHk6IG51bWJlcik6IFN1YkNodW5rIHwgbnVsbCB7XG4gICAgICAgIGlmICh5IDwgMCB8fCB5ID4gTUFYX1NVQkNIVU5LUykge1xuICAgICAgICAgICAgdGhyb3cgbmV3IEVycm9yKGBJbnZhbGlkIHN1YmNodW5rIGhlaWdodDogJHt5fWApO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiB0aGlzLnN1YkNodW5rcy5nZXQoeSkgPz8gbnVsbDtcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0T3JDcmVhdGVTdWJDaHVuayh5OiBudW1iZXIpOiBTdWJDaHVuayB7XG4gICAgICAgIGNvbnN0IHN1YkNodW5rID0gbmV3IFN1YkNodW5rKCk7XG4gICAgICAgIHRoaXMuc3ViQ2h1bmtzLnNldCh5LCBzdWJDaHVuayk7XG4gICAgICAgIHJldHVybiBzdWJDaHVuaztcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0U3ViQ2h1bmtzKCk6IE1hcDxudW1iZXIsIFN1YkNodW5rPiB7XG4gICAgICAgIHJldHVybiB0aGlzLnN1YkNodW5rcztcbiAgICB9XG5cbiAgICBwdWJsaWMgZ2V0SGlnaGVzdEJsb2NrQXQoX3g6IG51bWJlciwgX3o6IG51bWJlcik6IG51bWJlciB7XG4gICAgICAgIHJldHVybiAxMDA7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogUmV0dXJucyBibG9jayBsZWdhY3kgaWQgKERBVEEpIGluIHRoZSBjb3JyZXNwb25kaW5nIHN1YiBjaHVuay5cbiAgICAgKiBVc2Ugd29ybGQgdG8gZ2V0IHRoZSBhY3R1YWwgYmxvY2sgaW5zdGFuY2UgKHRoaXMgaXMgdG8ga2VlcCBjb2RlIGNsZWFuKS5cbiAgICAgKiBAcGFyYW0ge1ZlY3RvcjMgfCBudW1iZXJ9IHggLSBibG9jayB4LlxuICAgICAqIEBwYXJhbSB7bnVtYmVyfSBbeT0wXSAtIGJsb2NrIHkuXG4gICAgICogQHBhcmFtIHtudW1iZXJ9IFt6PTBdIC0gYmxvY2sgei5cbiAgICAgKiBAcGFyYW0ge251bWJlcn0gW2xheWVyPTBdIC0gYmxvY2sgc3RvcmFnZSBsYXllciAoMCBmb3IgYmxvY2tzLCAxIGZvciBsaXF1aWRzKS5cbiAgICAgKi9cbiAgICBwdWJsaWMgZ2V0QmxvY2soeDogVmVjdG9yMyB8IG51bWJlciwgeTogbnVtYmVyID0gMCwgejogbnVtYmVyID0gMCwgbGF5ZXIgPSAwKTogTGVnYWN5SWQge1xuICAgICAgICBpZiAoeCBpbnN0YW5jZW9mIFZlY3RvcjMpIHtcbiAgICAgICAgICAgIHJldHVybiB0aGlzLmdldEJsb2NrKHguZ2V0WCgpLCB4LmdldFkoKSwgeC5nZXRaKCksIGxheWVyKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNvbnN0IHN1YkNodW5rID0gdGhpcy5nZXRTdWJDaHVuayhNYXRoLmZsb29yKHkgLyAxNikpO1xuICAgICAgICBpZiAoIXN1YkNodW5rKSByZXR1cm4gQmxvY2tNYXBwaW5ncy5nZXRMZWdhY3lJZChCbG9ja01hcHBpbmdzLmdldFJ1bnRpbWVJZCgnbWluZWNyYWZ0OmFpcicpKTtcbiAgICAgICAgcmV0dXJuIHN1YkNodW5rLmdldEJsb2NrKHgsIHkgJiAweGYsIHosIGxheWVyKTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBTZXRzIGEgYmxvY2sgaW50byB0aGUgY2h1bmsgYnkgaXRzIHJ1bnRpbWUgSWQuXG4gICAgICogQHBhcmFtIHtudW1iZXJ9IHggLSBibG9jayB4XG4gICAgICogQHBhcmFtIHtudW1iZXJ9IHkgLSBibG9jayB5XG4gICAgICogQHBhcmFtIHtudW1iZXJ9IHogLSBibG9jayB6XG4gICAgICogQHBhcmFtIHtCbG9ja30gYmxvY2sgLSBibG9jayB0byBzZXRcbiAgICAgKiBAcGFyYW0ge251bWJlcn0gW2xheWVyPTBdIC0gYmxvY2sgc3RvcmFnZSBsYXllciAoMCBmb3IgYmxvY2tzLCAxIGZvciBsaXF1aWRzKVxuICAgICAqL1xuICAgIHB1YmxpYyBzZXRCbG9jayh4OiBudW1iZXIsIHk6IG51bWJlciwgejogbnVtYmVyLCBibG9jazogQmxvY2ssIGxheWVyID0gMCk6IHZvaWQge1xuICAgICAgICBsZXQgc3ViQ2h1bmsgPSB0aGlzLmdldFN1YkNodW5rKE1hdGguZmxvb3IoeSAvIDE2KSk7XG4gICAgICAgIGlmICghc3ViQ2h1bmspIHN1YkNodW5rID0gdGhpcy5nZXRPckNyZWF0ZVN1YkNodW5rKHkgPj4gNCk7XG4gICAgICAgIHN1YkNodW5rLnNldEJsb2NrKHgsIHkgJiAweGYsIHosIEJsb2NrTWFwcGluZ3MuZ2V0UnVudGltZUlkKGJsb2NrLmdldE5hbWUoKSksIGxheWVyKTtcblxuICAgICAgICB0aGlzLmhhc0NoYW5nZWQgPSB0cnVlO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIEhlbHBlciBtZXRob2QgdXNlZCB0byBoYXNoIGludG8gYSBzaW5nbGUgNjQgYml0cyBpbnRlZ2VyXG4gICAgICogYm90aCBDaHVuayBYIGFuZCBaIGNvb3JkaW5hdGVzLlxuICAgICAqIEBwYXJhbSB7bnVtYmVyfSBjaHVua1ggLSBUYXJnZXQgQ2h1bmsgWCBjb29yZGluYXRlLlxuICAgICAqIEBwYXJhbSB7bnVtYmVyfSBjaHVua1ogLSBUYXJnZXQgQ2h1bmsgWiBjb29yZGluYXRlLlxuICAgICAqIEByZXR1cm5zIHtiaWdpbnR9IEEgNjQgYml0IGludGdlciBjb250YWluaW5nIGEgaGFzaCBvZiBYIGFuZCBaLlxuICAgICAqL1xuICAgIHB1YmxpYyBzdGF0aWMgcGFja1haKGNodW5rWDogbnVtYmVyLCBjaHVua1o6IG51bWJlcik6IGJpZ2ludCB7XG4gICAgICAgIHJldHVybiAoKEJpZ0ludChjaHVua1gpICYgMHhmZmZmZmZmZm4pIDw8IDMybikgfCAoQmlnSW50KGNodW5rWikgJiAweGZmZmZmZmZmbik7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogSGVscGVyIG1ldGhvZCB1c2VkIHRvIGRlY29kZSBhIDY0IGJpdCBoYXNoIGNvbnRhaW5pbmdcbiAgICAgKiBib3RoIENodW5rIFggYW5kIFogY29vcmRpbmF0ZXMuXG4gICAgICogQHBhcmFtIHtiaWdpbnR9IHBhY2tlZCAtIFRhcmdldCBDaHVuayBjb29yZGluYXRlIGhhc2guXG4gICAgICogQHJldHVybnMge251bWJlcltdfSBBbiBhcnJheSBjb250YWluaW5nIGRlY29kZWQgQ2h1bmsgWCBhbmQgWiBjb29yZGluYXRlcy5cbiAgICAgKi9cbiAgICBwdWJsaWMgc3RhdGljIHVucGFja1haKHBhY2tlZDogYmlnaW50KTogbnVtYmVyW10ge1xuICAgICAgICByZXR1cm4gW051bWJlcihCaWdJbnQuYXNJbnROKDMyLCBwYWNrZWQgPj4gMzJuKSksIE51bWJlcihCaWdJbnQuYXNJbnROKDMyLCBwYWNrZWQgJiAweGZmZmZmZmZmbikpXTtcbiAgICB9XG5cbiAgICBwdWJsaWMgbmV0d29ya1NlcmlhbGl6ZSgpOiBCdWZmZXIge1xuICAgICAgICBjb25zdCBzdHJlYW0gPSBuZXcgQmluYXJ5U3RyZWFtKCk7XG5cbiAgICAgICAgLy8gRm9yIHNvbWUgcmVhc29ucyB3ZSBuZWVkIHRoaXMgaGFjayBzaW5jZSAxLjE4LFxuICAgICAgICAvLyBzZWVtcyBsaWtlIHRoZSBjbGllbnQgbm93IGhhcyBzb21lIG5lZ2F0aXZlIHNwYWNlLlxuICAgICAgICAvLyBUT0RPOiBmaWd1cmUgb3V0IHdoYXQgaXMgdGhpc1xuICAgICAgICBmb3IgKGxldCB5ID0gMDsgeSA8IDQ7ICsreSkge1xuICAgICAgICAgICAgc3RyZWFtLndyaXRlQnl0ZSg4KTsgLy8gc3ViY2h1bmsgdmVyc2lvbiA4XG4gICAgICAgICAgICBzdHJlYW0ud3JpdGVCeXRlKDApOyAvLyAwIGxheWVycyAoYWxsIGFpcilcbiAgICAgICAgfVxuXG4gICAgICAgIGZvciAobGV0IHkgPSAwOyB5IDwgdGhpcy5nZXRUb3BFbXB0eSgpOyArK3kpIHtcbiAgICAgICAgICAgICh0aGlzLnN1YkNodW5rcy5nZXQoeSkgPz8gQ2h1bmsuRU1QVFlfU1VCQ0hVTkspLm5ldHdvcmtTZXJpYWxpemUoc3RyZWFtKTtcbiAgICAgICAgfVxuXG4gICAgICAgIC8vIFRPRE86IDNEIGJpb21lc1xuICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IDI0OyBpKyspIHtcbiAgICAgICAgICAgIHN0cmVhbS53cml0ZUJ5dGUoMCk7IC8vIGZha2UgYmlvbWUgcGFsZXR0ZSwgbm9uIHBlcnNpc3RlbnRcbiAgICAgICAgICAgIHN0cmVhbS53cml0ZVVuc2lnbmVkVmFySW50KDEgPDwgMSk7IC8vIHBsYWluc1xuICAgICAgICB9XG5cbiAgICAgICAgc3RyZWFtLndyaXRlQnl0ZSgwKTsgLy8gYm9yZGVyID9cblxuICAgICAgICAvLyBUT0RPOiB0aWxlc1xuICAgICAgICByZXR1cm4gc3RyZWFtLmdldEJ1ZmZlcigpO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIERlc2VyaWFsaXplIG5ldHdvcmsgc3RyZWFtIGludG8gY2h1bmtcbiAgICAgKiB1c2VmdWwgZm9yIGNsaWVudCBhcHBsaWNhdGlvbnMgYW5kL29yIG91ciBGaWxlc3lzdGVtIGltcGxcbiAgICAgKiBAcGFyYW0ge0JpbmFyeVN0cmVhbX0gc3RyZWFtIC0gdGhlIG5ldHdvcmsgc3RyZWFtXG4gICAgICogQHBhcmFtIHtudW1iZXJ9IFt4XSAtIHRoZSBjaHVuayB4IGNvb3JkaW5hdGVcbiAgICAgKiBAcGFyYW0ge251bWJlcn0gW3pdIC0gdGhlIGNodW5rIHogY29vcmRpbmF0ZVxuICAgICAqL1xuICAgIHB1YmxpYyBzdGF0aWMgbmV0d29ya0Rlc2VyaWFsaXplKHN0cmVhbTogQmluYXJ5U3RyZWFtLCB4PzogbnVtYmVyLCB6PzogbnVtYmVyKTogQ2h1bmsge1xuICAgICAgICBzdHJlYW0ucmVhZCg4KTsgLy8gc2tpcCBmYWtlIHN1YmNodW5rc1xuXG4gICAgICAgIGNvbnN0IHN1YkNodW5rczogTWFwPG51bWJlciwgU3ViQ2h1bms+ID0gbmV3IE1hcCgpO1xuICAgICAgICBmb3IgKGxldCBpID0gMDsgaSA8IE1BWF9TVUJDSFVOS1M7IGkrKykge1xuICAgICAgICAgICAgc3ViQ2h1bmtzLnNldChpLCBTdWJDaHVuay5uZXR3b3JrRGVzZXJpYWxpemUoc3RyZWFtKSk7XG4gICAgICAgIH1cblxuICAgICAgICBjb25zdCBjaHVuayA9IG5ldyBDaHVuayh4LCB6LCBzdWJDaHVua3MpO1xuICAgICAgICByZXR1cm4gY2h1bms7XG4gICAgfVxufVxuIl0sIm1hcHBpbmdzIjoiOzs7OztBQVFBLElBQU0sZ0JBQWdCO0FBRXRCLElBQXFCLFFBQXJCLE1BQXFCLE1BQU07Q0FDdkI7Q0FDQTtDQUNBO0NBRUEsNEJBQTJDLElBQUksSUFBSTtDQUNuRCxPQUF3QixpQkFBaUIsSUFBSSxTQUFTO0NBRXRELFlBQW1CLFNBQVMsR0FBRyxTQUFTLEdBQUcsNkJBQW9DLElBQUksSUFBSSxHQUFHO0VBQ3RGLEtBQUssSUFBSTtFQUNULEtBQUssSUFBSTtFQUNULEtBQUssYUFBYTtDQUN0QjtDQUVBLE9BQXNCO0VBQ2xCLE9BQU8sS0FBSztDQUNoQjtDQUVBLE9BQXNCO0VBQ2xCLE9BQU8sS0FBSztDQUNoQjtDQUVBLGdCQUFnQztFQUM1QixPQUFPLEtBQUs7Q0FDaEI7Q0FFQSxZQUEyQjtFQUN2QixPQUFPLEtBQUssVUFBVTtDQUMxQjs7Ozs7Q0FNQSxjQUE2QjtFQUN6QixJQUFJLFdBQVcsZ0JBQWdCO0VBQy9CLE9BQ0ssWUFBWSxLQUFLLENBQUMsS0FBSyxVQUFVLElBQUksUUFBUSxLQUM3QyxLQUFLLFVBQVUsSUFBSSxRQUFRLEtBQUssS0FBSyxVQUFVLElBQUksUUFBUSxFQUFHLFFBQVEsR0FFdkU7RUFFSixPQUFPLEVBQUU7Q0FDYjs7Ozs7Q0FNQSxZQUFtQixHQUE0QjtFQUMzQyxJQUFJLElBQUksS0FBSyxJQUFJLGVBQ2IsTUFBTSxJQUFJLE1BQU0sNEJBQTRCLEdBQUc7RUFFbkQsT0FBTyxLQUFLLFVBQVUsSUFBSSxDQUFDLEtBQUs7Q0FDcEM7Q0FFQSxvQkFBMkIsR0FBcUI7RUFDNUMsTUFBTSxXQUFXLElBQUksU0FBUztFQUM5QixLQUFLLFVBQVUsSUFBSSxHQUFHLFFBQVE7RUFDOUIsT0FBTztDQUNYO0NBRUEsZUFBNkM7RUFDekMsT0FBTyxLQUFLO0NBQ2hCO0NBRUEsa0JBQXlCLElBQVksSUFBb0I7RUFDckQsT0FBTztDQUNYOzs7Ozs7Ozs7Q0FVQSxTQUFnQixHQUFxQixJQUFZLEdBQUcsSUFBWSxHQUFHLFFBQVEsR0FBYTtFQUNwRixJQUFJLGFBQWEsU0FDYixPQUFPLEtBQUssU0FBUyxFQUFFLEtBQUssR0FBRyxFQUFFLEtBQUssR0FBRyxFQUFFLEtBQUssR0FBRyxLQUFLO0VBRzVELE1BQU0sV0FBVyxLQUFLLFlBQVksS0FBSyxNQUFNLElBQUksRUFBRSxDQUFDO0VBQ3BELElBQUksQ0FBQyxVQUFVLE9BQU8sY0FBYyxZQUFZLGNBQWMsYUFBYSxlQUFlLENBQUM7RUFDM0YsT0FBTyxTQUFTLFNBQVMsR0FBRyxJQUFJLElBQUssR0FBRyxLQUFLO0NBQ2pEOzs7Ozs7Ozs7Q0FVQSxTQUFnQixHQUFXLEdBQVcsR0FBVyxPQUFjLFFBQVEsR0FBUztFQUM1RSxJQUFJLFdBQVcsS0FBSyxZQUFZLEtBQUssTUFBTSxJQUFJLEVBQUUsQ0FBQztFQUNsRCxJQUFJLENBQUMsVUFBVSxXQUFXLEtBQUssb0JBQW9CLEtBQUssQ0FBQztFQUN6RCxTQUFTLFNBQVMsR0FBRyxJQUFJLElBQUssR0FBRyxjQUFjLGFBQWEsTUFBTSxRQUFRLENBQUMsR0FBRyxLQUFLO0VBRW5GLEtBQUssYUFBYTtDQUN0Qjs7Ozs7Ozs7Q0FTQSxPQUFjLE9BQU8sUUFBZ0IsUUFBd0I7RUFDekQsUUFBUyxPQUFPLE1BQU0sSUFBSSxnQkFBZ0IsTUFBUSxPQUFPLE1BQU0sSUFBSTtDQUN2RTs7Ozs7OztDQVFBLE9BQWMsU0FBUyxRQUEwQjtFQUM3QyxPQUFPLENBQUMsT0FBTyxPQUFPLE9BQU8sSUFBSSxVQUFVLEdBQUcsQ0FBQyxHQUFHLE9BQU8sT0FBTyxPQUFPLElBQUksU0FBUyxXQUFXLENBQUMsQ0FBQztDQUNyRztDQUVBLG1CQUFrQztFQUM5QixNQUFNLFNBQVMsSUFBSSxhQUFhO0VBS2hDLEtBQUssSUFBSSxJQUFJLEdBQUcsSUFBSSxHQUFHLEVBQUUsR0FBRztHQUN4QixPQUFPLFVBQVUsQ0FBQztHQUNsQixPQUFPLFVBQVUsQ0FBQztFQUN0QjtFQUVBLEtBQUssSUFBSSxJQUFJLEdBQUcsSUFBSSxLQUFLLFlBQVksR0FBRyxFQUFFLEdBQ3RDLENBQUMsS0FBSyxVQUFVLElBQUksQ0FBQyxLQUFLLE1BQU0sZ0JBQWdCLGlCQUFpQixNQUFNO0VBSTNFLEtBQUssSUFBSSxJQUFJLEdBQUcsSUFBSSxJQUFJLEtBQUs7R0FDekIsT0FBTyxVQUFVLENBQUM7R0FDbEIsT0FBTyxvQkFBb0IsQ0FBTTtFQUNyQztFQUVBLE9BQU8sVUFBVSxDQUFDO0VBR2xCLE9BQU8sT0FBTyxVQUFVO0NBQzVCOzs7Ozs7OztDQVNBLE9BQWMsbUJBQW1CLFFBQXNCLEdBQVksR0FBbUI7RUFDbEYsT0FBTyxLQUFLLENBQUM7RUFFYixNQUFNLDRCQUFtQyxJQUFJLElBQUk7RUFDakQsS0FBSyxJQUFJLElBQUksR0FBRyxJQUFJLGVBQWUsS0FDL0IsVUFBVSxJQUFJLEdBQUcsU0FBUyxtQkFBbUIsTUFBTSxDQUFDO0VBSXhELE9BQU8sSUFEVyxNQUFNLEdBQUcsR0FBRyxTQUN2QjtDQUNYO0FBQ0oifQ==