forked from mirrors/NBTExplorer
391 lines
No EOL
13 KiB
C#
391 lines
No EOL
13 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Text;
|
|
using Substrate.Core;
|
|
using Substrate.NBT;
|
|
|
|
namespace Substrate
|
|
{
|
|
/// <summary>
|
|
/// The base Entity type for Minecraft Entities, providing access to data common to all Minecraft Entities.
|
|
/// </summary>
|
|
public class Entity : INBTObject<Entity>, ICopyable<Entity>
|
|
{
|
|
public class Vector3
|
|
{
|
|
public double X { get; set; }
|
|
public double Y { get; set; }
|
|
public double Z { get; set; }
|
|
}
|
|
|
|
public class Orientation
|
|
{
|
|
public double Pitch { get; set; }
|
|
public double Yaw { get; set; }
|
|
}
|
|
|
|
private static readonly SchemaNodeCompound _schema = new SchemaNodeCompound("")
|
|
{
|
|
new SchemaNodeList("Pos", TagType.TAG_DOUBLE, 3),
|
|
new SchemaNodeList("Motion", TagType.TAG_DOUBLE, 3),
|
|
new SchemaNodeList("Rotation", TagType.TAG_FLOAT, 2),
|
|
new SchemaNodeScaler("FallDistance", TagType.TAG_FLOAT),
|
|
new SchemaNodeScaler("Fire", TagType.TAG_SHORT),
|
|
new SchemaNodeScaler("Air", TagType.TAG_SHORT),
|
|
new SchemaNodeScaler("OnGround", TagType.TAG_BYTE),
|
|
};
|
|
|
|
private Vector3 _pos;
|
|
private Vector3 _motion;
|
|
private Orientation _rotation;
|
|
|
|
private float _fallDistance;
|
|
private short _fire;
|
|
private short _air;
|
|
private byte _onGround;
|
|
|
|
/// <summary>
|
|
/// Gets or sets the global position of the entity in fractional block coordinates.
|
|
/// </summary>
|
|
public Vector3 Position
|
|
{
|
|
get { return _pos; }
|
|
set { _pos = value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the velocity of the entity.
|
|
/// </summary>
|
|
public Vector3 Motion
|
|
{
|
|
get { return _motion; }
|
|
set { _motion = value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the orientation of the entity.
|
|
/// </summary>
|
|
public Orientation Rotation
|
|
{
|
|
get { return _rotation; }
|
|
set { _rotation = value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the distance that the entity has fallen, if it is falling.
|
|
/// </summary>
|
|
public double FallDistance
|
|
{
|
|
get { return _fallDistance; }
|
|
set { _fallDistance = (float)value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the fire counter of the entity.
|
|
/// </summary>
|
|
public int Fire
|
|
{
|
|
get { return _fire; }
|
|
set { _fire = (short)value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the reamining air availale to the entity.
|
|
/// </summary>
|
|
public int Air
|
|
{
|
|
get { return _air; }
|
|
set { _air = (short)value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets a value indicating whether the entity is currently touch the ground.
|
|
/// </summary>
|
|
public bool IsOnGround
|
|
{
|
|
get { return _onGround == 1; }
|
|
set { _onGround = (byte)(value ? 1 : 0); }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructs a new generic <see cref="Entity"/> with default values.
|
|
/// </summary>
|
|
public Entity ()
|
|
{
|
|
_pos = new Vector3();
|
|
_motion = new Vector3();
|
|
_rotation = new Orientation();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructs a new generic <see cref="Entity"/> by copying fields from another <see cref="Entity"/> object.
|
|
/// </summary>
|
|
/// <param name="e">An <see cref="Entity"/> to copy fields from.</param>
|
|
public Entity (Entity e)
|
|
{
|
|
_pos = new Vector3();
|
|
_pos.X = e._pos.X;
|
|
_pos.Y = e._pos.Y;
|
|
_pos.Z = e._pos.Z;
|
|
|
|
_motion = new Vector3();
|
|
_motion.X = e._motion.X;
|
|
_motion.Y = e._motion.Y;
|
|
_motion.Z = e._motion.Z;
|
|
|
|
_rotation = new Orientation();
|
|
_rotation.Pitch = e._rotation.Pitch;
|
|
_rotation.Yaw = e._rotation.Yaw;
|
|
|
|
_fallDistance = e._fallDistance;
|
|
_fire = e._fire;
|
|
_air = e._air;
|
|
_onGround = e._onGround;
|
|
}
|
|
|
|
|
|
#region INBTObject<Entity> Members
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="SchemaNode"/> representing the basic schema of an Entity.
|
|
/// </summary>
|
|
public static SchemaNodeCompound Schema
|
|
{
|
|
get { return _schema; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempt to load an Entity subtree into the <see cref="Entity"/> without validation.
|
|
/// </summary>
|
|
/// <param name="tree">The root node of an Entity subtree.</param>
|
|
/// <returns>The <see cref="Entity"/> returns itself on success, or null if the tree was unparsable.</returns>
|
|
public Entity LoadTree (TagNode tree)
|
|
{
|
|
TagNodeCompound ctree = tree as TagNodeCompound;
|
|
if (ctree == null) {
|
|
return null;
|
|
}
|
|
|
|
TagNodeList pos = ctree["Pos"].ToTagList();
|
|
_pos = new Vector3();
|
|
_pos.X = pos[0].ToTagDouble();
|
|
_pos.Y = pos[1].ToTagDouble();
|
|
_pos.Z = pos[2].ToTagDouble();
|
|
|
|
TagNodeList motion = ctree["Motion"].ToTagList();
|
|
_motion = new Vector3();
|
|
_motion.X = motion[0].ToTagDouble();
|
|
_motion.Y = motion[1].ToTagDouble();
|
|
_motion.Z = motion[2].ToTagDouble();
|
|
|
|
TagNodeList rotation = ctree["Rotation"].ToTagList();
|
|
_rotation = new Orientation();
|
|
_rotation.Yaw = rotation[0].ToTagFloat();
|
|
_rotation.Pitch = rotation[1].ToTagFloat();
|
|
|
|
_fire = ctree["Fire"].ToTagShort();
|
|
_air = ctree["Air"].ToTagShort();
|
|
_onGround = ctree["OnGround"].ToTagByte();
|
|
|
|
return this;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempt to load an Entity subtree into the <see cref="Entity"/> with validation.
|
|
/// </summary>
|
|
/// <param name="tree">The root node of an Entity subtree.</param>
|
|
/// <returns>The <see cref="Entity"/> returns itself on success, or null if the tree failed validation.</returns>
|
|
public Entity LoadTreeSafe (TagNode tree)
|
|
{
|
|
if (!ValidateTree(tree)) {
|
|
return null;
|
|
}
|
|
|
|
return LoadTree(tree);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Builds an Entity subtree from the current data.
|
|
/// </summary>
|
|
/// <returns>The root node of an Entity subtree representing the current data.</returns>
|
|
public TagNode BuildTree ()
|
|
{
|
|
TagNodeCompound tree = new TagNodeCompound();
|
|
|
|
TagNodeList pos = new TagNodeList(TagType.TAG_DOUBLE);
|
|
pos.Add(new TagNodeDouble(_pos.X));
|
|
pos.Add(new TagNodeDouble(_pos.Y));
|
|
pos.Add(new TagNodeDouble(_pos.Z));
|
|
tree["Pos"] = pos;
|
|
|
|
TagNodeList motion = new TagNodeList(TagType.TAG_DOUBLE);
|
|
motion.Add(new TagNodeDouble(_motion.X));
|
|
motion.Add(new TagNodeDouble(_motion.Y));
|
|
motion.Add(new TagNodeDouble(_motion.Z));
|
|
tree["Motion"] = motion;
|
|
|
|
TagNodeList rotation = new TagNodeList(TagType.TAG_FLOAT);
|
|
rotation.Add(new TagNodeFloat((float)_rotation.Yaw));
|
|
rotation.Add(new TagNodeFloat((float)_rotation.Pitch));
|
|
tree["Rotation"] = rotation;
|
|
|
|
tree["FallDistance"] = new TagNodeFloat(_fallDistance);
|
|
tree["Fire"] = new TagNodeShort(_fire);
|
|
tree["Air"] = new TagNodeShort(_air);
|
|
tree["OnGround"] = new TagNodeByte(_onGround);
|
|
|
|
return tree;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Validate an Entity subtree against a basic schema.
|
|
/// </summary>
|
|
/// <param name="tree">The root node of an Entity subtree.</param>
|
|
/// <returns>Status indicating whether the tree was valid against the internal schema.</returns>
|
|
public bool ValidateTree (TagNode tree)
|
|
{
|
|
return new NBTVerifier(tree, _schema).Verify();
|
|
}
|
|
|
|
#endregion
|
|
|
|
|
|
#region ICopyable<Entity> Members
|
|
|
|
/// <summary>
|
|
/// Creates a deep-copy of the <see cref="Entity"/>.
|
|
/// </summary>
|
|
/// <returns>A deep-copy of the <see cref="Entity"/>.</returns>
|
|
public Entity Copy ()
|
|
{
|
|
return new Entity(this);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
|
|
/// <summary>
|
|
/// A base entity type for all entities except <see cref="Player"/> entities.
|
|
/// </summary>
|
|
/// <remarks>Generally, this class should be subtyped into new concrete Entity types, as this generic type is unable to
|
|
/// capture any of the custom data fields. It is however still possible to create instances of <see cref="Entity"/> objects,
|
|
/// which may allow for graceful handling of unknown Entity types.</remarks>
|
|
public class EntityTyped : Entity, INBTObject<EntityTyped>, ICopyable<EntityTyped>
|
|
{
|
|
private static readonly SchemaNodeCompound _schema = Entity.Schema.MergeInto(new SchemaNodeCompound("")
|
|
{
|
|
new SchemaNodeScaler("id", TagType.TAG_STRING),
|
|
});
|
|
|
|
private string _id;
|
|
|
|
/// <summary>
|
|
/// Gets the id (type) of the entity.
|
|
/// </summary>
|
|
public string ID
|
|
{
|
|
get { return _id; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a new generic <see cref="EntityTyped"/> with the given id.
|
|
/// </summary>
|
|
/// <param name="id">The id (name) of the Entity.</param>
|
|
public EntityTyped (string id)
|
|
: base()
|
|
{
|
|
_id = id;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructs a new <see cref="EntityTyped"/> by copying an existing one.
|
|
/// </summary>
|
|
/// <param name="e">The <see cref="EntityTyped"/> to copy.</param>
|
|
public EntityTyped (EntityTyped e)
|
|
: base(e)
|
|
{
|
|
_id = e._id;
|
|
}
|
|
|
|
|
|
#region INBTObject<Entity> Members
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="SchemaNode"/> representing the basic schema of an Entity.
|
|
/// </summary>
|
|
public static SchemaNodeCompound Schema
|
|
{
|
|
get { return _schema; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempt to load an Entity subtree into the <see cref="EntityTyped"/> without validation.
|
|
/// </summary>
|
|
/// <param name="tree">The root node of an Entity subtree.</param>
|
|
/// <returns>The <see cref="EntityTyped"/> returns itself on success, or null if the tree was unparsable.</returns>
|
|
public virtual new EntityTyped LoadTree (TagNode tree)
|
|
{
|
|
TagNodeCompound ctree = tree as TagNodeCompound;
|
|
if (ctree == null || base.LoadTree(tree) == null) {
|
|
return null;
|
|
}
|
|
|
|
_id = ctree["id"].ToTagString();
|
|
|
|
return this;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Attempt to load an Entity subtree into the <see cref="EntityTyped"/> with validation.
|
|
/// </summary>
|
|
/// <param name="tree">The root node of an Entity subtree.</param>
|
|
/// <returns>The <see cref="EntityTyped"/> returns itself on success, or null if the tree failed validation.</returns>
|
|
public virtual new EntityTyped LoadTreeSafe (TagNode tree)
|
|
{
|
|
if (!ValidateTree(tree)) {
|
|
return null;
|
|
}
|
|
|
|
return LoadTree(tree);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Builds an Entity subtree from the current data.
|
|
/// </summary>
|
|
/// <returns>The root node of an Entity subtree representing the current data.</returns>
|
|
public virtual new TagNode BuildTree ()
|
|
{
|
|
TagNodeCompound tree = base.BuildTree() as TagNodeCompound;
|
|
tree["id"] = new TagNodeString(_id);
|
|
|
|
return tree;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Validate an Entity subtree against a basic schema.
|
|
/// </summary>
|
|
/// <param name="tree">The root node of an Entity subtree.</param>
|
|
/// <returns>Status indicating whether the tree was valid against the internal schema.</returns>
|
|
public virtual new bool ValidateTree (TagNode tree)
|
|
{
|
|
return new NBTVerifier(tree, _schema).Verify();
|
|
}
|
|
|
|
#endregion
|
|
|
|
|
|
#region ICopyable<Entity> Members
|
|
|
|
/// <summary>
|
|
/// Creates a deep-copy of the <see cref="EntityTyped"/>.
|
|
/// </summary>
|
|
/// <returns>A deep-copy of the <see cref="EntityTyped"/>.</returns>
|
|
public virtual new EntityTyped Copy ()
|
|
{
|
|
return new EntityTyped(this);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
|
|
} |