Major Block interface factoring to ease support of alternative block containers such as Classic Meps or MCSchematic files.

This commit is contained in:
Justin Aquadro 2011-04-08 02:16:06 +00:00
parent 134e979323
commit 1e029325bc
9 changed files with 993 additions and 165 deletions

View file

@ -7,7 +7,123 @@ namespace Substrate
using NBT; using NBT;
using Utility; using Utility;
public class Block : IBlock, ICopyable<Block> public class Block : IPropertyBlock, ICopyable<Block>
{
private int _id;
private int _data;
private TileEntity _tileEntity;
public Block (int id)
{
_id = id;
}
public Block (int id, int data)
{
_id = id;
_data = data;
}
public Block (IChunk chunk, int lx, int ly, int lz)
{
_id = chunk.GetBlockID(lx, ly, lz);
_data = chunk.GetBlockData(lx, ly, lz);
_tileEntity = chunk.GetTileEntity(lx, ly, lz).Copy();
}
#region IBlock Members
public BlockInfo Info
{
get { return BlockInfo.BlockTable[_id]; }
}
public int ID
{
get { return _id; }
set
{
BlockInfoEx info1 = BlockInfo.BlockTable[_id] as BlockInfoEx;
BlockInfoEx info2 = BlockInfo.BlockTable[value] as BlockInfoEx;
if (info1 != info2) {
if (info1 != null) {
_tileEntity = null;
}
if (info2 != null) {
_tileEntity = TileEntityFactory.Create(info2.TileEntityName);
}
}
_id = value;
}
}
public int Data
{
get { return _data; }
set
{
/*if (BlockManager.EnforceDataLimits && BlockInfo.BlockTable[_id] != null) {
if (!BlockInfo.BlockTable[_id].TestData(value)) {
return;
}
}*/
_data = value;
}
}
#endregion
#region IPropertyBlock Members
public TileEntity GetTileEntity ()
{
return _tileEntity;
}
public bool SetTileEntity (TileEntity te)
{
BlockInfoEx info = BlockInfo.BlockTable[_id] as BlockInfoEx;
if (info == null) {
return false;
}
if (te.GetType() != TileEntityFactory.Lookup(info.TileEntityName)) {
return false;
}
_tileEntity = te;
return true;
}
public bool ClearTileEntity ()
{
_tileEntity = null;
return true;
}
#endregion
#region ICopyable<Block> Members
public Block Copy ()
{
Block block = new Block(_id, _data);
block._tileEntity = _tileEntity.Copy();
return block;
}
#endregion
}
/*public class Block : IBlock, ICopyable<Block>
{ {
private int _id; private int _id;
private int _data; private int _data;
@ -128,5 +244,5 @@ namespace Substrate
} }
#endregion #endregion
} }*/
} }

View file

@ -4,8 +4,91 @@ using System.Text;
namespace Substrate namespace Substrate
{ {
public interface IBlock public interface IBlock
{
BlockInfo Info { get; }
int ID { get; set; }
int Data { get; set; }
}
public interface ILitBlock : IBlock
{
int BlockLight { get; set; }
int SkyLight { get; set; }
}
public interface IPropertyBlock : IBlock
{
TileEntity GetTileEntity ();
bool SetTileEntity (TileEntity te);
bool ClearTileEntity ();
}
public interface IBlockContainer
{
IBlock GetBlock (int x, int y, int z);
IBlock GetBlockRef (int x, int y, int z);
void SetBlock (int x, int y, int z, IBlock block);
BlockInfo GetBlockInfo (int x, int y, int z);
int GetBlockID (int x, int y, int z);
int GetBlockData (int x, int y, int z);
bool SetBlockID (int x, int y, int z, int id);
bool SetBlockData (int x, int y, int z, int data);
}
public interface IBoundedBlockContainer : IBlockContainer
{
int XDim { get; }
int YDim { get; }
int ZDim { get; }
}
public interface IResizableBlockContainer : IBoundedBlockContainer
{
new int XDim { get; set; }
new int YDim { get; set; }
new int ZDim { get; set; }
}
public interface ILitBlockContainer : IBlockContainer
{
new ILitBlock GetBlock (int x, int y, int z);
new ILitBlock GetBlockRef (int x, int y, int z);
void SetBlock (int x, int y, int z, ILitBlock block);
int GetBlockLight (int x, int y, int z);
int GetBlockSkyLight (int x, int y, int z);
bool SetBlockLight (int x, int y, int z, int light);
bool SetBlockSkyLight (int x, int y, int z, int light);
}
public interface IPropertyBlockContainer : IBlockContainer
{
new IPropertyBlock GetBlock (int x, int y, int z);
new IPropertyBlock GetBlockRef (int x, int y, int z);
void SetBlock (int x, int y, int z, IPropertyBlock block);
TileEntity GetTileEntity (int x, int y, int z);
bool SetTileEntity (int x, int y, int z, TileEntity te);
bool ClearTileEntity (int x, int y, int z);
}
public interface IAlphaBlockContainer : ILitBlockContainer, IPropertyBlockContainer
{
}
public interface IBlockManager : IAlphaBlockContainer
{
}
/*public interface IBlock
{ {
BlockInfo Info { get; } BlockInfo Info { get; }
int ID { get; set; } int ID { get; set; }
@ -53,5 +136,5 @@ namespace Substrate
public interface IBlockManager : IBlockContainer public interface IBlockManager : IBlockContainer
{ {
} }*/
} }

View file

@ -4,6 +4,286 @@ using System.Text;
namespace Substrate namespace Substrate
{ {
public class BlockManager : IBlockManager
{
public const int MIN_X = -32000000;
public const int MAX_X = 32000000;
public const int MIN_Y = 0;
public const int MAX_Y = 128;
public const int MIN_Z = -32000000;
public const int MAX_Z = 32000000;
protected int _chunkXDim;
protected int _chunkYDim;
protected int _chunkZDim;
protected int _chunkXMask;
protected int _chunkYMask;
protected int _chunkZMask;
protected int _chunkXLog;
protected int _chunkYLog;
protected int _chunkZLog;
protected IChunkManager _chunkMan;
protected ChunkRef _cache;
public BlockManager (IChunkManager cm)
{
_chunkMan = cm;
Chunk c = new Chunk(0, 0);
_chunkXDim = c.XDim;
_chunkYDim = c.YDim;
_chunkZDim = c.ZDim;
_chunkXMask = _chunkXDim - 1;
_chunkYMask = _chunkYDim - 1;
_chunkZMask = _chunkZDim - 1;
_chunkXLog = Log2(_chunkXDim);
_chunkYLog = Log2(_chunkYDim);
_chunkZLog = Log2(_chunkZDim);
}
public Block GetBlock (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return null;
}
return _cache.GetBlock(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public BlockRef GetBlockRef (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return null;
}
return _cache.GetBlockRef(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public void SetBlock (int x, int y, int z, Block block)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return;
}
_cache.SetBlock(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask, block);
}
protected ChunkRef GetChunk (int x, int y, int z)
{
x >>= _chunkXLog;
z >>= _chunkZLog;
return _chunkMan.GetChunkRef(x, z);
}
private int Log2 (int x)
{
int c = 0;
while (x > 1) {
x >>= 1;
c++;
}
return c;
}
/// <summary>
/// Called by other block-specific 'get' and 'set' functions to filter
/// out operations on some blocks. Override this method in derrived
/// classes to filter the entire BlockManager.
/// </summary>
protected virtual bool Check (int x, int y, int z)
{
return (x >= MIN_X) && (x < MAX_X) &&
(y >= MIN_Y) && (y < MAX_Y) &&
(z >= MIN_Z) && (z < MAX_Z);
}
#region IBlockContainer Members
IBlock IBlockContainer.GetBlock (int x, int y, int z)
{
return GetBlock(x, y, z);
}
IBlock IBlockContainer.GetBlockRef (int x, int y, int z)
{
return GetBlockRef(x, y, z);
}
public void SetBlock (int x, int y, int z, IBlock block)
{
_cache.SetBlock(x, y, z, block);
}
public BlockInfo GetBlockInfo (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return null;
}
return _cache.GetBlockInfo(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public int GetBlockID (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null) {
return 0;
}
return _cache.GetBlockID(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public int GetBlockData (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null) {
return 0;
}
return _cache.GetBlockData(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public bool SetBlockID (int x, int y, int z, int id)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return false;
}
return _cache.SetBlockID(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask, id);
}
public bool SetBlockData (int x, int y, int z, int data)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return false;
}
return _cache.SetBlockData(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask, data);
}
#endregion
#region ILitBlockContainer Members
ILitBlock ILitBlockContainer.GetBlock (int x, int y, int z)
{
throw new NotImplementedException();
}
ILitBlock ILitBlockContainer.GetBlockRef (int x, int y, int z)
{
return GetBlockRef(x, y, z);
}
public void SetBlock (int x, int y, int z, ILitBlock block)
{
_cache.SetBlock(x, y, z, block);
}
public int GetBlockLight (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null) {
return 0;
}
return _cache.GetBlockLight(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public int GetBlockSkyLight (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null) {
return 0;
}
return _cache.GetBlockSkyLight(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public bool SetBlockLight (int x, int y, int z, int light)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return false;
}
return _cache.SetBlockID(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask, light);
}
public bool SetBlockSkyLight (int x, int y, int z, int light)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return false;
}
return _cache.SetBlockSkyLight(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask, light);
}
#endregion
#region IPropertyBlockContainer Members
IPropertyBlock IPropertyBlockContainer.GetBlock (int x, int y, int z)
{
return GetBlock(x, y, z);
}
IPropertyBlock IPropertyBlockContainer.GetBlockRef (int x, int y, int z)
{
return GetBlockRef(x, y, z);
}
public void SetBlock (int x, int y, int z, IPropertyBlock block)
{
_cache.SetBlock(x, y, z, block);
}
public TileEntity GetTileEntity (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return null;
}
return _cache.GetTileEntity(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
public bool SetTileEntity (int x, int y, int z, TileEntity te)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return false;
}
return _cache.SetTileEntity(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask, te);
}
public bool ClearTileEntity (int x, int y, int z)
{
_cache = GetChunk(x, y, z);
if (_cache == null || !Check(x, y, z)) {
return false;
}
return _cache.ClearTileEntity(x & _chunkXMask, y & _chunkYMask, z & _chunkZMask);
}
#endregion
}
/*public interface IBlockManager : IBlockContainer /*public interface IBlockManager : IBlockContainer
{ {
Block GetBlock (int x, int y, int z); Block GetBlock (int x, int y, int z);
@ -13,7 +293,7 @@ namespace Substrate
bool SetBlock (int x, int y, int z, Block block); bool SetBlock (int x, int y, int z, Block block);
}*/ }*/
public class BlockManager : IBlockManager /*public class BlockManager : IBlockManager
{ {
public const int MIN_X = -32000000; public const int MIN_X = -32000000;
public const int MAX_X = 32000000; public const int MAX_X = 32000000;
@ -247,5 +527,5 @@ namespace Substrate
(y >= MIN_Y) && (y < MAX_Y) && (y >= MIN_Y) && (y < MAX_Y) &&
(z >= MIN_Z) && (z < MAX_Z); (z >= MIN_Z) && (z < MAX_Z);
} }
} }*/
} }

View file

@ -4,7 +4,114 @@ using System.Text;
namespace Substrate namespace Substrate
{ {
public class BlockRef : IBlock public class BlockRef : IPropertyBlock, ILitBlock
{
protected IAlphaBlockContainer _container;
protected int _x;
protected int _y;
protected int _z;
/*public int X
{
get { return _container.BlockGlobalX(_x); }
}
public int Y
{
get { return _container.BlockGlobalY(_y); }
}
public int Z
{
get { return _container.BlockGlobalZ(_z); }
}
public int LocalX
{
get { return _container.BlockLocalX(_x); }
}
public int LocalY
{
get { return _container.BlockLocalZ(_z); }
}
public int LocalZ
{
get { return _z; }
}*/
public BlockRef (IAlphaBlockContainer container, int x, int y, int z)
{
_container = container;
_x = x;
_y = y;
_z = z;
}
#region IBlock Members
public BlockInfo Info
{
get { return BlockInfo.BlockTable[_container.GetBlockID(_x, _y, _z)]; }
}
public int ID
{
get { return _container.GetBlockID(_x, _y, _z); }
set { _container.SetBlockID(_x, _y, _z, value); }
}
public int Data
{
get { return _container.GetBlockData(_x, _y, _z); }
set { _container.SetBlockData(_x, _y, _z, value); }
}
#endregion
#region ILitBlock Members
public int BlockLight
{
get { return _container.GetBlockLight(_x, _y, _z); }
set { _container.SetBlockLight(_x, _y, _z, value); }
}
public int SkyLight
{
get { return _container.GetBlockSkyLight(_x, _y, _z); }
set { _container.SetBlockSkyLight(_x, _y, _z, value); }
}
#endregion
#region IPropertyBlock Members
public TileEntity GetTileEntity ()
{
return _container.GetTileEntity(_x, _y, _z);
}
public bool SetTileEntity (TileEntity te)
{
return _container.SetTileEntity(_x, _y, _z, te);
}
public bool ClearTileEntity ()
{
return _container.ClearTileEntity(_x, _y, _z);
}
#endregion
}
/*public class BlockRef : IBlock
{ {
protected IBlockContainer _container; protected IBlockContainer _container;
@ -103,7 +210,7 @@ namespace Substrate
{ {
return _container.ClearTileEntity(_x, _y, _z); return _container.ClearTileEntity(_x, _y, _z);
} }
} }*/
/*public class BlockRef : IBlock /*public class BlockRef : IBlock
{ {

View file

@ -83,8 +83,8 @@ namespace Substrate
private void BuildNBTTree () private void BuildNBTTree ()
{ {
int elements2 = BlockManager.CHUNK_XLEN * BlockManager.CHUNK_ZLEN; int elements2 = XDim * ZDim;
int elements3 = elements2 * BlockManager.CHUNK_YLEN; int elements3 = elements2 *YDim;
_blocks = new NBT_ByteArray(new byte[elements3]); _blocks = new NBT_ByteArray(new byte[elements3]);
NBT_ByteArray data = new NBT_ByteArray(new byte[elements3 >> 1]); NBT_ByteArray data = new NBT_ByteArray(new byte[elements3 >> 1]);
@ -117,7 +117,7 @@ namespace Substrate
public int BlockGlobalX (int x) public int BlockGlobalX (int x)
{ {
return _cx * BlockManager.CHUNK_XLEN + x; return _cx * XDim + x;
} }
public int BlockGlobalY (int y) public int BlockGlobalY (int y)
@ -127,7 +127,7 @@ namespace Substrate
public int BlockGlobalZ (int z) public int BlockGlobalZ (int z)
{ {
return _cz * BlockManager.CHUNK_ZLEN + z; return _cz * XDim + z;
} }
public int BlockLocalX (int x) public int BlockLocalX (int x)
@ -167,11 +167,6 @@ namespace Substrate
return new BlockRef(this, lx, ly, lz); return new BlockRef(this, lx, ly, lz);
} }
public BlockInfo GetBlockInfo (int lx, int ly, int lz)
{
return BlockInfo.BlockTable[GetBlockID(lx, ly, lz)];
}
public void SetBlock (int lx, int ly, int lz, Block block) public void SetBlock (int lx, int ly, int lz, Block block)
{ {
int index = lx << 11 | lz << 7 | ly; int index = lx << 11 | lz << 7 | ly;
@ -179,12 +174,149 @@ namespace Substrate
SetBlockID(lx, ly, lz, block.ID); SetBlockID(lx, ly, lz, block.ID);
SetBlockData(lx, ly, lz, block.Data); SetBlockData(lx, ly, lz, block.Data);
_blockLight[index] = block.BlockLight;
_skyLight[index] = block.SkyLight;
SetTileEntity(lx, ly, lz, block.GetTileEntity().Copy()); 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<BlockKey, NBT_Compound>();
foreach (NBT_Compound te in _tileEntities) {
int tex = te["x"].ToNBTInt();
int tey = te["y"].ToNBTInt();
int tez = te["z"].ToNBTInt();
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) public int GetBlockID (int lx, int ly, int lz)
{ {
return _blocks.Data[lx << 11 | lz << 7 | ly]; return _blocks.Data[lx << 11 | lz << 7 | ly];
@ -195,16 +327,6 @@ namespace Substrate
return _data[lx << 11 | lz << 7 | ly]; return _data[lx << 11 | lz << 7 | ly];
} }
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 SetBlockID (int lx, int ly, int lz, int id) public bool SetBlockID (int lx, int ly, int lz, int id)
{ {
int index = lx << 11 | lz << 7 | ly; int index = lx << 11 | lz << 7 | ly;
@ -262,16 +384,65 @@ namespace Substrate
return false; return false;
} }
if (BlockManager.EnforceDataLimits && BlockInfo.BlockTable[_blocks[index]] != null) { /*if (BlockManager.EnforceDataLimits && BlockInfo.BlockTable[_blocks[index]] != null) {
if (!BlockInfo.BlockTable[_blocks[index]].TestData(data)) { if (!BlockInfo.BlockTable[_blocks[index]].TestData(data)) {
return false; return false;
} }
} }*/
_data[index] = data; _data[index] = data;
return true; 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) public bool SetBlockLight (int lx, int ly, int lz, int light)
{ {
int index = lx << 11 | lz << 7 | ly; int index = lx << 11 | lz << 7 | ly;
@ -294,57 +465,29 @@ namespace Substrate
return true; return true;
} }
public int CountBlockID (int id) #endregion
{
int c = 0;
for (int i = 0; i < _blocks.Length; i++) {
if (_blocks[i] == id) {
c++;
}
}
return c;
#region IPropertyBlockContainer Members
IPropertyBlock IPropertyBlockContainer.GetBlock (int lx, int ly, int lz)
{
return GetBlock(lx, ly, lz);
} }
public int CountBlockData (int id, int data) IPropertyBlock IPropertyBlockContainer.GetBlockRef (int lx, int ly, int lz)
{ {
int c = 0; return GetBlockRef(lx, ly, lz);
for (int i = 0; i < _blocks.Length; i++) {
if (_blocks[i] == id && _data[i] == data) {
c++;
}
}
return c;
} }
public int CountEntities () public void SetBlock (int lx, int ly, int lz, IPropertyBlock block)
{ {
return _entities.Count; int index = lx << 11 | lz << 7 | ly;
}
public int GetHeight (int lx, int lz) SetBlockID(lx, ly, lz, block.ID);
{ SetBlockData(lx, ly, lz, block.Data);
return _heightMap[lz << 4 | lx];
}
private void CreateTileEntity (int lx, int ly, int lz) SetTileEntity(lx, ly, lz, block.GetTileEntity().Copy());
{
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 TileEntity GetTileEntity (int lx, int ly, int lz) public TileEntity GetTileEntity (int lx, int ly, int lz)
@ -416,30 +559,8 @@ namespace Substrate
return true; return true;
} }
public virtual void SetLocation (int x, int z) #endregion
{
int diffx = x - _cx;
int diffz = z - _cz;
_cx = x;
_cz = z;
BuildTileEntityCache();
}
private void BuildTileEntityCache ()
{
_tileEntityTable = new Dictionary<BlockKey, NBT_Compound>();
foreach (NBT_Compound te in _tileEntities) {
int tex = te["x"].ToNBTInt();
int tey = te["y"].ToNBTInt();
int tez = te["z"].ToNBTInt();
BlockKey key = new BlockKey(tex, tey, tez);
_tileEntityTable[key] = te;
}
}
#region ICopyable<Chunk> Members #region ICopyable<Chunk> Members
@ -559,10 +680,10 @@ namespace Substrate
public bool AddEntity (Entity ent) public bool AddEntity (Entity ent)
{ {
double xlow = _cx * BlockManager.CHUNK_XLEN; double xlow = _cx * XDim;
double xhigh = xlow + BlockManager.CHUNK_XLEN; double xhigh = xlow + XDim;
double zlow = _cz * BlockManager.CHUNK_ZLEN; double zlow = _cz * ZDim;
double zhigh = zlow + BlockManager.CHUNK_ZLEN; double zhigh = zlow + ZDim;
Entity.Vector3 pos = ent.Position; Entity.Vector3 pos = ent.Position;
if (!(pos.X >= xlow && pos.X < xhigh && pos.Z >= zlow && pos.Z < zhigh)) { if (!(pos.X >= xlow && pos.X < xhigh && pos.Z >= zlow && pos.Z < zhigh)) {

View file

@ -6,7 +6,7 @@ using System.IO;
namespace Substrate namespace Substrate
{ {
public interface IChunk : IBlockContainer, IEntityContainer public interface IChunk : IBoundedBlockContainer, IAlphaBlockContainer, IEntityContainer
{ {
int X { get; } int X { get; }
int Z { get; } int Z { get; }

View file

@ -28,6 +28,8 @@ namespace Substrate
get { return _container.ChunkGlobalZ(_cz); } get { return _container.ChunkGlobalZ(_cz); }
} }
public int LocalX public int LocalX
{ {
get { return _container.ChunkLocalX(_cx); } get { return _container.ChunkLocalX(_cx); }
@ -52,7 +54,7 @@ namespace Substrate
public int BlockGlobalX (int x) public int BlockGlobalX (int x)
{ {
return _container.ChunkGlobalX(_cx) * BlockManager.CHUNK_XLEN + x; return _container.ChunkGlobalX(_cx) * XDim + x;
} }
public int BlockGlobalY (int y) public int BlockGlobalY (int y)
@ -62,7 +64,7 @@ namespace Substrate
public int BlockGlobalZ (int z) public int BlockGlobalZ (int z)
{ {
return _container.ChunkGlobalZ(_cz) * BlockManager.CHUNK_ZLEN + z; return _container.ChunkGlobalZ(_cz) * ZDim + z;
} }
public int BlockLocalX (int x) public int BlockLocalX (int x)
@ -185,62 +187,6 @@ namespace Substrate
GetChunk().SetBlock(lx, ly, lz, 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 int GetBlockLight (int lx, int ly, int lz)
{
return GetChunk().GetBlockSkyLight(lx, ly, lz);
}
public int GetBlockSkyLight (int lx, int ly, int lz)
{
return GetChunk().GetBlockSkyLight(lx, ly, lz);
}
public bool SetBlockID (int lx, int ly, int lz, int id)
{
if (GetChunk().SetBlockID(lx, ly, lz, id)) {
MarkDirty();
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;
}
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;
}
public int CountBlockID (int id) public int CountBlockID (int id)
{ {
return GetChunk().CountBlockID(id); return GetChunk().CountBlockID(id);
@ -261,14 +207,150 @@ namespace Substrate
return GetChunk().GetHeight(lx, lz); return GetChunk().GetHeight(lx, lz);
} }
#endregion
#region IBoundedBlockContainer Members
public int XDim
{
get { return _chunk.XDim; }
}
public int YDim
{
get { return _chunk.YDim; }
}
public int ZDim
{
get { return _chunk.ZDim; }
}
#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)
{
if (GetChunk().SetBlockID(lx, ly, lz, id)) {
MarkDirty();
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().GetBlockSkyLight(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) public TileEntity GetTileEntity (int lx, int ly, int lz)
{ {
return GetChunk().GetTileEntity(lx, ly, lz); return GetChunk().GetTileEntity(lx, ly, lz);
} }
public bool SetTileEntity (int lx, int ly, int lz, TileEntity te) public bool SetTileEntity (int lx, int ly, int lz, TileEntity te)
{ {
if (GetChunk().SetTileEntity(lx, ly, lz, te)) { if (GetChunk().SetTileEntity(lx, ly, lz, te)) {
MarkDirty(); MarkDirty();
return true; return true;
} }
@ -277,7 +359,7 @@ namespace Substrate
public bool ClearTileEntity (int lx, int ly, int lz) public bool ClearTileEntity (int lx, int ly, int lz)
{ {
if (GetChunk().ClearTileEntity(lx, ly, lz)) { if (GetChunk().ClearTileEntity(lx, ly, lz)) {
MarkDirty(); MarkDirty();
return true; return true;
} }

View file

@ -63,6 +63,9 @@
<Reference Include="System.Xml" /> <Reference Include="System.Xml" />
</ItemGroup> </ItemGroup>
<ItemGroup> <ItemGroup>
<Compile Include="Source\Level.cs" />
<Compile Include="Source\PlayerManager.cs" />
<Compile Include="Source\PlayerFile.cs" />
<Compile Include="Properties\AssemblyInfo.cs" /> <Compile Include="Properties\AssemblyInfo.cs" />
<Compile Include="Source\Block.cs" /> <Compile Include="Source\Block.cs" />
<Compile Include="Source\BlockInfo.cs" /> <Compile Include="Source\BlockInfo.cs" />
@ -78,18 +81,54 @@
<Compile Include="Source\ChunkKey.cs" /> <Compile Include="Source\ChunkKey.cs" />
<Compile Include="Source\ChunkManager.cs" /> <Compile Include="Source\ChunkManager.cs" />
<Compile Include="Source\ChunkRef.cs" /> <Compile Include="Source\ChunkRef.cs" />
<Compile Include="Source\Entities\EntityArrow.cs" />
<Compile Include="Source\Entities\EntityBoat.cs" />
<Compile Include="Source\Entities\EntityChicken.cs" />
<Compile Include="Source\Entities\EntityCow.cs" />
<Compile Include="Source\Entities\EntityCreeper.cs" />
<Compile Include="Source\Entities\EntityEgg.cs" />
<Compile Include="Source\Entities\EntityFallingSand.cs" />
<Compile Include="Source\Entities\EntityGhast.cs" />
<Compile Include="Source\Entities\EntityGiant.cs" />
<Compile Include="Source\Entities\EntityItem.cs" />
<Compile Include="Source\Entities\EntityMinecart.cs" />
<Compile Include="Source\Entities\EntityMinecartChest.cs" />
<Compile Include="Source\Entities\EntityMinecartFurnace.cs" />
<Compile Include="Source\Entities\EntityMob.cs" />
<Compile Include="Source\Entities\EntityMonster.cs" />
<Compile Include="Source\Entities\EntityPainting.cs" />
<Compile Include="Source\Entities\EntityPig.cs" />
<Compile Include="Source\Entities\EntityPigZombie.cs" />
<Compile Include="Source\Entities\EntityPrimedTnt.cs" />
<Compile Include="Source\Entities\EntitySheep.cs" />
<Compile Include="Source\Entities\EntitySkeleton.cs" />
<Compile Include="Source\Entities\EntitySlime.cs" />
<Compile Include="Source\Entities\EntitySnowball.cs" />
<Compile Include="Source\Entities\EntitySpider.cs" />
<Compile Include="Source\Entities\EntityThrowable.cs" />
<Compile Include="Source\Entities\EntityWolf.cs" />
<Compile Include="Source\Entities\EntityZombie.cs" />
<Compile Include="Source\Entity.cs" /> <Compile Include="Source\Entity.cs" />
<Compile Include="Source\EntityFactory.cs" />
<Compile Include="Source\Item.cs" /> <Compile Include="Source\Item.cs" />
<Compile Include="Source\NBTFile.cs" />
<Compile Include="Source\NBT\JSONSerializer.cs" /> <Compile Include="Source\NBT\JSONSerializer.cs" />
<Compile Include="Source\NBT\NBT.cs" /> <Compile Include="Source\NBT\NBT.cs" />
<Compile Include="Source\NBT\NBTSchema.cs" /> <Compile Include="Source\NBT\NBTSchema.cs" />
<Compile Include="Source\NBT\NBTValues.cs" /> <Compile Include="Source\NBT\NBTValues.cs" />
<Compile Include="Source\NBT\NBTVerifier.cs" /> <Compile Include="Source\NBT\NBTVerifier.cs" />
<Compile Include="Source\Player.cs" />
<Compile Include="Source\Region.cs" /> <Compile Include="Source\Region.cs" />
<Compile Include="Source\RegionEnumerator.cs" /> <Compile Include="Source\RegionEnumerator.cs" />
<Compile Include="Source\RegionFile.cs" /> <Compile Include="Source\RegionFile.cs" />
<Compile Include="Source\RegionKey.cs" /> <Compile Include="Source\RegionKey.cs" />
<Compile Include="Source\RegionManager.cs" /> <Compile Include="Source\RegionManager.cs" />
<Compile Include="Source\TileEntities\TileEntityChest.cs" />
<Compile Include="Source\TileEntities\TileEntityFurnace.cs" />
<Compile Include="Source\TileEntities\TileEntityMobSpawner.cs" />
<Compile Include="Source\TileEntities\TileEntityMusic.cs" />
<Compile Include="Source\TileEntities\TileEntitySign.cs" />
<Compile Include="Source\TileEntities\TileEntityTrap.cs" />
<Compile Include="Source\TileEntity.cs" /> <Compile Include="Source\TileEntity.cs" />
<Compile Include="Source\TileEntityFactory.cs" /> <Compile Include="Source\TileEntityFactory.cs" />
<Compile Include="Source\Utility\Interface.cs" /> <Compile Include="Source\Utility\Interface.cs" />

View file

@ -1,7 +1,7 @@
 
Microsoft Visual Studio Solution File, Format Version 10.00 Microsoft Visual Studio Solution File, Format Version 10.00
# Visual C# Express 2008 # Visual C# Express 2008
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "NBToolkit", "NBToolkit.csproj", "{AFE30E14-3F2F-4461-9F7D-147AB4DCA4C3}" Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Substrate", "Substrate.csproj", "{AFE30E14-3F2F-4461-9F7D-147AB4DCA4C3}"
EndProject EndProject
Global Global
GlobalSection(SolutionConfigurationPlatforms) = preSolution GlobalSection(SolutionConfigurationPlatforms) = preSolution