using System; using System.Collections.Generic; using System.Text; using System.IO; namespace Substrate { using NBT; using System.Collections; public class ChunkRef : IChunk { private IChunkContainer _container; private IChunkCache _cache; private Chunk _chunk; private int _cx; private int _cz; private bool _dirty; public int X { get { return _container.ChunkGlobalX(_cx); } } public int Z { get { return _container.ChunkGlobalZ(_cz); } } public int LocalX { get { return _container.ChunkLocalX(_cx); } } public int LocalZ { get { return _container.ChunkLocalZ(_cz); } } public ChunkRef (IChunkContainer container, IChunkCache cache, int cx, int cz) { _container = container; _cache = cache; _cx = cx; _cz = cz; if (!_container.ChunkExists(cx, cz)) { throw new MissingChunkException(); } } public int BlockGlobalX (int x) { return _container.ChunkGlobalX(_cx) * XDim + x; } public int BlockGlobalY (int y) { return y; } public int BlockGlobalZ (int z) { return _container.ChunkGlobalZ(_cz) * ZDim + z; } public int BlockLocalX (int x) { return x; } public int BlockLocalY (int y) { return y; } public int BlockLocalZ (int z) { return z; } private Chunk GetChunk () { if (_chunk == null) { _chunk = _container.GetChunk(_cx, _cz); _lightbit = new BitArray(XDim * 3 * ZDim * 3 * YDim); _update = new Queue(); } return _chunk; } private bool MarkDirty () { if (_dirty) { return false; } _cache.MarkChunkDirty(this); _dirty = true; return true; } public ChunkRef GetNorthNeighbor () { return _container.GetChunkRef(_cx - 1, _cz); } public ChunkRef GetSouthNeighbor () { return _container.GetChunkRef(_cx + 1, _cz); } public ChunkRef GetEastNeighbor () { return _container.GetChunkRef(_cx, _cz - 1); } public ChunkRef GetWestNeighbor () { return _container.GetChunkRef(_cx, _cz + 1); } public Chunk GetChunkCopy () { return GetChunk().Copy(); } public Chunk GetChunkRef () { Chunk chunk = GetChunk(); _chunk = null; return chunk; } public void SetChunkRef (Chunk chunk) { _chunk = chunk; _chunk.SetLocation(_cx, _cz); MarkDirty(); } #region IChunk Members public bool IsTerrainPopulated { get { return GetChunk().IsTerrainPopulated; } set { if (GetChunk().IsTerrainPopulated != value) { GetChunk().IsTerrainPopulated = value; MarkDirty(); } } } public bool Save (Stream outStream) { if (_dirty) { if (GetChunk().Save(outStream)) { _dirty = false; return true; } return false; } 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 BlockInfo GetBlockInfo (int lx, int ly, int lz) { return GetChunk().GetBlockInfo(lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, Block block) { GetChunk().SetBlock(lx, ly, lz, block); } public int CountBlockID (int id) { return GetChunk().CountBlockID(id); } public int CountBlockData (int id, int data) { return GetChunk().CountBlockData(id, data); } public int CountEntities () { return GetChunk().CountEntities(); } public int GetHeight (int lx, int lz) { return GetChunk().GetHeight(lx, lz); } #endregion #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 new Block(this, lx, ly, lz); } IBlock IBlockContainer.GetBlockRef (int lx, int ly, int lz) { return new BlockRef(this, lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, IBlock block) { GetChunk().SetBlock(lx, ly, lz, block); } public int GetBlockID (int lx, int ly, int lz) { return GetChunk().GetBlockID(lx, ly, lz); } public int GetBlockData (int lx, int ly, int lz) { return GetChunk().GetBlockData(lx, ly, lz); } public bool SetBlockID (int lx, int ly, int lz, int id) { BlockInfo info1 = GetChunk().GetBlockInfo(lx, ly, lz); if (GetChunk().SetBlockID(lx, ly, lz, id)) { MarkDirty(); BlockInfo info2 = GetChunk().GetBlockInfo(lx, ly, lz); if (info1.Luminance != info2.Luminance || info1.Opacity != info2.Opacity) { _update.Enqueue(new BlockKey(lx, ly, lz)); UpdateBlockLight(); } return true; } return false; } public bool SetBlockData (int lx, int ly, int lz, int data) { if (GetChunk().SetBlockData(lx, ly, lz, data)) { MarkDirty(); return true; } return false; } #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 new BlockRef(this, lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, ILitBlock block) { GetChunk().SetBlock(lx, ly, lz, block); } public int GetBlockLight (int lx, int ly, int lz) { return GetChunk().GetBlockLight(lx, ly, lz); } public int GetBlockSkyLight (int lx, int ly, int lz) { return GetChunk().GetBlockSkyLight(lx, ly, lz); } public bool SetBlockLight (int lx, int ly, int lz, int light) { if (GetChunk().SetBlockLight(lx, ly, lz, light)) { MarkDirty(); return true; } return false; } public bool SetBlockSkyLight (int lx, int ly, int lz, int light) { if (GetChunk().SetBlockSkyLight(lx, ly, lz, light)) { MarkDirty(); return true; } return false; } #endregion #region IPropertyBlockContainer Members IPropertyBlock IPropertyBlockContainer.GetBlock (int lx, int ly, int lz) { return new Block(this, lx, ly, lz); } IPropertyBlock IPropertyBlockContainer.GetBlockRef (int lx, int ly, int lz) { return new BlockRef(this, lx, ly, lz); } public void SetBlock (int lx, int ly, int lz, IPropertyBlock block) { GetChunk().SetBlock(lx, ly, lz, block); } public TileEntity GetTileEntity (int lx, int ly, int lz) { return GetChunk().GetTileEntity(lx, ly, lz); } public bool SetTileEntity (int lx, int ly, int lz, TileEntity te) { if (GetChunk().SetTileEntity(lx, ly, lz, te)) { MarkDirty(); return true; } return false; } public bool ClearTileEntity (int lx, int ly, int lz) { if (GetChunk().ClearTileEntity(lx, ly, lz)) { MarkDirty(); return true; } return false; } #endregion #region IEntityContainer Members public List FindEntities (string id) { return GetChunk().FindEntities(id); } public List FindEntities (Predicate match) { return GetChunk().FindEntities(match); } public bool AddEntity (Entity ent) { if (GetChunk().AddEntity(ent)) { MarkDirty(); return true; } return false; } public int RemoveEntities (string id) { int ret = GetChunk().RemoveEntities(id); if (ret > 0) { MarkDirty(); } return ret; } public int RemoveEntities (Predicate match) { int ret = GetChunk().RemoveEntities(match); if (ret > 0) { MarkDirty(); } return ret; } #endregion private BitArray _lightbit; private Queue _update; private void UpdateBlockLight () { while (_update.Count > 0) { BlockKey k = _update.Dequeue(); int index = LightBitmapIndex(k); _lightbit[index] = false; ChunkRef cc = LocalChunk(k.x, k.y, k.z); if (cc == null) { continue; } int lle = NeighborLight(k.x, k.y, k.z - 1); int lln = NeighborLight(k.x - 1, k.y, k.z); int lls = NeighborLight(k.x, k.y, k.z + 1); int llw = NeighborLight(k.x + 1, k.y, k.z); int lld = NeighborLight(k.x, k.y - 1, k.z); int llu = NeighborLight(k.x, k.y + 1, k.z); int x = (k.x + XDim) % XDim; int y = k.y; int z = (k.z + ZDim) % ZDim; int lightval = cc.GetBlockLight(x, y, z); BlockInfo info = cc.GetBlockInfo(x, y, z); int light = Math.Max(info.Luminance, 0); light = Math.Max(light, lle - 1); light = Math.Max(light, lln - 1); light = Math.Max(light, lls - 1); light = Math.Max(light, llw - 1); light = Math.Max(light, lld - 1); light = Math.Max(light, llu - 1); light = Math.Max(light - info.Opacity, 0); if (light != lightval) { //Console.WriteLine("Block Light: ({0},{1},{2}) " + lightval + " -> " + light, k.x, k.y, k.z); cc.SetBlockLight(x, y, z, light); QueueRelight(new BlockKey(k.x - 1, k.y, k.z)); QueueRelight(new BlockKey(k.x + 1, k.y, k.z)); QueueRelight(new BlockKey(k.x, k.y - 1, k.z)); QueueRelight(new BlockKey(k.x, k.y + 1, k.z)); QueueRelight(new BlockKey(k.x, k.y, k.z - 1)); QueueRelight(new BlockKey(k.x, k.y, k.z + 1)); } } } private int LightBitmapIndex (BlockKey key) { int x = key.x + XDim; int y = key.y; int z = key.z + ZDim; int zstride = YDim; int xstride = ZDim * 3 * zstride; return (x * xstride) + (z * zstride) + y; } private void QueueRelight (BlockKey key) { if (key.x < -15 || key.x >= 31 || key.z < -15 || key.z >= 31) { return; } int index = LightBitmapIndex(key); if (!_lightbit[index]) { _lightbit[index] = true; _update.Enqueue(key); } } private ChunkRef LocalChunk (int lx, int ly, int lz) { if (lx < 0) { if (lz < 0) { return _container.GetChunkRef(_cx - 1, _cz - 1); } else if (lz >= ZDim) { return _container.GetChunkRef(_cx - 1, _cz + 1); } return _container.GetChunkRef(_cx - 1, _cz); } else if (lx >= XDim) { if (lz < 0) { return _container.GetChunkRef(_cx + 1, _cz - 1); } else if (lz >= ZDim) { return _container.GetChunkRef(_cx + 1, _cz + 1); } return _container.GetChunkRef(_cx + 1, _cz); } else { if (lz < 0) { return _container.GetChunkRef(_cx, _cz - 1); } else if (lz >= ZDim) { return _container.GetChunkRef(_cx, _cz + 1); } return this; } } private int NeighborLight (int x, int y, int z) { if (y < 0 || y >= YDim) { return 0; } ChunkRef src = LocalChunk(x, y, z); if (src == null) { return 0; } x = (x + XDim) % XDim; z = (z + ZDim) % ZDim; BlockInfo info = src.GetBlockInfo(x, y, z); int light = src.GetBlockLight(x, y, z); return Math.Max(light, info.Luminance); } } }