#region License Information
/* HeuristicLab
* Copyright (C) 2002-2014 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.Drawing.Drawing2D;
using System.Linq;
using System.Windows.Forms;
using HeuristicLab.Common;
using HeuristicLab.Visualization;
namespace HeuristicLab.EvolutionTracking.Views {
public partial class GenealogyGraphChart : ChartControl {
private IGenealogyGraph genealogyGraph;
private const double XIncrement = 30;
private const double YIncrement = 30;
private const double Diameter = 20;
private readonly Brush defaultBrush;
private readonly Pen defaultPen;
private Dictionary nodeMap;
private Dictionary, VisualGenealogyGraphArc> arcMap;
#region chart modes
public bool SimpleLineages { get; set; }
public bool LockGenealogy { get; set; }
public bool TraceFragments { get; set; }
#endregion
private bool DrawInProgress { get; set; } // do not try to update the chart while the drawing is not finished
public IGenealogyGraph GenealogyGraph {
get { return genealogyGraph; }
set {
if (value == null) return;
genealogyGraph = value;
Clear();
DrawGraph(XIncrement, YIncrement, Diameter);
}
}
public IGenealogyGraphNode SelectedGraphNode {
get {
return SelectedVisualNode == null ? null : SelectedVisualNode.Data;
}
set {
if (value == null || value == SelectedGraphNode)
return;
SelectedVisualNode = GetMappedNode(value);
UpdateSelectedVisualNode();
}
}
private void Clear() {
nodeMap = new Dictionary();
arcMap = new Dictionary, VisualGenealogyGraphArc>();
Chart.Group.Clear();
}
// public bool UpdateEnabled {
// get { return Chart.UpdateEnabled; }
// set { Chart.UpdateEnabled = value; }
// }
// public void EnforceUpdate() {
// Chart.EnforceUpdate();
// }
private Visualization.Rectangle TargetRectangle { get; set; }
protected VisualGenealogyGraphNode SelectedVisualNode { get; set; }
public VisualGenealogyGraphNode GetMappedNode(IGenealogyGraphNode node) {
VisualGenealogyGraphNode v;
nodeMap.TryGetValue(node, out v);
return v;
}
public VisualGenealogyGraphArc GetMappedArc(IGenealogyGraphNode source, IGenealogyGraphNode target) {
VisualGenealogyGraphNode visualSource, visualTarget;
nodeMap.TryGetValue(source, out visualSource);
nodeMap.TryGetValue(target, out visualTarget);
if (visualSource == null || visualTarget == null) return null;
VisualGenealogyGraphArc arc;
arcMap.TryGetValue(new Tuple(visualSource, visualTarget), out arc);
return arc;
}
public GenealogyGraphChart() {
InitializeComponent();
defaultBrush = new SolidBrush(Color.Transparent);
defaultPen = new Pen(Color.DarkGray);
}
protected virtual void DrawGraph(double xIncrement, double yIncrement, double diameter) {
this.SuspendRendering();
DrawInProgress = true;
var ranks = GenealogyGraph.Ranks.Select(t => new { Rank = t.Key, Nodes = t.Value }).OrderBy(p => p.Rank).ToList();
double x = 0;
double y = PreferredSize.Height + yIncrement + 2 * diameter;
foreach (var rank in ranks) {
var nodes = rank.Nodes.ToList();
nodes.Sort((a, b) => b.CompareTo(a)); // sort descending by quality
var nl = Environment.NewLine;
foreach (var node in nodes) {
var brush = new SolidBrush(node.GetColor());
var visualNode = new VisualGenealogyGraphNode(Chart, x, y, x + diameter, y + diameter, defaultPen, brush) {
Data = node,
ToolTipText = "Rank: " + node.Rank + nl +
"Quality: " + String.Format("{0:0.0000}", node.Quality) + nl +
"IsElite: " + node.IsElite + nl +
"In/Out Degree: " + node.InDegree + " " + node.OutDegree
};
Chart.Group.Add(visualNode);
nodeMap.Add(node, visualNode);
x += xIncrement;
}
y -= yIncrement;
x = 0;
}
// add arcs
foreach (var node in GenealogyGraph.Vertices) {
if (!node.InArcs.Any()) continue;
var visualNode = GetMappedNode(node);
if (visualNode == null) continue;
foreach (var arc in node.InArcs) {
var parent = arc.Source;
var visualParent = GetMappedNode(parent);
if (visualParent == null) continue;
var pen = Pens.Transparent;
var visualArc = AddArc(Chart, visualParent, visualNode, pen);
Chart.OneLayerDown(visualArc); // send it behind the visual nodes
if (!arcMap.ContainsKey(Tuple.Create(visualParent, visualNode))) {
arcMap.Add(Tuple.Create(visualParent, visualNode), visualArc);
}
}
}
ResumeRendering();
DrawInProgress = false;
}
public event MouseEventHandler GenealogyGraphNodeClicked;
protected void OnGenealogyGraphNodeClicked(object sender, MouseEventArgs e) {
var clicked = GenealogyGraphNodeClicked;
if (clicked != null) clicked(sender, e);
}
public event MouseEventHandler GenealogyGraphNodeDoubleClicked;
protected void OnGenealogyGraphNodeDoubleClicked(object sender, MouseEventArgs e) {
var doubleClicked = GenealogyGraphNodeDoubleClicked;
if (doubleClicked != null)
doubleClicked(sender, e);
}
#region event handlers
protected override void PictureBoxOnMouseMove(object sender, MouseEventArgs e) {
if (!DrawInProgress) {
switch (e.Button) {
case MouseButtons.Left:
Mode = ChartMode.Select;
Cursor = Cursors.Default;
break;
case MouseButtons.Middle:
Mode = ChartMode.Move;
Cursor = Cursors.Hand;
break;
}
}
base.PictureBoxOnMouseMove(sender, e);
}
protected override void PictureBoxOnMouseUp(object sender, MouseEventArgs e) {
Cursor = Cursors.Default;
if (Mode == ChartMode.Move) {
Mode = ChartMode.Select;
return;
}
if (Mode != ChartMode.Select) {
base.PictureBoxOnMouseUp(sender, e);
return;
}
var primitive = Chart.GetAllPrimitives(e.Location).FirstOrDefault(p => p is VisualGenealogyGraphNode);
if (primitive == null) {
SelectedVisualNode = null;
return;
}
if (SelectedVisualNode == primitive) return;
SelectedVisualNode = primitive as VisualGenealogyGraphNode;
if (SelectedVisualNode == null) return;
UpdateSelectedVisualNode(); // redraw ancestries, mark node etc.
base.PictureBoxOnMouseUp(sender, e);
}
private void UpdateSelectedVisualNode() {
if (!LockGenealogy) {
// new node has been selected, clean up
SuspendRendering();
if (ModifierKeys != Keys.Shift) {
// clear colors
ClearPrimitives();
}
// use a rectangle to highlight the currently selected genealogy graph node
var graphNode = SelectedVisualNode.Data;
var visualNode = GetMappedNode(graphNode);
DrawLineage(visualNode, n => SimpleLineages ? n.IncomingArcs.Take(1) : n.IncomingArcs, a => a.Source);
((SolidBrush)visualNode.Brush).Color = Color.Transparent;
DrawLineage(visualNode, n => n.OutgoingArcs, a => a.Target);
}
MarkSelectedNode();
// update
ResumeRendering();
if (SelectedVisualNode != null)
OnGenealogyGraphNodeClicked(SelectedVisualNode, null); // emit clicked event
}
#endregion
#region drawing routines
private static void DrawLineage(VisualGenealogyGraphNode node, Func> arcSelector, Func nodeSelector) {
var brush = (SolidBrush)node.Brush;
if (brush.Color != Color.Transparent) return; // this lineage was already drawn (avoid redrawing common ancestors)
brush.Color = node.Data.GetColor();
var arcs = arcSelector(node);
var pen = new Pen(Color.Transparent);
foreach (var arc in arcs) {
var source = arc.Source.Data;
var target = arc.Target.Data;
var start = new Point((int)arc.Start.X, (int)arc.Start.Y);
var end = new Point((int)arc.End.X, (int)arc.End.Y);
arc.Pen = pen;
arc.Pen.Brush = new LinearGradientBrush(start, end, source.GetColor(), target.GetColor());
DrawLineage(nodeSelector(arc), arcSelector, nodeSelector);
}
}
void MarkSelectedNode() {
var center = SelectedVisualNode.Center;
var size = SelectedVisualNode.Size;
double x1 = center.X - size.Width / 2;
double x2 = x1 + size.Width;
double y1 = center.Y - size.Height / 2;
double y2 = y1 + size.Height;
if (TargetRectangle == null) {
var lowerLeft = new PointD(x1, y1);
var upperRight = new PointD(x2, y2);
TargetRectangle = new Visualization.Rectangle(Chart, lowerLeft, upperRight, new Pen(Color.Black), null);
Chart.Group.Add(TargetRectangle);
} else {
var lowerLeft = new PointD(x1, y1);
var upperRight = new PointD(x2, y2);
TargetRectangle.SetPosition(lowerLeft, upperRight);
}
}
private static VisualGenealogyGraphArc AddArc(IChart chart, VisualGenealogyGraphNode source, VisualGenealogyGraphNode target, Pen pen, Brush brush = null) {
var arc = new VisualGenealogyGraphArc(chart, source, target, pen) { Brush = brush };
arc.UpdatePosition();
source.OutgoingArcs.Add(arc);
target.IncomingArcs.Add(arc);
chart.Group.Add(arc);
return arc;
}
public void ClearArcs() {
foreach (var primitive in Chart.Group.Primitives.OfType()) {
primitive.Pen = Pens.Transparent;
}
}
public void ClearNodes() {
foreach (var primitive in Chart.Group.Primitives.OfType()) {
primitive.Brush = new SolidBrush(Color.Transparent);
primitive.Pen = new Pen(Color.LightGray);
}
}
public virtual void ClearPrimitives() {
foreach (var primitive in Chart.Group.Primitives) {
if (primitive is VisualGenealogyGraphArc) {
primitive.Pen = Pens.Transparent;
} else if (primitive is VisualGenealogyGraphNode) {
primitive.Brush = new SolidBrush(Color.Transparent);
primitive.Pen = new Pen(Color.DarkGray);
}
}
}
public void HighlightNodes(IEnumerable nodes, bool clearPrimitives = true) {
if (clearPrimitives)
ClearPrimitives();
foreach (var node in nodes) {
var graphNode = GetMappedNode(node);
graphNode.Brush = new SolidBrush(node.GetColor());
}
}
public void HighlightNodes(IEnumerable nodes, Color color, bool clearPrimitives = true) {
if (clearPrimitives)
ClearPrimitives();
foreach (var node in nodes.Select(GetMappedNode)) {
node.Brush = new SolidBrush(color);
}
}
public void HighlightAll() {
foreach (var visualNode in nodeMap.Values) {
visualNode.Brush = new SolidBrush(visualNode.Data.GetColor());
}
foreach (var arc in arcMap.Values) {
var source = arc.Source.Data;
var target = arc.Target.Data;
var start = new Point((int)arc.Start.X, (int)arc.Start.Y);
var end = new Point((int)arc.End.X, (int)arc.End.Y);
arc.Pen.Brush = new LinearGradientBrush(start, end, source.GetColor(), target.GetColor());
}
}
public void HighlightArc(IGenealogyGraphNode source, IGenealogyGraphNode target) {
var arc = GetMappedArc(source, target) ??
AddArc(Chart, GetMappedNode(source), GetMappedNode(target), new Pen(Color.Transparent));
var start = new Point((int)arc.Start.X, (int)arc.Start.Y);
var end = new Point((int)arc.End.X, (int)arc.End.Y);
arc.Pen = new Pen(Color.Transparent);
arc.Pen.Brush = new LinearGradientBrush(start, end, source.GetColor(), target.GetColor());
}
#endregion
}
internal static class Util {
public static Color GetColor(this IGenealogyGraphNode node) {
if (double.IsNaN(node.Quality))
return ColorGradient.Colors[0];
var colorIndex = (int)Math.Round(node.Quality * ColorGradient.Colors.Count);
if (colorIndex >= ColorGradient.Colors.Count) return ColorGradient.Colors.Last();
return ColorGradient.Colors[colorIndex];
}
}
}