using System; using System.Collections.Generic; using Substrate.Nbt; namespace NBTExplorer.Model { public abstract class TagDataNode : DataNode { public abstract class Container : TagDataNode, IMetaTagContainer { protected Container (TagNode tag) : base(tag) { } #region ITagContainer public virtual int TagCount { get { return 0; } } public virtual bool IsNamedContainer { get { return false; } } public virtual bool IsOrderedContainer { get { return false; } } public virtual INamedTagContainer NamedTagContainer { get { return null; } } public virtual IOrderedTagContainer OrderedTagContainer { get { return null; } } public virtual bool DeleteTag (TagNode tag) { return false; } #endregion protected override NodeCapabilities Capabilities { get { return NodeCapabilities.Copy | NodeCapabilities.CreateTag | NodeCapabilities.Cut | NodeCapabilities.Delete | NodeCapabilities.PasteInto | ((TagParent != null && TagParent.IsNamedContainer) ? NodeCapabilities.Rename : NodeCapabilities.None) | ((TagParent != null && TagParent.IsOrderedContainer) ? NodeCapabilities.Reorder : NodeCapabilities.None) | NodeCapabilities.Search; } } public override bool HasUnexpandedChildren { get { return !IsExpanded && TagCount > 0; } } public override string NodeDisplay { get { return NodeDisplayPrefix + TagCount + ((TagCount == 1) ? " entry" : " entries"); } } } private static Dictionary _tagRegistry; static TagDataNode () { _tagRegistry = new Dictionary(); _tagRegistry[TagType.TAG_BYTE] = typeof(TagByteDataNode); _tagRegistry[TagType.TAG_BYTE_ARRAY] = typeof(TagByteArrayDataNode); _tagRegistry[TagType.TAG_COMPOUND] = typeof(TagCompoundDataNode); _tagRegistry[TagType.TAG_DOUBLE] = typeof(TagDoubleDataNode); _tagRegistry[TagType.TAG_FLOAT] = typeof(TagFloatDataNode); _tagRegistry[TagType.TAG_INT] = typeof(TagIntDataNode); _tagRegistry[TagType.TAG_INT_ARRAY] = typeof(TagIntArrayDataNode); _tagRegistry[TagType.TAG_LIST] = typeof(TagListDataNode); _tagRegistry[TagType.TAG_LONG] = typeof(TagLongDataNode); _tagRegistry[TagType.TAG_SHORT] = typeof(TagShortDataNode); _tagRegistry[TagType.TAG_STRING] = typeof(TagStringDataNode); } static public TagDataNode CreateFromTag (TagNode tag) { if (tag == null || !_tagRegistry.ContainsKey(tag.GetTagType())) return null; return Activator.CreateInstance(_tagRegistry[tag.GetTagType()], tag) as TagDataNode; } static public TagNode DefaultTag (TagType type) { switch (type) { case TagType.TAG_BYTE: return new TagNodeByte(0); case TagType.TAG_BYTE_ARRAY: return new TagNodeByteArray(new byte[0]); case TagType.TAG_COMPOUND: return new TagNodeCompound(); case TagType.TAG_DOUBLE: return new TagNodeDouble(0); case TagType.TAG_FLOAT: return new TagNodeFloat(0); case TagType.TAG_INT: return new TagNodeInt(0); case TagType.TAG_INT_ARRAY: return new TagNodeIntArray(new int[0]); case TagType.TAG_LIST: return new TagNodeList(TagType.TAG_BYTE); case TagType.TAG_LONG: return new TagNodeLong(0); case TagType.TAG_SHORT: return new TagNodeShort(0); case TagType.TAG_STRING: return new TagNodeString(""); default: return new TagNodeByte(0); } } private TagNode _tag; protected TagDataNode (TagNode tag) { _tag = tag; } protected IMetaTagContainer TagParent { get { return base.Parent as IMetaTagContainer; } } public TagNode Tag { get { return _tag; } protected set { if (_tag.GetTagType() == value.GetTagType()) _tag = value; } } protected override NodeCapabilities Capabilities { get { return NodeCapabilities.Copy | NodeCapabilities.Cut | NodeCapabilities.Delete | NodeCapabilities.Edit | ((TagParent != null && TagParent.IsNamedContainer) ? NodeCapabilities.Rename : NodeCapabilities.None) | ((TagParent != null && TagParent.IsOrderedContainer) ? NodeCapabilities.Reorder : NodeCapabilities.None); } } public override bool CanMoveNodeUp { get { if (TagParent != null && TagParent.IsOrderedContainer) return TagParent.OrderedTagContainer.GetTagIndex(Tag) > 0; return false; } } public override bool CanMoveNodeDown { get { if (TagParent != null && TagParent.IsOrderedContainer) return TagParent.OrderedTagContainer.GetTagIndex(Tag) < (TagParent.TagCount - 1); return false; } } public override string NodeName { get { if (TagParent == null || !TagParent.IsNamedContainer) return null; return TagParent.NamedTagContainer.GetTagName(Tag); } } public override string NodePathName { get { if (Parent is TagDataNode.Container) { TagDataNode.Container container = Parent as TagDataNode.Container; if (container.IsOrderedContainer) return container.OrderedTagContainer.GetTagIndex(Tag).ToString(); } return base.NodePathName; } } protected string NodeDisplayPrefix { get { string name = NodeName; return String.IsNullOrEmpty(name) ? "" : name + ": "; } } public override string NodeDisplay { get { return NodeDisplayPrefix + Tag.ToString(); } } public override bool DeleteNode () { if (CanDeleteNode && TagParent != null) { TagParent.DeleteTag(Tag); IsParentModified = true; return Parent.Nodes.Remove(this); } return false; } public override bool RenameNode () { if (CanRenameNode && TagParent != null && TagParent.IsNamedContainer && FormRegistry.EditString != null) { RestrictedStringFormData data = new RestrictedStringFormData(TagParent.NamedTagContainer.GetTagName(Tag)); data.RestrictedValues.AddRange(TagParent.NamedTagContainer.TagNamesInUse); if (FormRegistry.RenameTag(data)) { if (TagParent.NamedTagContainer.RenameTag(Tag, data.Value)) { IsDataModified = true; return true; } } } return false; } public override bool CopyNode () { if (CanCopyNode) { NbtClipboardController.CopyToClipboard(new NbtClipboardData(NodeName, Tag)); return true; } return false; } public override bool CutNode () { if (CanCutNode && TagParent != null) { NbtClipboardController.CopyToClipboard(new NbtClipboardData(NodeName, Tag)); TagParent.DeleteTag(Tag); IsParentModified = true; Parent.Nodes.Remove(this); return true; } return false; } public override bool ChangeRelativePosition (int offset) { if (CanReoderNode && TagParent != null) { int curIndex = TagParent.OrderedTagContainer.GetTagIndex(Tag); int newIndex = curIndex + offset; if (newIndex < 0 || newIndex >= TagParent.OrderedTagContainer.TagCount) return false; TagParent.OrderedTagContainer.DeleteTag(Tag); TagParent.OrderedTagContainer.InsertTag(Tag, newIndex); DataNode parent = Parent; parent.Nodes.Remove(this); parent.Nodes.Insert(newIndex, this); IsParentModified = true; return true; } return false; } protected bool EditScalarValue (TagNode tag) { if (FormRegistry.EditTagScalar != null) { if (FormRegistry.EditTagScalar(new TagScalarFormData(tag))) { IsDataModified = true; return true; } } return false; } protected bool EditStringValue (TagNode tag) { if (FormRegistry.EditString != null) { StringFormData data = new StringFormData(tag.ToTagString().Data); if (FormRegistry.EditString(data)) { tag.ToTagString().Data = data.Value; IsDataModified = true; return true; } } return false; } protected bool EditByteHexValue (TagNode tag) { if (FormRegistry.EditByteArray != null) { byte[] byteData = new byte[tag.ToTagByteArray().Length]; Array.Copy(tag.ToTagByteArray().Data, byteData, byteData.Length); ByteArrayFormData data = new ByteArrayFormData() { NodeName = NodeName, BytesPerElement = 1, Data = byteData, }; if (FormRegistry.EditByteArray(data)) { tag.ToTagByteArray().Data = data.Data; //Array.Copy(data.Data, tag.ToTagByteArray().Data, tag.ToTagByteArray().Length); IsDataModified = true; return true; } } return false; } protected bool EditIntHexValue (TagNode tag) { if (FormRegistry.EditByteArray != null) { TagNodeIntArray iatag = tag.ToTagIntArray(); byte[] byteData = new byte[iatag.Length * 4]; for (int i = 0; i < iatag.Length; i++) { byte[] buf = BitConverter.GetBytes(iatag.Data[i]); Array.Copy(buf, 0, byteData, 4 * i, 4); } ByteArrayFormData data = new ByteArrayFormData() { NodeName = NodeName, BytesPerElement = 4, Data = byteData, }; if (FormRegistry.EditByteArray(data)) { iatag.Data = new int[data.Data.Length / 4]; for (int i = 0; i < iatag.Length; i++) { iatag.Data[i] = BitConverter.ToInt32(data.Data, i * 4); } IsDataModified = true; return true; } } return false; } public virtual void SyncTag () { } } }