Added entity access points into Chunks (as IEntityContainer interface). Added predicate-based find/remove capability to NBT Lists.

This commit is contained in:
Justin Aquadro 2011-04-06 23:54:41 +00:00
parent 50375d326d
commit 428c576925
7 changed files with 175 additions and 67 deletions

View file

@ -49,14 +49,4 @@ namespace Substrate
bool SetTileEntity (int lx, int ly, int lz, TileEntity te); bool SetTileEntity (int lx, int ly, int lz, TileEntity te);
bool ClearTileEntity (int lx, int ly, int lz); bool ClearTileEntity (int lx, int ly, int lz);
} }
public interface IEntity
{
}
public interface IEntityCollection
{
List<Entity> FindAll (Predicate<Entity> match);
}
} }

View file

@ -233,20 +233,6 @@ namespace Substrate
} }
} }
/*if (BlockInfo.SchemaTable[_blocks[index]] != BlockInfo.SchemaTable[id]) {
if (BlockInfo.SchemaTable[_blocks[index]] != null) {
ClearTileEntity(lx, ly, lz);
}
if (BlockInfo.SchemaTable[id] != null) {
TileEntity te = new TileEntity(BlockInfo.SchemaTable[id]);
te.X = BlockGlobalX(lx);
te.Y = BlockGlobalY(ly);
te.Z = BlockGlobalZ(lz);
_tileEntities.Add(te.Root);
}
}*/
// Update height map // Update height map
if (BlockInfo.BlockTable[id] != null) { if (BlockInfo.BlockTable[id] != null) {
@ -332,6 +318,11 @@ namespace Substrate
return c; return c;
} }
public int CountEntities ()
{
return _entities.Count;
}
public int GetHeight (int lx, int lz) public int GetHeight (int lx, int lz)
{ {
return _heightMap[lz << 4 | lx]; return _heightMap[lz << 4 | lx];
@ -459,6 +450,7 @@ namespace Substrate
#endregion #endregion
#region INBTObject<Chunk> Members #region INBTObject<Chunk> Members
public Chunk LoadTree (NBT_Value tree) public Chunk LoadTree (NBT_Value tree)
@ -520,5 +512,101 @@ namespace Substrate
} }
#endregion #endregion
#region IEntityContainer Members
public List<Entity> FindEntities (string id)
{
List<Entity> set = new List<Entity>();
foreach (NBT_Compound ent in _entities) {
NBT_Value eid;
if (!ent.TryGetValue("id", out eid)) {
continue;
}
if (eid.ToNBTString().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 (NBT_Compound 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 * BlockManager.CHUNK_XLEN;
double xhigh = xlow + BlockManager.CHUNK_XLEN;
double zlow = _cz * BlockManager.CHUNK_ZLEN;
double zhigh = zlow + BlockManager.CHUNK_ZLEN;
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 => {
NBT_Compound cval = val as NBT_Compound;
if (cval == null) {
return false;
}
NBT_Value sval;
if (!cval.TryGetValue("id", out sval)) {
return false;
}
return (sval.ToNBTString().Data == id);
});
}
public int RemoveEntities (Predicate<Entity> match)
{
return _entities.RemoveAll(val => {
NBT_Compound cval = val as NBT_Compound;
if (cval == null) {
return false;
}
Entity obj = EntityFactory.Create(cval);
if (obj == null) {
return false;
}
return match(obj);
});
}
#endregion
} }
} }

View file

@ -6,7 +6,7 @@ using System.IO;
namespace Substrate namespace Substrate
{ {
public interface IChunk : IBlockContainer public interface IChunk : IBlockContainer, IEntityContainer
{ {
int X { get; } int X { get; }
int Z { get; } int Z { get; }
@ -18,6 +18,8 @@ namespace Substrate
int CountBlockID (int id); int CountBlockID (int id);
int CountBlockData (int id, int data); int CountBlockData (int id, int data);
int CountEntities ();
int GetHeight (int lx, int lz); int GetHeight (int lx, int lz);
} }

View file

@ -251,6 +251,11 @@ namespace Substrate
return GetChunk().CountBlockData(id, data); return GetChunk().CountBlockData(id, data);
} }
public int CountEntities ()
{
return GetChunk().CountEntities();
}
public int GetHeight (int lx, int lz) public int GetHeight (int lx, int lz)
{ {
return GetChunk().GetHeight(lx, lz); return GetChunk().GetHeight(lx, lz);
@ -280,49 +285,48 @@ namespace Substrate
} }
#endregion #endregion
}
/*public bool VerifyTileEntities ()
#region IEntityContainer Members
public List<Entity> FindEntities (string id)
{ {
bool pass = true; return GetChunk().FindEntities(id);
NBT_List telist = GetTree().Root["Level"].ToNBTCompound()["TileEntities"].ToNBTList();
foreach (NBT_Value val in telist) {
NBT_Compound tree = val as NBT_Compound;
if (tree == null) {
pass = false;
continue;
}
if (new NBTVerifier(tree, TileEntity.BaseSchema).Verify() == false) {
pass = false;
continue;
}
int x = tree["x"].ToNBTInt() & BlockManager.CHUNK_XMASK;
int y = tree["y"].ToNBTInt() & BlockManager.CHUNK_YMASK;
int z = tree["z"].ToNBTInt() & BlockManager.CHUNK_ZMASK;
int id = GetBlockID(x, y, z);
NBTCompoundNode schema = BlockInfo.SchemaTable[id];
if (schema == null) {
pass = false;
continue;
}
pass = new NBTVerifier(tree, schema).Verify() && pass;
}
return pass;
} }
private static bool LocalBounds (int lx, int ly, int lz) public List<Entity> FindEntities (Predicate<Entity> match)
{ {
return lx >= 0 && lx < BlockManager.CHUNK_XLEN && return GetChunk().FindEntities(match);
ly >= 0 && ly < BlockManager.CHUNK_YLEN && }
lz >= 0 && lz < BlockManager.CHUNK_ZLEN;
}*/ 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<Entity> match)
{
int ret = GetChunk().RemoveEntities(match);
if (ret > 0) {
MarkDirty();
}
return ret;
}
#endregion
}
public class MalformedNBTTreeException : Exception { }
} }

View file

@ -7,6 +7,17 @@ namespace Substrate
using NBT; using NBT;
using Utility; using Utility;
public interface IEntityContainer
{
List<Entity> FindEntities (string id);
List<Entity> FindEntities (Predicate<Entity> match);
bool AddEntity (Entity ent);
int RemoveEntities (string id);
int RemoveEntities (Predicate<Entity> match);
}
public class Entity : INBTObject<Entity>, ICopyable<Entity> public class Entity : INBTObject<Entity>, ICopyable<Entity>
{ {
public class Vector3 public class Vector3

View file

@ -18,15 +18,18 @@ namespace Substrate
return null; return null;
} }
return Activator.CreateInstance(t, new object[] { type} ) as Entity; return Activator.CreateInstance(t, new object[] { type } ) as Entity;
} }
public static Entity Create (NBT_Compound tree) public static Entity Create (NBT_Compound tree)
{ {
string type = tree["id"].ToNBTString(); NBT_Value type;
if (!tree.TryGetValue("id", out type)) {
return null;
}
Type t; Type t;
if (!_registry.TryGetValue(type, out t)) { if (!_registry.TryGetValue(type.ToNBTString(), out t)) {
return null; return null;
} }

View file

@ -398,6 +398,16 @@ namespace Substrate.NBT {
return list; return list;
} }
public List<NBT_Value> FindAll(Predicate<NBT_Value> match)
{
return _items.FindAll(match);
}
public int RemoveAll (Predicate<NBT_Value> match)
{
return _items.RemoveAll(match);
}
#region IList<NBT_Value> Members #region IList<NBT_Value> Members
public int IndexOf (NBT_Value item) public int IndexOf (NBT_Value item)