using System; using System.IO; namespace Substrate { using NBT; using Utility; using System.Collections.Generic; public class Chunk : IChunk, INBTObject, ICopyable { public static NBTCompoundNode LevelSchema = new NBTCompoundNode() { new NBTCompoundNode("Level") { new NBTArrayNode("Blocks", 32768), new NBTArrayNode("Data", 16384), new NBTArrayNode("SkyLight", 16384), new NBTArrayNode("BlockLight", 16384), new NBTArrayNode("HeightMap", 256), new NBTListNode("Entities", TagType.TAG_COMPOUND), new NBTListNode("TileEntities", TagType.TAG_COMPOUND, TileEntity.BaseSchema), new NBTScalerNode("LastUpdate", TagType.TAG_LONG, NBTOptions.CREATE_ON_MISSING), new NBTScalerNode("xPos", TagType.TAG_INT), new NBTScalerNode("zPos", TagType.TAG_INT), new NBTScalerNode("TerrainPopulated", TagType.TAG_BYTE, NBTOptions.CREATE_ON_MISSING), }, }; private NBT_Tree _tree; private int _cx; private int _cz; protected TagByteArray _blocks; protected NibbleArray _data; protected NibbleArray _blockLight; protected NibbleArray _skyLight; protected TagByteArray _heightMap; protected TagList _entities; protected TagList _tileEntities; protected Dictionary _tileEntityTable; public int X { get { return _cx; } } public int Z { get { return _cz; } } public NBT_Tree Tree { get { return _tree; } } public bool IsTerrainPopulated { get { return _tree.Root["Level"].ToTagCompound()["TerrainPopulated"].ToTagByte() == 1; } set { _tree.Root["Level"].ToTagCompound()["TerrainPopulated"].ToTagByte().Data = (byte)(value ? 1 : 0); } } public Chunk (int x, int z) { _cx = x; _cz = z; BuildNBTTree(); BuildTileEntityCache(); } public Chunk (NBT_Tree tree) { if (LoadTreeSafe(tree.Root) == null) { throw new InvalidNBTObjectException(); } } private void BuildNBTTree () { int elements2 = XDim * ZDim; int elements3 = elements2 *YDim; _blocks = new TagByteArray(new byte[elements3]); TagByteArray data = new TagByteArray(new byte[elements3 >> 1]); TagByteArray blocklight = new TagByteArray(new byte[elements3 >> 1]); TagByteArray skylight = new TagByteArray(new byte[elements3 >> 1]); _heightMap = new TagByteArray(new byte[elements2]); _entities = new TagList(TagType.TAG_COMPOUND); _tileEntities = new TagList(TagType.TAG_COMPOUND); _data = new NibbleArray(data.Data); _blockLight = new NibbleArray(blocklight.Data); _skyLight = new NibbleArray(skylight.Data); TagCompound level = new TagCompound(); level.Add("Blocks", _blocks); level.Add("Data", data); level.Add("SkyLight", blocklight); level.Add("BlockLight", skylight); level.Add("HeightMap", _heightMap); level.Add("Entities", _entities); level.Add("TileEntities", _tileEntities); level.Add("LastUpdate", new TagLong(Timestamp())); level.Add("xPos", new TagInt(_cx)); level.Add("zPos", new TagInt(_cz)); level.Add("TerrainPopulated", new TagByte()); _tree = new NBT_Tree(); _tree.Root.Add("Level", level); } public int BlockGlobalX (int x) { return _cx * XDim + x; } public int BlockGlobalY (int y) { return y; } public int BlockGlobalZ (int z) { return _cz * XDim + z; } public int BlockLocalX (int x) { return x; } public int BlockLocalY (int y) { return y; } public int BlockLocalZ (int z) { return z; } public bool Save (Stream outStream) { if (outStream == null || !outStream.CanWrite) { return false; } _tree.WriteTo(outStream); outStream.Close(); return true; } public Block GetBlock (int lx, int ly, int lz) { return new Block(this, lx, ly, lz); } public BlockRef GetBlockRef (int lx, int ly, int lz) { return new BlockRef(this, lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, Block block) { int index = lx << 11 | lz << 7 | ly; SetBlockID(lx, ly, lz, block.ID); SetBlockData(lx, ly, lz, block.Data); SetTileEntity(lx, ly, lz, block.GetTileEntity().Copy()); } public int CountBlockID (int id) { int c = 0; for (int i = 0; i < _blocks.Length; i++) { if (_blocks[i] == id) { c++; } } return c; } public int CountBlockData (int id, int data) { int c = 0; for (int i = 0; i < _blocks.Length; i++) { if (_blocks[i] == id && _data[i] == data) { c++; } } return c; } public int CountEntities () { return _entities.Count; } public int GetHeight (int lx, int lz) { return _heightMap[lz << 4 | lx]; } private void CreateTileEntity (int lx, int ly, int lz) { BlockInfoEx info = GetBlockInfo(lx, ly, lz) as BlockInfoEx; if (info == null) { return; } TileEntity te = TileEntityFactory.Create(info.TileEntityName); if (te == null) { return; } te.X = BlockGlobalX(lx); te.Y = BlockGlobalY(ly); te.Z = BlockGlobalZ(lz); _tileEntities.Add(te.BuildTree()); } public virtual void SetLocation (int x, int z) { int diffx = x - _cx; int diffz = z - _cz; _cx = x; _cz = z; BuildTileEntityCache(); } private void BuildTileEntityCache () { _tileEntityTable = new Dictionary(); foreach (TagCompound te in _tileEntities) { int tex = te["x"].ToTagInt(); int tey = te["y"].ToTagInt(); int tez = te["z"].ToTagInt(); BlockKey key = new BlockKey(tex, tey, tez); _tileEntityTable[key] = te; } } #region IBoundedBlockContainer Members public int XDim { get { return 16; } } public int YDim { get { return 128; } } public int ZDim { get { return 16; } } #endregion #region IBlockContainer Members IBlock IBlockContainer.GetBlock (int lx, int ly, int lz) { return GetBlock(lx, ly, lz); } IBlock IBlockContainer.GetBlockRef (int lx, int ly, int lz) { return GetBlockRef(lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, IBlock block) { int index = lx << 11 | lz << 7 | ly; int oldid = _blocks.Data[index]; SetBlockID(lx, ly, lz, block.ID); SetBlockData(lx, ly, lz, block.Data); // Update tile entities BlockInfoEx info1 = BlockInfo.BlockTable[oldid] as BlockInfoEx; BlockInfoEx info2 = BlockInfo.BlockTable[block.ID] as BlockInfoEx; if (info1 != info2) { if (info1 != null) { ClearTileEntity(lx, ly, lz); } if (info2 != null) { CreateTileEntity(lx, ly, lz); } } } public BlockInfo GetBlockInfo (int lx, int ly, int lz) { return BlockInfo.BlockTable[GetBlockID(lx, ly, lz)]; } public int GetBlockID (int lx, int ly, int lz) { return _blocks.Data[lx << 11 | lz << 7 | ly]; } public int GetBlockData (int lx, int ly, int lz) { return _data[lx << 11 | lz << 7 | ly]; } public bool SetBlockID (int lx, int ly, int lz, int id) { int index = lx << 11 | lz << 7 | ly; int oldid = _blocks.Data[index]; if (oldid == id) { return false; } // Update value _blocks.Data[index] = (byte)id; // Update tile entities BlockInfoEx info1 = BlockInfo.BlockTable[oldid] as BlockInfoEx; BlockInfoEx info2 = BlockInfo.BlockTable[id] as BlockInfoEx; if (info1 != info2) { if (info1 != null) { ClearTileEntity(lx, ly, lz); } if (info2 != null) { CreateTileEntity(lx, ly, lz); } } // Update height map if (BlockInfo.BlockTable[id] != null) { int tileHeight = GetHeight(lx, lz); int newOpacity = BlockInfo.BlockTable[id].Opacity; if (ly > tileHeight && newOpacity > BlockInfo.MIN_OPACITY) { _heightMap[lz << 4 | lx] = (byte)ly; } else if (ly == tileHeight && newOpacity == BlockInfo.MIN_OPACITY) { for (int i = ly - 1; i >= 0; i--) { if (BlockInfo.BlockTable[GetBlockID(lx, i, lz)].Opacity > BlockInfo.MIN_OPACITY) { _heightMap[lz << 4 | lx] = (byte)i; break; } } } } // Light consistency return true; } public bool SetBlockData (int lx, int ly, int lz, int data) { int index = lx << 11 | lz << 7 | ly; if (_data[index] == data) { return false; } /*if (BlockManager.EnforceDataLimits && BlockInfo.BlockTable[_blocks[index]] != null) { if (!BlockInfo.BlockTable[_blocks[index]].TestData(data)) { return false; } }*/ _data[index] = data; return true; } #endregion #region ILitBlockContainer Members ILitBlock ILitBlockContainer.GetBlock (int lx, int ly, int lz) { throw new NotImplementedException(); } ILitBlock ILitBlockContainer.GetBlockRef (int lx, int ly, int lz) { return GetBlockRef(lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, ILitBlock block) { int index = lx << 11 | lz << 7 | ly; int oldid = _blocks.Data[index]; SetBlockID(lx, ly, lz, block.ID); SetBlockData(lx, ly, lz, block.Data); // Update tile entities BlockInfoEx info1 = BlockInfo.BlockTable[oldid] as BlockInfoEx; BlockInfoEx info2 = BlockInfo.BlockTable[block.ID] as BlockInfoEx; if (info1 != info2) { if (info1 != null) { ClearTileEntity(lx, ly, lz); } if (info2 != null) { CreateTileEntity(lx, ly, lz); } } } public int GetBlockLight (int lx, int ly, int lz) { return _blockLight[lx << 11 | lz << 7 | ly]; } public int GetBlockSkyLight (int lx, int ly, int lz) { return _skyLight[lx << 11 | lz << 7 | ly]; } public bool SetBlockLight (int lx, int ly, int lz, int light) { int index = lx << 11 | lz << 7 | ly; if (_blockLight[index] == light) { return false; } _blockLight[index] = light; return true; } public bool SetBlockSkyLight (int lx, int ly, int lz, int light) { int index = lx << 11 | lz << 7 | ly; if (_skyLight[index] == light) { return false; } _skyLight[index] = light; return true; } #endregion #region IPropertyBlockContainer Members IPropertyBlock IPropertyBlockContainer.GetBlock (int lx, int ly, int lz) { return GetBlock(lx, ly, lz); } IPropertyBlock IPropertyBlockContainer.GetBlockRef (int lx, int ly, int lz) { return GetBlockRef(lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, IPropertyBlock block) { int index = lx << 11 | lz << 7 | ly; SetBlockID(lx, ly, lz, block.ID); SetBlockData(lx, ly, lz, block.Data); SetTileEntity(lx, ly, lz, block.GetTileEntity().Copy()); } public TileEntity GetTileEntity (int lx, int ly, int lz) { int x = BlockGlobalX(lx); int y = BlockGlobalY(ly); int z = BlockGlobalZ(lz); BlockKey key = new BlockKey(x, y, z); TagCompound te; if (!_tileEntityTable.TryGetValue(key, out te)) { return null; } return TileEntityFactory.Create(te); } public bool SetTileEntity (int lx, int ly, int lz, TileEntity te) { BlockInfoEx info = GetBlockInfo(lx, ly, lz) as BlockInfoEx; if (info == null) { return false; } if (te.GetType() != TileEntityFactory.Lookup(info.TileEntityName)) { return false; } int x = BlockGlobalX(lx); int y = BlockGlobalY(ly); int z = BlockGlobalZ(lz); BlockKey key = new BlockKey(x, y, z); TagCompound oldte; if (_tileEntityTable.TryGetValue(key, out oldte)) { _tileEntities.Remove(oldte); } te.X = x; te.Y = y; te.Z = z; TagCompound tree = te.BuildTree() as TagCompound; _tileEntities.Add(tree); _tileEntityTable[key] = tree; return true; } public bool ClearTileEntity (int lx, int ly, int lz) { int x = BlockGlobalX(lx); int y = BlockGlobalY(ly); int z = BlockGlobalZ(lz); BlockKey key = new BlockKey(x, y, z); TagCompound te; if (!_tileEntityTable.TryGetValue(key, out te)) { return false; } _tileEntities.Remove(te); _tileEntityTable.Remove(key); return true; } #endregion #region ICopyable Members public Chunk Copy () { return new Chunk(_tree.Copy()); } #endregion #region INBTObject Members public Chunk LoadTree (TagValue tree) { TagCompound ctree = tree as TagCompound; if (ctree == null) { return null; } _tree = new NBT_Tree(ctree); TagCompound level = _tree.Root["Level"] as TagCompound; _blocks = level["Blocks"] as TagByteArray; _data = new NibbleArray(level["Data"].ToTagByteArray().Data); _blockLight = new NibbleArray(level["BlockLight"].ToTagByteArray().Data); _skyLight = new NibbleArray(level["SkyLight"].ToTagByteArray().Data); _heightMap = level["HeightMap"] as TagByteArray; _entities = level["Entities"] as TagList; _tileEntities = level["TileEntities"] as TagList; // List-type patch up if (_entities.Count == 0) { level["Entities"] = new TagList(TagType.TAG_COMPOUND); _entities = level["Entities"] as TagList; } if (_tileEntities.Count == 0) { level["TileEntities"] = new TagList(TagType.TAG_COMPOUND); _tileEntities = level["TileEntities"] as TagList; } _cx = level["xPos"].ToTagInt(); _cz = level["zPos"].ToTagInt(); BuildTileEntityCache(); return this; } public Chunk LoadTreeSafe (TagValue tree) { if (!ValidateTree(tree)) { return null; } return LoadTree(tree); } public TagValue BuildTree () { return _tree.Root; } public bool ValidateTree (TagValue tree) { return new NBTVerifier(tree, LevelSchema).Verify(); } #endregion #region IEntityContainer Members public List FindEntities (string id) { List set = new List(); foreach (TagCompound ent in _entities) { TagValue eid; if (!ent.TryGetValue("id", out eid)) { continue; } if (eid.ToTagString().Data != id) { continue; } Entity obj = EntityFactory.Create(ent); if (obj != null) { set.Add(obj); } } return set; } public List FindEntities (Predicate match) { List set = new List(); foreach (TagCompound ent in _entities) { Entity obj = EntityFactory.Create(ent); if (obj == null) { continue; } if (match(obj)) { set.Add(obj); } } return set; } public bool AddEntity (Entity ent) { double xlow = _cx * XDim; double xhigh = xlow + XDim; double zlow = _cz * ZDim; double zhigh = zlow + ZDim; Entity.Vector3 pos = ent.Position; if (!(pos.X >= xlow && pos.X < xhigh && pos.Z >= zlow && pos.Z < zhigh)) { return false; } _entities.Add(ent.BuildTree()); return true; } public int RemoveEntities (string id) { return _entities.RemoveAll(val => { TagCompound cval = val as TagCompound; if (cval == null) { return false; } TagValue sval; if (!cval.TryGetValue("id", out sval)) { return false; } return (sval.ToTagString().Data == id); }); } public int RemoveEntities (Predicate match) { return _entities.RemoveAll(val => { TagCompound cval = val as TagCompound; if (cval == null) { return false; } Entity obj = EntityFactory.Create(cval); if (obj == null) { return false; } return match(obj); }); } #endregion public void ResetBlockLight () { for (int i = 0; i < _blocks.Length; i++) { //BlockInfo info = BlockInfo.BlockTable[_blocks[i]]; _blockLight[i] = 0; // Math.Max(info.Luminance - info.Opacity, 0); } } public void ResetSkyLight () { for (int x = 0; x < XDim; x++) { for (int z = 0; z < ZDim; z++) { int height = GetHeight(x, z); int ystride = x << 11 | z << 7; for (int y = 0; y < YDim; y++ ) { int index = ystride + y; if (y >= height) { _skyLight[index] = BlockInfo.MIN_LUMINANCE; } else { _skyLight[index] = BlockInfo.MIN_LUMINANCE; } } } } } private int Timestamp () { DateTime epoch = new DateTime(1970, 1, 1, 0, 0, 0, 0); return (int)((DateTime.UtcNow - epoch).Ticks / (10000L * 1000L)); } } }