#region License Information /* HeuristicLab * Copyright (C) 2002-2010 Heuristic and Evolutionary Algorithms Laboratory (HEAL) * * This file is part of HeuristicLab. * * HeuristicLab is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * HeuristicLab is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with HeuristicLab. If not, see . */ #endregion using System; using System.Collections.Generic; using System.Drawing; using System.Linq; using System.Windows.Forms; using HeuristicLab.Core.Views; using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding; using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding.Views; using HeuristicLab.MainForm; using HeuristicLab.Problems.DataAnalysis.Symbolic; using HeuristicLab.Visualization; namespace HeuristicLab.EvolutionaryTracking.Views { [View("SymbolicExpressionTreeGenealogyGraph")] [Content(typeof(SymbolicExpressionTreeGenealogyGraph), IsDefaultView = true)] public sealed partial class GenealogyGraphView : ItemView { private VisualSymbolicExpressionTreeNode selectedVisualSymbExprTreeNode; private SymbolicExpressionTreeNodeSimilarityComparer comparer; private Dictionary clonedNodes; public new SymbolicExpressionTreeGenealogyGraph Content { get { return (SymbolicExpressionTreeGenealogyGraph)base.Content; } set { base.Content = value; } } public GenealogyGraphView() : base() { InitializeComponent(); // set button icons here because if set in the designer file, they get overwritten this.moveModeButton.Image = Common.Resources.VSImageLibrary.Pointer; this.zoomModeButton.Image = Common.Resources.VSImageLibrary.Zoom; this.selectModeButton.Image = Common.Resources.VSImageLibrary.Object; this.highlightAllButton.Image = Common.Resources.VSImageLibrary.Gradient; this.simpleLineagesCheckBox.Image = Common.Resources.VSImageLibrary.ArrowDown; this.lockGenealogyCheckBox.Image = Common.Resources.VSImageLibrary.ProtectForm; comparer = new SymbolicExpressionTreeNodeSimilarityComparer { MatchConstantValues = true, MatchVariableWeights = true, MatchVariableNames = true }; } protected override void DeregisterContentEvents() { // TODO: Deregister your event handlers here genealogyGraphChart.GenealogyGraphNodeClicked -= graphChart_GenealogyGraphNodeClicked; symbolicExpressionTreeChart.SymbolicExpressionTreeNodeClicked -= treeChart_SymbolicExpressionTreeNodeClicked; symbolicExpressionTreeChart.SymbolicExpressionTreeNodeDoubleClicked -= treeChart_SymbolicExpressionTreeNodeDoubleClicked; Content.GraphUpdated -= genealogyGraphUpdated; base.DeregisterContentEvents(); } protected override void RegisterContentEvents() { base.RegisterContentEvents(); // TODO: Register your event handlers here genealogyGraphChart.GenealogyGraphNodeClicked += graphChart_GenealogyGraphNodeClicked; symbolicExpressionTreeChart.SymbolicExpressionTreeNodeClicked += treeChart_SymbolicExpressionTreeNodeClicked; symbolicExpressionTreeChart.SymbolicExpressionTreeNodeDoubleClicked += treeChart_SymbolicExpressionTreeNodeDoubleClicked; Content.GraphUpdated += genealogyGraphUpdated; } #region Event Handlers (Content) // TODO: Put event handlers of the content here protected override void OnContentChanged() { base.OnContentChanged(); if (Content == null) genealogyGraphChart.Graph = null; else { genealogyGraphChart.Graph = Content; // var best = Content.Nodes.OrderByDescending(x => x.Quality).First(); // symbolicExpressionTreeChart.Tree = best.SymbolicExpressionTree; } } private void Content_GraphChanged(object sender, EventArgs e) { } protected override void SetEnabledStateOfControls() { base.SetEnabledStateOfControls(); genealogyGraphChart.Enabled = Content != null; } #endregion #region Event Handlers (child controls) // TODO: Put event handlers of child controls here. private void graphChart_GenealogyGraphNodeClicked(object sender, MouseEventArgs e) { // the type hierarchy goes like this: VisualGenealogyGraphNode --> GenealogyGraphNode --> SymbolicExpressionTree var visualGenealogyGraphNode = (VisualGenealogyGraphNode)sender; var genealogyGraphNode = visualGenealogyGraphNode.Data; symbolicExpressionTreeChart.Tree = genealogyGraphNode.SymbolicExpressionTree; if (genealogyGraphNode.InEdges == null) return; var arc = genealogyGraphNode.InEdges.Last(x => x.Source != x.Target); if (arc == null || arc.Data == null) return; var fragment = (IFragment)arc.Data; if (fragment.Root == null) return; foreach (var node in fragment.Root.IterateNodesBreadth()) { var visualSymbExprTreeNode = symbolicExpressionTreeChart.GetVisualSymbolicExpressionTreeNode(node); visualSymbExprTreeNode.LineColor = Color.OrangeRed; } symbolicExpressionTreeChart.Repaint(); } private void moveModeButton_CheckedChanged(object sender, EventArgs e) { var btn = (RadioButton)sender; if (btn.Checked) { genealogyGraphChart.Chart.Mode = ChartMode.Move; } } private void zoomModeButton_CheckedChanged(object sender, EventArgs e) { var btn = (RadioButton)sender; if (btn.Checked) { genealogyGraphChart.Chart.Mode = ChartMode.Zoom; } } private void selectModeButton_CheckedChanged(object sender, EventArgs e) { var btn = (RadioButton)sender; if (btn.Checked) { genealogyGraphChart.Chart.Mode = ChartMode.Select; } } private void genealogyGraphUpdated(object sender, EventArgs args) { genealogyGraphChart.Graph = Content; } private void treeChart_SymbolicExpressionTreeNodeClicked(object sender, MouseEventArgs e) { genealogyGraphChart.Chart.UpdateEnabled = false; switch (e.Button) { case MouseButtons.Left: { selectedVisualSymbExprTreeNode = (VisualSymbolicExpressionTreeNode)sender; if (selectedVisualSymbExprTreeNode == null) return; ISymbolicExpressionTreeNode selectedSubtree = selectedVisualSymbExprTreeNode.SymbolicExpressionTreeNode; var clonedSubtree = (ISymbolicExpressionTreeNode)selectedSubtree.Clone(); // map original nodes to cloned nodes clonedNodes = selectedSubtree.IterateNodesPrefix() .Zip(clonedSubtree.IterateNodesPrefix(), (original, clone) => new { original, clone }) .ToDictionary(p => p.original, p => p.clone); foreach (var node in symbolicExpressionTreeChart.Tree.IterateNodesPrefix()) { symbolicExpressionTreeChart.GetVisualSymbolicExpressionTreeNode(node).FillColor = Color.Transparent; } foreach (var node in selectedSubtree.IterateNodesPrefix()) { symbolicExpressionTreeChart.GetVisualSymbolicExpressionTreeNode(node).FillColor = Color.LightBlue; } break; } case MouseButtons.Middle: { var visualNode = (VisualSymbolicExpressionTreeNode)sender; if (selectedVisualSymbExprTreeNode == null || selectedVisualSymbExprTreeNode == visualNode) return; var selectedNode = visualNode.SymbolicExpressionTreeNode; if (clonedNodes.ContainsKey(selectedNode)) { var selectedClone = clonedNodes[selectedNode]; var parent = selectedClone.Parent; if (parent == null) break; int index = parent.IndexOfSubtree(selectedClone); if (index == -1) break; parent.RemoveSubtree(parent.IndexOfSubtree(selectedClone)); foreach (var node in selectedNode.IterateNodesPrefix()) { symbolicExpressionTreeChart.GetVisualSymbolicExpressionTreeNode(node).FillColor = Color.Transparent; } } break; } } if (selectedVisualSymbExprTreeNode == null || clonedNodes == null) return; // update visual graph nodes that contain matching trees MatchNodesAndRepaint(); // refresh the tree chart symbolicExpressionTreeChart.Repaint(); } void MatchNodesAndRepaint() { genealogyGraphChart.Chart.UpdateEnabled = false; genealogyGraphChart.ClearAllNodes(); // clear node colors var fragment = new Fragment(clonedNodes[selectedVisualSymbExprTreeNode.SymbolicExpressionTreeNode]); var fragmentLength = fragment.Length; // highlight nodes genealogyGraphChart.HighlightNodes(from node in genealogyGraphChart.Graph.Nodes let tree = node.SymbolicExpressionTree where tree.Length >= fragmentLength where tree.Root.ContainsFragment(fragment, comparer) select tree); // highlight edges that contain a matching fragment var fragmentSimilarityComparer = new SymbolicExpressionTreeFragmentSimilarityComparer { SimilarityComparer = comparer }; var matchingEdges = genealogyGraphChart.Graph.Nodes.Where(n => n.InEdges != null) .SelectMany(n => n.InEdges).Where(edge => edge.Data != null && ((IFragment)edge.Data).Root != null) .Where(edge => fragmentSimilarityComparer.Equals(fragment, (IFragment)edge.Data)); genealogyGraphChart.HighlightArcs(matchingEdges, Color.DodgerBlue); genealogyGraphChart.Chart.UpdateEnabled = true; genealogyGraphChart.Chart.EnforceUpdate(); } private void treeChart_SymbolicExpressionTreeNodeDoubleClicked(object sender, MouseEventArgs e) { } #endregion private void matchConstantsCheckBox_CheckedChanged(object sender, EventArgs e) { comparer.MatchConstantValues = matchConstantsCheckBox.Checked; if (selectedVisualSymbExprTreeNode != null) MatchNodesAndRepaint(); } private void matchVariableWeightsCheckBox_CheckedChanged(object sender, EventArgs e) { comparer.MatchVariableWeights = matchVariableWeightsCheckBox.Checked; if (selectedVisualSymbExprTreeNode != null) MatchNodesAndRepaint(); } private void matchVariableNamesCheckBox_CheckedChanged(object sender, EventArgs e) { comparer.MatchVariableNames = matchVariableNamesCheckBox.Checked; if (selectedVisualSymbExprTreeNode != null) MatchNodesAndRepaint(); } private void simpleLineagesCheckBox_CheckedChanged(object sender, EventArgs e) { genealogyGraphChart.SimpleLineages = simpleLineagesCheckBox.Checked; } private void highlightAllButton_Click(object sender, EventArgs e) { symbolicExpressionTreeChart.Repaint(); genealogyGraphChart.HighlightAll(); } private void lockGenealogyCheckBox_CheckedChanged(object sender, EventArgs e) { genealogyGraphChart.LockGenealogy = lockGenealogyCheckBox.Checked; } } }