using System; using System.Collections.Generic; using Substrate.Utility; using Substrate.NBT; namespace Substrate { public struct BlockData { public XZYByteArray blocks; public BlockData (XZYByteArray b) { blocks = b; } } public struct BlockDataData { public XZYNibbleArray data; public BlockDataData (XZYNibbleArray d) { data = d; } } public struct BlockLightData { public XZYNibbleArray light; public XZYNibbleArray skyLight; public ZXByteArray heightMap; public BlockLightData (XZYNibbleArray l, XZYNibbleArray s, ZXByteArray h) { light = l; skyLight = s; heightMap = h; } } public struct BlockPropertyData { public TagList tileEntities; public BlockPropertyData (TagList t) { tileEntities = t; } } public class AlphaBlockCollection : IBoundedAlphaBlockCollection, IEnumerable { private readonly int _xdim; private readonly int _ydim; private readonly int _zdim; private XZYByteArray _blocks; private XZYNibbleArray _data; private XZYNibbleArray _blockLight; private XZYNibbleArray _skyLight; private ZXByteArray _heightMap; private TagList _tileEntities; private BlockLight _lightManager; private BlockFluid _fluidManager; private BlockTileEntities _tileEntityManager; private bool _dirty = false; private bool _autoLight = true; private bool _autoFluid = false; public delegate AlphaBlockCollection NeighborLookupHandler (int relx, int rely, int relz); public event NeighborLookupHandler ResolveNeighbor { add { _lightManager.ResolveNeighbor += delegate(int relx, int rely, int relz) { return value(relx, rely, relz); }; _fluidManager.ResolveNeighbor += delegate(int relx, int rely, int relz) { return value(relx, rely, relz); }; } remove { _lightManager = new BlockLight(this); _fluidManager = new BlockFluid(this); } } public event BlockCoordinateHandler TranslateCoordinates { add { _tileEntityManager.TranslateCoordinates += value; } remove { _tileEntityManager.TranslateCoordinates -= value; } } public bool AutoLight { get { return _autoLight; } set { _autoLight = value; } } public bool AutoFluid { get { return _autoFluid; } set { _autoFluid = value; } } public bool IsDirty { get { return _dirty; } set { _dirty = value; } } public AlphaBlockCollection (BlockData root, BlockDataData data, BlockLightData light, BlockPropertyData properties) { _blocks = root.blocks; _data = data.data; _blockLight = light.light; _skyLight = light.skyLight; _heightMap = light.heightMap; _tileEntities = properties.tileEntities; _lightManager = new BlockLight(this); _fluidManager = new BlockFluid(this); _tileEntityManager = new BlockTileEntities(_blocks, _tileEntities); _xdim = _blocks.XDim; _ydim = _blocks.YDim; _zdim = _blocks.ZDim; } public Block GetBlock (int x, int y, int z) { return new Block(this, x, y, z); } public BlockRef GetBlockRef (int x, int y, int z) { return new BlockRef(this, x, y, z); } public void SetBlock (int x, int y, int z, Block block) { SetID(x, y, z, block.ID); SetData(x, y, z, block.Data); SetTileEntity(x, y, z, block.GetTileEntity().Copy()); } #region IBlockCollection Members public int XDim { get { return _xdim; } } public int YDim { get { return _ydim; } } public int ZDim { get { return _zdim; } } IBlock IBlockCollection.GetBlock (int x, int y, int z) { return GetBlock(x, y, z); } IBlock IBlockCollection.GetBlockRef (int x, int y, int z) { return GetBlockRef(x, y, z); } public void SetBlock (int x, int y, int z, IBlock block) { SetID(x, y, z, block.ID); } public BlockInfo GetInfo (int x, int y, int z) { return BlockInfo.BlockTable[_blocks[x, y, z]]; } internal BlockInfo GetInfo (int index) { return BlockInfo.BlockTable[_blocks[index]]; } public int GetID (int x, int y, int z) { return _blocks[x, y, z]; } internal int GetID (int index) { return _blocks[index]; } public void SetID (int x, int y, int z, int id) { int oldid = _blocks[x, y, z]; if (oldid == id) { return; } // Update value _blocks[x, y, z] = (byte)id; // Update tile entities BlockInfo info1 = BlockInfo.BlockTable[oldid]; BlockInfo info2 = BlockInfo.BlockTable[id]; BlockInfoEx einfo1 = info1 as BlockInfoEx; BlockInfoEx einfo2 = info2 as BlockInfoEx; if (einfo1 != einfo2) { if (einfo1 != null) { ClearTileEntity(x, y, z); } if (einfo2 != null) { CreateTileEntity(x, y, z); } } // Light consistency if (_autoLight) { if (info1.ObscuresLight != info2.ObscuresLight) { _lightManager.UpdateHeightMap(x, y, z); } if (info1.Luminance != info2.Luminance || info1.Opacity != info2.Opacity || info1.TransmitsLight != info2.TransmitsLight) { UpdateBlockLight(x, y, z); } if (info1.Opacity != info2.Opacity || info1.TransmitsLight != info2.TransmitsLight) { UpdateSkyLight(x, y, z); } } // Fluid consistency if (_autoFluid) { if (info1.State == BlockState.FLUID || info2.State == BlockState.FLUID) { UpdateFluid(x, y, z); } } _dirty = true; } public int CountByID (int id) { int c = 0; for (int i = 0; i < _blocks.Length; i++) { if (_blocks[i] == id) { c++; } } return c; } #endregion #region IDataBlockContainer Members IDataBlock IDataBlockCollection.GetBlock (int x, int y, int z) { return GetBlock(x, y, z); } IDataBlock IDataBlockCollection.GetBlockRef (int x, int y, int z) { return GetBlockRef(x, y, z); } public void SetBlock (int x, int y, int z, IDataBlock block) { SetID(x, y, z, block.ID); SetData(x, y, z, block.Data); } public int GetData (int x, int y, int z) { return _data[x, y, z]; } internal int GetData (int index) { return _data[index]; } public void SetData (int x, int y, int z, int data) { if (_data[x, y, z] != data) { _data[x, y, z] = (byte)data; _dirty = true; } /*if (BlockManager.EnforceDataLimits && BlockInfo.BlockTable[_blocks[index]] != null) { if (!BlockInfo.BlockTable[_blocks[index]].TestData(data)) { return false; } }*/ } internal void SetData (int index, int data) { if (_data[index] != data) { _data[index] = (byte)data; _dirty = true; } } public int CountByData (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; } #endregion #region ILitBlockCollection Members ILitBlock ILitBlockCollection.GetBlock (int x, int y, int z) { throw new NotImplementedException(); } ILitBlock ILitBlockCollection.GetBlockRef (int x, int y, int z) { return GetBlockRef(x, y, z); } public void SetBlock (int x, int y, int z, ILitBlock block) { SetID(x, y, z, block.ID); SetBlockLight(x, y, z, block.BlockLight); SetSkyLight(x, y, z, block.SkyLight); } public int GetBlockLight (int x, int y, int z) { return _blockLight[x, y, z]; } internal int GetBlockLight (int index) { return _blockLight[index]; } public int GetSkyLight (int x, int y, int z) { return _skyLight[x, y, z]; } internal int GetSkyLight (int index) { return _skyLight[index]; } public void SetBlockLight (int x, int y, int z, int light) { if (_blockLight[x, y, z] != light) { _blockLight[x, y, z] = (byte)light; _dirty = true; } } internal void SetBlockLight (int index, int light) { if (_blockLight[index] != light) { _blockLight[index] = (byte)light; _dirty = true; } } public void SetSkyLight (int x, int y, int z, int light) { if (_skyLight[x, y, z] != light) { _skyLight[x, y, z] = (byte)light; _dirty = true; } } internal void SetSkyLight (int index, int light) { if (_skyLight[index] != light) { _skyLight[index] = (byte)light; _dirty = true; } } public int GetHeight (int x, int z) { return _heightMap[x, z]; } public void SetHeight (int x, int z, int height) { _heightMap[x, z] = (byte)height; } public void UpdateBlockLight (int x, int y, int z) { _lightManager.UpdateBlockLight(x, y, z); _dirty = true; } public void UpdateSkyLight (int x, int y, int z) { _lightManager.UpdateBlockSkyLight(x, y, z); _dirty = true; } public void ResetBlockLight () { _blockLight.Clear(); _dirty = true; } public void ResetSkyLight () { _skyLight.Clear(); _dirty = true; } public void RebuildBlockLight () { _lightManager.RebuildBlockLight(); _dirty = true; } public void RebuildSkyLight () { _lightManager.RebuildBlockSkyLight(); _dirty = true; } public void RebuildHeightMap () { _lightManager.RebuildHeightMap(); _dirty = true; } public void StitchBlockLight () { _lightManager.StitchBlockLight(); _dirty = true; } public void StitchSkyLight () { _lightManager.StitchBlockSkyLight(); _dirty = true; } public void StitchBlockLight (IBoundedLitBlockCollection blockset, BlockCollectionEdge edge) { _lightManager.StitchBlockLight(blockset, edge); _dirty = true; } public void StitchSkyLight (IBoundedLitBlockCollection blockset, BlockCollectionEdge edge) { _lightManager.StitchBlockSkyLight(blockset, edge); _dirty = true; } #endregion #region IPropertyBlockCollection Members IPropertyBlock IPropertyBlockCollection.GetBlock (int x, int y, int z) { return GetBlock(x, y, z); } IPropertyBlock IPropertyBlockCollection.GetBlockRef (int x, int y, int z) { return GetBlockRef(x, y, z); } public void SetBlock (int x, int y, int z, IPropertyBlock block) { SetID(x, y, z, block.ID); SetTileEntity(x, y, z, block.GetTileEntity().Copy()); } public TileEntity GetTileEntity (int x, int y, int z) { return _tileEntityManager.GetTileEntity(x, y, z); } public void SetTileEntity (int x, int y, int z, TileEntity te) { _tileEntityManager.SetTileEntity(x, y, z, te); _dirty = true; } public void CreateTileEntity (int x, int y, int z) { _tileEntityManager.CreateTileEntity(x, y, z); _dirty = true; } public void ClearTileEntity (int x, int y, int z) { _tileEntityManager.ClearTileEntity(x, y, z); _dirty = true; } #endregion public void ResetFluid () { _fluidManager.ResetWater(_blocks, _data); _fluidManager.ResetLava(_blocks, _data); _dirty = true; } public void RebuildFluid () { _fluidManager.RebuildWater(); _fluidManager.RebuildLava(); _dirty = true; } public void UpdateFluid (int x, int y, int z) { bool autofluid = _autoFluid; _autoFluid = false; int blocktype = _blocks[x, y, z]; if (blocktype == BlockType.WATER || blocktype == BlockType.STATIONARY_WATER) { _fluidManager.UpdateWater(x, y, z); _dirty = true; } else if (blocktype == BlockType.LAVA || blocktype == BlockType.STATIONARY_LAVA) { _fluidManager.UpdateLava(x, y, z); _dirty = true; } _autoFluid = autofluid; } #region IEnumerable Members public IEnumerator GetEnumerator () { return new AlphaBlockEnumerator(this); } #endregion #region IEnumerable Members System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator () { return new AlphaBlockEnumerator(this); } #endregion public class AlphaBlockEnumerator : IEnumerator { private AlphaBlockCollection _collection; private int _index; private int _size; public AlphaBlockEnumerator (AlphaBlockCollection collection) { _collection = collection; _index = -1; _size = collection.XDim * collection.YDim * collection.ZDim; } #region IEnumerator Members public AlphaBlockRef Current { get { if (_index == -1 || _index == _size) { throw new InvalidOperationException(); } return new AlphaBlockRef(_collection, _index); } } #endregion #region IDisposable Members public void Dispose () { } #endregion #region IEnumerator Members object System.Collections.IEnumerator.Current { get { return Current; } } public bool MoveNext () { if (++_index == _size) { return false; } return true; } public void Reset () { _index = -1; } #endregion } } }