1 | #region License Information
|
---|
2 | /* HeuristicLab
|
---|
3 | * Copyright (C) 2002-2008 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 |
|
---|
22 | using System;
|
---|
23 | using System.Collections.Generic;
|
---|
24 | using System.Text;
|
---|
25 | using System.Drawing;
|
---|
26 | using System.Linq;
|
---|
27 | using HeuristicLab.Charting;
|
---|
28 | using System.Windows.Forms;
|
---|
29 |
|
---|
30 | namespace HeuristicLab.CEDMA.Charting {
|
---|
31 | public class Histogram : Chart {
|
---|
32 | private static readonly Color defaultColor = Color.Blue;
|
---|
33 | private static readonly Color selectionColor = Color.Red;
|
---|
34 | private static readonly Pen defaultPen = new Pen(defaultColor);
|
---|
35 | private static readonly Brush defaultBrush = defaultPen.Brush;
|
---|
36 | private static readonly Pen selectionPen = new Pen(selectionColor);
|
---|
37 | private static readonly Brush selectionBrush = selectionPen.Brush;
|
---|
38 |
|
---|
39 | private double minX;
|
---|
40 | private double maxX;
|
---|
41 | private double maxFrequency;
|
---|
42 | private const int N_BUCKETS = 50;
|
---|
43 | private const int MAX_BUCKETS = 100;
|
---|
44 | private List<Record> records;
|
---|
45 | private ResultList results;
|
---|
46 | private Dictionary<IPrimitive, List<Record>> primitiveToRecordsDictionary;
|
---|
47 | private Dictionary<Record, IPrimitive> recordToPrimitiveDictionary;
|
---|
48 | private Group bars;
|
---|
49 | private string dimension;
|
---|
50 |
|
---|
51 | public Histogram(ResultList results, double x1, double y1, double x2, double y2)
|
---|
52 | : this(results, new PointD(x1, y1), new PointD(x2, y2)) {
|
---|
53 | }
|
---|
54 |
|
---|
55 | public Histogram(ResultList results, PointD lowerLeft, PointD upperRight)
|
---|
56 | : base(lowerLeft, upperRight) {
|
---|
57 | records = new List<Record>();
|
---|
58 | primitiveToRecordsDictionary = new Dictionary<IPrimitive, List<Record>>();
|
---|
59 | recordToPrimitiveDictionary = new Dictionary<Record, IPrimitive>();
|
---|
60 | this.results = results;
|
---|
61 | foreach(Record r in results.Records) {
|
---|
62 | records.Add(r);
|
---|
63 | }
|
---|
64 | results.OnRecordAdded += new EventHandler<RecordAddedEventArgs>(results_OnRecordAdded);
|
---|
65 | results.Changed += new EventHandler(results_Changed);
|
---|
66 | }
|
---|
67 |
|
---|
68 | void results_Changed(object sender, EventArgs e) {
|
---|
69 | ResetViewSize();
|
---|
70 | Repaint();
|
---|
71 | EnforceUpdate();
|
---|
72 | }
|
---|
73 |
|
---|
74 | void results_OnRecordAdded(object sender, RecordAddedEventArgs e) {
|
---|
75 | lock(records) {
|
---|
76 | records.Add(e.Record);
|
---|
77 | }
|
---|
78 | }
|
---|
79 |
|
---|
80 | public void ShowFrequency(string dimension) {
|
---|
81 | if(this.dimension != dimension) {
|
---|
82 | this.dimension = dimension;
|
---|
83 | ResetViewSize();
|
---|
84 | Repaint();
|
---|
85 | ZoomToViewSize();
|
---|
86 | }
|
---|
87 | }
|
---|
88 |
|
---|
89 | private void Repaint() {
|
---|
90 | lock(records) {
|
---|
91 | if(dimension == null) return;
|
---|
92 | UpdateEnabled = false;
|
---|
93 | Group.Clear();
|
---|
94 | primitiveToRecordsDictionary.Clear();
|
---|
95 | recordToPrimitiveDictionary.Clear();
|
---|
96 | bars = new Group(this);
|
---|
97 | Group.Add(new Axis(this, 0, 0, AxisType.Both));
|
---|
98 | UpdateViewSize(0, 0);
|
---|
99 | PaintHistogram(records);
|
---|
100 | Group.Add(bars);
|
---|
101 | UpdateEnabled = true;
|
---|
102 | }
|
---|
103 | }
|
---|
104 |
|
---|
105 | private void PaintHistogram(IEnumerable<Record> records) {
|
---|
106 | var values = records.Select(r => new { Record = r, Value = r.Get(dimension) }).Where(
|
---|
107 | x => !double.IsNaN(x.Value) && !double.IsInfinity(x.Value) && x.Value != double.MinValue && x.Value != double.MaxValue).OrderBy(x => x.Value);
|
---|
108 | if(values.Count() == 0) return;
|
---|
109 | double bucketSize = 1.0;
|
---|
110 | var frequencies = values.GroupBy(x => x.Value);
|
---|
111 | if(frequencies.Count() > MAX_BUCKETS) {
|
---|
112 | double min = values.ElementAt((int)(values.Count() * 0.05)).Value;
|
---|
113 | double max = values.ElementAt((int)(values.Count() * 0.95)).Value;
|
---|
114 | bucketSize = (max - min) / N_BUCKETS;
|
---|
115 | frequencies = values.GroupBy(x => Math.Min(Math.Max(min, Math.Floor((x.Value - min) / bucketSize) * bucketSize + min), max));
|
---|
116 | }
|
---|
117 | foreach(var g in frequencies) {
|
---|
118 | double freq = g.Count();
|
---|
119 | double selectedFreq = g.Where(r=>r.Record.Selected).Count();
|
---|
120 | double lower = g.Key;
|
---|
121 | double upper = g.Key + bucketSize;
|
---|
122 | HeuristicLab.Charting.Rectangle bar = new HeuristicLab.Charting.Rectangle(this, lower, 0, upper, freq, defaultPen, defaultBrush);
|
---|
123 | primitiveToRecordsDictionary[bar] = g.Select(r => r.Record).ToList();
|
---|
124 | primitiveToRecordsDictionary[bar].ForEach(x => recordToPrimitiveDictionary[x] = bar);
|
---|
125 | HeuristicLab.Charting.Rectangle selectedBar = new HeuristicLab.Charting.Rectangle(this, lower, 0, upper, selectedFreq, selectionPen, selectionBrush);
|
---|
126 | primitiveToRecordsDictionary[selectedBar] = g.Select(r => r.Record).Where(r=>r.Selected).ToList();
|
---|
127 | primitiveToRecordsDictionary[selectedBar].ForEach(x => recordToPrimitiveDictionary[x] = bar);
|
---|
128 | if(lower == frequencies.First().Key) {
|
---|
129 | selectedBar.ToolTipText = " x < " + upper + " : " + selectedFreq;
|
---|
130 | bar.ToolTipText = " x < " + upper + " : " + freq;
|
---|
131 | } else if(lower == frequencies.Last().Key) {
|
---|
132 | selectedBar.ToolTipText = "x >= " + lower + " : " + selectedFreq;
|
---|
133 | bar.ToolTipText = "x >= " + lower + " : " + freq;
|
---|
134 | } else {
|
---|
135 | selectedBar.ToolTipText = "x in [" + lower + " .. " + upper + "[ : " + selectedFreq;
|
---|
136 | bar.ToolTipText = "x in [" + lower + " .. " + upper + "[ : " + freq;
|
---|
137 | }
|
---|
138 | bars.Add(bar);
|
---|
139 | bars.Add(selectedBar);
|
---|
140 | UpdateViewSize(lower, freq);
|
---|
141 | UpdateViewSize(upper, freq);
|
---|
142 | }
|
---|
143 | }
|
---|
144 |
|
---|
145 | private void ZoomToViewSize() {
|
---|
146 | if(minX < maxX) {
|
---|
147 | // enlarge view by 5% on each side
|
---|
148 | double width = maxX - minX;
|
---|
149 | minX = minX - width * 0.05;
|
---|
150 | maxX = maxX + width * 0.05;
|
---|
151 | double minY = 0 - maxFrequency * 0.05;
|
---|
152 | double maxY = maxFrequency + maxFrequency * 0.05;
|
---|
153 | ZoomIn(minX, minY, maxX, maxY);
|
---|
154 | }
|
---|
155 | }
|
---|
156 |
|
---|
157 | private void UpdateViewSize(double x, double freq) {
|
---|
158 | if(x < minX) minX = x;
|
---|
159 | if(x > maxX) maxX = x;
|
---|
160 | if(freq > maxFrequency) maxFrequency = freq;
|
---|
161 | }
|
---|
162 |
|
---|
163 | private void ResetViewSize() {
|
---|
164 | minX = double.PositiveInfinity;
|
---|
165 | maxX = double.NegativeInfinity;
|
---|
166 | maxFrequency = double.NegativeInfinity;
|
---|
167 | }
|
---|
168 |
|
---|
169 | internal List<Record> GetRecords(Point point) {
|
---|
170 | List<Record> records = null;
|
---|
171 | IPrimitive p = bars.GetPrimitive(TransformPixelToWorld(point));
|
---|
172 | if(p != null) {
|
---|
173 | primitiveToRecordsDictionary.TryGetValue(p, out records);
|
---|
174 | }
|
---|
175 | return records;
|
---|
176 | }
|
---|
177 |
|
---|
178 | public override void MouseClick(Point point, MouseButtons button) {
|
---|
179 | if(button == MouseButtons.Left) {
|
---|
180 | lock(records) {
|
---|
181 | List<Record> rs = GetRecords(point);
|
---|
182 | UpdateEnabled = false;
|
---|
183 | if(rs != null) rs.ForEach(r => r.ToggleSelected());
|
---|
184 | UpdateEnabled = true;
|
---|
185 | }
|
---|
186 | results.FireChanged();
|
---|
187 | } else {
|
---|
188 | base.MouseClick(point, button);
|
---|
189 | }
|
---|
190 | }
|
---|
191 | }
|
---|
192 | }
|
---|