forked from mirrors/NBTExplorer
334 lines
9.6 KiB
C#
334 lines
9.6 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Text;
|
|
using Substrate.Core;
|
|
using Substrate.Nbt;
|
|
|
|
namespace Substrate
|
|
{
|
|
/// <summary>
|
|
/// Represents an item (or item stack) within an item slot.
|
|
/// </summary>
|
|
public class Item : INbtObject<Item>, ICopyable<Item>
|
|
{
|
|
private static readonly SchemaNodeCompound _schema = new SchemaNodeCompound("")
|
|
{
|
|
new SchemaNodeScaler("id", TagType.TAG_SHORT),
|
|
new SchemaNodeScaler("Damage", TagType.TAG_SHORT),
|
|
new SchemaNodeScaler("Count", TagType.TAG_BYTE),
|
|
};
|
|
|
|
private TagNodeCompound _source;
|
|
|
|
private short _id;
|
|
private byte _count;
|
|
private short _damage;
|
|
|
|
/// <summary>
|
|
/// Constructs an empty <see cref="Item"/> instance.
|
|
/// </summary>
|
|
public Item ()
|
|
{
|
|
}
|
|
|
|
/// <summary>
|
|
/// Constructs an <see cref="Item"/> instance representing the given item id.
|
|
/// </summary>
|
|
/// <param name="id">An item id.</param>
|
|
public Item (int id)
|
|
{
|
|
_id = (short)id;
|
|
}
|
|
|
|
#region Properties
|
|
|
|
/// <summary>
|
|
/// Gets an <see cref="ItemInfo"/> entry for this item's type.
|
|
/// </summary>
|
|
public ItemInfo Info
|
|
{
|
|
get { return ItemInfo.ItemTable[_id]; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the current type (id) of the item.
|
|
/// </summary>
|
|
public int ID
|
|
{
|
|
get { return _id; }
|
|
set { _id = (short)value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the damage value of the item.
|
|
/// </summary>
|
|
/// <remarks>The damage value may represent a generic data value for some items.</remarks>
|
|
public int Damage
|
|
{
|
|
get { return _damage; }
|
|
set { _damage = (short)value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets the number of this item stacked together in an item slot.
|
|
/// </summary>
|
|
public int Count
|
|
{
|
|
get { return _count; }
|
|
set { _count = (byte)value; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="SchemaNode"/> representing the schema of an item.
|
|
/// </summary>
|
|
public static SchemaNodeCompound Schema
|
|
{
|
|
get { return _schema; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ICopyable<Item> Members
|
|
|
|
/// <inheritdoc/>
|
|
public Item Copy ()
|
|
{
|
|
Item item = new Item();
|
|
item._id = _id;
|
|
item._count = _count;
|
|
item._damage = _damage;
|
|
|
|
if (_source != null) {
|
|
item._source = _source.Copy() as TagNodeCompound;
|
|
}
|
|
|
|
return item;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region INBTObject<Item> Members
|
|
|
|
/// <inheritdoc/>
|
|
public Item LoadTree (TagNode tree)
|
|
{
|
|
TagNodeCompound ctree = tree as TagNodeCompound;
|
|
if (ctree == null) {
|
|
return null;
|
|
}
|
|
|
|
_id = ctree["id"].ToTagShort();
|
|
_count = ctree["Count"].ToTagByte();
|
|
_damage = ctree["Damage"].ToTagShort();
|
|
|
|
_source = ctree.Copy() as TagNodeCompound;
|
|
|
|
return this;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public Item LoadTreeSafe (TagNode tree)
|
|
{
|
|
if (!ValidateTree(tree)) {
|
|
return null;
|
|
}
|
|
|
|
return LoadTree(tree);
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public TagNode BuildTree ()
|
|
{
|
|
TagNodeCompound tree = new TagNodeCompound();
|
|
tree["id"] = new TagNodeShort(_id);
|
|
tree["Count"] = new TagNodeByte(_count);
|
|
tree["Damage"] = new TagNodeShort(_damage);
|
|
|
|
if (_source != null) {
|
|
tree.MergeFrom(_source);
|
|
}
|
|
|
|
return tree;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public bool ValidateTree (TagNode tree)
|
|
{
|
|
return new NbtVerifier(tree, _schema).Verify();
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
|
|
/// <summary>
|
|
/// Represents a collection of items, such as a chest or an inventory.
|
|
/// </summary>
|
|
/// <remarks>ItemCollections have a limited number of slots that depends on where they are used.</remarks>
|
|
public class ItemCollection : INbtObject<ItemCollection>, ICopyable<ItemCollection>
|
|
{
|
|
private static readonly SchemaNodeCompound _schema = Item.Schema.MergeInto(new SchemaNodeCompound("")
|
|
{
|
|
new SchemaNodeScaler("Slot", TagType.TAG_BYTE),
|
|
});
|
|
|
|
private static readonly SchemaNodeList _listSchema = new SchemaNodeList("", TagType.TAG_COMPOUND, _schema);
|
|
|
|
private Dictionary<int, Item> _items;
|
|
private int _capacity;
|
|
|
|
/// <summary>
|
|
/// Constructs an <see cref="ItemCollection"/> with at most <paramref name="capacity"/> item slots.
|
|
/// </summary>
|
|
/// <param name="capacity">The upper bound on item slots available.</param>
|
|
/// <remarks>The <paramref name="capacity"/> parameter does not necessarily indicate the true capacity of an item collection.
|
|
/// The player object, for example, contains a conventional inventory, a range of invalid slots, and then equipment. Capacity in
|
|
/// this case would refer to the highest equipment slot.</remarks>
|
|
public ItemCollection (int capacity)
|
|
{
|
|
_capacity = capacity;
|
|
_items = new Dictionary<int, Item>();
|
|
}
|
|
|
|
#region Properties
|
|
|
|
/// <summary>
|
|
/// Gets the capacity of the item collection.
|
|
/// </summary>
|
|
public int Capacity
|
|
{
|
|
get { return _capacity; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the current number of item slots actually used in the collection.
|
|
/// </summary>
|
|
public int Count
|
|
{
|
|
get { return _items.Count; }
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets or sets an item in a given item slot.
|
|
/// </summary>
|
|
/// <param name="slot">The item slot to query or insert an item or item stack into.</param>
|
|
public Item this [int slot]
|
|
{
|
|
get
|
|
{
|
|
Item item;
|
|
_items.TryGetValue(slot, out item);
|
|
return item;
|
|
}
|
|
|
|
set
|
|
{
|
|
if (slot < 0 || slot >= _capacity) {
|
|
return;
|
|
}
|
|
_items[slot] = value;
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets a <see cref="SchemaNode"/> representing the schema of an item collection.
|
|
/// </summary>
|
|
public static SchemaNodeCompound Schema
|
|
{
|
|
get { return _schema; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
/// <summary>
|
|
/// Checks if an item exists in the given item slot.
|
|
/// </summary>
|
|
/// <param name="slot">The item slot to check.</param>
|
|
/// <returns>True if an item or stack of items exists in the given slot.</returns>
|
|
public bool ItemExists (int slot)
|
|
{
|
|
return _items.ContainsKey(slot);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Removes an item from the given item slot, if it exists.
|
|
/// </summary>
|
|
/// <param name="slot">The item slot to clear.</param>
|
|
/// <returns>True if an item was removed; false otherwise.</returns>
|
|
public bool Clear (int slot)
|
|
{
|
|
return _items.Remove(slot);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Removes all items from the item collection.
|
|
/// </summary>
|
|
public void ClearAllItems ()
|
|
{
|
|
_items.Clear();
|
|
}
|
|
|
|
#region ICopyable<ItemCollection> Members
|
|
|
|
/// <inheritdoc/>
|
|
public ItemCollection Copy ()
|
|
{
|
|
ItemCollection ic = new ItemCollection(_capacity);
|
|
foreach (KeyValuePair<int, Item> item in _items) {
|
|
ic[item.Key] = item.Value.Copy();
|
|
}
|
|
return ic;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region INBTObject<ItemCollection> Members
|
|
|
|
/// <inheritdoc/>
|
|
public ItemCollection LoadTree (TagNode tree)
|
|
{
|
|
TagNodeList ltree = tree as TagNodeList;
|
|
if (ltree == null) {
|
|
return null;
|
|
}
|
|
|
|
foreach (TagNodeCompound item in ltree) {
|
|
int slot = item["Slot"].ToTagByte();
|
|
_items[slot] = new Item().LoadTree(item);
|
|
}
|
|
|
|
return this;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public ItemCollection LoadTreeSafe (TagNode tree)
|
|
{
|
|
if (!ValidateTree(tree)) {
|
|
return null;
|
|
}
|
|
|
|
return LoadTree(tree);
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public TagNode BuildTree ()
|
|
{
|
|
TagNodeList list = new TagNodeList(TagType.TAG_COMPOUND);
|
|
|
|
foreach (KeyValuePair<int, Item> item in _items) {
|
|
TagNodeCompound itemtree = item.Value.BuildTree() as TagNodeCompound;
|
|
itemtree["Slot"] = new TagNodeByte((byte)item.Key);
|
|
list.Add(itemtree);
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
/// <inheritdoc/>
|
|
public bool ValidateTree (TagNode tree)
|
|
{
|
|
return new NbtVerifier(tree, _listSchema).Verify();
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|