forked from mirrors/NBTExplorer
I really messed this transaction up.
This commit is contained in:
parent
999f4f96fa
commit
ad3f3d336d
25 changed files with 3701 additions and 0 deletions
38
Substrate/SubstrateCS/Source/Nbt/INBTObject.cs
Normal file
38
Substrate/SubstrateCS/Source/Nbt/INBTObject.cs
Normal file
|
@ -0,0 +1,38 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// Defines methods for loading or extracting an NBT tree.
|
||||
/// </summary>
|
||||
/// <typeparam name="T">Object type that supports this interface.</typeparam>
|
||||
public interface INBTObject<T>
|
||||
{
|
||||
/// <summary>
|
||||
/// Attempt to load an NBT tree into the object without validation.
|
||||
/// </summary>
|
||||
/// <param name="tree">The root node of an NBT tree.</param>
|
||||
/// <returns>The object returns itself on success, or null if the tree was unparsable.</returns>
|
||||
T LoadTree (TagNode tree);
|
||||
|
||||
/// <summary>
|
||||
/// Attempt to load an NBT tree into the object with validation.
|
||||
/// </summary>
|
||||
/// <param name="tree">The root node of an NBT tree.</param>
|
||||
/// <returns>The object returns itself on success, or null if the tree failed validation.</returns>
|
||||
T LoadTreeSafe (TagNode tree);
|
||||
|
||||
/// <summary>
|
||||
/// Builds an NBT tree from the object's data.
|
||||
/// </summary>
|
||||
/// <returns>The root node of an NBT tree representing the object's data.</returns>
|
||||
TagNode BuildTree ();
|
||||
|
||||
/// <summary>
|
||||
/// Validate an NBT tree, usually against an object-supplied schema.
|
||||
/// </summary>
|
||||
/// <param name="tree">The root node of an NBT tree.</param>
|
||||
/// <returns>Status indicating whether the tree was valid for this object.</returns>
|
||||
bool ValidateTree (TagNode tree);
|
||||
}
|
||||
}
|
173
Substrate/SubstrateCS/Source/Nbt/JSONSerializer.cs
Normal file
173
Substrate/SubstrateCS/Source/Nbt/JSONSerializer.cs
Normal file
|
@ -0,0 +1,173 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Text;
|
||||
using Substrate.Core;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
public class JSONSerializer
|
||||
{
|
||||
public static string Serialize (TagNode tag)
|
||||
{
|
||||
return Serialize(tag, 0);
|
||||
}
|
||||
|
||||
public static string Serialize (TagNode tag, int level)
|
||||
{
|
||||
StringBuilder str = new StringBuilder();
|
||||
|
||||
if (tag.GetTagType() == TagType.TAG_COMPOUND) {
|
||||
SerializeCompound(tag as TagNodeCompound, str, level);
|
||||
}
|
||||
else if (tag.GetTagType() == TagType.TAG_LIST) {
|
||||
SerializeList(tag as TagNodeList, str, level);
|
||||
}
|
||||
else {
|
||||
SerializeScaler(tag, str);
|
||||
}
|
||||
|
||||
return str.ToString();
|
||||
}
|
||||
|
||||
private static void SerializeCompound (TagNodeCompound tag, StringBuilder str, int level)
|
||||
{
|
||||
if (tag.Count == 0) {
|
||||
str.Append("{ }");
|
||||
return;
|
||||
}
|
||||
|
||||
str.AppendLine();
|
||||
AddLine(str, "{", level);
|
||||
|
||||
IEnumerator<KeyValuePair<string, TagNode>> en = tag.GetEnumerator();
|
||||
bool first = true;
|
||||
while (en.MoveNext()) {
|
||||
if (!first) {
|
||||
str.Append(",");
|
||||
str.AppendLine();
|
||||
}
|
||||
|
||||
KeyValuePair<string, TagNode> item = en.Current;
|
||||
Add(str, "\"" + item.Key + "\": ", level + 1);
|
||||
|
||||
if (item.Value.GetTagType() == TagType.TAG_COMPOUND) {
|
||||
SerializeCompound(item.Value as TagNodeCompound, str, level + 1);
|
||||
}
|
||||
else if (item.Value.GetTagType() == TagType.TAG_LIST) {
|
||||
SerializeList(item.Value as TagNodeList, str, level + 1);
|
||||
}
|
||||
else {
|
||||
SerializeScaler(item.Value, str);
|
||||
}
|
||||
|
||||
first = false;
|
||||
}
|
||||
|
||||
str.AppendLine();
|
||||
Add(str, "}", level);
|
||||
}
|
||||
|
||||
private static void SerializeList (TagNodeList tag, StringBuilder str, int level)
|
||||
{
|
||||
if (tag.Count == 0) {
|
||||
str.Append("[ ]");
|
||||
return;
|
||||
}
|
||||
|
||||
str.AppendLine();
|
||||
AddLine(str, "[", level);
|
||||
|
||||
IEnumerator<TagNode> en = tag.GetEnumerator();
|
||||
bool first = true;
|
||||
while (en.MoveNext()) {
|
||||
if (!first) {
|
||||
str.Append(",");
|
||||
}
|
||||
|
||||
TagNode item = en.Current;
|
||||
|
||||
if (item.GetTagType() == TagType.TAG_COMPOUND) {
|
||||
SerializeCompound(item as TagNodeCompound, str, level + 1);
|
||||
}
|
||||
else if (item.GetTagType() == TagType.TAG_LIST) {
|
||||
SerializeList(item as TagNodeList, str, level + 1);
|
||||
}
|
||||
else {
|
||||
if (!first) {
|
||||
str.AppendLine();
|
||||
}
|
||||
Indent(str, level + 1);
|
||||
SerializeScaler(item, str);
|
||||
}
|
||||
|
||||
|
||||
first = false;
|
||||
}
|
||||
|
||||
str.AppendLine();
|
||||
Add(str, "]", level);
|
||||
}
|
||||
|
||||
private static void SerializeScaler (TagNode tag, StringBuilder str)
|
||||
{
|
||||
switch (tag.GetTagType()) {
|
||||
case TagType.TAG_STRING:
|
||||
str.Append("\"" + tag.ToTagString().Data + "\"");
|
||||
break;
|
||||
|
||||
case TagType.TAG_BYTE:
|
||||
str.Append(tag.ToTagByte().Data);
|
||||
break;
|
||||
|
||||
case TagType.TAG_SHORT:
|
||||
str.Append(tag.ToTagShort().Data);
|
||||
break;
|
||||
|
||||
case TagType.TAG_INT:
|
||||
str.Append(tag.ToTagInt().Data);
|
||||
break;
|
||||
|
||||
case TagType.TAG_LONG:
|
||||
str.Append(tag.ToTagLong().Data);
|
||||
break;
|
||||
|
||||
case TagType.TAG_FLOAT:
|
||||
str.Append(tag.ToTagFloat().Data);
|
||||
break;
|
||||
|
||||
case TagType.TAG_DOUBLE:
|
||||
str.Append(tag.ToTagDouble().Data);
|
||||
break;
|
||||
|
||||
case TagType.TAG_BYTE_ARRAY:
|
||||
str.Append(Convert.ToBase64String(tag.ToTagByteArray().Data));
|
||||
/*if (tag.ToTagByteArray().Length == (16 * 16 * 128 / 2)) {
|
||||
str.Append(Base16.Encode(tag.ToTagByteArray().Data, 1));
|
||||
}
|
||||
else {
|
||||
str.Append(Base16.Encode(tag.ToTagByteArray().Data, 2));
|
||||
}*/
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private static void AddLine (StringBuilder str, string line, int level)
|
||||
{
|
||||
Indent(str, level);
|
||||
str.AppendLine(line);
|
||||
}
|
||||
|
||||
private static void Add (StringBuilder str, string line, int level)
|
||||
{
|
||||
Indent(str, level);
|
||||
str.Append(line);
|
||||
}
|
||||
|
||||
private static void Indent (StringBuilder str, int count)
|
||||
{
|
||||
for (int i = 0; i < count; i++) {
|
||||
str.Append("\t");
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
534
Substrate/SubstrateCS/Source/Nbt/NBT.cs
Normal file
534
Substrate/SubstrateCS/Source/Nbt/NBT.cs
Normal file
|
@ -0,0 +1,534 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Text;
|
||||
using System.IO;
|
||||
using System.IO.Compression;
|
||||
using Substrate.Core;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// Contains the root node of an NBT tree and handles IO of tree nodes.
|
||||
/// </summary>
|
||||
/// <remarks>
|
||||
/// NBT, or Named Byte Tag, is a tree-based data structure for storing most Minecraft data.
|
||||
/// NBT_Tree is more of a helper class for NBT trees that handles reading and writing nodes to data streams.
|
||||
/// Most of the API takes a TagValue or derived node as the root of the tree, rather than an NBT_Tree object itself.
|
||||
/// </remarks>
|
||||
public class NbtTree : ICopyable<NbtTree>
|
||||
{
|
||||
private Stream _stream = null;
|
||||
private TagNodeCompound _root = null;
|
||||
|
||||
private static TagNodeNull _nulltag = new TagNodeNull();
|
||||
|
||||
/// <summary>
|
||||
/// Gets the root node of this tree.
|
||||
/// </summary>
|
||||
public TagNodeCompound Root
|
||||
{
|
||||
get { return _root; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a wrapper around a new NBT tree with an empty root node.
|
||||
/// </summary>
|
||||
public NbtTree ()
|
||||
{
|
||||
_root = new TagNodeCompound();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a wrapper around another NBT tree.
|
||||
/// </summary>
|
||||
/// <param name="tree">The root node of an NBT tree.</param>
|
||||
public NbtTree (TagNodeCompound tree)
|
||||
{
|
||||
_root = tree;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs and wrapper around a new NBT tree parsed from a source data stream.
|
||||
/// </summary>
|
||||
/// <param name="s">An open, readable data stream containing NBT data.</param>
|
||||
public NbtTree (Stream s)
|
||||
{
|
||||
ReadFrom(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Rebuild the internal NBT tree from a source data stream.
|
||||
/// </summary>
|
||||
/// <param name="s">An open, readable data stream containing NBT data.</param>
|
||||
public void ReadFrom (Stream s)
|
||||
{
|
||||
if (s != null) {
|
||||
_stream = s;
|
||||
_root = ReadRoot();
|
||||
_stream = null;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Writes out the internal NBT tree to a destination data stream.
|
||||
/// </summary>
|
||||
/// <param name="s">An open, writable data stream.</param>
|
||||
public void WriteTo (Stream s)
|
||||
{
|
||||
if (s != null) {
|
||||
_stream = s;
|
||||
|
||||
if (_root != null) {
|
||||
WriteTag("", _root);
|
||||
}
|
||||
|
||||
_stream = null;
|
||||
}
|
||||
}
|
||||
|
||||
private TagNode ReadValue (TagType type)
|
||||
{
|
||||
switch (type) {
|
||||
case TagType.TAG_END:
|
||||
return null;
|
||||
|
||||
case TagType.TAG_BYTE:
|
||||
return ReadByte();
|
||||
|
||||
case TagType.TAG_SHORT:
|
||||
return ReadShort();
|
||||
|
||||
case TagType.TAG_INT:
|
||||
return ReadInt();
|
||||
|
||||
case TagType.TAG_LONG:
|
||||
return ReadLong();
|
||||
|
||||
case TagType.TAG_FLOAT:
|
||||
return ReadFloat();
|
||||
|
||||
case TagType.TAG_DOUBLE:
|
||||
return ReadDouble();
|
||||
|
||||
case TagType.TAG_BYTE_ARRAY:
|
||||
return ReadByteArray();
|
||||
|
||||
case TagType.TAG_STRING:
|
||||
return ReadString();
|
||||
|
||||
case TagType.TAG_LIST:
|
||||
return ReadList();
|
||||
|
||||
case TagType.TAG_COMPOUND:
|
||||
return ReadCompound();
|
||||
}
|
||||
|
||||
throw new Exception();
|
||||
}
|
||||
|
||||
private TagNode ReadByte ()
|
||||
{
|
||||
int gzByte = _stream.ReadByte();
|
||||
if (gzByte == -1) {
|
||||
throw new NBTException(NBTException.MSG_GZIP_ENDOFSTREAM);
|
||||
}
|
||||
|
||||
TagNodeByte val = new TagNodeByte((byte)gzByte);
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadShort ()
|
||||
{
|
||||
byte[] gzBytes = new byte[2];
|
||||
_stream.Read(gzBytes, 0, 2);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
TagNodeShort val = new TagNodeShort(BitConverter.ToInt16(gzBytes, 0));
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadInt ()
|
||||
{
|
||||
byte[] gzBytes = new byte[4];
|
||||
_stream.Read(gzBytes, 0, 4);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
TagNodeInt val = new TagNodeInt(BitConverter.ToInt32(gzBytes, 0));
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadLong ()
|
||||
{
|
||||
byte[] gzBytes = new byte[8];
|
||||
_stream.Read(gzBytes, 0, 8);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
TagNodeLong val = new TagNodeLong(BitConverter.ToInt64(gzBytes, 0));
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadFloat ()
|
||||
{
|
||||
byte[] gzBytes = new byte[4];
|
||||
_stream.Read(gzBytes, 0, 4);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
TagNodeFloat val = new TagNodeFloat(BitConverter.ToSingle(gzBytes, 0));
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadDouble ()
|
||||
{
|
||||
byte[] gzBytes = new byte[8];
|
||||
_stream.Read(gzBytes, 0, 8);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
TagNodeDouble val = new TagNodeDouble(BitConverter.ToDouble(gzBytes, 0));
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadByteArray ()
|
||||
{
|
||||
byte[] lenBytes = new byte[4];
|
||||
_stream.Read(lenBytes, 0, 4);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(lenBytes);
|
||||
}
|
||||
|
||||
int length = BitConverter.ToInt32(lenBytes, 0);
|
||||
if (length < 0) {
|
||||
throw new NBTException(NBTException.MSG_READ_NEG);
|
||||
}
|
||||
|
||||
byte[] data = new byte[length];
|
||||
_stream.Read(data, 0, length);
|
||||
|
||||
TagNodeByteArray val = new TagNodeByteArray(data);
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadString ()
|
||||
{
|
||||
byte[] lenBytes = new byte[2];
|
||||
_stream.Read(lenBytes, 0, 2);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(lenBytes);
|
||||
}
|
||||
|
||||
short len = BitConverter.ToInt16(lenBytes, 0);
|
||||
if (len < 0) {
|
||||
throw new NBTException(NBTException.MSG_READ_NEG);
|
||||
}
|
||||
|
||||
byte[] strBytes = new byte[len];
|
||||
_stream.Read(strBytes, 0, len);
|
||||
|
||||
System.Text.Encoding str = Encoding.GetEncoding(28591);
|
||||
|
||||
TagNodeString val = new TagNodeString(str.GetString(strBytes));
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadList ()
|
||||
{
|
||||
int gzByte = _stream.ReadByte();
|
||||
if (gzByte == -1) {
|
||||
throw new NBTException(NBTException.MSG_GZIP_ENDOFSTREAM);
|
||||
}
|
||||
|
||||
TagNodeList val = new TagNodeList((TagType)gzByte);
|
||||
if (val.ValueType > (TagType)Enum.GetValues(typeof(TagType)).GetUpperBound(0)) {
|
||||
throw new NBTException(NBTException.MSG_READ_TYPE);
|
||||
}
|
||||
|
||||
byte[] lenBytes = new byte[4];
|
||||
_stream.Read(lenBytes, 0, 4);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(lenBytes);
|
||||
}
|
||||
|
||||
int length = BitConverter.ToInt32(lenBytes, 0);
|
||||
if (length < 0) {
|
||||
throw new NBTException(NBTException.MSG_READ_NEG);
|
||||
}
|
||||
|
||||
for (int i = 0; i < length; i++) {
|
||||
val.Add(ReadValue(val.ValueType));
|
||||
}
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNode ReadCompound ()
|
||||
{
|
||||
TagNodeCompound val = new TagNodeCompound();
|
||||
|
||||
while (ReadTag(val)) ;
|
||||
|
||||
return val;
|
||||
}
|
||||
|
||||
private TagNodeCompound ReadRoot ()
|
||||
{
|
||||
TagType type = (TagType)_stream.ReadByte();
|
||||
if (type == TagType.TAG_COMPOUND) {
|
||||
ReadString(); // name
|
||||
return ReadValue(type) as TagNodeCompound;
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
|
||||
private bool ReadTag (TagNodeCompound parent)
|
||||
{
|
||||
//NBT_Tag tag = new NBT_Tag();
|
||||
|
||||
TagType type = (TagType)_stream.ReadByte();
|
||||
if (type != TagType.TAG_END) {
|
||||
string name = ReadString().ToTagString().Data;
|
||||
parent[name] = ReadValue(type);
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
|
||||
//tag.Value = ReadValue(type);
|
||||
|
||||
//return tag;
|
||||
}
|
||||
|
||||
private void WriteValue (TagNode val)
|
||||
{
|
||||
switch (val.GetTagType()) {
|
||||
case TagType.TAG_END:
|
||||
break;
|
||||
|
||||
case TagType.TAG_BYTE:
|
||||
WriteByte(val.ToTagByte());
|
||||
break;
|
||||
|
||||
case TagType.TAG_SHORT:
|
||||
WriteShort(val.ToTagShort());
|
||||
break;
|
||||
|
||||
case TagType.TAG_INT:
|
||||
WriteInt(val.ToTagInt());
|
||||
break;
|
||||
|
||||
case TagType.TAG_LONG:
|
||||
WriteLong(val.ToTagLong());
|
||||
break;
|
||||
|
||||
case TagType.TAG_FLOAT:
|
||||
WriteFloat(val.ToTagFloat());
|
||||
break;
|
||||
|
||||
case TagType.TAG_DOUBLE:
|
||||
WriteDouble(val.ToTagDouble());
|
||||
break;
|
||||
|
||||
case TagType.TAG_BYTE_ARRAY:
|
||||
WriteByteArray(val.ToTagByteArray());
|
||||
break;
|
||||
|
||||
case TagType.TAG_STRING:
|
||||
WriteString(val.ToTagString());
|
||||
break;
|
||||
|
||||
case TagType.TAG_LIST:
|
||||
WriteList(val.ToTagList());
|
||||
break;
|
||||
|
||||
case TagType.TAG_COMPOUND:
|
||||
WriteCompound(val.ToTagCompound());
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
private void WriteByte (TagNodeByte val)
|
||||
{
|
||||
_stream.WriteByte(val.Data);
|
||||
}
|
||||
|
||||
private void WriteShort (TagNodeShort val)
|
||||
{
|
||||
byte[] gzBytes = BitConverter.GetBytes(val.Data);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
_stream.Write(gzBytes, 0, 2);
|
||||
}
|
||||
|
||||
private void WriteInt (TagNodeInt val)
|
||||
{
|
||||
byte[] gzBytes = BitConverter.GetBytes(val.Data);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
_stream.Write(gzBytes, 0, 4);
|
||||
}
|
||||
|
||||
private void WriteLong (TagNodeLong val)
|
||||
{
|
||||
byte[] gzBytes = BitConverter.GetBytes(val.Data);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
_stream.Write(gzBytes, 0, 8);
|
||||
}
|
||||
|
||||
private void WriteFloat (TagNodeFloat val)
|
||||
{
|
||||
byte[] gzBytes = BitConverter.GetBytes(val.Data);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
_stream.Write(gzBytes, 0, 4);
|
||||
}
|
||||
|
||||
private void WriteDouble (TagNodeDouble val)
|
||||
{
|
||||
byte[] gzBytes = BitConverter.GetBytes(val.Data);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(gzBytes);
|
||||
}
|
||||
|
||||
_stream.Write(gzBytes, 0, 8);
|
||||
}
|
||||
|
||||
private void WriteByteArray (TagNodeByteArray val)
|
||||
{
|
||||
byte[] lenBytes = BitConverter.GetBytes(val.Length);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(lenBytes);
|
||||
}
|
||||
|
||||
_stream.Write(lenBytes, 0, 4);
|
||||
_stream.Write(val.Data, 0, val.Length);
|
||||
}
|
||||
|
||||
private void WriteString (TagNodeString val)
|
||||
{
|
||||
byte[] lenBytes = BitConverter.GetBytes((short)val.Length);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(lenBytes);
|
||||
}
|
||||
|
||||
_stream.Write(lenBytes, 0, 2);
|
||||
|
||||
System.Text.Encoding str = Encoding.GetEncoding(28591);
|
||||
byte[] gzBytes = str.GetBytes(val.Data);
|
||||
|
||||
_stream.Write(gzBytes, 0, gzBytes.Length);
|
||||
}
|
||||
|
||||
private void WriteList (TagNodeList val)
|
||||
{
|
||||
byte[] lenBytes = BitConverter.GetBytes(val.Count);
|
||||
|
||||
if (BitConverter.IsLittleEndian) {
|
||||
Array.Reverse(lenBytes);
|
||||
}
|
||||
|
||||
_stream.WriteByte((byte)val.ValueType);
|
||||
_stream.Write(lenBytes, 0, 4);
|
||||
|
||||
foreach (TagNode v in val) {
|
||||
WriteValue(v);
|
||||
}
|
||||
}
|
||||
|
||||
private void WriteCompound (TagNodeCompound val)
|
||||
{
|
||||
foreach (KeyValuePair<string, TagNode> item in val) {
|
||||
WriteTag(item.Key, item.Value);
|
||||
}
|
||||
|
||||
WriteTag(null, _nulltag);
|
||||
}
|
||||
|
||||
private void WriteTag (string name, TagNode val)
|
||||
{
|
||||
_stream.WriteByte((byte)val.GetTagType());
|
||||
|
||||
if (val.GetTagType() != TagType.TAG_END) {
|
||||
WriteString(name);
|
||||
WriteValue(val);
|
||||
}
|
||||
}
|
||||
|
||||
#region ICopyable<NBT_Tree> Members
|
||||
|
||||
/// <summary>
|
||||
/// Creates a deep copy of the NBT_Tree and underlying nodes.
|
||||
/// </summary>
|
||||
/// <returns>A new NBT_tree.</returns>
|
||||
public NbtTree Copy ()
|
||||
{
|
||||
NbtTree tree = new NbtTree();
|
||||
tree._root = _root.Copy() as TagNodeCompound;
|
||||
|
||||
return tree;
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
|
||||
// TODO: Revise exceptions?
|
||||
public class NBTException : Exception
|
||||
{
|
||||
public const String MSG_GZIP_ENDOFSTREAM = "Gzip Error: Unexpected end of stream";
|
||||
|
||||
public const String MSG_READ_NEG = "Read Error: Negative length";
|
||||
public const String MSG_READ_TYPE = "Read Error: Invalid value type";
|
||||
|
||||
public NBTException () { }
|
||||
|
||||
public NBTException (String msg) : base(msg) { }
|
||||
|
||||
public NBTException (String msg, Exception innerException) : base(msg, innerException) { }
|
||||
}
|
||||
|
||||
public class InvalidNBTObjectException : Exception { }
|
||||
|
||||
public class InvalidTagException : Exception { }
|
||||
|
||||
public class InvalidValueException : Exception { }
|
||||
}
|
386
Substrate/SubstrateCS/Source/Nbt/NBTVerifier.cs
Normal file
386
Substrate/SubstrateCS/Source/Nbt/NBTVerifier.cs
Normal file
|
@ -0,0 +1,386 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using Substrate.Core;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// Indicates how an <see cref="NBTVerifier"/> event processor should respond to returning event handler.
|
||||
/// </summary>
|
||||
public enum TagEventCode
|
||||
{
|
||||
/// <summary>
|
||||
/// The event processor should process the next event in the chian.
|
||||
/// </summary>
|
||||
NEXT,
|
||||
|
||||
/// <summary>
|
||||
/// The event processor should ignore the verification failure and stop processing any remaining events.
|
||||
/// </summary>
|
||||
PASS,
|
||||
|
||||
/// <summary>
|
||||
/// The event processor should fail and stop processing any remaining events.
|
||||
/// </summary>
|
||||
FAIL,
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Event arguments for <see cref="NBTVerifier"/> failure events.
|
||||
/// </summary>
|
||||
public class TagEventArgs : EventArgs
|
||||
{
|
||||
private string _tagName;
|
||||
private TagNode _parent;
|
||||
private TagNode _tag;
|
||||
private SchemaNode _schema;
|
||||
|
||||
/// <summary>
|
||||
/// Gets the expected name of the <see cref="TagNode"/> referenced by this event.
|
||||
/// </summary>
|
||||
public string TagName
|
||||
{
|
||||
get { return _tagName; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the parent <see cref="TagNode"/> of the <see cref="TagNode"/> referenced by this event, if it exists.
|
||||
/// </summary>
|
||||
public TagNode Parent
|
||||
{
|
||||
get { return _parent; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the <see cref="TagNode"/> referenced by this event.
|
||||
/// </summary>
|
||||
public TagNode Tag
|
||||
{
|
||||
get { return _tag; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the <see cref="SchemaNode"/> corresponding to the <see cref="TagNode"/> referenced by this event.
|
||||
/// </summary>
|
||||
public SchemaNode Schema
|
||||
{
|
||||
get { return _schema; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new event argument set.
|
||||
/// </summary>
|
||||
/// <param name="tagName">The expected name of a <see cref="TagNode"/>.</param>
|
||||
public TagEventArgs (string tagName)
|
||||
: base()
|
||||
{
|
||||
_tagName = tagName;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new event argument set.
|
||||
/// </summary>
|
||||
/// <param name="tagName">The expected name of a <see cref="TagNode"/>.</param>
|
||||
/// <param name="tag">The <see cref="TagNode"/> involved in this event.</param>
|
||||
public TagEventArgs (string tagName, TagNode tag)
|
||||
: base()
|
||||
{
|
||||
_tag = tag;
|
||||
_tagName = tagName;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new event argument set.
|
||||
/// </summary>
|
||||
/// <param name="schema">The <see cref="SchemaNode"/> corresponding to the <see cref="TagNode"/> involved in this event.</param>
|
||||
/// <param name="tag">The <see cref="TagNode"/> involved in this event.</param>
|
||||
public TagEventArgs (SchemaNode schema, TagNode tag)
|
||||
: base()
|
||||
{
|
||||
_tag = tag;
|
||||
_schema = schema;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// An event handler for intercepting and responding to verification failures of NBT trees.
|
||||
/// </summary>
|
||||
/// <param name="eventArgs">Information relating to a verification event.</param>
|
||||
/// <returns>A <see cref="TagEventCode"/> determining how the event processor should respond.</returns>
|
||||
public delegate TagEventCode VerifierEventHandler (TagEventArgs eventArgs);
|
||||
|
||||
/// <summary>
|
||||
/// Verifies the integrity of an NBT tree against a schema definition.
|
||||
/// </summary>
|
||||
public class NBTVerifier
|
||||
{
|
||||
private TagNode _root;
|
||||
private SchemaNode _schema;
|
||||
|
||||
/// <summary>
|
||||
/// An event that gets fired whenever an expected <see cref="TagNode"/> is not found.
|
||||
/// </summary>
|
||||
public static event VerifierEventHandler MissingTag;
|
||||
|
||||
/// <summary>
|
||||
/// An event that gets fired whenever an expected <see cref="TagNode"/> is of the wrong type and cannot be cast.
|
||||
/// </summary>
|
||||
public static event VerifierEventHandler InvalidTagType;
|
||||
|
||||
/// <summary>
|
||||
/// An event that gets fired whenever an expected <see cref="TagNode"/> has a value that violates the schema.
|
||||
/// </summary>
|
||||
public static event VerifierEventHandler InvalidTagValue;
|
||||
|
||||
private Dictionary<string, TagNode> _scratch = new Dictionary<string,TagNode>();
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="NBTVerifier"/> object for a given NBT tree and schema.
|
||||
/// </summary>
|
||||
/// <param name="root">A <see cref="TagNode"/> representing the root of an NBT tree.</param>
|
||||
/// <param name="schema">A <see cref="SchemaNode"/> representing the root of a schema definition for the NBT tree.</param>
|
||||
public NBTVerifier (TagNode root, SchemaNode schema)
|
||||
{
|
||||
_root = root;
|
||||
_schema = schema;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Invokes the verifier.
|
||||
/// </summary>
|
||||
/// <returns>Status indicating whether the NBT tree is valid for the given schema.</returns>
|
||||
public virtual bool Verify ()
|
||||
{
|
||||
return Verify(null, _root, _schema);
|
||||
}
|
||||
|
||||
private bool Verify (TagNode parent, TagNode tag, SchemaNode schema)
|
||||
{
|
||||
if (tag == null) {
|
||||
return OnMissingTag(new TagEventArgs(schema.Name));
|
||||
}
|
||||
|
||||
SchemaNodeScaler scaler = schema as SchemaNodeScaler;
|
||||
if (scaler != null) {
|
||||
return VerifyScaler(tag, scaler);
|
||||
}
|
||||
|
||||
SchemaNodeString str = schema as SchemaNodeString;
|
||||
if (str != null) {
|
||||
return VerifyString(tag, str);
|
||||
}
|
||||
|
||||
SchemaNodeArray array = schema as SchemaNodeArray;
|
||||
if (array != null) {
|
||||
return VerifyArray(tag, array);
|
||||
}
|
||||
|
||||
SchemaNodeList list = schema as SchemaNodeList;
|
||||
if (list != null) {
|
||||
return VerifyList(tag, list);
|
||||
}
|
||||
|
||||
SchemaNodeCompound compound = schema as SchemaNodeCompound;
|
||||
if (compound != null) {
|
||||
return VerifyCompound(tag, compound);
|
||||
}
|
||||
|
||||
return OnInvalidTagType(new TagEventArgs(schema.Name, tag));
|
||||
}
|
||||
|
||||
private bool VerifyScaler (TagNode tag, SchemaNodeScaler schema)
|
||||
{
|
||||
if (!tag.IsCastableTo(schema.Type)) {
|
||||
if (!OnInvalidTagType(new TagEventArgs(schema.Name, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private bool VerifyString (TagNode tag, SchemaNodeString schema)
|
||||
{
|
||||
TagNodeString stag = tag as TagNodeString;
|
||||
if (stag == null) {
|
||||
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (schema.Length > 0 && stag.Length > schema.Length) {
|
||||
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (schema.Value != null && stag.Data != schema.Value) {
|
||||
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
private bool VerifyArray (TagNode tag, SchemaNodeArray schema)
|
||||
{
|
||||
TagNodeByteArray atag = tag as TagNodeByteArray;
|
||||
if (atag == null) {
|
||||
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (schema.Length > 0 && atag.Length != schema.Length) {
|
||||
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private bool VerifyList (TagNode tag, SchemaNodeList schema)
|
||||
{
|
||||
TagNodeList ltag = tag as TagNodeList;
|
||||
if (ltag == null) {
|
||||
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (ltag.Count > 0 && ltag.ValueType != schema.Type) {
|
||||
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
if (schema.Length > 0 && ltag.Count != schema.Length) {
|
||||
if (!OnInvalidTagValue(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
// Patch up empty lists
|
||||
//if (schema.Length == 0) {
|
||||
// tag = new NBT_List(schema.Type);
|
||||
//}
|
||||
|
||||
bool pass = true;
|
||||
|
||||
// If a subschema is set, test all items in list against it
|
||||
|
||||
if (schema.SubSchema != null) {
|
||||
foreach (TagNode v in ltag) {
|
||||
pass = Verify(tag, v, schema.SubSchema) && pass;
|
||||
}
|
||||
}
|
||||
|
||||
return pass;
|
||||
}
|
||||
|
||||
private bool VerifyCompound (TagNode tag, SchemaNodeCompound schema)
|
||||
{
|
||||
TagNodeCompound ctag = tag as TagNodeCompound;
|
||||
if (ctag == null) {
|
||||
if (!OnInvalidTagType(new TagEventArgs(schema, tag))) {
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
bool pass = true;
|
||||
|
||||
foreach (SchemaNode node in schema) {
|
||||
TagNode value;
|
||||
ctag.TryGetValue(node.Name, out value);
|
||||
|
||||
if (value == null) {
|
||||
if ((node.Options & SchemaOptions.CREATE_ON_MISSING) == SchemaOptions.CREATE_ON_MISSING) {
|
||||
_scratch[node.Name] = node.BuildDefaultTree();
|
||||
continue;
|
||||
}
|
||||
else if ((node.Options & SchemaOptions.OPTIONAL) == SchemaOptions.OPTIONAL) {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
pass = Verify(tag, value, node) && pass;
|
||||
}
|
||||
|
||||
foreach (KeyValuePair<string, TagNode> item in _scratch) {
|
||||
ctag[item.Key] = item.Value;
|
||||
}
|
||||
|
||||
_scratch.Clear();
|
||||
|
||||
return pass;
|
||||
}
|
||||
|
||||
#region Event Handlers
|
||||
|
||||
/// <summary>
|
||||
/// Processes registered events for <see cref="MissingTag"/> whenever an expected <see cref="TagNode"/> is not found.
|
||||
/// </summary>
|
||||
/// <param name="e">Arguments for this event.</param>
|
||||
/// <returns>Status indicating whether this event can be ignored.</returns>
|
||||
protected virtual bool OnMissingTag (TagEventArgs e)
|
||||
{
|
||||
if (MissingTag != null) {
|
||||
foreach (VerifierEventHandler func in MissingTag.GetInvocationList()) {
|
||||
TagEventCode code = func(e);
|
||||
switch (code) {
|
||||
case TagEventCode.FAIL:
|
||||
return false;
|
||||
case TagEventCode.PASS:
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Processes registered events for <see cref="InvalidTagType"/> whenever an expected <see cref="TagNode"/> is of the wrong type and cannot be cast.
|
||||
/// </summary>
|
||||
/// <param name="e">Arguments for this event.</param>
|
||||
/// <returns>Status indicating whether this event can be ignored.</returns>
|
||||
protected virtual bool OnInvalidTagType (TagEventArgs e)
|
||||
{
|
||||
if (InvalidTagType != null) {
|
||||
foreach (VerifierEventHandler func in InvalidTagType.GetInvocationList()) {
|
||||
TagEventCode code = func(e);
|
||||
switch (code) {
|
||||
case TagEventCode.FAIL:
|
||||
return false;
|
||||
case TagEventCode.PASS:
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Processes registered events for <see cref="InvalidTagValue"/> whenever an expected <see cref="TagNode"/> has a value that violates the schema.
|
||||
/// </summary>
|
||||
/// <param name="e">Arguments for this event.</param>
|
||||
/// <returns>Status indicating whether this event can be ignored.</returns>
|
||||
protected virtual bool OnInvalidTagValue (TagEventArgs e)
|
||||
{
|
||||
if (InvalidTagValue != null) {
|
||||
foreach (VerifierEventHandler func in InvalidTagValue.GetInvocationList()) {
|
||||
TagEventCode code = func(e);
|
||||
switch (code) {
|
||||
case TagEventCode.FAIL:
|
||||
return false;
|
||||
case TagEventCode.PASS:
|
||||
return true;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
60
Substrate/SubstrateCS/Source/Nbt/SchemaNode.cs
Normal file
60
Substrate/SubstrateCS/Source/Nbt/SchemaNode.cs
Normal file
|
@ -0,0 +1,60 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Text;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// A node in an NBT schema definition, used to define what values are considered valid for a given NBT node.
|
||||
/// </summary>
|
||||
public abstract class SchemaNode
|
||||
{
|
||||
private string _name;
|
||||
private SchemaOptions _options;
|
||||
|
||||
/// <summary>
|
||||
/// Gets the name of an expected NBT node.
|
||||
/// </summary>
|
||||
public string Name
|
||||
{
|
||||
get { return _name; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets additional schema options defined for this node.
|
||||
/// </summary>
|
||||
public SchemaOptions Options
|
||||
{
|
||||
get { return _options; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNode"/> representing a <see cref="TagNode"/> named <paramref name="name"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNode"/>.</param>
|
||||
protected SchemaNode (string name)
|
||||
{
|
||||
_name = name;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNode"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNode"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
protected SchemaNode (string name, SchemaOptions options)
|
||||
{
|
||||
_name = name;
|
||||
_options = options;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Construct a sensible default NBT tree representative of this schema node.
|
||||
/// </summary>
|
||||
/// <returns>A <see cref="TagNode"/> that is valid for this schema node.</returns>
|
||||
public virtual TagNode BuildDefaultTree ()
|
||||
{
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
81
Substrate/SubstrateCS/Source/Nbt/SchemaNodeArray.cs
Normal file
81
Substrate/SubstrateCS/Source/Nbt/SchemaNodeArray.cs
Normal file
|
@ -0,0 +1,81 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// A concrete <see cref="SchemaNode"/> representing a <see cref="TagNodeByteArray"/>.
|
||||
/// </summary>
|
||||
public sealed class SchemaNodeArray : SchemaNode
|
||||
{
|
||||
private int _length;
|
||||
|
||||
/// <summary>
|
||||
/// Gets the expected length of the corresponding byte array.
|
||||
/// </summary>
|
||||
public int Length
|
||||
{
|
||||
get { return _length; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Indicates whether there is an expected length of the corresponding byte array.
|
||||
/// </summary>
|
||||
public bool HasExpectedLength
|
||||
{
|
||||
get { return _length > 0; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeArray"/> representing a <see cref="TagNodeByteArray"/> named <paramref name="name"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeByteArray"/>.</param>
|
||||
public SchemaNodeArray (string name)
|
||||
: base(name)
|
||||
{
|
||||
_length = 0;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeArray"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeByteArray"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeArray (string name, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_length = 0;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeArray"/> representing a <see cref="TagNodeByteArray"/> named <paramref name="name"/> with expected length <paramref name="length"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeByteArray"/>.</param>
|
||||
/// <param name="length">The expected length of corresponding byte array.</param>
|
||||
public SchemaNodeArray (string name, int length)
|
||||
: base(name)
|
||||
{
|
||||
_length = length;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeArray"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeByteArray"/>.</param>
|
||||
/// <param name="length">The expected length of corresponding byte array.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeArray (string name, int length, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_length = length;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a default <see cref="TagNodeByteArray"/> satisfying the constraints of this node.
|
||||
/// </summary>
|
||||
/// <returns>A <see cref="TagNodeString"/> with a sensible default value.</returns>
|
||||
public override TagNode BuildDefaultTree ()
|
||||
{
|
||||
return new TagNodeByteArray(new byte[_length]);
|
||||
}
|
||||
}
|
||||
}
|
219
Substrate/SubstrateCS/Source/Nbt/SchemaNodeCompound.cs
Normal file
219
Substrate/SubstrateCS/Source/Nbt/SchemaNodeCompound.cs
Normal file
|
@ -0,0 +1,219 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// A concrete <see cref="SchemaNode"/> representing a <see cref="TagNodeCompound"/>.
|
||||
/// </summary>
|
||||
public sealed class SchemaNodeCompound : SchemaNode, ICollection<SchemaNode>
|
||||
{
|
||||
private List<SchemaNode> _subnodes;
|
||||
|
||||
#region ICollection<NBTSchemaNode> Members
|
||||
|
||||
/// <summary>
|
||||
/// Adds a <see cref="SchemaNode"/> as a child of this node.
|
||||
/// </summary>
|
||||
/// <param name="item">The <see cref="SchemaNode"/> to add.</param>
|
||||
public void Add (SchemaNode item)
|
||||
{
|
||||
_subnodes.Add(item);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes all <see cref="SchemaNode"/> objects from the node.
|
||||
/// </summary>
|
||||
public void Clear ()
|
||||
{
|
||||
_subnodes.Clear();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if a <see cref="SchemaNode"/> is a child of this node.
|
||||
/// </summary>
|
||||
/// <param name="item">The <see cref="SchemaNode"/> to check for existance.</param>
|
||||
/// <returns>Status indicating if the <see cref="SchemaNode"/> exists as a child of this node.</returns>
|
||||
public bool Contains (SchemaNode item)
|
||||
{
|
||||
return _subnodes.Contains(item);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Copies all child <see cref="SchemaNode"/> objects of this node to a compatible one-dimensional array, starting at the specified index of the target array.
|
||||
/// </summary>
|
||||
/// <param name="array">The one-dimensional <see cref="Array"/> that is the destination of the subnodes copied. The Array must have zero-based indexing.</param>
|
||||
/// <param name="arrayIndex">The zero-based index in <paramref name="array"/> at which copying begins.</param>
|
||||
public void CopyTo (SchemaNode[] array, int arrayIndex)
|
||||
{
|
||||
_subnodes.CopyTo(array, arrayIndex);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the number of child <see cref="SchemaNode"/> objects in this node.
|
||||
/// </summary>
|
||||
public int Count
|
||||
{
|
||||
get { return _subnodes.Count; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a value indicating whether the node is readonly.
|
||||
/// </summary>
|
||||
public bool IsReadOnly
|
||||
{
|
||||
get { return false; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes the first occurance of a <see cref="SchemaNode"/> from this node.
|
||||
/// </summary>
|
||||
/// <param name="item">The <see cref="SchemaNode"/> to remove.</param>
|
||||
/// <returns>Status indicating whether a <see cref="SchemaNode"/> was removed.</returns>
|
||||
public bool Remove (SchemaNode item)
|
||||
{
|
||||
return _subnodes.Remove(item);
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region IEnumerable<SchemaNode> Members
|
||||
|
||||
/// <summary>
|
||||
/// Iterates through all of the <see cref="SchemaNode"/> objects in this <see cref="SchemaNodeCompound"/>.
|
||||
/// </summary>
|
||||
/// <returns>An enumerator for this node.</returns>
|
||||
public IEnumerator<SchemaNode> GetEnumerator ()
|
||||
{
|
||||
return _subnodes.GetEnumerator();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region IEnumerable Members
|
||||
|
||||
/// <summary>
|
||||
/// Iterates through all of the <see cref="SchemaNode"/> objects in this <see cref="SchemaNodeCompound"/>.
|
||||
/// </summary>
|
||||
/// <returns>An enumerator for this node.</returns>
|
||||
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator ()
|
||||
{
|
||||
return _subnodes.GetEnumerator();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeCompound"/> representing a root <see cref="TagNodeCompound"/>.
|
||||
/// </summary>
|
||||
public SchemaNodeCompound ()
|
||||
: base("")
|
||||
{
|
||||
_subnodes = new List<SchemaNode>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeCompound"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeCompound (SchemaOptions options)
|
||||
: base("", options)
|
||||
{
|
||||
_subnodes = new List<SchemaNode>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeCompound"/> representing a <see cref="TagNodeCompound"/> named <paramref name="name"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeCompound"/>.</param>
|
||||
public SchemaNodeCompound (string name)
|
||||
: base(name)
|
||||
{
|
||||
_subnodes = new List<SchemaNode>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeCompound"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeCompound"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeCompound (string name, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_subnodes = new List<SchemaNode>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeCompound"/> representing a <see cref="TagNodeCompound"/> named <paramref name="name"/> matching the given schema.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeCompound"/>.</param>
|
||||
/// <param name="subschema">A <see cref="SchemaNodeCompound"/> representing a schema to verify against the corresponding <see cref="TagNodeCompound"/>.</param>
|
||||
public SchemaNodeCompound (string name, SchemaNode subschema)
|
||||
: base(name)
|
||||
{
|
||||
_subnodes = new List<SchemaNode>();
|
||||
|
||||
SchemaNodeCompound schema = subschema as SchemaNodeCompound;
|
||||
if (schema == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
foreach (SchemaNode node in schema._subnodes) {
|
||||
_subnodes.Add(node);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeCompound"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeCompound"/>.</param>
|
||||
/// <param name="subschema">A <see cref="SchemaNodeCompound"/> representing a schema to verify against the corresponding <see cref="TagNodeCompound"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeCompound (string name, SchemaNode subschema, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_subnodes = new List<SchemaNode>();
|
||||
|
||||
SchemaNodeCompound schema = subschema as SchemaNodeCompound;
|
||||
if (schema == null) {
|
||||
return;
|
||||
}
|
||||
|
||||
foreach (SchemaNode node in schema._subnodes) {
|
||||
_subnodes.Add(node);
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Copies all the elements of this <see cref="SchemaNodeCompound"/> into <paramref name="tree"/>.
|
||||
/// </summary>
|
||||
/// <param name="tree">The destination <see cref="SchemaNodeCompound"/> to copy <see cref="SchemaNode"/> elements into.</param>
|
||||
/// <returns>A reference to <paramref name="tree"/>.</returns>
|
||||
public SchemaNodeCompound MergeInto (SchemaNodeCompound tree)
|
||||
{
|
||||
foreach (SchemaNode node in _subnodes) {
|
||||
SchemaNode f = tree._subnodes.Find(n => n.Name == node.Name);
|
||||
if (f != null) {
|
||||
continue;
|
||||
}
|
||||
tree.Add(node);
|
||||
}
|
||||
|
||||
return tree;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a default <see cref="TagNodeCompound"/> satisfying the constraints of this node.
|
||||
/// </summary>
|
||||
/// <returns>A <see cref="TagNodeCompound"/> with a sensible default value. A default child <see cref="TagNode"/> is created for every <see cref="SchemaNode"/> contained in this <see cref="SchemaNodeCompound"/>.</returns>
|
||||
public override TagNode BuildDefaultTree ()
|
||||
{
|
||||
TagNodeCompound list = new TagNodeCompound();
|
||||
foreach (SchemaNode node in _subnodes) {
|
||||
list[node.Name] = node.BuildDefaultTree();
|
||||
}
|
||||
|
||||
return list;
|
||||
}
|
||||
}
|
||||
}
|
172
Substrate/SubstrateCS/Source/Nbt/SchemaNodeList.cs
Normal file
172
Substrate/SubstrateCS/Source/Nbt/SchemaNodeList.cs
Normal file
|
@ -0,0 +1,172 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// A concrete <see cref="SchemaNode"/> representing a <see cref="TagNodeList"/>.
|
||||
/// </summary>
|
||||
public sealed class SchemaNodeList : SchemaNode
|
||||
{
|
||||
private TagType _type;
|
||||
private int _length;
|
||||
private SchemaNode _subschema;
|
||||
|
||||
/// <summary>
|
||||
/// Gets the expected number of items contained in the corresponding <see cref="TagNodeList"/>.
|
||||
/// </summary>
|
||||
public int Length
|
||||
{
|
||||
get { return _length; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the expected <see cref="TagType"/> of the items contained in the corresponding <see cref="TagNodeList"/>.
|
||||
/// </summary>
|
||||
public TagType Type
|
||||
{
|
||||
get { return _type; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a <see cref="SchemaNode"/> representing a schema that items contained in the corresponding <see cref="TagNodeList"/> should be verified against.
|
||||
/// </summary>
|
||||
public SchemaNode SubSchema
|
||||
{
|
||||
get { return _subschema; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Indicates whether there is an expected number of items of the corresponding <see cref="TagNodeList"/>.
|
||||
/// </summary>
|
||||
public bool HasExpectedLength
|
||||
{
|
||||
get { return _length > 0; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> representing a <see cref="TagNodeList"/> named <paramref name="name"/> containing items of type <paramref name="type"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
public SchemaNodeList (string name, TagType type)
|
||||
: base(name)
|
||||
{
|
||||
_type = type;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeList (string name, TagType type, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_type = type;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> representing a <see cref="TagNodeList"/> named <paramref name="name"/> containing <paramref name="length"/> items of type <paramref name="type"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="length">The number of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
public SchemaNodeList (string name, TagType type, int length)
|
||||
: base(name)
|
||||
{
|
||||
_type = type;
|
||||
_length = length;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="length">The number of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeList (string name, TagType type, int length, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_type = type;
|
||||
_length = length;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> representing a <see cref="TagNodeList"/> named <paramref name="name"/> containing items of type <paramref name="type"/> matching the given schema.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="subschema">A <see cref="SchemaNode"/> representing a schema to verify against items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
public SchemaNodeList (string name, TagType type, SchemaNode subschema)
|
||||
: base(name)
|
||||
{
|
||||
_type = type;
|
||||
_subschema = subschema;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="subschema">A <see cref="SchemaNode"/> representing a schema to verify against items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeList (string name, TagType type, SchemaNode subschema, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_type = type;
|
||||
_subschema = subschema;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> representing a <see cref="TagNodeList"/> named <paramref name="name"/> containing <paramref name="length"/> items of type <paramref name="type"/> matching the given schema.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="length">The number of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="subschema">A <see cref="SchemaNode"/> representing a schema to verify against items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
public SchemaNodeList (string name, TagType type, int length, SchemaNode subschema)
|
||||
: base(name)
|
||||
{
|
||||
_type = type;
|
||||
_length = length;
|
||||
_subschema = subschema;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeList"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="type">The type of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="length">The number of items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="subschema">A <see cref="SchemaNode"/> representing a schema to verify against items contained in the corresponding <see cref="TagNodeList"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeList (string name, TagType type, int length, SchemaNode subschema, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_type = type;
|
||||
_length = length;
|
||||
_subschema = subschema;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a default <see cref="TagNodeList"/> satisfying the constraints of this node.
|
||||
/// </summary>
|
||||
/// <returns>A <see cref="TagNodeList"/> with a sensible default value. If a length is specified, default child <see cref="TagNode"/> objects of the necessary type will be created and added to the <see cref="TagNodeList"/>.</returns>
|
||||
public override TagNode BuildDefaultTree ()
|
||||
{
|
||||
if (_length == 0) {
|
||||
return new TagNodeList(_type);
|
||||
}
|
||||
|
||||
TagNodeList list = new TagNodeList(_type);
|
||||
for (int i = 0; i < _length; i++) {
|
||||
list.Add(_subschema.BuildDefaultTree());
|
||||
}
|
||||
|
||||
return list;
|
||||
}
|
||||
}
|
||||
}
|
75
Substrate/SubstrateCS/Source/Nbt/SchemaNodeScaler.cs
Normal file
75
Substrate/SubstrateCS/Source/Nbt/SchemaNodeScaler.cs
Normal file
|
@ -0,0 +1,75 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// A concrete <see cref="SchemaNode"/> representing a scaler-type <see cref="TagNode"/>.
|
||||
/// </summary>
|
||||
public sealed class SchemaNodeScaler : SchemaNode
|
||||
{
|
||||
private TagType _type;
|
||||
|
||||
/// <summary>
|
||||
/// Gets the scaler <see cref="TagType"/> that this node represents.
|
||||
/// </summary>
|
||||
public TagType Type
|
||||
{
|
||||
get { return _type; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeScaler"/> representing a <see cref="TagNode"/> named <paramref name="name"/> and of type <paramref name="type"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNode"/>.</param>
|
||||
/// <param name="type">The type of the corresponding <see cref="TagNode"/>, restricted to scaler types.</param>
|
||||
public SchemaNodeScaler (string name, TagType type)
|
||||
: base(name)
|
||||
{
|
||||
_type = type;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeScaler"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNode"/>.</param>
|
||||
/// <param name="type">The type of the corresponding <see cref="TagNode"/>, restricted to scaler types.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeScaler (string name, TagType type, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_type = type;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a default <see cref="TagNode"/> according to the <see cref="TagType"/> this node represents.
|
||||
/// </summary>
|
||||
/// <returns>A <see cref="TagNode"/> with a sensible default value.</returns>
|
||||
public override TagNode BuildDefaultTree ()
|
||||
{
|
||||
switch (_type) {
|
||||
case TagType.TAG_STRING:
|
||||
return new TagNodeString();
|
||||
|
||||
case TagType.TAG_BYTE:
|
||||
return new TagNodeByte();
|
||||
|
||||
case TagType.TAG_SHORT:
|
||||
return new TagNodeShort();
|
||||
|
||||
case TagType.TAG_INT:
|
||||
return new TagNodeInt();
|
||||
|
||||
case TagType.TAG_LONG:
|
||||
return new TagNodeLong();
|
||||
|
||||
case TagType.TAG_FLOAT:
|
||||
return new TagNodeFloat();
|
||||
|
||||
case TagType.TAG_DOUBLE:
|
||||
return new TagNodeDouble();
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
116
Substrate/SubstrateCS/Source/Nbt/SchemaNodeString.cs
Normal file
116
Substrate/SubstrateCS/Source/Nbt/SchemaNodeString.cs
Normal file
|
@ -0,0 +1,116 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// A concrete <see cref="SchemaNode"/> representing a <see cref="TagNodeString"/>.
|
||||
/// </summary>
|
||||
public sealed class SchemaNodeString : SchemaNode
|
||||
{
|
||||
private string _value = "";
|
||||
private int _length;
|
||||
|
||||
/// <summary>
|
||||
/// Gets the maximum length of a valid string.
|
||||
/// </summary>
|
||||
public int Length
|
||||
{
|
||||
get { return _length; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the expected value of a valid string.
|
||||
/// </summary>
|
||||
/// <remarks>A <see cref="TagNodeString"/> must be set to this value to be considered valid.</remarks>
|
||||
public string Value
|
||||
{
|
||||
get { return _value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Indicates whether there is a maximum-length constraint on strings in this node.
|
||||
/// </summary>
|
||||
public bool HasMaxLength
|
||||
{
|
||||
get { return _length > 0; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeString"/> representing a <see cref="TagNodeString"/> named <paramref name="name"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeString"/>.</param>
|
||||
public SchemaNodeString (string name)
|
||||
: base(name)
|
||||
{
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeString"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeString"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeString (string name, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeString"/> representing a <see cref="TagNodeString"/> named <paramref name="name"/> set to <paramref name="value"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeString"/>.</param>
|
||||
/// <param name="value">The value that the corresponding <see cref="TagNodeString"/> must be set to.</param>
|
||||
public SchemaNodeString (string name, string value)
|
||||
: base(name)
|
||||
{
|
||||
_value = value;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeString"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeString"/>.</param>
|
||||
/// <param name="value">The value that the corresponding <see cref="TagNodeString"/> must be set to.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeString (string name, string value, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_value = value;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeString"/> representing a <see cref="TagNodeString"/> named <paramref name="name"/> with maximum length <paramref name="length"/>.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeString"/>.</param>
|
||||
/// <param name="length">The maximum length of strings in the corresponding <see cref="TagNodeString"/>.</param>
|
||||
public SchemaNodeString (string name, int length)
|
||||
: base(name)
|
||||
{
|
||||
_length = length;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new <see cref="SchemaNodeString"/> with additional options.
|
||||
/// </summary>
|
||||
/// <param name="name">The name of the corresponding <see cref="TagNodeString"/>.</param>
|
||||
/// <param name="length">The maximum length of strings in the corresponding <see cref="TagNodeString"/>.</param>
|
||||
/// <param name="options">One or more option flags modifying the processing of this node.</param>
|
||||
public SchemaNodeString (string name, int length, SchemaOptions options)
|
||||
: base(name, options)
|
||||
{
|
||||
_length = length;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a default <see cref="TagNodeString"/> satisfying the constraints of this node.
|
||||
/// </summary>
|
||||
/// <returns>A <see cref="TagNodeString"/> with a sensible default value. If this node represents a particular string, the <see cref="TagNodeString"/> constructed will be set to that string.</returns>
|
||||
public override TagNode BuildDefaultTree ()
|
||||
{
|
||||
if (_value.Length > 0) {
|
||||
return new TagNodeString(_value);
|
||||
}
|
||||
|
||||
return new TagNodeString();
|
||||
}
|
||||
}
|
||||
}
|
21
Substrate/SubstrateCS/Source/Nbt/SchemaOptions.cs
Normal file
21
Substrate/SubstrateCS/Source/Nbt/SchemaOptions.cs
Normal file
|
@ -0,0 +1,21 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// Additional options that modify the processing of a <see cref="SchemaNode"/>.
|
||||
/// </summary>
|
||||
[Flags]
|
||||
public enum SchemaOptions
|
||||
{
|
||||
/// <summary>
|
||||
/// Any <see cref="SchemaNode"/> with this option will not throw an error if the corresponding <see cref="TagNode"/> is missing.
|
||||
/// </summary>
|
||||
OPTIONAL = 0x1,
|
||||
|
||||
/// <summary>
|
||||
/// If a <see cref="TagNode"/> cannot be found for a <see cref="SchemaNode"/> marked with this option, a sensible default <see cref="TagNode"/> will be created and inserted into the tree.
|
||||
/// </summary>
|
||||
CREATE_ON_MISSING = 0x2,
|
||||
}
|
||||
}
|
140
Substrate/SubstrateCS/Source/Nbt/TagNode.cs
Normal file
140
Substrate/SubstrateCS/Source/Nbt/TagNode.cs
Normal file
|
@ -0,0 +1,140 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
using System.Text;
|
||||
using Substrate.Core;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An abstract base class representing a node in an NBT tree.
|
||||
/// </summary>
|
||||
public abstract class TagNode : ICopyable<TagNode>
|
||||
{
|
||||
/// <summary>
|
||||
/// Convert this node to a null tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new null node.</returns>
|
||||
public virtual TagNodeNull ToTagNull ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a byte tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new byte node.</returns>
|
||||
public virtual TagNodeByte ToTagByte ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a short tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new short node.</returns>
|
||||
public virtual TagNodeShort ToTagShort ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to an int tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new int node.</returns>
|
||||
public virtual TagNodeInt ToTagInt ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a long tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new long node.</returns>
|
||||
public virtual TagNodeLong ToTagLong ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a float tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new float node.</returns>
|
||||
public virtual TagNodeFloat ToTagFloat ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a double tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new double node.</returns>
|
||||
public virtual TagNodeDouble ToTagDouble ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a byte array tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new byte array node.</returns>
|
||||
public virtual TagNodeByteArray ToTagByteArray ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a string tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new string node.</returns>
|
||||
public virtual TagNodeString ToTagString ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a list tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new list node.</returns>
|
||||
public virtual TagNodeList ToTagList ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Convert this node to a compound tag type if supported.
|
||||
/// </summary>
|
||||
/// <returns>A new compound node.</returns>
|
||||
public virtual TagNodeCompound ToTagCompound ()
|
||||
{
|
||||
throw new InvalidCastException();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the underlying tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>An NBT tag type.</returns>
|
||||
public virtual TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_END;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if this node is castable to another node of a given tag type.
|
||||
/// </summary>
|
||||
/// <param name="type">An NBT tag type.</param>
|
||||
/// <returns>Status indicating whether this object could be cast to a node type represented by the given tag type.</returns>
|
||||
public virtual bool IsCastableTo (TagType type)
|
||||
{
|
||||
return type == GetTagType();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the NBT node.
|
||||
/// </summary>
|
||||
/// <returns>A new NBT node.</returns>
|
||||
public virtual TagNode Copy ()
|
||||
{
|
||||
return null;
|
||||
}
|
||||
}
|
||||
}
|
161
Substrate/SubstrateCS/Source/Nbt/TagNodeByte.cs
Normal file
161
Substrate/SubstrateCS/Source/Nbt/TagNodeByte.cs
Normal file
|
@ -0,0 +1,161 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a signed byte tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeByte : TagNode
|
||||
{
|
||||
private byte _data = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeByte ToTagByte ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to a new short node.
|
||||
/// </summary>
|
||||
/// <returns>A short node representing the same data.</returns>
|
||||
public override TagNodeShort ToTagShort ()
|
||||
{
|
||||
return new TagNodeShort(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to a new int node.
|
||||
/// </summary>
|
||||
/// <returns>An int node representing the same data.</returns>
|
||||
public override TagNodeInt ToTagInt ()
|
||||
{
|
||||
return new TagNodeInt(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to a new long node.
|
||||
/// </summary>
|
||||
/// <returns>A long node representing the same data.</returns>
|
||||
public override TagNodeLong ToTagLong ()
|
||||
{
|
||||
return new TagNodeLong(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_BYTE tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_BYTE;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if the node is castable to another node of a given tag type.
|
||||
/// </summary>
|
||||
/// <param name="type">An NBT tag type.</param>
|
||||
/// <returns>Status indicating whether this object could be cast to a node type represented by the given tag type.</returns>
|
||||
public override bool IsCastableTo (TagType type)
|
||||
{
|
||||
return (type == TagType.TAG_BYTE ||
|
||||
type == TagType.TAG_SHORT ||
|
||||
type == TagType.TAG_INT ||
|
||||
type == TagType.TAG_LONG);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a byte of tag data.
|
||||
/// </summary>
|
||||
public byte Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new byte node with a data value of 0.
|
||||
/// </summary>
|
||||
public TagNodeByte () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new byte node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeByte (byte d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new byte node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeByte(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system byte to a byte node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte value.</param>
|
||||
/// <returns>A new byte node containing the given value.</returns>
|
||||
public static implicit operator TagNodeByte (byte b)
|
||||
{
|
||||
return new TagNodeByte(b);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a byte node to a system byte representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte node.</param>
|
||||
/// <returns>A system byte set to the node's data value.</returns>
|
||||
public static implicit operator byte (TagNodeByte b)
|
||||
{
|
||||
return b._data;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a byte node to a system short representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte node.</param>
|
||||
/// <returns>A system short set to the node's data value.</returns>
|
||||
public static implicit operator short (TagNodeByte b)
|
||||
{
|
||||
return b._data;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a byte node to a system int representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte node.</param>
|
||||
/// <returns>A system int set to the node's data value.</returns>
|
||||
public static implicit operator int (TagNodeByte b)
|
||||
{
|
||||
return b._data;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a byte node to a system long representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte node.</param>
|
||||
/// <returns>A system long set to the node's data value.</returns>
|
||||
public static implicit operator long (TagNodeByte b)
|
||||
{
|
||||
return b._data;
|
||||
}
|
||||
}
|
||||
}
|
113
Substrate/SubstrateCS/Source/Nbt/TagNodeByteArray.cs
Normal file
113
Substrate/SubstrateCS/Source/Nbt/TagNodeByteArray.cs
Normal file
|
@ -0,0 +1,113 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing an unsigned byte array tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeByteArray : TagNode
|
||||
{
|
||||
private byte[] _data = null;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeByteArray ToTagByteArray ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_BYTE_ARRAY tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_BYTE_ARRAY;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a byte array of tag data.
|
||||
/// </summary>
|
||||
public byte[] Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the length of the stored byte array.
|
||||
/// </summary>
|
||||
public int Length
|
||||
{
|
||||
get { return _data.Length; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new byte array node with a null data value.
|
||||
/// </summary>
|
||||
public TagNodeByteArray () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new byte array node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeByteArray (byte[] d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new byte array node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
byte[] arr = new byte[_data.Length];
|
||||
_data.CopyTo(arr, 0);
|
||||
|
||||
return new TagNodeByteArray(arr);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a single byte at the specified index.
|
||||
/// </summary>
|
||||
/// <param name="index">Valid index within stored byte array.</param>
|
||||
/// <returns>The byte value at the given index of the stored byte array.</returns>
|
||||
public byte this[int index]
|
||||
{
|
||||
get { return _data[index]; }
|
||||
set { _data[index] = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system byte array to a byte array node representing the same data.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte array.</param>
|
||||
/// <returns>A new byte array node containing the given value.</returns>
|
||||
public static implicit operator TagNodeByteArray (byte[] b)
|
||||
{
|
||||
return new TagNodeByteArray(b);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a byte array node to a system byte array representing the same data.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte array node.</param>
|
||||
/// <returns>A system byte array set to the node's data.</returns>
|
||||
public static implicit operator byte[] (TagNodeByteArray b)
|
||||
{
|
||||
return b._data;
|
||||
}
|
||||
}
|
||||
}
|
265
Substrate/SubstrateCS/Source/Nbt/TagNodeCompound.cs
Normal file
265
Substrate/SubstrateCS/Source/Nbt/TagNodeCompound.cs
Normal file
|
@ -0,0 +1,265 @@
|
|||
using System;
|
||||
using System.Collections;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a compound tag type containing other nodes.
|
||||
/// </summary>
|
||||
public sealed class TagNodeCompound : TagNode, IDictionary<string, TagNode>
|
||||
{
|
||||
private Dictionary<string, TagNode> _tags;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeCompound ToTagCompound ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_STRING tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_COMPOUND;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the number of subnodes contained in node.
|
||||
/// </summary>
|
||||
public int Count
|
||||
{
|
||||
get { return _tags.Count; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new empty compound node.
|
||||
/// </summary>
|
||||
public TagNodeCompound ()
|
||||
{
|
||||
_tags = new Dictionary<string, TagNode>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new compound node containing new subnodes representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
TagNodeCompound list = new TagNodeCompound();
|
||||
foreach (KeyValuePair<string, TagNode> item in _tags) {
|
||||
list[item.Key] = item.Value.Copy();
|
||||
}
|
||||
return list;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _tags.ToString();
|
||||
}
|
||||
|
||||
#region IDictionary<string,NBT_Value> Members
|
||||
|
||||
/// <summary>
|
||||
/// Adds a named subnode to the set.
|
||||
/// </summary>
|
||||
/// <param name="key">The name of the subnode.</param>
|
||||
/// <param name="value">The subnode to add.</param>
|
||||
/// <exception cref="ArgumentNullException"><paramref name="key"/> is null.</exception>
|
||||
/// <exception cref="ArgumentException">A subnode with the same key already exists in the set.</exception>
|
||||
public void Add (string key, TagNode value)
|
||||
{
|
||||
_tags.Add(key, value);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if a subnode exists in the set with the specified name.
|
||||
/// </summary>
|
||||
/// <param name="key">The name of a subnode to check.</param>
|
||||
/// <returns>Status indicating whether a subnode with the specified name exists.</returns>
|
||||
/// <exception cref="ArgumentNullException"><paramref name="key"/> is null.</exception>
|
||||
public bool ContainsKey (string key)
|
||||
{
|
||||
return _tags.ContainsKey(key);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a collection containing all the names of subnodes in this set.
|
||||
/// </summary>
|
||||
public ICollection<string> Keys
|
||||
{
|
||||
get { return _tags.Keys; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes a subnode with the specified name.
|
||||
/// </summary>
|
||||
/// <param name="key">The name of the subnode to remove.</param>
|
||||
/// <returns>Status indicating whether a subnode was removed.</returns>
|
||||
/// <exception cref="ArgumentNullException"><paramref name="key"/> is null.</exception>
|
||||
public bool Remove (string key)
|
||||
{
|
||||
return _tags.Remove(key);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the subnode associated with the given name.
|
||||
/// </summary>
|
||||
/// <param name="key">The name of the subnode to get.</param>
|
||||
/// <param name="value">When the function returns, contains the subnode assicated with the specified key. If no subnode was found, contains a default value.</param>
|
||||
/// <returns>Status indicating whether a subnode was found.</returns>
|
||||
/// <exception cref="ArgumentNullException"><paramref name="key"/> is null.</exception>
|
||||
public bool TryGetValue (string key, out TagNode value)
|
||||
{
|
||||
return _tags.TryGetValue(key, out value);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a collection containing all the subnodes in this set.
|
||||
/// </summary>
|
||||
public ICollection<TagNode> Values
|
||||
{
|
||||
get { return _tags.Values; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the subnode with the associated name.
|
||||
/// </summary>
|
||||
/// <param name="key">The name of the subnode to get or set.</param>
|
||||
/// <exception cref="ArgumentNullException"><paramref name="key"/> is null.</exception>
|
||||
/// <exception cref="KeyNotFoundException">The property is retrieved and key does not exist in the collection.</exception>
|
||||
public TagNode this[string key]
|
||||
{
|
||||
get
|
||||
{
|
||||
return _tags[key];
|
||||
}
|
||||
set
|
||||
{
|
||||
_tags[key] = value;
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region ICollection<KeyValuePair<string,NBT_Value>> Members
|
||||
|
||||
/// <summary>
|
||||
/// Adds a subnode to the to the set with the specified name.
|
||||
/// </summary>
|
||||
/// <param name="item">The <see cref="KeyValuePair{TKey, TVal}"/> structure representing the key and subnode to add to the set.</param>
|
||||
/// <exception cref="ArgumentNullException">The key of <paramref name="item"/> is null.</exception>
|
||||
/// <exception cref="ArgumentException">A subnode with the same key already exists in the set.</exception>
|
||||
public void Add (KeyValuePair<string, TagNode> item)
|
||||
{
|
||||
_tags.Add(item.Key, item.Value);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes all of the subnodes from this node.
|
||||
/// </summary>
|
||||
public void Clear ()
|
||||
{
|
||||
_tags.Clear();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if a specific subnode with a specific name is contained in the set.
|
||||
/// </summary>
|
||||
/// <param name="item">The <see cref="KeyValuePair{TKey, TValue}"/> structure representing the key and subnode to look for.</param>
|
||||
/// <returns>Status indicating if the subnode and key combination exists in the set.</returns>
|
||||
public bool Contains (KeyValuePair<string, TagNode> item)
|
||||
{
|
||||
TagNode value;
|
||||
if (!_tags.TryGetValue(item.Key, out value)) {
|
||||
return false;
|
||||
}
|
||||
return value == item.Value;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Copies the elements of the <see cref="ICollection{T}"/> to an array of type <see cref="KeyValuePair{TKey, TVal}"/>, starting at the specified array index.
|
||||
/// </summary>
|
||||
/// <param name="array">The one-dimensional <see cref="Array"/> that is the destination of the subnodes copied. The Array must have zero-based indexing.</param>
|
||||
/// <param name="arrayIndex">The zero-based index in <paramref name="array"/> at which copying begins.</param>
|
||||
/// <exception cref="ArgumentNullException"><paramref name="array"/> is null.</exception>
|
||||
/// <exception cref="ArgumentOutOfRangeException"><paramref name="arrayIndex"/> is less than 0.</exception>
|
||||
/// <exception cref="ArgumentException">The number of elements in the source <see cref="ICollection{T}"/> is greater than the available space from <paramref name="arrayIndex"/> to the end of the destination <paramref name="array"/>.</exception>
|
||||
public void CopyTo (KeyValuePair<string, TagNode>[] array, int arrayIndex)
|
||||
{
|
||||
if (array == null) {
|
||||
throw new ArgumentNullException();
|
||||
}
|
||||
if (arrayIndex < 0) {
|
||||
throw new ArgumentOutOfRangeException();
|
||||
}
|
||||
if (array.Length - arrayIndex < _tags.Count) {
|
||||
throw new ArgumentException();
|
||||
}
|
||||
|
||||
foreach (KeyValuePair<string, TagNode> item in _tags) {
|
||||
array[arrayIndex] = item;
|
||||
arrayIndex++;
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a value indicating whether the node is readonly.
|
||||
/// </summary>
|
||||
public bool IsReadOnly
|
||||
{
|
||||
get { return false; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes the specified key and subnode combination from the set.
|
||||
/// </summary>
|
||||
/// <param name="item">The <see cref="KeyValuePair{TKey, TVal}"/> structure representing the key and value to remove from the set.</param>
|
||||
/// <returns>Status indicating whether a subnode was removed.</returns>
|
||||
public bool Remove (KeyValuePair<string, TagNode> item)
|
||||
{
|
||||
if (Contains(item)) {
|
||||
_tags.Remove(item.Key);
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region IEnumerable<KeyValuePair<string,NBT_Value>> Members
|
||||
|
||||
/// <summary>
|
||||
/// Returns an enumerator that iterates through all of the subnodes in the set.
|
||||
/// </summary>
|
||||
/// <returns>An enumerator for this node.</returns>
|
||||
public IEnumerator<KeyValuePair<string, TagNode>> GetEnumerator ()
|
||||
{
|
||||
return _tags.GetEnumerator();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region IEnumerable Members
|
||||
|
||||
/// <summary>
|
||||
/// Returns an enumerator that iterates through all of the subnodes in the set.
|
||||
/// </summary>
|
||||
/// <returns>An enumerator for this node.</returns>
|
||||
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator ()
|
||||
{
|
||||
return _tags.GetEnumerator();
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
101
Substrate/SubstrateCS/Source/Nbt/TagNodeDouble.cs
Normal file
101
Substrate/SubstrateCS/Source/Nbt/TagNodeDouble.cs
Normal file
|
@ -0,0 +1,101 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a double-precision floating point tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeDouble : TagNode
|
||||
{
|
||||
private double _data = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeDouble ToTagDouble ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_DOUBLE tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_DOUBLE;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a double of tag data.
|
||||
/// </summary>
|
||||
public double Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new double node with a data value of 0.0.
|
||||
/// </summary>
|
||||
public TagNodeDouble () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new double node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeDouble (double d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new double node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeDouble(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system float to a double node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="f">A float value.</param>
|
||||
/// <returns>A new double node containing the given value.</returns>
|
||||
public static implicit operator TagNodeDouble (float f)
|
||||
{
|
||||
return new TagNodeDouble(f);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system double to a double node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="d">A double value.</param>
|
||||
/// <returns>A new double node containing the given value.</returns>
|
||||
public static implicit operator TagNodeDouble (double d)
|
||||
{
|
||||
return new TagNodeDouble(d);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a double node to a system double representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="d">A double node.</param>
|
||||
/// <returns>A system double set to the node's data value.</returns>
|
||||
public static implicit operator double (TagNodeDouble d)
|
||||
{
|
||||
return d._data;
|
||||
}
|
||||
}
|
||||
}
|
121
Substrate/SubstrateCS/Source/Nbt/TagNodeFloat.cs
Normal file
121
Substrate/SubstrateCS/Source/Nbt/TagNodeFloat.cs
Normal file
|
@ -0,0 +1,121 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a single-precision floating point tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeFloat : TagNode
|
||||
{
|
||||
private float _data = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeFloat ToTagFloat ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to a new double node.
|
||||
/// </summary>
|
||||
/// <returns>A double node representing the same data.</returns>
|
||||
public override TagNodeDouble ToTagDouble ()
|
||||
{
|
||||
return new TagNodeDouble(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_FLOAT tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_FLOAT;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if the node is castable to another node of a given tag type.
|
||||
/// </summary>
|
||||
/// <param name="type">An NBT tag type.</param>
|
||||
/// <returns>Status indicating whether this object could be cast to a node type represented by the given tag type.</returns>
|
||||
public override bool IsCastableTo (TagType type)
|
||||
{
|
||||
return (type == TagType.TAG_FLOAT ||
|
||||
type == TagType.TAG_DOUBLE);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a float of tag data.
|
||||
/// </summary>
|
||||
public float Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new float node with a data value of 0.0.
|
||||
/// </summary>
|
||||
public TagNodeFloat () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new float node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeFloat (float d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new float node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeFloat(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system float to a float node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="f">A float value.</param>
|
||||
/// <returns>A new float node containing the given value.</returns>
|
||||
public static implicit operator TagNodeFloat (float f)
|
||||
{
|
||||
return new TagNodeFloat(f);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a float node to a system float representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="f">A float node.</param>
|
||||
/// <returns>A system float set to the node's data value.</returns>
|
||||
public static implicit operator float (TagNodeFloat f)
|
||||
{
|
||||
return f._data;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a float node to a system double representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="f">A float node.</param>
|
||||
/// <returns>A system double set to the node's data value.</returns>
|
||||
public static implicit operator double (TagNodeFloat f)
|
||||
{
|
||||
return f._data;
|
||||
}
|
||||
}
|
||||
}
|
141
Substrate/SubstrateCS/Source/Nbt/TagNodeInt.cs
Normal file
141
Substrate/SubstrateCS/Source/Nbt/TagNodeInt.cs
Normal file
|
@ -0,0 +1,141 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a signed int tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeInt : TagNode
|
||||
{
|
||||
private int _data = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeInt ToTagInt ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to a new long node.
|
||||
/// </summary>
|
||||
/// <returns>A long node representing the same data.</returns>
|
||||
public override TagNodeLong ToTagLong ()
|
||||
{
|
||||
return new TagNodeLong(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_INT tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_INT;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if the node is castable to another node of a given tag type.
|
||||
/// </summary>
|
||||
/// <param name="type">An NBT tag type.</param>
|
||||
/// <returns>Status indicating whether this object could be cast to a node type represented by the given tag type.</returns>
|
||||
public override bool IsCastableTo (TagType type)
|
||||
{
|
||||
return (type == TagType.TAG_INT ||
|
||||
type == TagType.TAG_LONG);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets an int of tag data.
|
||||
/// </summary>
|
||||
public int Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new int node with a data value of 0.
|
||||
/// </summary>
|
||||
public TagNodeInt () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new int node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeInt (int d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new int node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeInt(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system byte to an int node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte value.</param>
|
||||
/// <returns>A new int node containing the given value.</returns>
|
||||
public static implicit operator TagNodeInt (byte b)
|
||||
{
|
||||
return new TagNodeInt(b);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system short to an int node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="s">A short value.</param>
|
||||
/// <returns>A new int node containing the given value.</returns>
|
||||
public static implicit operator TagNodeInt (short s)
|
||||
{
|
||||
return new TagNodeInt(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system int to an int node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="i">An int value.</param>
|
||||
/// <returns>A new int node containing the given value.</returns>
|
||||
public static implicit operator TagNodeInt (int i)
|
||||
{
|
||||
return new TagNodeInt(i);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts an int node to a system int representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="i">An int node.</param>
|
||||
/// <returns>A system int set to the node's data value.</returns>
|
||||
public static implicit operator int (TagNodeInt i)
|
||||
{
|
||||
return i._data;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts an int node to a system long representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="i">An int node.</param>
|
||||
/// <returns>A system long set to the node's data value.</returns>
|
||||
public static implicit operator long (TagNodeInt i)
|
||||
{
|
||||
return i._data;
|
||||
}
|
||||
}
|
||||
}
|
264
Substrate/SubstrateCS/Source/Nbt/TagNodeList.cs
Normal file
264
Substrate/SubstrateCS/Source/Nbt/TagNodeList.cs
Normal file
|
@ -0,0 +1,264 @@
|
|||
using System;
|
||||
using System.Collections.Generic;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a list tag type containing other nodes.
|
||||
/// </summary>
|
||||
/// <remarks>
|
||||
/// A list node contains 0 or more nodes of the same type. The nodes are unnamed
|
||||
/// but can be accessed by sequential index.
|
||||
/// </remarks>
|
||||
public sealed class TagNodeList : TagNode, IList<TagNode>
|
||||
{
|
||||
private TagType _type = TagType.TAG_END;
|
||||
|
||||
private List<TagNode> _items = null;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeList ToTagList ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_STRING tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_LIST;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the number of subnodes contained in the list.
|
||||
/// </summary>
|
||||
public int Count
|
||||
{
|
||||
get { return _items.Count; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the subnodes contained in the list.
|
||||
/// </summary>
|
||||
public TagType ValueType
|
||||
{
|
||||
get { return _type; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new empty list node.
|
||||
/// </summary>
|
||||
/// <param name="type">The tag type of the list's subnodes.</param>
|
||||
public TagNodeList (TagType type)
|
||||
{
|
||||
_type = type;
|
||||
_items = new List<TagNode>();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new list node from a list of nodes.
|
||||
/// </summary>
|
||||
/// <param name="type">The tag type of the list's subnodes.</param>
|
||||
/// <param name="items">A list containing node objects matching the type parameter.</param>
|
||||
public TagNodeList (TagType type, List<TagNode> items)
|
||||
{
|
||||
_type = type;
|
||||
_items = items;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new list node containing new subnodes representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
TagNodeList list = new TagNodeList(_type);
|
||||
foreach (TagNode item in _items) {
|
||||
list.Add(item.Copy());
|
||||
}
|
||||
return list;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Retrieves all the subnodes that match the conditions defined by the specified predicate.
|
||||
/// </summary>
|
||||
/// <param name="match">The <see cref="Predicate{TagNode}"/> delegate that defines the conditions of the subnode to search for.</param>
|
||||
/// <returns>A list of all subnodes matching the predicate.</returns>
|
||||
public List<TagNode> FindAll (Predicate<TagNode> match)
|
||||
{
|
||||
return _items.FindAll(match);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes all subnodes that match the conditions defined by the specified predicate.
|
||||
/// </summary>
|
||||
/// <param name="match">The <see cref="Predicate{TagNode}"/> delegate that defines the conditions of the subnode to search for.</param>
|
||||
/// <returns>The number of subnodes removed from the node.</returns>
|
||||
public int RemoveAll (Predicate<TagNode> match)
|
||||
{
|
||||
return _items.RemoveAll(match);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _items.ToString();
|
||||
}
|
||||
|
||||
#region IList<NBT_Value> Members
|
||||
|
||||
/// <summary>
|
||||
/// Searches for the specified subnode and returns the zero-based index of the first occurrence within the entire node's list.
|
||||
/// </summary>
|
||||
/// <param name="item">The subnode to locate.</param>
|
||||
/// <returns>The zero-based index of the subnode within the node's list if found, or -1 otherwise.</returns>
|
||||
public int IndexOf (TagNode item)
|
||||
{
|
||||
return _items.IndexOf(item);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Inserts a subnode into the node's list at the specified index.
|
||||
/// </summary>
|
||||
/// <param name="index">The zero-based index at which the subnode should be inserted.</param>
|
||||
/// <param name="item">The subnode to insert.</param>
|
||||
/// <exception cref="ArgumentException">Thrown when a subnode being inserted has the wrong tag type.</exception>
|
||||
public void Insert (int index, TagNode item)
|
||||
{
|
||||
if (item.GetTagType() != _type) {
|
||||
throw new ArgumentException("The tag type of item is invalid for this node");
|
||||
}
|
||||
_items.Insert(index, item);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes the subnode from the node's list at the specified index.
|
||||
/// </summary>
|
||||
/// <param name="index">The zero-based index to remove a subnode at.</param>
|
||||
public void RemoveAt (int index)
|
||||
{
|
||||
_items.RemoveAt(index);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets the subnode in the node's list at the specified index.
|
||||
/// </summary>
|
||||
/// <param name="index">The zero-based index to get or set from.</param>
|
||||
/// <returns>The subnode at the specified index.</returns>
|
||||
/// <exception cref="ArgumentException">Thrown when a subnode being assigned has the wrong tag type.</exception>
|
||||
public TagNode this[int index]
|
||||
{
|
||||
get
|
||||
{
|
||||
return _items[index];
|
||||
}
|
||||
set
|
||||
{
|
||||
if (value.GetTagType() != _type) {
|
||||
throw new ArgumentException("The tag type of the assigned subnode is invalid for this node");
|
||||
}
|
||||
_items[index] = value;
|
||||
}
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region ICollection<NBT_Value> Members
|
||||
|
||||
/// <summary>
|
||||
/// Adds a subnode to the end of the node's list.
|
||||
/// </summary>
|
||||
/// <param name="item">The subnode to add.</param>
|
||||
/// <exception cref="ArgumentException">Thrown when a subnode being added has the wrong tag type.</exception>
|
||||
public void Add (TagNode item)
|
||||
{
|
||||
if (item.GetTagType() != _type) {
|
||||
throw new ArgumentException("The tag type of item is invalid for this node");
|
||||
}
|
||||
|
||||
_items.Add(item);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes all subnode's from the node's list.
|
||||
/// </summary>
|
||||
public void Clear ()
|
||||
{
|
||||
_items.Clear();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if a subnode is contained within the node's list.
|
||||
/// </summary>
|
||||
/// <param name="item">The subnode to check for existance.</param>
|
||||
/// <returns>Status indicating if the subnode exists in the node's list.</returns>
|
||||
public bool Contains (TagNode item)
|
||||
{
|
||||
return _items.Contains(item);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Copies the entire node's list to a compatible one-dimensional array, starting at the specified index of the target array.
|
||||
/// </summary>
|
||||
/// <param name="array">The one-dimensional <see cref="Array"/> that is the destination of the subnodes copied. The Array must have zero-based indexing.</param>
|
||||
/// <param name="arrayIndex">The zero-based index in <paramref name="array"/> at which copying begins.</param>
|
||||
public void CopyTo (TagNode[] array, int arrayIndex)
|
||||
{
|
||||
_items.CopyTo(array, arrayIndex);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a value indicating whether the node is readonly.
|
||||
/// </summary>
|
||||
public bool IsReadOnly
|
||||
{
|
||||
get { return false; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Removes the first occurance of a subnode from the node's list.
|
||||
/// </summary>
|
||||
/// <param name="item">The subnode to remove.</param>
|
||||
/// <returns>Status indicating whether a subnode was removed.</returns>
|
||||
public bool Remove (TagNode item)
|
||||
{
|
||||
return _items.Remove(item);
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region IEnumerable<NBT_Value> Members
|
||||
|
||||
/// <summary>
|
||||
/// Returns an enumerator that iterates through all of the subnodes in the node's list.
|
||||
/// </summary>
|
||||
/// <returns>An enumerator for this node.</returns>
|
||||
public IEnumerator<TagNode> GetEnumerator ()
|
||||
{
|
||||
return _items.GetEnumerator();
|
||||
}
|
||||
|
||||
#endregion
|
||||
|
||||
#region IEnumerable Members
|
||||
|
||||
/// <summary>
|
||||
/// Returns an enumerator that iterates through all of the subnodes in the node's list.
|
||||
/// </summary>
|
||||
/// <returns>An enumerator for this node.</returns>
|
||||
System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator ()
|
||||
{
|
||||
return _items.GetEnumerator();
|
||||
}
|
||||
|
||||
#endregion
|
||||
}
|
||||
}
|
121
Substrate/SubstrateCS/Source/Nbt/TagNodeLong.cs
Normal file
121
Substrate/SubstrateCS/Source/Nbt/TagNodeLong.cs
Normal file
|
@ -0,0 +1,121 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a signed long tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeLong : TagNode
|
||||
{
|
||||
private long _data = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeLong ToTagLong ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_LONG tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_LONG;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a long of tag data.
|
||||
/// </summary>
|
||||
public long Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new long node with a data value of 0.
|
||||
/// </summary>
|
||||
public TagNodeLong () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new long node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeLong (long d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new long node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeLong(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system byte to a long node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte value.</param>
|
||||
/// <returns>A new long node containing the given value.</returns>
|
||||
public static implicit operator TagNodeLong (byte b)
|
||||
{
|
||||
return new TagNodeLong(b);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system shprt to a long node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="s">A short value.</param>
|
||||
/// <returns>A new long node containing the given value.</returns>
|
||||
public static implicit operator TagNodeLong (short s)
|
||||
{
|
||||
return new TagNodeLong(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system int to a long node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="i">An int value.</param>
|
||||
/// <returns>A new long node containing the given value.</returns>
|
||||
public static implicit operator TagNodeLong (int i)
|
||||
{
|
||||
return new TagNodeLong(i);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system long to a long node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="l">A long value.</param>
|
||||
/// <returns>A new long node containing the given value.</returns>
|
||||
public static implicit operator TagNodeLong (long l)
|
||||
{
|
||||
return new TagNodeLong(l);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a long node to a system long representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="l">A long node.</param>
|
||||
/// <returns>A system long set to the node's data value.</returns>
|
||||
public static implicit operator long (TagNodeLong l)
|
||||
{
|
||||
return l._data;
|
||||
}
|
||||
}
|
||||
}
|
37
Substrate/SubstrateCS/Source/Nbt/TagNodeNull.cs
Normal file
37
Substrate/SubstrateCS/Source/Nbt/TagNodeNull.cs
Normal file
|
@ -0,0 +1,37 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a null tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeNull : TagNode
|
||||
{
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeNull ToTagNull ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_END tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_END;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new null node.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeNull();
|
||||
}
|
||||
}
|
||||
}
|
151
Substrate/SubstrateCS/Source/Nbt/TagNodeShort.cs
Normal file
151
Substrate/SubstrateCS/Source/Nbt/TagNodeShort.cs
Normal file
|
@ -0,0 +1,151 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a signed short tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeShort : TagNode
|
||||
{
|
||||
private short _data = 0;
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeShort ToTagShort ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to a new int node.
|
||||
/// </summary>
|
||||
/// <returns>An int node representing the same data.</returns>
|
||||
public override TagNodeInt ToTagInt ()
|
||||
{
|
||||
return new TagNodeInt(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to a new long node.
|
||||
/// </summary>
|
||||
/// <returns>A long node representing the same data.</returns>
|
||||
public override TagNodeLong ToTagLong ()
|
||||
{
|
||||
return new TagNodeLong(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_SHORT tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_SHORT;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Checks if the node is castable to another node of a given tag type.
|
||||
/// </summary>
|
||||
/// <param name="type">An NBT tag type.</param>
|
||||
/// <returns>Status indicating whether this object could be cast to a node type represented by the given tag type.</returns>
|
||||
public override bool IsCastableTo (TagType type)
|
||||
{
|
||||
return (type == TagType.TAG_SHORT ||
|
||||
type == TagType.TAG_INT ||
|
||||
type == TagType.TAG_LONG);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a short of tag data.
|
||||
/// </summary>
|
||||
public short Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new short node with a data value of 0.
|
||||
/// </summary>
|
||||
public TagNodeShort () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new short node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeShort (short d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new short node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeShort(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system byte to a short node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="b">A byte value.</param>
|
||||
/// <returns>A new short node containing the given value.</returns>
|
||||
public static implicit operator TagNodeShort (byte b)
|
||||
{
|
||||
return new TagNodeShort(b);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system short to a short node representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="s">A short value.</param>
|
||||
/// <returns>A new short node containing the given value.</returns>
|
||||
public static implicit operator TagNodeShort (short s)
|
||||
{
|
||||
return new TagNodeShort(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a short node to a system short representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="s">A short node.</param>
|
||||
/// <returns>A system short set to the node's data value.</returns>
|
||||
public static implicit operator short (TagNodeShort s)
|
||||
{
|
||||
return s._data;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a short node to a system int representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="s">A short node.</param>
|
||||
/// <returns>A system int set to the node's data value.</returns>
|
||||
public static implicit operator int (TagNodeShort s)
|
||||
{
|
||||
return s._data;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a short node to a system long representing the same value.
|
||||
/// </summary>
|
||||
/// <param name="s">A short node.</param>
|
||||
/// <returns>A system long set to the node's data value.</returns>
|
||||
public static implicit operator long (TagNodeShort s)
|
||||
{
|
||||
return s._data;
|
||||
}
|
||||
}
|
||||
}
|
99
Substrate/SubstrateCS/Source/Nbt/TagNodeString.cs
Normal file
99
Substrate/SubstrateCS/Source/Nbt/TagNodeString.cs
Normal file
|
@ -0,0 +1,99 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// An NBT node representing a string tag type.
|
||||
/// </summary>
|
||||
public sealed class TagNodeString : TagNode
|
||||
{
|
||||
private string _data = "";
|
||||
|
||||
/// <summary>
|
||||
/// Converts the node to itself.
|
||||
/// </summary>
|
||||
/// <returns>A reference to itself.</returns>
|
||||
public override TagNodeString ToTagString ()
|
||||
{
|
||||
return this;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the tag type of the node.
|
||||
/// </summary>
|
||||
/// <returns>The TAG_STRING tag type.</returns>
|
||||
public override TagType GetTagType ()
|
||||
{
|
||||
return TagType.TAG_STRING;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets or sets a string of tag data.
|
||||
/// </summary>
|
||||
public string Data
|
||||
{
|
||||
get { return _data; }
|
||||
set { _data = value; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the length of the stored string.
|
||||
/// </summary>
|
||||
public int Length
|
||||
{
|
||||
get { return _data.Length; }
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new byte array node with an empty string.
|
||||
/// </summary>
|
||||
public TagNodeString () { }
|
||||
|
||||
/// <summary>
|
||||
/// Constructs a new string node.
|
||||
/// </summary>
|
||||
/// <param name="d">The value to set the node's tag data value.</param>
|
||||
public TagNodeString (string d)
|
||||
{
|
||||
_data = d;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Makes a deep copy of the node.
|
||||
/// </summary>
|
||||
/// <returns>A new string node representing the same data.</returns>
|
||||
public override TagNode Copy ()
|
||||
{
|
||||
return new TagNodeString(_data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets a string representation of the node's data.
|
||||
/// </summary>
|
||||
/// <returns>String representation of the node's data.</returns>
|
||||
public override string ToString ()
|
||||
{
|
||||
return _data.ToString();
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a system string to a string node representing the same data.
|
||||
/// </summary>
|
||||
/// <param name="s">A string.</param>
|
||||
/// <returns>A new string node containing the given value.</returns>
|
||||
public static implicit operator TagNodeString (string s)
|
||||
{
|
||||
return new TagNodeString(s);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Converts a string node to a system string representing the same data.
|
||||
/// </summary>
|
||||
/// <param name="s">A string node.</param>
|
||||
/// <returns>A system string set to the node's data.</returns>
|
||||
public static implicit operator string (TagNodeString s)
|
||||
{
|
||||
return s._data;
|
||||
}
|
||||
}
|
||||
}
|
65
Substrate/SubstrateCS/Source/Nbt/TagType.cs
Normal file
65
Substrate/SubstrateCS/Source/Nbt/TagType.cs
Normal file
|
@ -0,0 +1,65 @@
|
|||
using System;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// Defines the type of an NBT tag.
|
||||
/// </summary>
|
||||
public enum TagType
|
||||
{
|
||||
/// <summary>
|
||||
/// A null tag, used to terminate lists.
|
||||
/// </summary>
|
||||
TAG_END = 0,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing an 8-bit signed integer.
|
||||
/// </summary>
|
||||
TAG_BYTE = 1,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a 16-bit signed integer.
|
||||
/// </summary>
|
||||
TAG_SHORT = 2,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a 32-bit signed integer.
|
||||
/// </summary>
|
||||
TAG_INT = 3,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a 64-bit signed integer.
|
||||
/// </summary>
|
||||
TAG_LONG = 4,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a 32-bit (single precision) floating-point value.
|
||||
/// </summary>
|
||||
TAG_FLOAT = 5,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a 64-bit (double precision) floating-point value.
|
||||
/// </summary>
|
||||
TAG_DOUBLE = 6,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing an array of unsigned 8-bit byte values.
|
||||
/// </summary>
|
||||
TAG_BYTE_ARRAY = 7,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a string of text.
|
||||
/// </summary>
|
||||
TAG_STRING = 8,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a sequential list of tags, where all tags of of the same type.
|
||||
/// </summary>
|
||||
TAG_LIST = 9,
|
||||
|
||||
/// <summary>
|
||||
/// A tag containing a key-value store of tags, where each tag can be of any type.
|
||||
/// </summary>
|
||||
TAG_COMPOUND = 10
|
||||
}
|
||||
}
|
47
Substrate/SubstrateCS/Source/Nbt/VerifierLogger.cs
Normal file
47
Substrate/SubstrateCS/Source/Nbt/VerifierLogger.cs
Normal file
|
@ -0,0 +1,47 @@
|
|||
using System;
|
||||
using Substrate.Core;
|
||||
|
||||
namespace Substrate.Nbt
|
||||
{
|
||||
/// <summary>
|
||||
/// A collection of static methods that can be hooked into <see cref="NBTVerifier"/> events for logging NBT errors to the console.
|
||||
/// </summary>
|
||||
public static class VerifierLogger
|
||||
{
|
||||
/// <summary>
|
||||
/// Logs an occurance of a missing tag error, and advances to the next event in the event chain.
|
||||
/// </summary>
|
||||
/// <param name="e">Data about the NBT node being verified.</param>
|
||||
/// <returns>A <see cref="TagEventCode"/> indicating whether event processing should pass, fail, or advance.</returns>
|
||||
public static TagEventCode MissingTagHandler (TagEventArgs e)
|
||||
{
|
||||
Console.WriteLine("Missing Tag Error: '{0}'", e.TagName);
|
||||
|
||||
return TagEventCode.NEXT;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Logs an occurance of an invalid tag type error, and advances to the next event in the event chain.
|
||||
/// </summary>
|
||||
/// <param name="e">Data about the NBT node being verified.</param>
|
||||
/// <returns>A <see cref="TagEventCode"/> indicating whether event processing should pass, fail, or advance.</returns>
|
||||
public static TagEventCode InvalidTagTypeHandler (TagEventArgs e)
|
||||
{
|
||||
Console.WriteLine("Invalid Tag Type Error: '{0}' has type '{1}', expected '{2}'", e.TagName, e.Tag.GetTagType(), e.Schema.ToString());
|
||||
|
||||
return TagEventCode.NEXT;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Logs an occurance of an invalid tag value error, and advances to the next event in the event chain.
|
||||
/// </summary>
|
||||
/// <param name="e">Data about the NBT node being verified.</param>
|
||||
/// <returns>A <see cref="TagEventCode"/> indicating whether event processing should pass, fail, or advance.</returns>
|
||||
public static TagEventCode InvalidTagValueHandler (TagEventArgs e)
|
||||
{
|
||||
Console.WriteLine("Invalid Tag Value Error: '{0}' of type '{1}' is set to invalid value '{2}'", e.TagName, e.Tag.GetTagType(), e.Tag.ToString());
|
||||
|
||||
return TagEventCode.NEXT;
|
||||
}
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue