Free cookie consent management tool by TermsFeed Policy Generator

source: trunk/sources/HeuristicLab.Optimization.Views/3.3/ExperimentTreeView.cs @ 6552

Last change on this file since 6552 was 6552, checked in by mkommend, 13 years ago

#1555: Forbid renaming of nodes during experiment execution.

File size: 39.6 KB
Line 
1#region License Information
2/* HeuristicLab
3 * Copyright (C) 2002-2011 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
4 *
5 * This file is part of HeuristicLab.
6 *
7 * HeuristicLab is free software: you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation, either version 3 of the License, or
10 * (at your option) any later version.
11 *
12 * HeuristicLab is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with HeuristicLab. If not, see <http://www.gnu.org/licenses/>.
19 */
20#endregion
21
22using System;
23using System.Collections;
24using System.Collections.Generic;
25using System.Drawing;
26using System.Linq;
27using System.Windows.Forms;
28using HeuristicLab.Collections;
29using HeuristicLab.Common;
30using HeuristicLab.Core;
31using HeuristicLab.Core.Views;
32using HeuristicLab.MainForm;
33using HeuristicLab.PluginInfrastructure;
34
35namespace HeuristicLab.Optimization.Views {
36  public sealed partial class ExperimentTreeView : ItemView {
37    private TypeSelectorDialog typeSelectorDialog;
38    private Dictionary<INamedItem, List<TreeNode>> treeNodeTagMapping;
39
40    public ExperimentTreeView() {
41      InitializeComponent();
42      treeNodeTagMapping = new Dictionary<INamedItem, List<TreeNode>>();
43    }
44
45    protected override void Dispose(bool disposing) {
46      if (disposing) {
47        if (typeSelectorDialog != null) typeSelectorDialog.Dispose();
48        if (components != null) components.Dispose();
49      }
50      base.Dispose(disposing);
51    }
52
53    #region necessary code to handle dock correctly regarding the expanded nodes
54    bool[] expandendedState;
55    protected override void OnHandleCreated(EventArgs e) {
56      base.OnHandleCreated(e);
57      if (expandendedState == null) return;
58      var nodes = IterateTreeNodes().ToList();
59      for (int i = 0; i < nodes.Count; i++)
60        if (expandendedState[i]) nodes[i].Expand();
61    }
62    protected override void OnHandleDestroyed(EventArgs e) {
63      base.OnHandleDestroyed(e);
64      var nodes = IterateTreeNodes().ToList();
65      expandendedState = new bool[nodes.Count];
66      for (int i = 0; i < nodes.Count; i++)
67        expandendedState[i] = nodes[i].IsExpanded;
68    }
69    #endregion
70
71    public new Experiment Content {
72      get { return (Experiment)base.Content; }
73      set { base.Content = value; }
74    }
75
76    #region events registration
77    protected override void RegisterContentEvents() {
78      base.RegisterContentEvents();
79      Content.ExecutionStateChanged += new EventHandler(Content_ExecutionStateChanged);
80      Content.Optimizers.ItemsAdded += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsAdded);
81      Content.Optimizers.ItemsMoved += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsMoved);
82      Content.Optimizers.ItemsRemoved += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsRemoved);
83      Content.Optimizers.ItemsReplaced += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsReplaced);
84      Content.Optimizers.CollectionReset += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_CollectionReset);
85    }
86
87    protected override void DeregisterContentEvents() {
88      Content.ExecutionStateChanged -= new EventHandler(Content_ExecutionStateChanged);
89      Content.Optimizers.ItemsAdded -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsAdded);
90      Content.Optimizers.ItemsMoved -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsMoved);
91      Content.Optimizers.ItemsRemoved -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsRemoved);
92      Content.Optimizers.ItemsReplaced -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsReplaced);
93      Content.Optimizers.CollectionReset -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_CollectionReset);
94      foreach (var optimizer in treeNodeTagMapping.Keys)
95        DeregisterNamedItemEvents(optimizer);
96      base.DeregisterContentEvents();
97    }
98
99    private void RegisterNamedItemEvents(INamedItem namedItem) {
100      namedItem.ToStringChanged += new EventHandler(namedItem_ToStringChanged);
101      namedItem.ItemImageChanged += new EventHandler(namedItem_ItemImageChanged);
102
103      var algorithm = namedItem as IAlgorithm;
104      var batchRun = namedItem as BatchRun;
105      var experiment = namedItem as Experiment;
106
107      if (algorithm != null) {
108        algorithm.Prepared += new EventHandler(algorithm_Prepared);
109        algorithm.ProblemChanged += new EventHandler(algorithm_ProblemChanged);
110      } else if (batchRun != null) {
111        batchRun.OptimizerChanged += new EventHandler(batchRun_OptimizerChanged);
112      } else if (experiment != null) {
113        experiment.Optimizers.ItemsAdded += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsAdded);
114        experiment.Optimizers.ItemsMoved += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsMoved);
115        experiment.Optimizers.ItemsRemoved += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsRemoved);
116        experiment.Optimizers.ItemsReplaced += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsReplaced);
117        experiment.Optimizers.CollectionReset += new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_CollectionReset);
118      }
119    }
120
121    private void DeregisterNamedItemEvents(INamedItem namedItem) {
122      namedItem.ToStringChanged -= new EventHandler(namedItem_ToStringChanged);
123      namedItem.ItemImageChanged -= new EventHandler(namedItem_ItemImageChanged);
124
125      var algorithm = namedItem as IAlgorithm;
126      var batchRun = namedItem as BatchRun;
127      var experiment = namedItem as Experiment;
128
129      if (algorithm != null) {
130        algorithm.Prepared -= new EventHandler(algorithm_Prepared);
131        algorithm.ProblemChanged -= new EventHandler(algorithm_ProblemChanged);
132      } else if (batchRun != null) {
133        batchRun.OptimizerChanged -= new EventHandler(batchRun_OptimizerChanged);
134      } else if (experiment != null) {
135        experiment.Optimizers.ItemsAdded -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsAdded);
136        experiment.Optimizers.ItemsMoved -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsMoved);
137        experiment.Optimizers.ItemsRemoved -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsRemoved);
138        experiment.Optimizers.ItemsReplaced -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_ItemsReplaced);
139        experiment.Optimizers.CollectionReset -= new CollectionItemsChangedEventHandler<IndexedItem<IOptimizer>>(Optimizers_CollectionReset);
140      }
141    }
142    #endregion
143
144    protected override void OnContentChanged() {
145      base.OnContentChanged();
146      if (Content == null) {
147        treeView.Nodes.Clear();
148      } else {
149        UpdateOptimizerTreeView();
150        treeView.ExpandAll();
151      }
152    }
153
154    #region content events
155    private void Content_ExecutionStateChanged(object sender, EventArgs e) {
156      if (InvokeRequired) {
157        Invoke((Action<object, EventArgs>)Content_ExecutionStateChanged, sender, e);
158        return;
159      }
160      SetEnabledStateOfControls();
161    }
162
163    private void algorithm_Prepared(object sender, EventArgs e) {
164      var algorithm = (IAlgorithm)sender;
165      foreach (TreeNode node in treeNodeTagMapping[algorithm]) {
166        TreeNode resultsNode = node.Nodes.OfType<TreeNode>().Where(x => x.Tag is ResultCollection).Single();
167        if (detailsViewHost.Content == resultsNode.Tag)
168          detailsViewHost.Content = algorithm.Results;
169        resultsNode.Tag = algorithm.Results;
170      }
171    }
172
173    private void algorithm_ProblemChanged(object sender, EventArgs e) {
174      if (InvokeRequired) {
175        Invoke((Action<object, EventArgs>)algorithm_ProblemChanged, sender, e);
176        return;
177      }
178
179      var algorithm = (IAlgorithm)sender;
180      foreach (TreeNode node in treeNodeTagMapping[algorithm]) {
181        foreach (TreeNode childNode in node.Nodes.OfType<TreeNode>().ToList()) {
182          DisposeTreeNode(childNode);
183          childNode.Remove();
184        }
185        List<TreeNode> nodes;
186        foreach (TreeNode childNode in CreateAlgorithmChildNodes(algorithm)) {
187          node.Nodes.Add(childNode);
188          NamedItem namedItem = childNode.Tag as NamedItem;
189          if (namedItem != null) {
190            if (!treeNodeTagMapping.TryGetValue(namedItem, out nodes)) {
191              nodes = new List<TreeNode>();
192              treeNodeTagMapping.Add(namedItem, nodes);
193              RegisterNamedItemEvents(namedItem);
194            }
195            nodes.Add(childNode);
196          }
197        }
198
199        node.Expand();
200      }
201
202      RebuildImageList();
203      UpdateDetailsViewHost();
204    }
205
206    private void batchRun_OptimizerChanged(object sender, EventArgs e) {
207      if (InvokeRequired) {
208        Invoke((Action<object, EventArgs>)batchRun_OptimizerChanged, sender, e);
209        return;
210      }
211      var batchRun = (BatchRun)sender;
212      foreach (TreeNode node in treeNodeTagMapping[batchRun]) {
213        foreach (TreeNode childNode in node.Nodes.OfType<TreeNode>().ToList()) {
214          DisposeTreeNode(childNode);
215          childNode.Remove();
216        }
217
218        if (batchRun.Optimizer != null) {
219          UpdateChildTreeNodes(node.Nodes, batchRun);
220          node.Expand();
221        }
222      }
223      RebuildImageList();
224      UpdateDetailsViewHost();
225    }
226
227    private void Optimizers_ItemsAdded(object sender, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>> e) {
228      if (InvokeRequired) {
229        Invoke((Action<object, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>>>)Optimizers_ItemsAdded, sender, e);
230        return;
231      }
232
233      var optimizerList = (OptimizerList)sender;
234      IEnumerable<TreeNodeCollection> parentNodes;
235      if (optimizerList == Content.Optimizers) parentNodes = new List<TreeNodeCollection>() { treeView.Nodes };
236      else {
237        Experiment experiment = treeNodeTagMapping.Keys.OfType<Experiment>().Where(exp => exp.Optimizers == optimizerList).First();
238        parentNodes = treeNodeTagMapping[experiment].Select(node => node.Nodes);
239      }
240
241      foreach (TreeNodeCollection parentNode in parentNodes) {
242        foreach (var childOptimizer in e.Items) {
243          TreeNode childNode = CreateTreeNode(childOptimizer.Value);
244          UpdateChildTreeNodes(childNode.Nodes, childOptimizer.Value);
245          parentNode.Insert(childOptimizer.Index, childNode);
246          childNode.ExpandAll();
247          if (childNode.Parent != null) childNode.Parent.ExpandAll();
248        }
249      }
250      RebuildImageList();
251    }
252    private void Optimizers_ItemsMoved(object sender, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>> e) {
253      if (InvokeRequired) {
254        Invoke((Action<object, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>>>)Optimizers_ItemsMoved, sender, e);
255        return;
256      }
257
258      var optimizerList = (OptimizerList)sender;
259      IEnumerable<TreeNodeCollection> parentNodes;
260      if (optimizerList == Content.Optimizers) parentNodes = new List<TreeNodeCollection>() { treeView.Nodes };
261      else {
262        Experiment experiment = treeNodeTagMapping.Keys.OfType<Experiment>().Where(exp => exp.Optimizers == optimizerList).First();
263        parentNodes = treeNodeTagMapping[experiment].Select(node => node.Nodes);
264      }
265
266      foreach (TreeNodeCollection parentNode in parentNodes) {
267        //get all effected child nodes
268        foreach (TreeNode childNode in parentNode.OfType<TreeNode>()
269          .Where(n => e.OldItems.Select(x => x.Value).Contains((IOptimizer)n.Tag)).ToList()) {
270          DisposeTreeNode(childNode);
271          childNode.Remove();
272        }
273
274        foreach (var childOptimizer in e.Items) {
275          TreeNode childNode = CreateTreeNode(childOptimizer.Value);
276          UpdateChildTreeNodes(childNode.Nodes, childOptimizer.Value);
277          childNode.ExpandAll();
278          parentNode.Insert(childOptimizer.Index, childNode);
279        }
280      }
281      RebuildImageList();
282      UpdateDetailsViewHost();
283    }
284    private void Optimizers_ItemsRemoved(object sender, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>> e) {
285      if (InvokeRequired) {
286        Invoke((Action<object, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>>>)Optimizers_ItemsRemoved, sender, e);
287        return;
288      }
289
290      var optimizerList = (OptimizerList)sender;
291      IEnumerable<TreeNodeCollection> parentNodes;
292      if (optimizerList == Content.Optimizers) parentNodes = new List<TreeNodeCollection>() { treeView.Nodes };
293      else {
294        Experiment experiment = treeNodeTagMapping.Keys.OfType<Experiment>().Where(exp => exp.Optimizers == optimizerList).First();
295        parentNodes = treeNodeTagMapping[experiment].Select(node => node.Nodes);
296      }
297
298      foreach (TreeNodeCollection parentNode in parentNodes) {
299        foreach (var childOptimizer in e.Items) {
300          TreeNode childNode = parentNode[childOptimizer.Index];
301          DisposeTreeNode(childNode);
302          childNode.Remove();
303        }
304      }
305      RebuildImageList();
306      UpdateDetailsViewHost();
307    }
308    private void Optimizers_ItemsReplaced(object sender, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>> e) {
309      if (InvokeRequired) {
310        Invoke((Action<object, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>>>)Optimizers_ItemsReplaced, sender, e);
311        return;
312      }
313
314      var optimizerList = (OptimizerList)sender;
315      IEnumerable<TreeNodeCollection> parentNodes;
316      if (optimizerList == Content.Optimizers) parentNodes = new List<TreeNodeCollection>() { treeView.Nodes };
317      else {
318        Experiment experiment = treeNodeTagMapping.Keys.OfType<Experiment>().Where(exp => exp.Optimizers == optimizerList).First();
319        parentNodes = treeNodeTagMapping[experiment].Select(node => node.Nodes);
320      }
321
322      foreach (TreeNodeCollection parentNode in parentNodes) {
323        foreach (var childOptimizer in e.OldItems) {
324          TreeNode childNode = parentNode.Cast<TreeNode>().Where(n => n.Tag == childOptimizer.Value && n.Index == childOptimizer.Index).First();
325          DisposeTreeNode(childNode);
326          childNode.Remove();
327        }
328        foreach (var childOptimizer in e.Items) {
329          TreeNode childNode = CreateTreeNode(childOptimizer.Value);
330          UpdateChildTreeNodes(childNode.Nodes, childOptimizer.Value);
331          parentNode.Insert(childOptimizer.Index, childNode);
332        }
333      }
334      RebuildImageList();
335      UpdateDetailsViewHost();
336    }
337    private void Optimizers_CollectionReset(object sender, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>> e) {
338      if (InvokeRequired) {
339        Invoke((Action<object, CollectionItemsChangedEventArgs<IndexedItem<IOptimizer>>>)Optimizers_CollectionReset, sender, e);
340        return;
341      }
342
343      var optimizerList = (OptimizerList)sender;
344      IEnumerable<TreeNodeCollection> parentNodes;
345      if (optimizerList == Content.Optimizers) parentNodes = new List<TreeNodeCollection>() { treeView.Nodes };
346      else {
347        Experiment experiment = treeNodeTagMapping.Keys.OfType<Experiment>().Where(exp => exp.Optimizers == optimizerList).First();
348        parentNodes = treeNodeTagMapping[experiment].Select(node => node.Nodes);
349      }
350
351      foreach (TreeNodeCollection parentNode in parentNodes) {
352        foreach (var childOptimizer in e.OldItems) {
353          TreeNode childNode = parentNode.Cast<TreeNode>().Where(n => n.Tag == childOptimizer.Value && n.Index == childOptimizer.Index).First();
354          DisposeTreeNode(childNode);
355          childNode.Remove();
356        }
357        foreach (var childOptimizer in e.Items) {
358          TreeNode childNode = CreateTreeNode(childOptimizer.Value);
359          UpdateChildTreeNodes(childNode.Nodes, childOptimizer.Value);
360          parentNode.Insert(childOptimizer.Index, childNode);
361        }
362      }
363      RebuildImageList();
364      UpdateDetailsViewHost();
365    }
366
367    private void namedItem_ToStringChanged(object sender, EventArgs e) {
368      if (InvokeRequired) {
369        Invoke((Action<object, EventArgs>)namedItem_ToStringChanged, sender, e);
370        return;
371      }
372      var namedItem = (INamedItem)sender;
373      foreach (TreeNode node in treeNodeTagMapping[namedItem])
374        node.Text = namedItem.ToString();
375    }
376
377    private void namedItem_ItemImageChanged(object sender, EventArgs e) {
378      if (InvokeRequired) {
379        Invoke((Action<object, EventArgs>)namedItem_ItemImageChanged, sender, e);
380        return;
381      }
382      INamedItem namedItem = (INamedItem)sender;
383      foreach (TreeNode node in treeNodeTagMapping[namedItem]) {
384        treeView.ImageList.Images[node.ImageIndex] = namedItem.ItemImage;
385        node.ImageIndex = node.ImageIndex;
386      }
387      SetEnabledStateOfControls();
388    }
389    #endregion
390
391    protected override void PropagateStateChanges(Control control, Type type, System.Reflection.PropertyInfo propertyInfo) {
392      return;
393    }
394
395    protected override void SetEnabledStateOfControls() {
396      base.SetEnabledStateOfControls();
397      IOptimizer optimizer = null;
398      IAlgorithm algorithm = null;
399      BatchRun batchRun = null;
400      Experiment experiment = null;
401
402      IOptimizer parentOptimizer = null;
403      Experiment parentExperiment = null;
404
405      if (treeView.SelectedNode != null) {
406        optimizer = treeView.SelectedNode.Tag as IOptimizer;
407        algorithm = optimizer as IAlgorithm;
408        batchRun = optimizer as BatchRun;
409        experiment = optimizer as Experiment;
410
411        if (treeView.SelectedNode.Parent != null) parentOptimizer = treeView.SelectedNode.Parent.Tag as IOptimizer;
412        else parentOptimizer = Content;
413
414        parentExperiment = parentOptimizer as Experiment;
415      }
416
417      treeView.Enabled = Content != null;
418      if (parentOptimizer != null) {
419        detailsViewHost.ReadOnly = parentOptimizer.ExecutionState == ExecutionState.Started;
420        detailsViewHost.Locked = parentOptimizer.ExecutionState == ExecutionState.Started;
421      }
422
423      addButton.Enabled = Content != null && !Locked && !ReadOnly &&
424        (treeView.SelectedNode == null || experiment != null || batchRun != null || algorithm != null);
425      moveUpButton.Enabled = Content != null && !Locked && !ReadOnly &&
426        treeView.SelectedNode != null && treeView.SelectedNode.PrevNode != null && parentExperiment != null;
427      moveDownButton.Enabled = Content != null && !Locked && !ReadOnly &&
428        treeView.SelectedNode != null && treeView.SelectedNode.NextNode != null && parentExperiment != null;
429      removeButton.Enabled = Content != null && !Locked && !ReadOnly && optimizer != null;
430    }
431
432    private void UpdateOptimizerTreeView() {
433      treeView.Nodes.Clear();
434      UpdateChildTreeNodes(treeView.Nodes, Content);
435      RebuildImageList();
436    }
437
438    private void UpdateChildTreeNodes(TreeNodeCollection collection, IOptimizer optimizer) {
439      var batchRun = optimizer as BatchRun;
440      var experiment = optimizer as Experiment;
441
442      if (batchRun != null && batchRun.Optimizer != null) UpdateChildTreeNodes(collection, new List<IOptimizer>() { batchRun.Optimizer });
443      else if (experiment != null) UpdateChildTreeNodes(collection, experiment.Optimizers);
444    }
445    private void UpdateChildTreeNodes(TreeNodeCollection collection, IEnumerable<IOptimizer> optimizers) {
446      foreach (IOptimizer optimizer in optimizers) {
447        var node = CreateTreeNode(optimizer);
448        collection.Add(node);
449        UpdateChildTreeNodes(node.Nodes, optimizer);
450      }
451    }
452
453
454    #region drag & drop
455    private void optimizerTreeView_ItemDrag(object sender, ItemDragEventArgs e) {
456      if (Locked) return;
457
458      TreeNode selectedNode = (TreeNode)e.Item;
459      var item = (IItem)selectedNode.Tag;
460      if (item == null) return;
461
462      DataObject data = new DataObject();
463      data.SetData(HeuristicLab.Common.Constants.DragDropDataFormat, item);
464      validDragOperation = true;
465
466      if (ReadOnly || !(item is IOptimizer)) {
467        DoDragDrop(data, DragDropEffects.Copy | DragDropEffects.Link);
468      } else {
469        DragDropEffects result = DoDragDrop(data, DragDropEffects.Copy | DragDropEffects.Link | DragDropEffects.Move);
470        if ((result & DragDropEffects.Move) == DragDropEffects.Move) {
471          var optimizer = (IOptimizer)item;
472          if (selectedNode.Parent == null) Content.Optimizers.Remove(optimizer);
473          else {
474            var parentOptimizer = (IOptimizer)selectedNode.Parent.Tag;
475            var parentBatchRun = parentOptimizer as BatchRun;
476            var parentExperiment = parentOptimizer as Experiment;
477            if (parentBatchRun != null) parentBatchRun.Optimizer = null;
478            else if (parentExperiment != null) parentExperiment.Optimizers.Remove(optimizer);
479            else throw new NotSupportedException("Handling for specific type not implemented" + parentOptimizer.GetType());
480          }
481        }
482      }
483    }
484
485    private bool validDragOperation = false;
486    private void optimizerTreeView_DragEnter(object sender, DragEventArgs e) {
487      validDragOperation = false;
488      if (!ReadOnly) {
489        var data = e.Data.GetData(HeuristicLab.Common.Constants.DragDropDataFormat);
490        if (data is IOptimizer) validDragOperation = true;
491        else if (data is IProblem) validDragOperation = true;
492        else if (data is IEnumerable) {
493          IEnumerable items = (IEnumerable)data;
494          validDragOperation = true;
495          foreach (object item in items)
496            validDragOperation = validDragOperation && (item is IOptimizer);
497        }
498      }
499    }
500    private void optimizerTreeView_DragOver(object sender, DragEventArgs e) {
501      e.Effect = DragDropEffects.None;
502      if (!validDragOperation) return;
503      Point coordinates = treeView.PointToClient(new Point(e.X, e.Y));
504      TreeNode node = treeView.GetNodeAt(coordinates);
505      Experiment experiment = null;
506      BatchRun batchRun = null;
507      Algorithm algorithm = null;
508
509      if (node == null) experiment = Content;
510      else {
511        experiment = node.Tag as Experiment;
512        batchRun = node.Tag as BatchRun;
513        algorithm = node.Tag as Algorithm;
514      }
515
516      if (batchRun == null && experiment == null && algorithm == null) return;
517
518      var data = e.Data.GetData(HeuristicLab.Common.Constants.DragDropDataFormat);
519
520      if (algorithm != null) {
521        var problem = data as IProblem;
522        if (problem == null) return;
523        if (!algorithm.ProblemType.IsAssignableFrom(problem.GetType())) return;
524      } else if (batchRun != null) {
525        var optimizer = data as IOptimizer;
526        if (optimizer == null) return;
527        if (batchRun == optimizer) return;
528        if (optimizer.NestedOptimizers.Contains(batchRun)) return;
529      } //do not allow recursive nesting of contents
530      else if (experiment != null) {
531        var optimizer = data as IOptimizer;
532        IEnumerable<IOptimizer> optimizers = null;
533        var enumerable = data as IEnumerable;
534        if (enumerable != null) optimizers = enumerable.Cast<IOptimizer>();
535        if (experiment == optimizer) return;
536        if (optimizer != null && optimizer.NestedOptimizers.Contains(experiment)) return;
537        if (optimizers != null && optimizers.Any(x => x.NestedOptimizers.Contains(experiment))) return;
538      }
539
540      if ((e.KeyState & 32) == 32 && e.AllowedEffect.HasFlag(DragDropEffects.Link)) e.Effect = DragDropEffects.Link;  // ALT key
541      else if ((e.KeyState & 4) == 4 && e.AllowedEffect.HasFlag(DragDropEffects.Move)) e.Effect = DragDropEffects.Move;  // SHIFT key
542      else if (e.AllowedEffect.HasFlag(DragDropEffects.Copy)) e.Effect = DragDropEffects.Copy;
543      else if (e.AllowedEffect.HasFlag(DragDropEffects.Move)) e.Effect = DragDropEffects.Move;
544      else if (e.AllowedEffect.HasFlag(DragDropEffects.Link)) e.Effect = DragDropEffects.Link;
545    }
546
547    private void optimizerTreeView_DragDrop(object sender, DragEventArgs e) {
548      Point coordinates = treeView.PointToClient(new Point(e.X, e.Y));
549      TreeNode node = treeView.GetNodeAt(coordinates);
550      Algorithm algorithm = null;
551      BatchRun batchRun = null;
552      Experiment experiment = null;
553
554      if (node == null) experiment = Content;
555      else {
556        algorithm = node.Tag as Algorithm;
557        batchRun = node.Tag as BatchRun;
558        experiment = node.Tag as Experiment;
559      }
560
561      var data = e.Data.GetData(HeuristicLab.Common.Constants.DragDropDataFormat);
562      if (data is IProblem) {
563        var problem = (IProblem)data;
564        if (e.Effect.HasFlag(DragDropEffects.Copy)) problem = (IProblem)problem.Clone();
565        algorithm.Problem = problem;
566      } else if (data is IOptimizer) {
567        IOptimizer optimizer = (IOptimizer)data;
568        if (e.Effect.HasFlag(DragDropEffects.Copy)) optimizer = (IOptimizer)optimizer.Clone();
569        if (batchRun != null) batchRun.Optimizer = optimizer;
570        else if (experiment != null) experiment.Optimizers.Add(optimizer);
571        else throw new NotSupportedException("Handling for specific type not implemented" + node.Tag.GetType());
572      } else if (data is IEnumerable) {
573        IEnumerable<IOptimizer> optimizers = ((IEnumerable)data).Cast<IOptimizer>();
574        if (e.Effect.HasFlag(DragDropEffects.Copy)) {
575          Cloner cloner = new Cloner();
576          optimizers = optimizers.Select(o => cloner.Clone(o));
577        }
578        if (experiment != null) experiment.Optimizers.AddRange(optimizers);
579        else throw new NotSupportedException("Handling for specific type not implemented" + node.Tag.GetType());
580      }
581    }
582    #endregion
583
584    #region control events
585    private void treeView_NodeMouseDoubleClick(object sender, TreeNodeMouseClickEventArgs e) {
586      if (rightClickOccured) return;
587      if (e.X < e.Node.Bounds.Left || e.X > e.Node.Bounds.Right) return;
588      e.Node.Toggle();
589      IContent optimizer = (IContent)e.Node.Tag;
590      MainFormManager.MainForm.ShowContent(optimizer);
591    }
592    private void treeview_NodeMouseClick(object sender, TreeNodeMouseClickEventArgs e) {
593      if (e.X < e.Node.Bounds.Left || e.X > e.Node.Bounds.Right) return;
594      treeView.SelectedNode = e.Node;
595      detailsViewHost.Content = (IContent)e.Node.Tag;
596      SetEnabledStateOfControls();
597    }
598    private void treeView_MouseDown(object sender, MouseEventArgs e) {
599      // enables deselection of treeNodes
600      if (e.Button != MouseButtons.Right) rightClickOccured = false;
601      if (treeView.SelectedNode == null) return;
602      Point coordinates = new Point(e.X, e.Y);
603      TreeNode node = treeView.GetNodeAt(coordinates);
604      if (node == null || coordinates.X < node.Bounds.Left || coordinates.X > node.Bounds.Right) {
605        treeView.SelectedNode = null;
606        detailsViewHost.Content = null;
607        SetEnabledStateOfControls();
608      }
609    }
610
611
612    private void treeView_KeyDown(object sender, KeyEventArgs e) {
613      if (Locked || ReadOnly) return;
614      if (treeView.SelectedNode == null) return;
615      if (!(treeView.SelectedNode.Tag is INamedItem)) return;
616
617      var treeNode = treeView.SelectedNode;
618      var namedItem = (NamedItem)treeNode.Tag;
619      var optimizer = namedItem as IOptimizer;
620
621      if (e.KeyCode == Keys.F2 && !treeNode.IsEditing) {
622        treeNode.BeginEdit();
623        return;
624      }
625
626      if (e.KeyCode == Keys.Delete && optimizer != null) {
627        if (treeNode.Parent == null)
628          Content.Optimizers.Remove(optimizer);
629        else {
630          var batchRun = treeNode.Parent.Tag as BatchRun;
631          var experiment = treeNode.Parent.Tag as Experiment;
632          if (batchRun != null) batchRun.Optimizer = null;
633          else if (experiment != null) experiment.Optimizers.Remove(optimizer);
634          else throw new NotSupportedException("Handling for specific type not implemented" + treeView.SelectedNode.Tag.GetType());
635        }
636        SetEnabledStateOfControls();
637        UpdateDetailsViewHost();
638        RebuildImageList();
639      }
640    }
641
642    private bool rightClickOccured = true;
643    private TreeNode toolStripMenuNode = null;
644    private void treeView_RightClick(object sender, EventArgs e) {
645      rightClickOccured = true;
646      Point coordinates = treeView.PointToClient(Cursor.Position);
647      toolStripMenuNode = treeView.GetNodeAt(coordinates);
648
649      if (toolStripMenuNode != null && coordinates.X >= toolStripMenuNode.Bounds.Left && coordinates.X <= toolStripMenuNode.Bounds.Right) {
650        treeView.SelectedNode = toolStripMenuNode;
651        detailsViewHost.Content = (IContent)toolStripMenuNode.Tag;
652        SetEnabledStateOfControls();
653
654        ExpandToolStripMenuItem.Enabled = !toolStripMenuNode.IsExpanded && toolStripMenuNode.Nodes.Count > 0;
655        ExpandToolStripMenuItem.Visible = !toolStripMenuNode.IsExpanded && toolStripMenuNode.Nodes.Count > 0;
656        CollapseToolStripMenuItem.Enabled = toolStripMenuNode.IsExpanded;
657        CollapseToolStripMenuItem.Visible = toolStripMenuNode.IsExpanded;
658        EditNodeLabelToolStripMenuItem.Enabled = !Locked && !ReadOnly && toolStripMenuNode.Tag != null && toolStripMenuNode.Tag is INamedItem && ((INamedItem)toolStripMenuNode.Tag).CanChangeName;
659        EditNodeLabelToolStripMenuItem.Visible = !Locked && !ReadOnly && toolStripMenuNode.Tag != null && toolStripMenuNode.Tag is INamedItem && ((INamedItem)toolStripMenuNode.Tag).CanChangeName;
660        if (contextMenuStrip.Items.Cast<ToolStripMenuItem>().Any(item => item.Enabled))
661          contextMenuStrip.Show(Cursor.Position);
662      }
663    }
664
665    private void treeView_BeforeLabelEdit(object sender, NodeLabelEditEventArgs e) {
666      if (Locked) e.CancelEdit = true;
667      if (ReadOnly) e.CancelEdit = true;
668      if (e.Node.Tag == null) e.CancelEdit = true;
669      var namedItem = e.Node.Tag as INamedItem;
670      if (namedItem == null) e.CancelEdit = true;
671      else if (!namedItem.CanChangeName) e.CancelEdit = true;
672    }
673    private void treeView_AfterLabelEdit(object sender, NodeLabelEditEventArgs e) {
674      if (e.Label == null) return;
675      e.Node.EndEdit(false);
676      var namedItem = (INamedItem)e.Node.Tag;
677      namedItem.Name = e.Label;
678    }
679
680    private void ExpandToolStripMenuItem_Click(object sender, EventArgs e) {
681      if (toolStripMenuNode != null) toolStripMenuNode.ExpandAll();
682    }
683    private void CollapseToolStripMenuItem_Click(object sender, EventArgs e) {
684      if (toolStripMenuNode != null) toolStripMenuNode.Collapse();
685    }
686    private void EditNodeLabelToolStripMenuItem_Click(object sender, EventArgs e) {
687      if (toolStripMenuNode != null) {
688        if (!toolStripMenuNode.IsEditing) toolStripMenuNode.BeginEdit();
689      }
690    }
691
692    private void addButton_Click(object sender, System.EventArgs e) {
693      if (typeSelectorDialog == null) typeSelectorDialog = new TypeSelectorDialog();
694
695      IAlgorithm algorithm = null;
696      if (treeView.SelectedNode != null && (treeView.SelectedNode.Tag is IAlgorithm))
697        algorithm = (IAlgorithm)treeView.SelectedNode.Tag;
698
699      if (algorithm == null) {
700        typeSelectorDialog.Caption = "Select Optimizer";
701        typeSelectorDialog.TypeSelector.Caption = "Available Optimizers";
702        typeSelectorDialog.TypeSelector.Configure(typeof(IOptimizer), false, true);
703      } else {
704        typeSelectorDialog.Caption = "Select Problem";
705        typeSelectorDialog.TypeSelector.Caption = "Available Problems";
706        typeSelectorDialog.TypeSelector.Configure(algorithm.ProblemType, false, true);
707      }
708
709      if (typeSelectorDialog.ShowDialog(this) == DialogResult.OK) {
710        try {
711          if (algorithm == null) {
712            IOptimizer optimizer = (IOptimizer)typeSelectorDialog.TypeSelector.CreateInstanceOfSelectedType();
713            if (treeView.SelectedNode == null) Content.Optimizers.Add(optimizer);
714            else {
715              var batchRun = treeView.SelectedNode.Tag as BatchRun;
716              var experiment = treeView.SelectedNode.Tag as Experiment;
717              if (batchRun != null) batchRun.Optimizer = optimizer;
718              else if (experiment != null) experiment.Optimizers.Add(optimizer);
719              else throw new NotSupportedException("Handling for specific type not implemented" + treeView.SelectedNode.Tag.GetType());
720            }
721          } else {
722            IProblem problem = (IProblem)typeSelectorDialog.TypeSelector.CreateInstanceOfSelectedType();
723            algorithm.Problem = problem;
724          }
725        }
726        catch (Exception ex) {
727          ErrorHandling.ShowErrorDialog(this, ex);
728        }
729      }
730    }
731
732    private void moveUpButton_Click(object sender, EventArgs e) {
733      var optimizer = (IOptimizer)treeView.SelectedNode.Tag;
734      Experiment experiment = null;
735      if (treeView.SelectedNode.Parent == null) experiment = Content;
736      else experiment = (Experiment)treeView.SelectedNode.Parent.Tag;
737
738      int index = treeView.SelectedNode.Index;
739      experiment.Optimizers.Reverse(index - 1, 2);
740      treeView.SelectedNode = treeNodeTagMapping[optimizer].First();
741      SetEnabledStateOfControls();
742      UpdateDetailsViewHost();
743      RebuildImageList();
744    }
745    private void moveDownButton_Click(object sender, EventArgs e) {
746      var optimizer = (IOptimizer)treeView.SelectedNode.Tag;
747      Experiment experiment = null;
748      if (treeView.SelectedNode.Parent == null) experiment = Content;
749      else experiment = (Experiment)treeView.SelectedNode.Parent.Tag;
750
751      int index = treeView.SelectedNode.Index;
752      experiment.Optimizers.Reverse(index, 2);
753      treeView.SelectedNode = treeNodeTagMapping[optimizer].First();
754      SetEnabledStateOfControls();
755      UpdateDetailsViewHost();
756      RebuildImageList();
757    }
758
759    private void removeButton_Click(object sender, EventArgs e) {
760      var treeNode = treeView.SelectedNode;
761      var optimizer = treeNode.Tag as IOptimizer;
762
763      if (treeNode.Parent == null)
764        Content.Optimizers.Remove(optimizer);
765      else {
766        var batchRun = treeNode.Parent.Tag as BatchRun;
767        var experiment = treeNode.Parent.Tag as Experiment;
768        if (batchRun != null) batchRun.Optimizer = null;
769        else if (experiment != null) experiment.Optimizers.Remove(optimizer);
770        else throw new NotSupportedException("Handling for specific type not implemented" + treeView.SelectedNode.Tag.GetType());
771      }
772      SetEnabledStateOfControls();
773      UpdateDetailsViewHost();
774      RebuildImageList();
775    }
776
777    private void showDetailsCheckBox_CheckedChanged(object sender, System.EventArgs e) {
778      if (showDetailsCheckBox.Checked) {
779        splitContainer.Panel2Collapsed = false;
780        detailsGroupBox.Enabled = treeView.SelectedNode != null;
781        detailsViewHost.Content = treeView.SelectedNode != null ? (IContent)treeView.SelectedNode.Tag : null;
782      } else {
783        splitContainer.Panel2Collapsed = true;
784        detailsViewHost.Content = null;
785      }
786    }
787    #endregion
788
789    #region helpers
790    private void UpdateDetailsViewHost() {
791      if (treeView.SelectedNode != null)
792        detailsViewHost.Content = (IContent)treeView.SelectedNode.Tag;
793      else
794        detailsViewHost.Content = null;
795    }
796
797    private TreeNode CreateTreeNode(IOptimizer optimizer) {
798      TreeNode node = new TreeNode(optimizer.ToString());
799      node.Tag = optimizer;
800
801      var algorithm = optimizer as IAlgorithm;
802      if (algorithm != null) {
803        foreach (TreeNode childNode in CreateAlgorithmChildNodes(algorithm))
804          node.Nodes.Add(childNode);
805      }
806
807      List<TreeNode> nodes;
808      if (!treeNodeTagMapping.TryGetValue(optimizer, out nodes)) {
809        nodes = new List<TreeNode>();
810        treeNodeTagMapping.Add(optimizer, nodes);
811        RegisterNamedItemEvents(optimizer);
812      }
813      nodes.Add(node);
814
815      foreach (TreeNode childNode in node.Nodes) {
816        NamedItem namedItem = childNode.Tag as NamedItem;
817        if (namedItem != null) {
818          if (!treeNodeTagMapping.TryGetValue(namedItem, out nodes)) {
819            nodes = new List<TreeNode>();
820            treeNodeTagMapping.Add(namedItem, nodes);
821            RegisterNamedItemEvents(namedItem);
822          }
823          nodes.Add(childNode);
824        }
825      }
826      return node;
827    }
828
829    private IEnumerable<TreeNode> CreateAlgorithmChildNodes(IAlgorithm algorithm) {
830      TreeNode problemNode;
831      if (algorithm.Problem != null) {
832        problemNode = new TreeNode(algorithm.Problem.Name);
833        problemNode.Tag = algorithm.Problem;
834      } else {
835        problemNode = new TreeNode("No Problem");
836        problemNode.Tag = null;
837      }
838      TreeNode parametersNode = new TreeNode("Parameters");
839      parametersNode.Tag = algorithm.Parameters;
840      TreeNode resultsNode = new TreeNode("Results");
841      resultsNode.Tag = algorithm.Results;
842
843      yield return problemNode;
844      yield return parametersNode;
845      yield return resultsNode;
846    }
847
848    private void DisposeTreeNode(TreeNode node) {
849      var namedItem = node.Tag as INamedItem;
850      if (namedItem == null) return;
851
852      List<TreeNode> nodes;
853      if (!treeNodeTagMapping.TryGetValue(namedItem, out nodes))
854        throw new ArgumentException();
855      nodes.Remove(node);
856      if (nodes.Count == 0) {
857        treeNodeTagMapping.Remove(namedItem);
858        DeregisterNamedItemEvents(namedItem);
859      }
860    }
861
862    private IEnumerable<TreeNode> IterateTreeNodes(TreeNode node = null) {
863      TreeNodeCollection nodes;
864      if (node == null)
865        nodes = treeView.Nodes;
866      else {
867        nodes = node.Nodes;
868        yield return node;
869      }
870
871      foreach (var childNode in nodes.OfType<TreeNode>())
872        foreach (var n in IterateTreeNodes(childNode))
873          yield return n;
874    }
875
876    private void RebuildImageList() {
877      if (InvokeRequired) {
878        Invoke((Action)RebuildImageList);
879        return;
880      }
881
882      treeView.BeginUpdate();
883
884      treeView.ImageList.Images.Clear();
885      var topLevelNodes = treeView.Nodes.OfType<TreeNode>().ToArray();
886      var nodes = IterateTreeNodes().ToList();
887      treeView.Nodes.Clear();
888
889      foreach (TreeNode treeNode in nodes) {
890        var item = (IItem)treeNode.Tag;
891        treeView.ImageList.Images.Add(item == null ? HeuristicLab.Common.Resources.VSImageLibrary.Nothing : item.ItemImage);
892        treeNode.ImageIndex = treeView.ImageList.Images.Count - 1;
893        treeNode.SelectedImageIndex = treeNode.ImageIndex;
894      }
895      treeView.Nodes.AddRange(topLevelNodes);
896      treeView.EndUpdate();
897    }
898    #endregion
899
900
901    public sealed class CustomTreeView : System.Windows.Forms.TreeView {
902      protected override void WndProc(ref System.Windows.Forms.Message m) {
903        const int WM_RBUTTONDOWN = 0x204;
904        if (m.Msg == WM_RBUTTONDOWN) {
905          //Raise your custom event right click event to prevent node highlighting
906          OnRightClick();
907          return;
908        }
909        base.WndProc(ref m);
910      }
911
912      public event EventHandler RightClick;
913      private void OnRightClick() {
914        var handler = RightClick;
915        if (handler != null) RightClick(this, EventArgs.Empty);
916      }
917    }
918  }
919}
Note: See TracBrowser for help on using the repository browser.