mirror of
https://github.com/jaquadro/NBTExplorer.git
synced 2025-01-25 00:36:26 +00:00
1ccf299f9c
Conflicts: Vendor/MultiSelectTreeView/MultiSelectTreeview.cs Windows/MainForm.cs
1034 lines
33 KiB
C#
1034 lines
33 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Text;
|
|
using System.Windows.Forms;
|
|
using NBTExplorer.Model;
|
|
using Substrate.Nbt;
|
|
using NBTExplorer.Windows;
|
|
using NBTExplorer.Vendor.MultiSelectTreeView;
|
|
using System.IO;
|
|
|
|
namespace NBTExplorer.Controllers
|
|
{
|
|
public class MessageBoxEventArgs : EventArgs
|
|
{
|
|
public bool Cancel { get; set; }
|
|
public string Message { get; set; }
|
|
|
|
public MessageBoxEventArgs (string message)
|
|
{
|
|
Message = message;
|
|
}
|
|
}
|
|
|
|
public class NodeTreeController
|
|
{
|
|
private TreeView _nodeTree;
|
|
private MultiSelectTreeView _multiTree;
|
|
|
|
private IconRegistry _iconRegistry;
|
|
|
|
//private TagCompoundDataNode _rootData;
|
|
private RootDataNode _rootData;
|
|
|
|
public NodeTreeController (TreeView nodeTree)
|
|
{
|
|
_nodeTree = nodeTree;
|
|
_multiTree = nodeTree as MultiSelectTreeView;
|
|
|
|
InitializeIconRegistry();
|
|
ShowVirtualRoot = true;
|
|
|
|
//_rootData = new TagCompoundDataNode(new TagNodeCompound());
|
|
_rootData = new RootDataNode();
|
|
RefreshRootNodes();
|
|
}
|
|
|
|
public RootDataNode Root
|
|
{
|
|
get { return _rootData; }
|
|
}
|
|
|
|
public TreeView Tree
|
|
{
|
|
get { return _nodeTree; }
|
|
}
|
|
|
|
public bool ShowVirtualRoot { get; set; }
|
|
|
|
public string VirtualRootDisplay
|
|
{
|
|
get { return _rootData.NodeDisplay; }
|
|
set
|
|
{
|
|
_rootData.SetDisplayName(value);
|
|
if (ShowVirtualRoot && _nodeTree.Nodes.Count > 0)
|
|
UpdateNodeText(_nodeTree.Nodes[0]);
|
|
}
|
|
}
|
|
|
|
public event EventHandler SelectionInvalidated;
|
|
|
|
protected virtual void OnSelectionInvalidated (EventArgs e)
|
|
{
|
|
if (SelectionInvalidated != null)
|
|
SelectionInvalidated(this, e);
|
|
}
|
|
|
|
private void OnSelectionInvalidated ()
|
|
{
|
|
OnSelectionInvalidated(EventArgs.Empty);
|
|
}
|
|
|
|
public event EventHandler<MessageBoxEventArgs> ConfirmAction;
|
|
|
|
protected virtual void OnConfirmAction (MessageBoxEventArgs e)
|
|
{
|
|
if (ConfirmAction != null)
|
|
ConfirmAction(this, e);
|
|
}
|
|
|
|
private bool OnConfirmAction (string message)
|
|
{
|
|
MessageBoxEventArgs e = new MessageBoxEventArgs(message);
|
|
OnConfirmAction(e);
|
|
return !e.Cancel;
|
|
}
|
|
|
|
|
|
private TreeNode SelectedNode
|
|
{
|
|
get
|
|
{
|
|
if (_multiTree != null)
|
|
return _multiTree.SelectedNode;
|
|
else
|
|
return _nodeTree.SelectedNode;
|
|
}
|
|
}
|
|
|
|
#region Load / Save
|
|
|
|
public void Save ()
|
|
{
|
|
foreach (TreeNode node in _nodeTree.Nodes) {
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (dataNode != null)
|
|
dataNode.Save();
|
|
}
|
|
|
|
OnSelectionInvalidated();
|
|
}
|
|
|
|
public int OpenPaths (string[] paths)
|
|
{
|
|
_nodeTree.Nodes.Clear();
|
|
|
|
int failCount = 0;
|
|
|
|
foreach (string path in paths) {
|
|
if (Directory.Exists(path)) {
|
|
DirectoryDataNode node = new DirectoryDataNode(path);
|
|
_nodeTree.Nodes.Add(CreateUnexpandedNode(node));
|
|
}
|
|
else if (File.Exists(path)) {
|
|
DataNode node = null;
|
|
foreach (var item in FileTypeRegistry.RegisteredTypes) {
|
|
if (item.Value.NamePatternTest(path))
|
|
node = item.Value.NodeCreate(path);
|
|
}
|
|
|
|
if (node != null)
|
|
_nodeTree.Nodes.Add(CreateUnexpandedNode(node));
|
|
else
|
|
failCount++;
|
|
}
|
|
}
|
|
|
|
if (_nodeTree.Nodes.Count > 0) {
|
|
_nodeTree.Nodes[0].Expand();
|
|
}
|
|
|
|
OnSelectionInvalidated();
|
|
|
|
return failCount;
|
|
}
|
|
|
|
#endregion
|
|
|
|
public void CreateNode (TreeNode node, TagType type)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanCreateTag(type))
|
|
return;
|
|
|
|
if (dataNode.CreateNode(type)) {
|
|
node.Text = dataNode.NodeDisplay;
|
|
RefreshChildNodes(node, dataNode);
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
public void CreateNode (TagType type)
|
|
{
|
|
if (_nodeTree.SelectedNode == null)
|
|
return;
|
|
|
|
CreateNode(_nodeTree.SelectedNode, type);
|
|
}
|
|
|
|
public void DeleteNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanDeleteNode)
|
|
return;
|
|
|
|
if (dataNode.DeleteNode()) {
|
|
UpdateNodeText(node.Parent);
|
|
node.Remove();
|
|
|
|
RemoveNodeFromSelection(node);
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
private bool DeleteNode (IList<TreeNode> nodes)
|
|
{
|
|
bool? elideChildren = null;
|
|
if (!CanOperateOnNodesEx(nodes, Predicates.DeleteNodePred, out elideChildren))
|
|
return false;
|
|
|
|
if (elideChildren == true)
|
|
nodes = ElideChildren(nodes);
|
|
|
|
bool selectionModified = false;
|
|
foreach (TreeNode node in nodes) {
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (dataNode.DeleteNode()) {
|
|
UpdateNodeText(node.Parent);
|
|
node.Remove();
|
|
|
|
RemoveNodeFromSelection(node);
|
|
selectionModified = true;
|
|
}
|
|
}
|
|
|
|
if (selectionModified)
|
|
OnSelectionInvalidated();
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool RemoveNodeFromSelection (TreeNode node)
|
|
{
|
|
bool selectionModified = false;
|
|
|
|
if (_multiTree != null) {
|
|
if (_multiTree.SelectedNodes.Contains(node)) {
|
|
_multiTree.SelectedNodes.Remove(node);
|
|
selectionModified = true;
|
|
}
|
|
|
|
if (_multiTree.SelectedNode == node) {
|
|
_multiTree.SelectedNode = null;
|
|
selectionModified = true;
|
|
}
|
|
}
|
|
|
|
if (_nodeTree.SelectedNode == node) {
|
|
_nodeTree.SelectedNode = null;
|
|
selectionModified = true;
|
|
}
|
|
|
|
return selectionModified;
|
|
}
|
|
|
|
public void DeleteSelection ()
|
|
{
|
|
if (_multiTree != null)
|
|
DeleteNode(_multiTree.SelectedNodes);
|
|
else
|
|
DeleteNode(SelectedNode);
|
|
}
|
|
|
|
public void EditNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanEditNode)
|
|
return;
|
|
|
|
if (dataNode.EditNode()) {
|
|
node.Text = dataNode.NodeDisplay;
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
public void EditSelection ()
|
|
{
|
|
EditNode(SelectedNode);
|
|
}
|
|
|
|
private void RenameNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanRenameNode)
|
|
return;
|
|
|
|
if (dataNode.RenameNode()) {
|
|
node.Text = dataNode.NodeDisplay;
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
public void RenameSelection ()
|
|
{
|
|
RenameNode(SelectedNode);
|
|
}
|
|
|
|
private void RefreshNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanRefreshNode)
|
|
return;
|
|
|
|
if (!OnConfirmAction("Refresh data anyway?"))
|
|
return;
|
|
|
|
if (dataNode.RefreshNode()) {
|
|
RefreshChildNodes(node, dataNode);
|
|
ExpandToEdge(node);
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
public void RefreshSelection ()
|
|
{
|
|
RefreshNode(SelectedNode);
|
|
}
|
|
|
|
private void ExpandToEdge (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (dataNode.IsExpanded) {
|
|
if (!node.IsExpanded)
|
|
node.Expand();
|
|
|
|
foreach (TreeNode child in node.Nodes)
|
|
ExpandToEdge(child);
|
|
}
|
|
}
|
|
|
|
private void CopyNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanCopyNode)
|
|
return;
|
|
|
|
dataNode.CopyNode();
|
|
}
|
|
|
|
private void CutNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanCutNode)
|
|
return;
|
|
|
|
if (dataNode.CutNode()) {
|
|
UpdateNodeText(node.Parent);
|
|
node.Remove();
|
|
|
|
RemoveNodeFromSelection(node);
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
private void PasteNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (!dataNode.CanPasteIntoNode)
|
|
return;
|
|
|
|
if (dataNode.PasteNode()) {
|
|
node.Text = dataNode.NodeDisplay;
|
|
RefreshChildNodes(node, dataNode);
|
|
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
public void CopySelection ()
|
|
{
|
|
CopyNode(SelectedNode);
|
|
}
|
|
|
|
public void CutSelection ()
|
|
{
|
|
CutNode(SelectedNode);
|
|
}
|
|
|
|
public void PasteIntoSelection ()
|
|
{
|
|
PasteNode(SelectedNode);
|
|
}
|
|
|
|
public void MoveNodeUp (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
return;
|
|
|
|
DataNode datanode = node.Tag as DataNode;
|
|
if (datanode == null || !datanode.CanMoveNodeUp)
|
|
return;
|
|
|
|
if (datanode.ChangeRelativePosition(-1)) {
|
|
RefreshChildNodes(node.Parent, datanode.Parent);
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
public void MoveNodeDown (TreeNode node)
|
|
{
|
|
if (node == null)
|
|
return;
|
|
|
|
DataNode datanode = node.Tag as DataNode;
|
|
if (datanode == null || !datanode.CanMoveNodeDown)
|
|
return;
|
|
|
|
if (datanode.ChangeRelativePosition(1)) {
|
|
RefreshChildNodes(node.Parent, datanode.Parent);
|
|
OnSelectionInvalidated();
|
|
}
|
|
}
|
|
|
|
public void MoveSelectionUp ()
|
|
{
|
|
MoveNodeUp(SelectedNode);
|
|
}
|
|
|
|
public void MoveSelectionDown ()
|
|
{
|
|
MoveNodeDown(SelectedNode);
|
|
}
|
|
|
|
/*public void CreateRootNode (TagType type)
|
|
{
|
|
if (_rootData.CreateNode(type)) {
|
|
RefreshRootNodes();
|
|
UpdateUI(_rootData);
|
|
}
|
|
}*/
|
|
|
|
public void ExpandNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
if (node.IsExpanded)
|
|
return;
|
|
|
|
node.Nodes.Clear();
|
|
|
|
DataNode backNode = node.Tag as DataNode;
|
|
if (!backNode.IsExpanded) {
|
|
backNode.Expand();
|
|
node.Text = backNode.NodeDisplay;
|
|
}
|
|
|
|
foreach (DataNode child in backNode.Nodes)
|
|
node.Nodes.Add(CreateUnexpandedNode(child));
|
|
}
|
|
|
|
public void ExpandSelectedNode ()
|
|
{
|
|
ExpandNode(SelectedNode);
|
|
if (SelectedNode != null)
|
|
SelectedNode.Expand();
|
|
}
|
|
|
|
public void CollapseNode (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode backNode = node.Tag as DataNode;
|
|
if (backNode.IsModified)
|
|
return;
|
|
|
|
backNode.Collapse();
|
|
node.Name = backNode.NodeDisplay;
|
|
|
|
node.Nodes.Clear();
|
|
if (backNode.HasUnexpandedChildren)
|
|
node.Nodes.Add(new TreeNode());
|
|
}
|
|
|
|
public void CollapseSelectedNode ()
|
|
{
|
|
CollapseNode(SelectedNode);
|
|
}
|
|
|
|
private TreeNode GetRootFromDataNodePath (DataNode node, out Stack<DataNode> hierarchy)
|
|
{
|
|
hierarchy = new Stack<DataNode>();
|
|
while (node != null) {
|
|
hierarchy.Push(node);
|
|
node = node.Parent;
|
|
}
|
|
|
|
DataNode rootDataNode = hierarchy.Pop();
|
|
TreeNode frontNode = null;
|
|
foreach (TreeNode child in _nodeTree.Nodes) {
|
|
if (child.Tag == rootDataNode)
|
|
frontNode = child;
|
|
}
|
|
|
|
return frontNode;
|
|
}
|
|
|
|
private TreeNode FindFrontNode (DataNode node)
|
|
{
|
|
Stack<DataNode> hierarchy;
|
|
TreeNode frontNode = GetRootFromDataNodePath(node, out hierarchy);
|
|
|
|
if (frontNode == null)
|
|
return null;
|
|
|
|
while (hierarchy.Count > 0) {
|
|
if (!frontNode.IsExpanded) {
|
|
frontNode.Nodes.Add(new TreeNode());
|
|
frontNode.Expand();
|
|
}
|
|
|
|
DataNode childData = hierarchy.Pop();
|
|
foreach (TreeNode childFront in frontNode.Nodes) {
|
|
if (childFront.Tag == childData) {
|
|
frontNode = childFront;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
return frontNode;
|
|
}
|
|
|
|
public void CollapseBelow (DataNode node)
|
|
{
|
|
Stack<DataNode> hierarchy;
|
|
TreeNode frontNode = GetRootFromDataNodePath(node, out hierarchy);
|
|
|
|
if (frontNode == null)
|
|
return;
|
|
|
|
while (hierarchy.Count > 0) {
|
|
if (!frontNode.IsExpanded)
|
|
return;
|
|
|
|
DataNode childData = hierarchy.Pop();
|
|
foreach (TreeNode childFront in frontNode.Nodes) {
|
|
if (childFront.Tag == childData) {
|
|
frontNode = childFront;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (frontNode.IsExpanded)
|
|
frontNode.Collapse();
|
|
}
|
|
|
|
public void SelectNode (DataNode node)
|
|
{
|
|
if (_multiTree != null) {
|
|
_multiTree.SelectedNode = FindFrontNode(node);
|
|
}
|
|
else {
|
|
_nodeTree.SelectedNode = FindFrontNode(node);
|
|
}
|
|
}
|
|
|
|
public void RefreshRootNodes ()
|
|
{
|
|
if (ShowVirtualRoot) {
|
|
_nodeTree.Nodes.Clear();
|
|
_nodeTree.Nodes.Add(CreateUnexpandedNode(_rootData));
|
|
return;
|
|
}
|
|
|
|
if (_rootData.HasUnexpandedChildren)
|
|
_rootData.Expand();
|
|
|
|
Dictionary<DataNode, TreeNode> currentNodes = new Dictionary<DataNode, TreeNode>();
|
|
foreach (TreeNode child in _nodeTree.Nodes) {
|
|
if (child.Tag is DataNode)
|
|
currentNodes.Add(child.Tag as DataNode, child);
|
|
}
|
|
|
|
_nodeTree.Nodes.Clear();
|
|
foreach (DataNode child in _rootData.Nodes) {
|
|
if (!currentNodes.ContainsKey(child))
|
|
_nodeTree.Nodes.Add(CreateUnexpandedNode(child));
|
|
else
|
|
_nodeTree.Nodes.Add(currentNodes[child]);
|
|
}
|
|
|
|
//if (_nodeTree.Nodes.Count == 0 && _rootData.HasUnexpandedChildren) {
|
|
// _rootData.Expand();
|
|
// RefreshRootNodes();
|
|
//}
|
|
}
|
|
|
|
public void RefreshChildNodes (TreeNode node, DataNode dataNode)
|
|
{
|
|
Dictionary<DataNode, TreeNode> currentNodes = new Dictionary<DataNode, TreeNode>();
|
|
foreach (TreeNode child in node.Nodes) {
|
|
if (child.Tag is DataNode)
|
|
currentNodes.Add(child.Tag as DataNode, child);
|
|
}
|
|
|
|
node.Nodes.Clear();
|
|
foreach (DataNode child in dataNode.Nodes) {
|
|
if (!currentNodes.ContainsKey(child))
|
|
node.Nodes.Add(CreateUnexpandedNode(child));
|
|
else
|
|
node.Nodes.Add(currentNodes[child]);
|
|
}
|
|
|
|
foreach (TreeNode child in node.Nodes)
|
|
child.ContextMenuStrip = BuildNodeContextMenu(child, child.Tag as DataNode);
|
|
|
|
if (node.Nodes.Count == 0 && dataNode.HasUnexpandedChildren) {
|
|
ExpandNode(node);
|
|
node.Expand();
|
|
}
|
|
}
|
|
|
|
public void RefreshTreeNode (DataNode dataNode)
|
|
{
|
|
TreeNode node = FindFrontNode(dataNode);
|
|
RefreshChildNodes(node, dataNode);
|
|
}
|
|
|
|
private void InitializeIconRegistry ()
|
|
{
|
|
_iconRegistry = new IconRegistry();
|
|
_iconRegistry.DefaultIcon = 15;
|
|
|
|
_iconRegistry.Register(typeof(TagByteDataNode), 0);
|
|
_iconRegistry.Register(typeof(TagShortDataNode), 1);
|
|
_iconRegistry.Register(typeof(TagIntDataNode), 2);
|
|
_iconRegistry.Register(typeof(TagLongDataNode), 3);
|
|
_iconRegistry.Register(typeof(TagFloatDataNode), 4);
|
|
_iconRegistry.Register(typeof(TagDoubleDataNode), 5);
|
|
_iconRegistry.Register(typeof(TagByteArrayDataNode), 6);
|
|
_iconRegistry.Register(typeof(TagStringDataNode), 7);
|
|
_iconRegistry.Register(typeof(TagListDataNode), 8);
|
|
_iconRegistry.Register(typeof(TagCompoundDataNode), 9);
|
|
_iconRegistry.Register(typeof(RegionChunkDataNode), 9);
|
|
_iconRegistry.Register(typeof(DirectoryDataNode), 10);
|
|
_iconRegistry.Register(typeof(RegionFileDataNode), 11);
|
|
_iconRegistry.Register(typeof(CubicRegionDataNode), 11);
|
|
_iconRegistry.Register(typeof(NbtFileDataNode), 12);
|
|
_iconRegistry.Register(typeof(TagIntArrayDataNode), 14);
|
|
_iconRegistry.Register(typeof(RootDataNode), 16);
|
|
}
|
|
|
|
private void UpdateNodeText (TreeNode node)
|
|
{
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
node.Text = dataNode.NodeDisplay;
|
|
}
|
|
|
|
public bool CheckModifications ()
|
|
{
|
|
foreach (TreeNode node in _nodeTree.Nodes) {
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (dataNode != null && dataNode.IsModified)
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private TreeNode CreateUnexpandedNode (DataNode node)
|
|
{
|
|
TreeNode frontNode = new TreeNode(node.NodeDisplay);
|
|
frontNode.ImageIndex = _iconRegistry.Lookup(node.GetType());
|
|
frontNode.SelectedImageIndex = frontNode.ImageIndex;
|
|
frontNode.Tag = node;
|
|
//frontNode.ContextMenuStrip = BuildNodeContextMenu(node);
|
|
|
|
if (node.HasUnexpandedChildren || node.Nodes.Count > 0)
|
|
frontNode.Nodes.Add(new TreeNode());
|
|
|
|
return frontNode;
|
|
}
|
|
|
|
public ContextMenuStrip BuildNodeContextMenu (TreeNode frontNode, DataNode node)
|
|
{
|
|
if (node == null)
|
|
return null;
|
|
|
|
ContextMenuStrip menu = new ContextMenuStrip();
|
|
|
|
if (node.HasUnexpandedChildren || node.Nodes.Count > 0) {
|
|
if (frontNode.IsExpanded) {
|
|
ToolStripMenuItem itemCollapse = new ToolStripMenuItem("&Collapse", null, _contextCollapse_Click);
|
|
itemCollapse.Font = new System.Drawing.Font(itemCollapse.Font, System.Drawing.FontStyle.Bold);
|
|
|
|
ToolStripMenuItem itemExpandChildren = new ToolStripMenuItem("Expand C&hildren", null, _contextExpandChildren_Click);
|
|
ToolStripMenuItem itemExpandTree = new ToolStripMenuItem("Expand &Tree", null, _contextExpandTree_Click);
|
|
|
|
menu.Items.AddRange(new ToolStripItem[] {
|
|
itemCollapse, new ToolStripSeparator(), itemExpandChildren, itemExpandTree,
|
|
});
|
|
}
|
|
else {
|
|
ToolStripMenuItem itemExpand = new ToolStripMenuItem("&Expand", null, _contextExpand_Click);
|
|
itemExpand.Font = new System.Drawing.Font(itemExpand.Font, System.Drawing.FontStyle.Bold);
|
|
|
|
menu.Items.Add(itemExpand);
|
|
}
|
|
}
|
|
|
|
if (node.CanReoderNode) {
|
|
ToolStripMenuItem itemUp = new ToolStripMenuItem("Move &Up", Properties.Resources.ArrowUp, _contextMoveUp_Click);
|
|
ToolStripMenuItem itemDn = new ToolStripMenuItem("Move &Down", Properties.Resources.ArrowDown, _contextMoveDown_Click);
|
|
|
|
itemUp.Enabled = node.CanMoveNodeUp;
|
|
itemDn.Enabled = node.CanMoveNodeDown;
|
|
|
|
menu.Items.Add(itemUp);
|
|
menu.Items.Add(itemDn);
|
|
}
|
|
|
|
return (menu.Items.Count > 0) ? menu : null;
|
|
}
|
|
|
|
private void _contextCollapse_Click (object sender, EventArgs e)
|
|
{
|
|
if (_multiTree.SelectedNode != null)
|
|
_multiTree.SelectedNode.Collapse();
|
|
}
|
|
|
|
private void _contextExpand_Click (object sender, EventArgs e)
|
|
{
|
|
if (_multiTree.SelectedNode != null)
|
|
_multiTree.SelectedNode.Expand();
|
|
}
|
|
|
|
private void _contextExpandChildren_Click (object sender, EventArgs e)
|
|
{
|
|
if (_multiTree.SelectedNode != null) {
|
|
foreach (TreeNode node in _multiTree.SelectedNode.Nodes)
|
|
node.Expand();
|
|
}
|
|
}
|
|
|
|
private void _contextExpandTree_Click (object sender, EventArgs e)
|
|
{
|
|
if (_multiTree.SelectedNode != null) {
|
|
_multiTree.SelectedNode.ExpandAll();
|
|
}
|
|
}
|
|
|
|
private void _contextMoveUp_Click (object sender, EventArgs e)
|
|
{
|
|
MoveSelectionUp();
|
|
}
|
|
|
|
private void _contextMoveDown_Click (object sender, EventArgs e)
|
|
{
|
|
MoveSelectionDown();
|
|
}
|
|
|
|
#region Capability Checking
|
|
|
|
#region Capability Predicates
|
|
|
|
public delegate bool DataNodePredicate (DataNode dataNode, out GroupCapabilities caps);
|
|
|
|
public static class Predicates
|
|
{
|
|
public static bool CreateByteNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_BYTE);
|
|
}
|
|
|
|
public static bool CreateShortNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_SHORT);
|
|
}
|
|
|
|
public static bool CreateIntNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_INT);
|
|
}
|
|
|
|
public static bool CreateLongNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_LONG);
|
|
}
|
|
|
|
public static bool CreateFloatNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_FLOAT);
|
|
}
|
|
|
|
public static bool CreateDoubleNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_DOUBLE);
|
|
}
|
|
|
|
public static bool CreateByteArrayNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_BYTE_ARRAY);
|
|
}
|
|
|
|
public static bool CreateIntArrayNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_INT_ARRAY);
|
|
}
|
|
|
|
public static bool CreateStringNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_STRING);
|
|
}
|
|
|
|
public static bool CreateListNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_LIST);
|
|
}
|
|
|
|
public static bool CreateCompoundNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = GroupCapabilities.Single;
|
|
return (dataNode != null) && dataNode.CanCreateTag(TagType.TAG_COMPOUND);
|
|
}
|
|
|
|
public static bool RenameNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.RenameNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanRenameNode;
|
|
}
|
|
|
|
public static bool EditNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.EditNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanEditNode;
|
|
}
|
|
|
|
public static bool DeleteNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.DeleteNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanDeleteNode;
|
|
}
|
|
|
|
public static bool MoveNodeUpPred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.ReorderNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanMoveNodeUp;
|
|
}
|
|
|
|
public static bool MoveNodeDownPred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.ReorderNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanMoveNodeDown;
|
|
}
|
|
|
|
public static bool CutNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.CutNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanCutNode;
|
|
}
|
|
|
|
public static bool CopyNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.CopyNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanCopyNode;
|
|
}
|
|
|
|
public static bool PasteIntoNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.PasteIntoNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanPasteIntoNode;
|
|
}
|
|
|
|
public static bool SearchNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.SearchNodeCapabilites;
|
|
return (dataNode != null) && dataNode.CanSearchNode;
|
|
}
|
|
|
|
public static bool ReorderNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.ReorderNodeCapabilities;
|
|
return (dataNode != null) && dataNode.CanReoderNode;
|
|
}
|
|
|
|
public static bool RefreshNodePred (DataNode dataNode, out GroupCapabilities caps)
|
|
{
|
|
caps = dataNode.RefreshNodeCapabilites;
|
|
return (dataNode != null) && dataNode.CanRefreshNode;
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
public bool CanOperateOnSelection (DataNodePredicate pred)
|
|
{
|
|
if (_multiTree != null)
|
|
return CanOperateOnNodes(_multiTree.SelectedNodes, pred);
|
|
else
|
|
return CanOperateOnNodes(new List<TreeNode>() { _nodeTree.SelectedNode }, pred);
|
|
}
|
|
|
|
private bool CanOperateOnNodes (IList<TreeNode> nodes, DataNodePredicate pred)
|
|
{
|
|
bool? elideChildren = null;
|
|
return CanOperateOnNodesEx(nodes, pred, out elideChildren);
|
|
}
|
|
|
|
private bool CanOperateOnNodesEx (IList<TreeNode> nodes, DataNodePredicate pred, out bool? elideChildren)
|
|
{
|
|
GroupCapabilities caps = GroupCapabilities.All;
|
|
elideChildren = null;
|
|
|
|
foreach (TreeNode node in nodes) {
|
|
if (node == null || !(node.Tag is DataNode))
|
|
return false;
|
|
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
GroupCapabilities dataCaps;
|
|
if (!pred(dataNode, out dataCaps))
|
|
return false;
|
|
|
|
caps &= dataCaps;
|
|
|
|
bool elideChildrenFlag = (dataNode.DeleteNodeCapabilities & GroupCapabilities.ElideChildren) == GroupCapabilities.ElideChildren;
|
|
if (elideChildren == null)
|
|
elideChildren = elideChildrenFlag;
|
|
if (elideChildren != elideChildrenFlag)
|
|
return false;
|
|
}
|
|
|
|
if (nodes.Count > 1 && !SufficientCapabilities(nodes, caps))
|
|
return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
private IList<TreeNode> ElideChildren (IList<TreeNode> nodes)
|
|
{
|
|
List<TreeNode> filtered = new List<TreeNode>();
|
|
|
|
foreach (TreeNode node in nodes) {
|
|
TreeNode parent = node.Parent;
|
|
bool foundParent = false;
|
|
|
|
while (parent != null) {
|
|
if (nodes.Contains(parent)) {
|
|
foundParent = true;
|
|
break;
|
|
}
|
|
parent = parent.Parent;
|
|
}
|
|
|
|
if (!foundParent)
|
|
filtered.Add(node);
|
|
}
|
|
|
|
return filtered;
|
|
}
|
|
|
|
private bool CommonContainer (IEnumerable<TreeNode> nodes)
|
|
{
|
|
object container = null;
|
|
foreach (TreeNode node in nodes) {
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (container == null)
|
|
container = dataNode.Parent;
|
|
|
|
if (container != dataNode.Parent)
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool CommonType (IEnumerable<TreeNode> nodes)
|
|
{
|
|
Type datatype = null;
|
|
foreach (TreeNode node in nodes) {
|
|
DataNode dataNode = node.Tag as DataNode;
|
|
if (datatype == null)
|
|
datatype = dataNode.GetType();
|
|
|
|
if (datatype != dataNode.GetType())
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
private bool SufficientCapabilities (IEnumerable<TreeNode> nodes, GroupCapabilities commonCaps)
|
|
{
|
|
bool commonContainer = CommonContainer(nodes);
|
|
bool commonType = CommonType(nodes);
|
|
|
|
bool pass = true;
|
|
if (commonContainer && commonType)
|
|
pass &= ((commonCaps & GroupCapabilities.SiblingSameType) == GroupCapabilities.SiblingSameType);
|
|
else if (commonContainer && !commonType)
|
|
pass &= ((commonCaps & GroupCapabilities.SiblingMixedType) == GroupCapabilities.SiblingMixedType);
|
|
else if (!commonContainer && commonType)
|
|
pass &= ((commonCaps & GroupCapabilities.MultiSameType) == GroupCapabilities.MultiSameType);
|
|
else if (!commonContainer && !commonType)
|
|
pass &= ((commonCaps & GroupCapabilities.MultiMixedType) == GroupCapabilities.MultiMixedType);
|
|
|
|
return pass;
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|