2011-04-06 04:43:54 +00:00
|
|
|
|
using System;
|
|
|
|
|
using System.IO;
|
|
|
|
|
|
2011-04-06 21:20:35 +00:00
|
|
|
|
namespace Substrate
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
|
|
|
|
using NBT;
|
|
|
|
|
using Utility;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
|
|
|
|
|
public class Chunk : IChunk, INBTObject<Chunk>, ICopyable<Chunk>
|
|
|
|
|
{
|
|
|
|
|
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),
|
2011-04-09 02:50:42 +00:00
|
|
|
|
new NBTListNode("Entities", TagType.TAG_COMPOUND),
|
|
|
|
|
new NBTListNode("TileEntities", TagType.TAG_COMPOUND, TileEntity.BaseSchema),
|
2011-04-13 05:04:32 +00:00
|
|
|
|
new NBTScalerNode("LastUpdate", TagType.TAG_LONG, NBTOptions.CREATE_ON_MISSING),
|
2011-04-09 02:50:42 +00:00
|
|
|
|
new NBTScalerNode("xPos", TagType.TAG_INT),
|
|
|
|
|
new NBTScalerNode("zPos", TagType.TAG_INT),
|
2011-04-13 05:04:32 +00:00
|
|
|
|
new NBTScalerNode("TerrainPopulated", TagType.TAG_BYTE, NBTOptions.CREATE_ON_MISSING),
|
2011-04-06 04:43:54 +00:00
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
private NBT_Tree _tree;
|
|
|
|
|
|
|
|
|
|
private int _cx;
|
|
|
|
|
private int _cz;
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
protected TagByteArray _blocks;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
protected NibbleArray _data;
|
|
|
|
|
protected NibbleArray _blockLight;
|
|
|
|
|
protected NibbleArray _skyLight;
|
2011-04-09 02:50:42 +00:00
|
|
|
|
protected TagByteArray _heightMap;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
protected TagList _entities;
|
|
|
|
|
protected TagList _tileEntities;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
protected Dictionary<BlockKey, TagCompound> _tileEntityTable;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
public int X
|
|
|
|
|
{
|
|
|
|
|
get { return _cx; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int Z
|
|
|
|
|
{
|
|
|
|
|
get { return _cz; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public NBT_Tree Tree
|
|
|
|
|
{
|
|
|
|
|
get { return _tree; }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool IsTerrainPopulated
|
|
|
|
|
{
|
2011-04-09 02:50:42 +00:00
|
|
|
|
get { return _tree.Root["Level"].ToTagCompound()["TerrainPopulated"].ToTagByte() == 1; }
|
|
|
|
|
set { _tree.Root["Level"].ToTagCompound()["TerrainPopulated"].ToTagByte().Data = (byte)(value ? 1 : 0); }
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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 ()
|
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
int elements2 = XDim * ZDim;
|
|
|
|
|
int elements3 = elements2 *YDim;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
_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);
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
_data = new NibbleArray(data.Data);
|
|
|
|
|
_blockLight = new NibbleArray(blocklight.Data);
|
|
|
|
|
_skyLight = new NibbleArray(skylight.Data);
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound level = new TagCompound();
|
2011-04-06 04:43:54 +00:00
|
|
|
|
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);
|
2011-04-13 08:46:06 +00:00
|
|
|
|
level.Add("LastUpdate", new TagLong(Timestamp()));
|
|
|
|
|
level.Add("xPos", new TagInt(_cx));
|
|
|
|
|
level.Add("zPos", new TagInt(_cz));
|
2011-04-09 02:50:42 +00:00
|
|
|
|
level.Add("TerrainPopulated", new TagByte());
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
_tree = new NBT_Tree();
|
|
|
|
|
_tree.Root.Add("Level", level);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int BlockGlobalX (int x)
|
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
return _cx * XDim + x;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int BlockGlobalY (int y)
|
|
|
|
|
{
|
|
|
|
|
return y;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int BlockGlobalZ (int z)
|
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
return _cz * XDim + z;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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());
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
public int CountBlockID (int id)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
int c = 0;
|
|
|
|
|
for (int i = 0; i < _blocks.Length; i++) {
|
|
|
|
|
if (_blocks[i] == id) {
|
|
|
|
|
c++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return c;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
public int CountBlockData (int id, int data)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
int c = 0;
|
|
|
|
|
for (int i = 0; i < _blocks.Length; i++) {
|
|
|
|
|
if (_blocks[i] == id && _data[i] == data) {
|
|
|
|
|
c++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return c;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
public int CountEntities ()
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
return _entities.Count;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
public int GetHeight (int lx, int lz)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
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 ()
|
|
|
|
|
{
|
2011-04-09 02:50:42 +00:00
|
|
|
|
_tileEntityTable = new Dictionary<BlockKey, TagCompound>();
|
2011-04-08 02:16:06 +00:00
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
foreach (TagCompound te in _tileEntities) {
|
|
|
|
|
int tex = te["x"].ToTagInt();
|
|
|
|
|
int tey = te["y"].ToTagInt();
|
|
|
|
|
int tez = te["z"].ToTagInt();
|
2011-04-08 02:16:06 +00:00
|
|
|
|
|
|
|
|
|
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];
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-11 06:13:33 +00:00
|
|
|
|
// Light consistency
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-04-06 04:43:54 +00:00
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
/*if (BlockManager.EnforceDataLimits && BlockInfo.BlockTable[_blocks[index]] != null) {
|
2011-04-06 04:43:54 +00:00
|
|
|
|
if (!BlockInfo.BlockTable[_blocks[index]].TestData(data)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2011-04-08 02:16:06 +00:00
|
|
|
|
}*/
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
_data[index] = data;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
#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];
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-06 04:43:54 +00:00
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
#endregion
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
#region IPropertyBlockContainer Members
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
IPropertyBlock IPropertyBlockContainer.GetBlock (int lx, int ly, int lz)
|
2011-04-06 23:54:41 +00:00
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
return GetBlock(lx, ly, lz);
|
2011-04-06 23:54:41 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
IPropertyBlock IPropertyBlockContainer.GetBlockRef (int lx, int ly, int lz)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
return GetBlockRef(lx, ly, lz);
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
public void SetBlock (int lx, int ly, int lz, IPropertyBlock block)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
int index = lx << 11 | lz << 7 | ly;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
SetBlockID(lx, ly, lz, block.ID);
|
|
|
|
|
SetBlockData(lx, ly, lz, block.Data);
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
SetTileEntity(lx, ly, lz, block.GetTileEntity().Copy());
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
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);
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound te;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
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);
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound oldte;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
if (_tileEntityTable.TryGetValue(key, out oldte)) {
|
|
|
|
|
_tileEntities.Remove(oldte);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
te.X = x;
|
|
|
|
|
te.Y = y;
|
|
|
|
|
te.Z = z;
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound tree = te.BuildTree() as TagCompound;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
_tileEntities.Add(tree);
|
2011-04-08 02:16:06 +00:00
|
|
|
|
_tileEntityTable[key] = tree;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
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);
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound te;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
if (!_tileEntityTable.TryGetValue(key, out te)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_tileEntities.Remove(te);
|
|
|
|
|
_tileEntityTable.Remove(key);
|
|
|
|
|
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-08 02:16:06 +00:00
|
|
|
|
#endregion
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#region ICopyable<Chunk> Members
|
|
|
|
|
|
|
|
|
|
public Chunk Copy ()
|
|
|
|
|
{
|
|
|
|
|
return new Chunk(_tree.Copy());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
|
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
2011-04-06 04:43:54 +00:00
|
|
|
|
#region INBTObject<Chunk> Members
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
public Chunk LoadTree (TagValue tree)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound ctree = tree as TagCompound;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
if (ctree == null) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
_tree = new NBT_Tree(ctree);
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound level = _tree.Root["Level"] as TagCompound;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
_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;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
_entities = level["Entities"] as TagList;
|
|
|
|
|
_tileEntities = level["TileEntities"] as TagList;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
// List-type patch up
|
|
|
|
|
if (_entities.Count == 0) {
|
2011-04-09 02:50:42 +00:00
|
|
|
|
level["Entities"] = new TagList(TagType.TAG_COMPOUND);
|
|
|
|
|
_entities = level["Entities"] as TagList;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (_tileEntities.Count == 0) {
|
2011-04-09 02:50:42 +00:00
|
|
|
|
level["TileEntities"] = new TagList(TagType.TAG_COMPOUND);
|
|
|
|
|
_tileEntities = level["TileEntities"] as TagList;
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
_cx = level["xPos"].ToTagInt();
|
|
|
|
|
_cz = level["zPos"].ToTagInt();
|
2011-04-06 04:43:54 +00:00
|
|
|
|
|
|
|
|
|
BuildTileEntityCache();
|
|
|
|
|
|
|
|
|
|
return this;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
public Chunk LoadTreeSafe (TagValue tree)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
|
|
|
|
if (!ValidateTree(tree)) {
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return LoadTree(tree);
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
public TagValue BuildTree ()
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
|
|
|
|
return _tree.Root;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
public bool ValidateTree (TagValue tree)
|
2011-04-06 04:43:54 +00:00
|
|
|
|
{
|
|
|
|
|
return new NBTVerifier(tree, LevelSchema).Verify();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#region IEntityContainer Members
|
|
|
|
|
|
|
|
|
|
public List<Entity> FindEntities (string id)
|
|
|
|
|
{
|
|
|
|
|
List<Entity> set = new List<Entity>();
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
foreach (TagCompound ent in _entities) {
|
|
|
|
|
TagValue eid;
|
2011-04-06 23:54:41 +00:00
|
|
|
|
if (!ent.TryGetValue("id", out eid)) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
if (eid.ToTagString().Data != id) {
|
2011-04-06 23:54:41 +00:00
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Entity obj = EntityFactory.Create(ent);
|
|
|
|
|
if (obj != null) {
|
|
|
|
|
set.Add(obj);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return set;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public List<Entity> FindEntities (Predicate<Entity> match)
|
|
|
|
|
{
|
|
|
|
|
List<Entity> set = new List<Entity>();
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
foreach (TagCompound ent in _entities) {
|
2011-04-06 23:54:41 +00:00
|
|
|
|
Entity obj = EntityFactory.Create(ent);
|
|
|
|
|
if (obj == null) {
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (match(obj)) {
|
|
|
|
|
set.Add(obj);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return set;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool AddEntity (Entity ent)
|
|
|
|
|
{
|
2011-04-08 02:16:06 +00:00
|
|
|
|
double xlow = _cx * XDim;
|
|
|
|
|
double xhigh = xlow + XDim;
|
|
|
|
|
double zlow = _cz * ZDim;
|
|
|
|
|
double zhigh = zlow + ZDim;
|
2011-04-06 23:54:41 +00:00
|
|
|
|
|
|
|
|
|
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 => {
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound cval = val as TagCompound;
|
2011-04-06 23:54:41 +00:00
|
|
|
|
if (cval == null) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagValue sval;
|
2011-04-06 23:54:41 +00:00
|
|
|
|
if (!cval.TryGetValue("id", out sval)) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-09 02:50:42 +00:00
|
|
|
|
return (sval.ToTagString().Data == id);
|
2011-04-06 23:54:41 +00:00
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int RemoveEntities (Predicate<Entity> match)
|
|
|
|
|
{
|
|
|
|
|
return _entities.RemoveAll(val => {
|
2011-04-09 02:50:42 +00:00
|
|
|
|
TagCompound cval = val as TagCompound;
|
2011-04-06 23:54:41 +00:00
|
|
|
|
if (cval == null) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Entity obj = EntityFactory.Create(cval);
|
|
|
|
|
if (obj == null) {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return match(obj);
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endregion
|
2011-04-11 06:13:33 +00:00
|
|
|
|
|
2011-04-14 02:30:40 +00:00
|
|
|
|
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;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2011-04-13 08:46:06 +00:00
|
|
|
|
private int Timestamp ()
|
|
|
|
|
{
|
|
|
|
|
DateTime epoch = new DateTime(1970, 1, 1, 0, 0, 0, 0);
|
|
|
|
|
return (int)((DateTime.UtcNow - epoch).Ticks / (10000L * 1000L));
|
|
|
|
|
}
|
2011-04-06 04:43:54 +00:00
|
|
|
|
}
|
|
|
|
|
}
|