NBTExplorer/Substrate/SubstrateCS/Source/Chunk.cs
2011-04-12 00:30:17 +00:00

803 lines
23 KiB
C#

using System;
using System.IO;
namespace Substrate
{
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),
new NBTListNode("Entities", TagType.TAG_COMPOUND),
new NBTListNode("TileEntities", TagType.TAG_COMPOUND, TileEntity.BaseSchema),
//new NBTScalerNode("LastUpdate", TagType.TAG_LONG),
new NBTScalerNode("xPos", TagType.TAG_INT),
new NBTScalerNode("zPos", TagType.TAG_INT),
new NBTScalerNode("TerrainPopulated", TagType.TAG_BYTE),
},
};
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<BlockKey, TagCompound> _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());
level.Add("xPos", new TagInt());
level.Add("zPos", new TagInt());
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<BlockKey, TagCompound>();
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<Chunk> Members
public Chunk Copy ()
{
return new Chunk(_tree.Copy());
}
#endregion
#region INBTObject<Chunk> 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<Entity> FindEntities (string id)
{
List<Entity> set = new List<Entity>();
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<Entity> FindEntities (Predicate<Entity> match)
{
List<Entity> set = new List<Entity>();
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<Entity> 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]];
if (info == null) {
_blockLight[i] = 0;
}
else {
_blockLight[i] = Math.Max(info.Luminance - info.Opacity, 0);
}
}
}
public void StepBlockLight (IChunk east, IChunk north, IChunk south, IChunk west)
{
for (int x = 0; x < XDim; x++) {
for (int z = 0; z < ZDim; z++) {
for (int y = 0; y < YDim; y++) {
int lle = NeighborLight(x, y, z - 1, east);
int lln = NeighborLight(x - 1, y, z, north);
int lls = NeighborLight(x, y, z + 1, south);
int llw = NeighborLight(x + 1, y, z, west);
int lld = NeighborLight(x, y - 1, z, null);
int llu = NeighborLight(x, y + 1, z, null);
BlockInfo info = 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);
SetBlockLight(x, y, z, light);
}
}
}
}
private int NeighborLight (int x, int y, int z, IChunk n)
{
if (y < 0 || y >= YDim) {
return 0;
}
IChunk src = this;
if (x < 0 || x >= XDim || z < 0 || z >= ZDim) {
src = n;
x = (x + XDim) % XDim;
z = (z + ZDim) % ZDim;
}
if (src == null) {
return 0;
}
BlockInfo info = src.GetBlockInfo(x, y, z);
int light = src.GetBlockLight(x, y, z);
return Math.Max(light, info.Luminance);
}
}
}