mirror of
https://github.com/jaquadro/NBTExplorer.git
synced 2025-01-10 01:46:24 +00:00
1289 lines
38 KiB
C#
1289 lines
38 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.ComponentModel;
|
|
using System.Data;
|
|
using System.Drawing;
|
|
using System.Text;
|
|
using System.Windows.Forms;
|
|
using Substrate;
|
|
using Substrate.Nbt;
|
|
using System.IO;
|
|
using Substrate.Core;
|
|
|
|
namespace NBTExplorer
|
|
{
|
|
public partial class Form1 : Form
|
|
{
|
|
private static Dictionary<TagType, int> _tagIconIndex;
|
|
|
|
public Form1 ()
|
|
{
|
|
InitializeComponent();
|
|
|
|
_nodeTree.BeforeExpand += NodeExpand;
|
|
_nodeTree.AfterCollapse += NodeCollapse;
|
|
_nodeTree.AfterSelect += NodeSelected;
|
|
_nodeTree.NodeMouseClick += NodeClicked;
|
|
_nodeTree.NodeMouseDoubleClick += NodeDoubleClicked;
|
|
|
|
string path = Environment.GetFolderPath(Environment.SpecialFolder.ApplicationData);
|
|
path = Path.Combine(path, ".minecraft");
|
|
path = Path.Combine(path, "saves");
|
|
|
|
if (!Directory.Exists(path)) {
|
|
path = Environment.GetFolderPath(Environment.SpecialFolder.MyComputer);
|
|
}
|
|
|
|
LoadDirectory(path);
|
|
_nodeTree.Nodes[0].Expand();
|
|
}
|
|
|
|
public void LoadNbtStream (TreeNodeCollection parent, Stream stream)
|
|
{
|
|
LoadNbtStream(parent, stream, null);
|
|
}
|
|
|
|
public void LoadNbtStream (TreeNodeCollection parent, Stream stream, string name)
|
|
{
|
|
string text = !String.IsNullOrEmpty(name) ? name : null;
|
|
text = text ?? "[root]";
|
|
|
|
TreeNode root = new TreeNode(text, 9, 9);
|
|
|
|
LoadNbtStream(root, stream);
|
|
|
|
parent.Add(root);
|
|
}
|
|
|
|
public void LoadNbtStream (TreeNode node, Stream stream)
|
|
{
|
|
NbtTree tree = new NbtTree();
|
|
tree.ReadFrom(stream);
|
|
|
|
if (node.Tag != null && node.Tag is NbtDataNode) {
|
|
(node.Tag as NbtDataNode).Tree = tree;
|
|
}
|
|
|
|
PopulateNodeFromTag(node, tree.Root);
|
|
}
|
|
|
|
private TreeNode NodeFromTag (TagNode tag)
|
|
{
|
|
return NodeFromTag(tag, null);
|
|
}
|
|
|
|
private TreeNode NodeFromTag (TagNode tag, string name)
|
|
{
|
|
TreeNode node = new TreeNode();
|
|
node.ImageIndex = _tagIconIndex[tag.GetTagType()];
|
|
node.SelectedImageIndex = _tagIconIndex[tag.GetTagType()];
|
|
node.Tag = tag;
|
|
node.Text = GetNodeText(name, tag);
|
|
|
|
if (tag.GetTagType() == TagType.TAG_LIST) {
|
|
PopulateNodeFromTag(node, tag.ToTagList());
|
|
}
|
|
else if (tag.GetTagType() == TagType.TAG_COMPOUND) {
|
|
PopulateNodeFromTag(node, tag.ToTagCompound());
|
|
}
|
|
|
|
return node;
|
|
}
|
|
|
|
private string GetTagNodeText (TagNode tag)
|
|
{
|
|
if (tag == null)
|
|
return null;
|
|
|
|
switch (tag.GetTagType()) {
|
|
case TagType.TAG_BYTE:
|
|
case TagType.TAG_SHORT:
|
|
case TagType.TAG_INT:
|
|
case TagType.TAG_LONG:
|
|
case TagType.TAG_FLOAT:
|
|
case TagType.TAG_DOUBLE:
|
|
case TagType.TAG_STRING:
|
|
return tag.ToString();
|
|
|
|
case TagType.TAG_BYTE_ARRAY:
|
|
return tag.ToTagByteArray().Length + " bytes";
|
|
|
|
case TagType.TAG_LIST:
|
|
return tag.ToTagList().Count + " entries";
|
|
|
|
case TagType.TAG_COMPOUND:
|
|
return tag.ToTagCompound().Count + " entries";
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private string GetNodeText (string name, string value)
|
|
{
|
|
name = String.IsNullOrEmpty(name) ? "" : name + ": ";
|
|
value = value ?? "";
|
|
|
|
return name + value;
|
|
}
|
|
|
|
private string GetNodeText (string name, TagNode tag)
|
|
{
|
|
return GetNodeText(name, GetTagNodeText(tag));
|
|
}
|
|
|
|
private string GetNodeText (TreeNode node)
|
|
{
|
|
return GetNodeText(GetTagNodeName(node), GetTagNodeText(node));
|
|
}
|
|
|
|
private string GetTagNodeText (TreeNode node)
|
|
{
|
|
TagNode tag = GetTagNode(node);
|
|
if (tag == null)
|
|
return null;
|
|
|
|
return GetTagNodeText(tag);
|
|
}
|
|
|
|
private void PopulateNodeFromTag (TreeNode node, TagNodeList list)
|
|
{
|
|
foreach (TagNode tag in list) {
|
|
node.Nodes.Add(NodeFromTag(tag));
|
|
}
|
|
}
|
|
|
|
private void PopulateNodeFromTag (TreeNode node, TagNodeCompound dict)
|
|
{
|
|
var list = new SortedList<TagKey, TagNode>();
|
|
foreach (KeyValuePair<string, TagNode> kv in dict) {
|
|
list.Add(new TagKey(kv.Key, kv.Value.GetTagType()), kv.Value);
|
|
}
|
|
|
|
foreach (KeyValuePair<TagKey, TagNode> kv in list) {
|
|
node.Nodes.Add(NodeFromTag(kv.Value, kv.Key.Name));
|
|
}
|
|
}
|
|
|
|
public void LoadWorld (string path)
|
|
{
|
|
NbtWorld world = NbtWorld.Open(path);
|
|
|
|
BetaWorld beta = world as BetaWorld;
|
|
if (beta != null) {
|
|
LoadBetaWorld(beta);
|
|
}
|
|
}
|
|
|
|
private void LoadBetaWorld (BetaWorld world)
|
|
{
|
|
RegionManager rm = world.GetRegionManager();
|
|
}
|
|
|
|
public void LoadRegion (TreeNodeCollection parent, string path)
|
|
{
|
|
TreeNode root = new TreeNode(Path.GetFileName(path), 11, 11);
|
|
LoadRegion(root, path);
|
|
parent.Add(root);
|
|
}
|
|
|
|
public void LoadRegion (TreeNode node, string path)
|
|
{
|
|
RegionFile rf = new RegionFile(path);
|
|
|
|
for (int x = 0; x < 32; x++) {
|
|
for (int y = 0; y < 32; y++) {
|
|
if (rf.HasChunk(x, y)) {
|
|
TreeNode child = CreateLazyChunk(rf, x, y);
|
|
node.Nodes.Add(child);
|
|
LinkDataNodeParent(child, node);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
public TreeNode CreateLazyChunk (RegionFile rf, int x, int z)
|
|
{
|
|
TreeNode node = new TreeNode();
|
|
node.ImageIndex = 9;
|
|
node.SelectedImageIndex = 9;
|
|
node.Text = "Chunk [" + x + ", " + z + "]";
|
|
node.Tag = new RegionChunkData(rf, x, z);
|
|
node.Nodes.Add(new TreeNode());
|
|
|
|
return node;
|
|
}
|
|
|
|
public TreeNode CreateLazyRegion (string path)
|
|
{
|
|
TreeNode node = new TreeNode();
|
|
node.ImageIndex = 11;
|
|
node.SelectedImageIndex = 11;
|
|
node.Text = Path.GetFileName(path);
|
|
node.Tag = new RegionData(path);
|
|
node.Nodes.Add(new TreeNode());
|
|
|
|
return node;
|
|
}
|
|
|
|
public TreeNode CreateLazyNbt (string path, CompressionType cztype)
|
|
{
|
|
TreeNode node = new TreeNode();
|
|
node.ImageIndex = 12;
|
|
node.SelectedImageIndex = 12;
|
|
node.Text = Path.GetFileName(path);
|
|
node.Tag = new NbtFileData(path, cztype);
|
|
node.Nodes.Add(new TreeNode());
|
|
|
|
return node;
|
|
}
|
|
|
|
public TreeNode CreateLazyDirectory (string path)
|
|
{
|
|
TreeNode node = new TreeNode();
|
|
node.ImageIndex = 10;
|
|
node.SelectedImageIndex = 10;
|
|
node.Text = Path.GetFileName(path);
|
|
node.Tag = new DirectoryData(path);
|
|
node.Nodes.Add(new TreeNode());
|
|
|
|
return node;
|
|
}
|
|
|
|
public TreeNode CreateLazyDirectory (string path, TreeNode parent)
|
|
{
|
|
TreeNode node = CreateLazyDirectory(path);
|
|
LinkDataNodeParent(node, parent);
|
|
|
|
return node;
|
|
}
|
|
|
|
public TreeNode CreateLazyNbt (string path, CompressionType cztype, TreeNode parent)
|
|
{
|
|
TreeNode node = CreateLazyNbt(path, cztype);
|
|
LinkDataNodeParent(node, parent);
|
|
|
|
return node;
|
|
}
|
|
|
|
private void LinkDataNodeParent (TreeNode node, TreeNode parent)
|
|
{
|
|
if (node != null && parent != null && node.Tag != null && parent.Tag != null) {
|
|
DataNode nodeDn = node.Tag as DataNode;
|
|
DataNode parentDn = parent.Tag as DataNode;
|
|
|
|
if (nodeDn != null && parentDn != null) {
|
|
nodeDn.Parent = parentDn;
|
|
}
|
|
}
|
|
}
|
|
|
|
public void LoadLazyChunk (TreeNode node)
|
|
{
|
|
RegionChunkData data = node.Tag as RegionChunkData;
|
|
if (data == null || data.Modified)
|
|
return;
|
|
|
|
node.Nodes.Clear();
|
|
LoadNbtStream(node, data.Region.GetChunkDataInputStream(data.X, data.Z));
|
|
|
|
data.Expanded = true;
|
|
}
|
|
|
|
public void LoadLazyNbt (TreeNode node)
|
|
{
|
|
NbtFileData data = node.Tag as NbtFileData;
|
|
if (data == null || data.Modified)
|
|
return;
|
|
|
|
node.Nodes.Clear();
|
|
|
|
NBTFile file = new NBTFile(data.Path);
|
|
LoadNbtStream(node, file.GetDataInputStream(data.CompressionType));
|
|
|
|
data.Expanded = true;
|
|
}
|
|
|
|
public void LoadLazyDirectory (TreeNode node)
|
|
{
|
|
DirectoryData data = node.Tag as DirectoryData;
|
|
if (data == null || data.Modified)
|
|
return;
|
|
|
|
node.Nodes.Clear();
|
|
|
|
foreach (string dirpath in Directory.EnumerateDirectories(data.Path)) {
|
|
node.Nodes.Add(CreateLazyDirectory(dirpath, node));
|
|
}
|
|
|
|
foreach (string filepath in Directory.EnumerateFiles(data.Path)) {
|
|
TryLoadFile(node.Nodes, filepath);
|
|
}
|
|
|
|
data.Expanded = true;
|
|
}
|
|
|
|
public void LoadLazyRegion (TreeNode node)
|
|
{
|
|
RegionData data = node.Tag as RegionData;
|
|
if (data == null || data.Modified)
|
|
return;
|
|
|
|
node.Nodes.Clear();
|
|
LoadRegion(node, data.Path);
|
|
|
|
data.Expanded = true;
|
|
}
|
|
|
|
public void UnloadLazyDataNode (TreeNode node)
|
|
{
|
|
DataNode data = node.Tag as DataNode;
|
|
if (data == null || data.Modified)
|
|
return;
|
|
|
|
node.Nodes.Clear();
|
|
node.Nodes.Add(new TreeNode());
|
|
|
|
data.Expanded = false;
|
|
}
|
|
|
|
private void ExpandNode (TreeNode node)
|
|
{
|
|
if (node.Tag == null)
|
|
return;
|
|
|
|
if (node.Tag is DataNode) {
|
|
if ((node.Tag as DataNode).Expanded)
|
|
return;
|
|
}
|
|
|
|
if (node.Tag is RegionChunkData) {
|
|
LoadLazyChunk(node);
|
|
}
|
|
else if (node.Tag is NbtFileData) {
|
|
LoadLazyNbt(node);
|
|
}
|
|
else if (node.Tag is DirectoryData) {
|
|
LoadLazyDirectory(node);
|
|
}
|
|
else if (node.Tag is RegionData) {
|
|
LoadLazyRegion(node);
|
|
}
|
|
}
|
|
|
|
private void CollapseNode (TreeNode node)
|
|
{
|
|
if (node.Tag == null)
|
|
return;
|
|
|
|
if (node.Tag is DataNode) {
|
|
UnloadLazyDataNode(node);
|
|
}
|
|
}
|
|
|
|
private void NodeExpand (object sender, TreeViewCancelEventArgs e)
|
|
{
|
|
ExpandNode(e.Node);
|
|
}
|
|
|
|
private void NodeCollapse (object sender, TreeViewEventArgs e)
|
|
{
|
|
CollapseNode(e.Node);
|
|
}
|
|
|
|
private void NodeSelected (object sender, TreeViewEventArgs e)
|
|
{
|
|
UpdateToolbar();
|
|
}
|
|
|
|
private void NodeClicked (object sender, TreeNodeMouseClickEventArgs e)
|
|
{
|
|
|
|
}
|
|
|
|
private void NodeDoubleClicked (object sender, TreeNodeMouseClickEventArgs e)
|
|
{
|
|
EditNodeValue(_nodeTree.SelectedNode);
|
|
}
|
|
|
|
private void UpdateToolbar ()
|
|
{
|
|
TreeNode node = _nodeTree.SelectedNode;
|
|
TagNode tag = node.Tag as TagNode;
|
|
|
|
if (tag == null && node.Tag is NbtDataNode) {
|
|
NbtDataNode data = node.Tag as NbtDataNode;
|
|
if (data.Tree != null)
|
|
tag = data.Tree.Root;
|
|
}
|
|
|
|
_buttonRename.Enabled = tag != null && node.Tag is TagNode;
|
|
_buttonDelete.Enabled = tag != null && node.Tag is TagNode;
|
|
_buttonEdit.Enabled = tag != null
|
|
&& node.Tag is TagNode
|
|
&& tag.GetTagType() != TagType.TAG_BYTE_ARRAY
|
|
&& tag.GetTagType() != TagType.TAG_COMPOUND
|
|
&& tag.GetTagType() != TagType.TAG_LIST;
|
|
|
|
if (tag == null || tag.GetTagType() != TagType.TAG_COMPOUND)
|
|
SetTagButtons(false);
|
|
if (tag != null && tag.GetTagType() == TagType.TAG_COMPOUND)
|
|
SetTagButtons(true);
|
|
if (tag != null && tag.GetTagType() == TagType.TAG_LIST && tag.ToTagList().Count == 0)
|
|
SetTagButtons(true);
|
|
|
|
if (tag != null && tag.GetTagType() == TagType.TAG_LIST) {
|
|
switch (tag.ToTagList().ValueType) {
|
|
case TagType.TAG_BYTE:
|
|
_buttonAddTagByte.Enabled = true;
|
|
break;
|
|
case TagType.TAG_SHORT:
|
|
_buttonAddTagShort.Enabled = true;
|
|
break;
|
|
case TagType.TAG_INT:
|
|
_buttonAddTagInt.Enabled = true;
|
|
break;
|
|
case TagType.TAG_LONG:
|
|
_buttonAddTagLong.Enabled = true;
|
|
break;
|
|
case TagType.TAG_FLOAT:
|
|
_buttonAddTagFloat.Enabled = true;
|
|
break;
|
|
case TagType.TAG_DOUBLE:
|
|
_buttonAddTagDouble.Enabled = true;
|
|
break;
|
|
case TagType.TAG_BYTE_ARRAY:
|
|
_buttonAddTagByteArray.Enabled = true;
|
|
break;
|
|
case TagType.TAG_STRING:
|
|
_buttonAddTagString.Enabled = true;
|
|
break;
|
|
case TagType.TAG_LIST:
|
|
_buttonAddTagList.Enabled = true;
|
|
break;
|
|
case TagType.TAG_COMPOUND:
|
|
_buttonAddTagCompound.Enabled = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
private void SetTagButtons (bool state)
|
|
{
|
|
_buttonAddTagByte.Enabled = state;
|
|
_buttonAddTagShort.Enabled = state;
|
|
_buttonAddTagInt.Enabled = state;
|
|
_buttonAddTagLong.Enabled = state;
|
|
_buttonAddTagFloat.Enabled = state;
|
|
_buttonAddTagDouble.Enabled = state;
|
|
_buttonAddTagByteArray.Enabled = state;
|
|
_buttonAddTagString.Enabled = state;
|
|
_buttonAddTagList.Enabled = state;
|
|
_buttonAddTagCompound.Enabled = state;
|
|
}
|
|
|
|
public void LoadDirectory (string path)
|
|
{
|
|
LoadDirectory(path, _nodeTree.Nodes);
|
|
}
|
|
|
|
public void LoadDirectory (string path, TreeNodeCollection parent)
|
|
{
|
|
LoadDirectory(path, parent, path);
|
|
}
|
|
|
|
public void LoadDirectory (string path, TreeNodeCollection parent, string name)
|
|
{
|
|
TreeNode root = new TreeNode(name, 10, 10);
|
|
|
|
foreach (string dirpath in Directory.EnumerateDirectories(path)) {
|
|
root.Nodes.Add(CreateLazyDirectory(dirpath, root));
|
|
}
|
|
|
|
foreach (string filepath in Directory.EnumerateFiles(path)) {
|
|
TryLoadFile(root.Nodes, filepath);
|
|
}
|
|
|
|
parent.Add(root);
|
|
}
|
|
|
|
public void TryLoadFile (TreeNodeCollection parent, string path)
|
|
{
|
|
if (Path.GetExtension(path) == ".mcr") {
|
|
TreeNode node = CreateLazyRegion(path);
|
|
parent.Add(node);
|
|
LinkDataNodeParent(node, node.Parent);
|
|
return;
|
|
}
|
|
|
|
if (Path.GetExtension(path) == ".dat") {
|
|
try {
|
|
NBTFile file = new NBTFile(path);
|
|
NbtTree tree = new NbtTree();
|
|
tree.ReadFrom(file.GetDataInputStream());
|
|
TreeNode node = CreateLazyNbt(path, CompressionType.GZip);
|
|
parent.Add(node);
|
|
LinkDataNodeParent(node, node.Parent);
|
|
return;
|
|
}
|
|
catch { }
|
|
|
|
try {
|
|
NBTFile file = new NBTFile(path);
|
|
NbtTree tree = new NbtTree();
|
|
tree.ReadFrom(file.GetDataInputStream(CompressionType.None));
|
|
TreeNode node = CreateLazyNbt(path, CompressionType.None);
|
|
parent.Add(node);
|
|
LinkDataNodeParent(node, node.Parent);
|
|
return;
|
|
}
|
|
catch { }
|
|
}
|
|
}
|
|
|
|
static Form1 ()
|
|
{
|
|
_tagIconIndex = new Dictionary<TagType, int>();
|
|
_tagIconIndex[TagType.TAG_BYTE] = 0;
|
|
_tagIconIndex[TagType.TAG_SHORT] = 1;
|
|
_tagIconIndex[TagType.TAG_INT] = 2;
|
|
_tagIconIndex[TagType.TAG_LONG] = 3;
|
|
_tagIconIndex[TagType.TAG_FLOAT] = 4;
|
|
_tagIconIndex[TagType.TAG_DOUBLE] = 5;
|
|
_tagIconIndex[TagType.TAG_BYTE_ARRAY] = 6;
|
|
_tagIconIndex[TagType.TAG_STRING] = 7;
|
|
_tagIconIndex[TagType.TAG_LIST] = 8;
|
|
_tagIconIndex[TagType.TAG_COMPOUND] = 9;
|
|
}
|
|
|
|
private void OpenFile ()
|
|
{
|
|
OpenFileDialog ofd = new OpenFileDialog();
|
|
ofd.RestoreDirectory = true;
|
|
ofd.Multiselect = false;
|
|
|
|
if (ofd.ShowDialog() == DialogResult.OK) {
|
|
_nodeTree.Nodes.Clear();
|
|
TryLoadFile(_nodeTree.Nodes, ofd.FileName);
|
|
}
|
|
}
|
|
|
|
private void SaveAll ()
|
|
{
|
|
if (_nodeTree.Nodes.Count > 0) {
|
|
SaveNode(_nodeTree.Nodes[0]);
|
|
}
|
|
}
|
|
|
|
private void SaveNode (TreeNode node)
|
|
{
|
|
foreach (TreeNode sub in node.Nodes) {
|
|
if (sub.Tag != null && sub.Tag is DataNode) {
|
|
SaveNode(sub);
|
|
}
|
|
}
|
|
|
|
if (node.Tag is NbtFileData) {
|
|
SaveNbtFileNode(node);
|
|
}
|
|
else if (node.Tag is RegionChunkData) {
|
|
SaveRegionChunkNode(node);
|
|
}
|
|
}
|
|
|
|
private void SaveNbtFileNode (TreeNode node)
|
|
{
|
|
NbtFileData data = node.Tag as NbtFileData;
|
|
if (data == null || !data.Modified)
|
|
return;
|
|
|
|
NBTFile file = new NBTFile(data.Path);
|
|
using (Stream str = file.GetDataOutputStream(data.CompressionType)) {
|
|
data.Tree.WriteTo(str);
|
|
}
|
|
data.Modified = false;
|
|
}
|
|
|
|
private void SaveRegionChunkNode (TreeNode node)
|
|
{
|
|
RegionChunkData data = node.Tag as RegionChunkData;
|
|
if (data == null || !data.Modified)
|
|
return;
|
|
|
|
using (Stream str = data.Region.GetChunkDataOutputStream(data.X, data.Z)) {
|
|
data.Tree.WriteTo(str);
|
|
}
|
|
data.Modified = false;
|
|
}
|
|
|
|
private void aboutToolStripMenuItem_Click (object sender, EventArgs e)
|
|
{
|
|
new About().Show();
|
|
}
|
|
|
|
private void openToolStripMenuItem_Click (object sender, EventArgs e)
|
|
{
|
|
OpenFile();
|
|
}
|
|
|
|
private void openFolderToolStripMenuItem_Click (object sender, EventArgs e)
|
|
{
|
|
FolderBrowserDialog ofd = new FolderBrowserDialog();
|
|
if (ofd.ShowDialog() == DialogResult.OK) {
|
|
_nodeTree.Nodes.Clear();
|
|
LoadDirectory(ofd.SelectedPath);
|
|
}
|
|
}
|
|
|
|
private void toolStripButton1_Click (object sender, EventArgs e)
|
|
{
|
|
OpenFile();
|
|
}
|
|
|
|
private TreeNode BaseNode (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
return null;
|
|
|
|
TreeNode baseNode = node;
|
|
while (baseNode.Tag == null || !(baseNode.Tag is DataNode)) {
|
|
baseNode = baseNode.Parent;
|
|
}
|
|
|
|
return baseNode;
|
|
}
|
|
|
|
#region Tag Deletion
|
|
|
|
private void DeleteNode (TreeNode node)
|
|
{
|
|
TreeNode baseNode = BaseNode(node);
|
|
|
|
// Can only delete internal NBT nodes
|
|
if (baseNode == null || baseNode == node)
|
|
return;
|
|
|
|
(baseNode.Tag as DataNode).Modified = true;
|
|
|
|
DeleteNodeNbtTag(node);
|
|
|
|
if (node.Parent != null) {
|
|
node.Parent.Text = GetNodeText(node.Parent);
|
|
}
|
|
|
|
node.Remove();
|
|
}
|
|
|
|
private void _buttonDelete_Click (object sender, EventArgs e)
|
|
{
|
|
DeleteNode(_nodeTree.SelectedNode);
|
|
}
|
|
|
|
private void DeleteNodeNbtTag (TreeNode node)
|
|
{
|
|
TagNode tag = node.Tag as TagNode;
|
|
if (tag == null)
|
|
return;
|
|
|
|
TagNode parentTag = node.Parent.Tag as TagNode;
|
|
if (parentTag == null) {
|
|
NbtDataNode parentData = node.Parent.Tag as NbtDataNode;
|
|
if (parentData == null)
|
|
return;
|
|
|
|
parentTag = parentData.Tree.Root;
|
|
if (parentTag == null)
|
|
return;
|
|
}
|
|
|
|
switch (parentTag.GetTagType()) {
|
|
case TagType.TAG_LIST:
|
|
parentTag.ToTagList().Remove(tag);
|
|
break;
|
|
|
|
case TagType.TAG_COMPOUND:
|
|
DeleteTagFromCompound(parentTag.ToTagCompound(), tag);
|
|
break;
|
|
}
|
|
}
|
|
|
|
private void DeleteTagFromCompound (TagNodeCompound parent, TagNode target)
|
|
{
|
|
string match = "";
|
|
foreach (KeyValuePair<string, TagNode> kv in parent) {
|
|
if (kv.Value == target) {
|
|
match = kv.Key;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (match != null) {
|
|
parent.Remove(match);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
private void _buttonSave_Click (object sender, EventArgs e)
|
|
{
|
|
SaveAll();
|
|
}
|
|
|
|
private TagNode GetTagNode (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
return null;
|
|
|
|
if (node.Tag is TagNode) {
|
|
return node.Tag as TagNode;
|
|
}
|
|
else if (node.Tag is NbtDataNode) {
|
|
NbtDataNode data = node.Tag as NbtDataNode;
|
|
if (data == null)
|
|
return null;
|
|
|
|
return data.Tree.Root;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private TagNode GetParentTagNode (TreeNode node)
|
|
{
|
|
if (GetTagNode(node) == null)
|
|
return null;
|
|
|
|
return GetTagNode(node.Parent);
|
|
}
|
|
|
|
private string GetTagNodeName (TreeNode node)
|
|
{
|
|
TagNode tag = GetTagNode(node);
|
|
if (tag == null)
|
|
return null;
|
|
|
|
TagNode parentTag = GetTagNode(node.Parent);
|
|
if (parentTag == null)
|
|
return null;
|
|
|
|
if (parentTag.GetTagType() != TagType.TAG_COMPOUND)
|
|
return null;
|
|
|
|
foreach (KeyValuePair<string, TagNode> sub in parentTag.ToTagCompound()) {
|
|
if (sub.Value == tag)
|
|
return sub.Key;
|
|
}
|
|
|
|
return null;
|
|
}
|
|
|
|
private bool SetTagNodeName (TreeNode node, string name)
|
|
{
|
|
TagNode tag = GetTagNode(node);
|
|
if (tag == null)
|
|
return false;
|
|
|
|
TagNode parentTag = GetTagNode(node.Parent);
|
|
if (parentTag == null)
|
|
return false;
|
|
|
|
if (parentTag.GetTagType() != TagType.TAG_COMPOUND)
|
|
return false;
|
|
|
|
if (parentTag.ToTagCompound().ContainsKey(name))
|
|
return false;
|
|
|
|
string oldName = null;
|
|
foreach (KeyValuePair<string, TagNode> sub in parentTag.ToTagCompound()) {
|
|
if (sub.Value == tag) {
|
|
oldName = sub.Key;
|
|
break;
|
|
}
|
|
}
|
|
|
|
parentTag.ToTagCompound().Remove(oldName);
|
|
parentTag.ToTagCompound().Add(name, tag);
|
|
|
|
return true;
|
|
}
|
|
|
|
private void EditNodeValue (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
return;
|
|
|
|
TagNode tag = GetTagNode(node);
|
|
if (tag == null)
|
|
return;
|
|
|
|
if (tag.GetTagType() == TagType.TAG_BYTE_ARRAY ||
|
|
tag.GetTagType() == TagType.TAG_LIST ||
|
|
tag.GetTagType() == TagType.TAG_COMPOUND)
|
|
return;
|
|
|
|
EditValue form = new EditValue(tag);
|
|
if (form.ShowDialog() == DialogResult.OK) {
|
|
TreeNode baseNode = BaseNode(node);
|
|
if (baseNode != null) {
|
|
(baseNode.Tag as DataNode).Modified = true;
|
|
}
|
|
|
|
node.Text = GetNodeText(node);
|
|
}
|
|
}
|
|
|
|
private void _buttonEdit_Click (object sender, EventArgs e)
|
|
{
|
|
EditNodeValue(_nodeTree.SelectedNode);
|
|
}
|
|
|
|
private void EditNodeName (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
return;
|
|
|
|
TagNode tag = GetTagNode(node);
|
|
if (tag == null)
|
|
return;
|
|
|
|
string name = GetTagNodeName(node);
|
|
if (name == null)
|
|
return;
|
|
|
|
EditValue form = new EditValue(name);
|
|
|
|
TagNode parentTag = GetParentTagNode(node);
|
|
foreach (string key in parentTag.ToTagCompound().Keys) {
|
|
form.InvalidNames.Add(key);
|
|
}
|
|
|
|
if (form.ShowDialog() == DialogResult.OK) {
|
|
TreeNode baseNode = BaseNode(node);
|
|
if (baseNode != null) {
|
|
(baseNode.Tag as DataNode).Modified = true;
|
|
}
|
|
|
|
SetTagNodeName(node, form.NodeName);
|
|
node.Text = GetNodeText(node);
|
|
}
|
|
}
|
|
|
|
private void _buttonRename_Click (object sender, EventArgs e)
|
|
{
|
|
EditNodeName(_nodeTree.SelectedNode);
|
|
}
|
|
|
|
private void AddTagToNode (TreeNode node, TagType type)
|
|
{
|
|
TagNode tag = GetTagNode(node);
|
|
if (tag == null)
|
|
return;
|
|
|
|
if (tag.GetTagType() != TagType.TAG_COMPOUND &&
|
|
tag.GetTagType() != TagType.TAG_LIST)
|
|
return;
|
|
|
|
if (tag.GetTagType() == TagType.TAG_LIST &&
|
|
tag.ToTagList().ValueType != type &&
|
|
tag.ToTagList().Count > 0)
|
|
return;
|
|
|
|
TagNode newNode = null;
|
|
switch (type) {
|
|
case TagType.TAG_BYTE:
|
|
newNode = new TagNodeByte();
|
|
break;
|
|
case TagType.TAG_SHORT:
|
|
newNode = new TagNodeShort();
|
|
break;
|
|
case TagType.TAG_INT:
|
|
newNode = new TagNodeInt();
|
|
break;
|
|
case TagType.TAG_LONG:
|
|
newNode = new TagNodeLong();
|
|
break;
|
|
case TagType.TAG_FLOAT:
|
|
newNode = new TagNodeFloat();
|
|
break;
|
|
case TagType.TAG_DOUBLE:
|
|
newNode = new TagNodeDouble();
|
|
break;
|
|
case TagType.TAG_BYTE_ARRAY:
|
|
newNode = new TagNodeByteArray();
|
|
break;
|
|
case TagType.TAG_STRING:
|
|
newNode = new TagNodeString();
|
|
break;
|
|
case TagType.TAG_LIST:
|
|
newNode = new TagNodeList(TagType.TAG_BYTE);
|
|
break;
|
|
case TagType.TAG_COMPOUND:
|
|
newNode = new TagNodeCompound();
|
|
break;
|
|
}
|
|
|
|
if (tag is TagNodeCompound) {
|
|
TagNodeCompound ctag = tag as TagNodeCompound;
|
|
|
|
EditValue form = new EditValue("");
|
|
foreach (string key in ctag.Keys) {
|
|
form.InvalidNames.Add(key);
|
|
}
|
|
|
|
if (form.ShowDialog() != DialogResult.OK)
|
|
return;
|
|
|
|
ctag.Add(form.NodeName, newNode);
|
|
|
|
TreeNode tnode = NodeFromTag(newNode, form.NodeName);
|
|
node.Nodes.Add(tnode);
|
|
|
|
_nodeTree.SelectedNode = tnode;
|
|
tnode.Expand();
|
|
}
|
|
else if (tag is TagNodeList) {
|
|
TagNodeList ltag = tag as TagNodeList;
|
|
if (ltag.ValueType != type)
|
|
ltag.ChangeValueType(type);
|
|
|
|
ltag.Add(newNode);
|
|
|
|
TreeNode tnode = NodeFromTag(newNode);
|
|
node.Nodes.Add(tnode);
|
|
|
|
_nodeTree.SelectedNode = tnode;
|
|
tnode.Expand();
|
|
}
|
|
|
|
node.Text = GetNodeText(node);
|
|
|
|
TreeNode baseNode = BaseNode(node);
|
|
if (baseNode != null) {
|
|
(baseNode.Tag as DataNode).Modified = true;
|
|
}
|
|
}
|
|
|
|
private void _buttonAddTagByte_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_BYTE);
|
|
}
|
|
|
|
private void _buttonAddTagShort_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_SHORT);
|
|
}
|
|
|
|
private void _buttonAddTagInt_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_INT);
|
|
}
|
|
|
|
private void _buttonAddTagLong_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_LONG);
|
|
}
|
|
|
|
private void _buttonAddTagFloat_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_FLOAT);
|
|
}
|
|
|
|
private void _buttonAddTagDouble_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_DOUBLE);
|
|
}
|
|
|
|
private void _buttonAddTagByteArray_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_BYTE_ARRAY);
|
|
}
|
|
|
|
private void _buttonAddTagString_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_STRING);
|
|
}
|
|
|
|
private void _buttonAddTagList_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_LIST);
|
|
}
|
|
|
|
private void _buttonAddTagCompound_Click (object sender, EventArgs e)
|
|
{
|
|
AddTagToNode(_nodeTree.SelectedNode, TagType.TAG_COMPOUND);
|
|
}
|
|
|
|
private void saveToolStripMenuItem_Click (object sender, EventArgs e)
|
|
{
|
|
SaveAll();
|
|
}
|
|
|
|
private void exitToolStripMenuItem_Click (object sender, EventArgs e)
|
|
{
|
|
Application.Exit();
|
|
}
|
|
|
|
private TreeNode _rootSearchNode;
|
|
private string _searchName;
|
|
private string _searchValue;
|
|
private IEnumerator<TreeNode> _search;
|
|
|
|
private void Find (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
return;
|
|
|
|
Find form = new Find();
|
|
if (form.ShowDialog() == DialogResult.OK) {
|
|
_rootSearchNode = node;
|
|
|
|
_searchName = form.MatchName ? form.NameToken : null;
|
|
_searchValue = form.MatchValue ? form.ValueToken : null;
|
|
|
|
_search = FindNode(node).GetEnumerator();
|
|
|
|
FindNext();
|
|
}
|
|
}
|
|
|
|
private void FindNext ()
|
|
{
|
|
if (_search == null)
|
|
return;
|
|
|
|
statusStrip1.Visible = true;
|
|
Application.DoEvents();
|
|
|
|
_nodeTree.SelectedNode = null;
|
|
|
|
if (!_search.MoveNext()) {
|
|
_nodeTree.SelectedNode = _rootSearchNode;
|
|
if (_rootSearchNode != null && _rootSearchNode.Tag is DataNode) {
|
|
if (_rootSearchNode.IsExpanded && !(_rootSearchNode.Tag as DataNode).Expanded) {
|
|
_rootSearchNode.Collapse();
|
|
}
|
|
}
|
|
|
|
statusStrip1.Visible = false;
|
|
|
|
MessageBox.Show("End of results");
|
|
_search = null;
|
|
}
|
|
|
|
statusStrip1.Visible = false;
|
|
}
|
|
|
|
private IEnumerable<TreeNode> FindNode (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
yield break;
|
|
|
|
bool expand = false;
|
|
|
|
if (node.Tag is DataNode) {
|
|
DataNode data = node.Tag as DataNode;
|
|
if (!data.Expanded) {
|
|
ExpandNode(node);
|
|
expand = true;
|
|
}
|
|
}
|
|
|
|
TagNode tag = GetTagNode(node);
|
|
if (tag != null) {
|
|
bool mName = _searchName == null;
|
|
bool mValue = _searchValue == null;
|
|
|
|
if (_searchName != null) {
|
|
string tagName = GetTagNodeName(node);
|
|
if (tagName != null)
|
|
mName = tagName.Contains(_searchName);
|
|
}
|
|
if (_searchValue != null) {
|
|
string tagValue = GetTagNodeText(node);
|
|
if (tagValue != null)
|
|
mValue = tagValue.Contains(_searchValue);
|
|
}
|
|
|
|
if (mName && mValue) {
|
|
_nodeTree.SelectedNode = node;
|
|
yield return node;
|
|
}
|
|
}
|
|
|
|
foreach (TreeNode sub in node.Nodes) {
|
|
foreach (TreeNode s in FindNode(sub))
|
|
yield return s;
|
|
}
|
|
|
|
if (expand) {
|
|
DataNode data = node.Tag as DataNode;
|
|
if (!data.Modified)
|
|
CollapseNode(node);
|
|
}
|
|
}
|
|
|
|
private void findToolStripMenuItem_Click (object sender, EventArgs e)
|
|
{
|
|
Find(_nodeTree.SelectedNode);
|
|
}
|
|
|
|
private void findNextToolStripMenuItem_Click (object sender, EventArgs e)
|
|
{
|
|
FindNext();
|
|
}
|
|
|
|
private void _buttonFindNext_Click (object sender, EventArgs e)
|
|
{
|
|
if (_search == null)
|
|
Find(_nodeTree.SelectedNode);
|
|
else
|
|
FindNext();
|
|
}
|
|
}
|
|
|
|
public class TagKey : IComparable<TagKey>
|
|
{
|
|
public TagKey (string name, TagType type)
|
|
{
|
|
Name = name;
|
|
TagType = type;
|
|
}
|
|
|
|
public string Name { get; set; }
|
|
public TagType TagType { get; set; }
|
|
|
|
#region IComparer<TagKey> Members
|
|
|
|
public int Compare(TagKey x, TagKey y)
|
|
{
|
|
int typeDiff = (int)x.TagType - (int)y.TagType;
|
|
if (typeDiff != 0)
|
|
return typeDiff;
|
|
|
|
return String.Compare(x.Name, y.Name, true);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IComparable<TagKey> Members
|
|
|
|
public int CompareTo (TagKey other)
|
|
{
|
|
return Compare(this, other);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
|
|
public class DataNode
|
|
{
|
|
public DataNode ()
|
|
{
|
|
}
|
|
|
|
public DataNode (DataNode parent)
|
|
{
|
|
Parent = parent;
|
|
}
|
|
|
|
public DataNode Parent { get; set; }
|
|
|
|
private bool _modified;
|
|
public bool Modified
|
|
{
|
|
get { return _modified; }
|
|
set
|
|
{
|
|
if (value && Parent != null) {
|
|
Parent.Modified = value;
|
|
}
|
|
_modified = value;
|
|
}
|
|
}
|
|
|
|
public bool Expanded { get; set; }
|
|
}
|
|
|
|
public class NbtDataNode : DataNode
|
|
{
|
|
public NbtDataNode ()
|
|
{
|
|
}
|
|
|
|
public NbtDataNode (DataNode parent)
|
|
: base(parent)
|
|
{
|
|
}
|
|
|
|
public NbtTree Tree { get; set; }
|
|
}
|
|
|
|
public class RegionChunkData : NbtDataNode
|
|
{
|
|
public RegionChunkData (RegionFile file, int x, int z)
|
|
: this(null, file, x, z)
|
|
{
|
|
}
|
|
|
|
public RegionChunkData (DataNode parent, RegionFile file, int x, int z)
|
|
: base(parent)
|
|
{
|
|
Region = file;
|
|
X = x;
|
|
Z = z;
|
|
}
|
|
|
|
public RegionFile Region { get; private set; }
|
|
public int X { get; private set; }
|
|
public int Z { get; private set; }
|
|
}
|
|
|
|
public class RegionData : DataNode
|
|
{
|
|
public RegionData (string path)
|
|
: this(null, path)
|
|
{
|
|
}
|
|
|
|
public RegionData (DataNode parent, string path)
|
|
: base(parent)
|
|
{
|
|
Path = path;
|
|
}
|
|
|
|
public string Path { get; private set; }
|
|
}
|
|
|
|
public class NbtFileData : NbtDataNode
|
|
{
|
|
public NbtFileData (string path, CompressionType cztype)
|
|
: this(null, path, cztype)
|
|
{
|
|
}
|
|
|
|
public NbtFileData (DataNode parent, string path, CompressionType cztype)
|
|
: base(parent)
|
|
{
|
|
Path = path;
|
|
CompressionType = cztype;
|
|
}
|
|
|
|
public string Path { get; private set; }
|
|
public CompressionType CompressionType { get; private set; }
|
|
}
|
|
|
|
public class DirectoryData : DataNode
|
|
{
|
|
public DirectoryData (string path)
|
|
: this(null, path)
|
|
{
|
|
}
|
|
|
|
public DirectoryData (DataNode parent, string path)
|
|
: base(parent)
|
|
{
|
|
Path = path;
|
|
}
|
|
|
|
public string Path { get; private set; }
|
|
}
|
|
}
|