Free cookie consent management tool by TermsFeed Policy Generator

source: branches/histogram/HeuristicLab.Analysis.Views/3.3/DataTableView.cs @ 6032

Last change on this file since 6032 was 6032, checked in by abeham, 13 years ago

#1465

  • updated from trunk
  • fixed the bug regarding the -1 x-axis start
File size: 29.3 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.Generic;
24using System.Drawing;
25using System.Linq;
26using System.Windows.Forms;
27using System.Windows.Forms.DataVisualization.Charting;
28using HeuristicLab.Collections;
29using HeuristicLab.Core.Views;
30using HeuristicLab.MainForm;
31
32namespace HeuristicLab.Analysis.Views {
33  /// <summary>
34  /// The visual representation of a <see cref="Variable"/>.
35  /// </summary>
36  [View("DataTable View")]
37  [Content(typeof(DataTable), true)]
38  public partial class DataTableView : NamedItemView {
39    protected List<Series> invisibleSeries;
40    protected Dictionary<IObservableList<double>, DataRow> valuesRowsTable;
41    /// <summary>
42    /// Gets or sets the variable to represent visually.
43    /// </summary>
44    /// <remarks>Uses property <see cref="ViewBase.Item"/> of base class <see cref="ViewBase"/>.
45    /// No own data storage present.</remarks>
46    public new DataTable Content {
47      get { return (DataTable)base.Content; }
48      set { base.Content = value; }
49    }
50
51    /// <summary>
52    /// Initializes a new instance of <see cref="VariableView"/> with caption "Variable".
53    /// </summary>
54    public DataTableView() {
55      InitializeComponent();
56      valuesRowsTable = new Dictionary<IObservableList<double>, DataRow>();
57      invisibleSeries = new List<Series>();
58      chart.CustomizeAllChartAreas();
59      chart.ChartAreas[0].CursorX.Interval = 1;
60    }
61
62    #region Event Handler Registration
63    /// <summary>
64    /// Removes the eventhandlers from the underlying <see cref="Variable"/>.
65    /// </summary>
66    /// <remarks>Calls <see cref="ViewBase.RemoveItemEvents"/> of base class <see cref="ViewBase"/>.</remarks>
67    protected override void DeregisterContentEvents() {
68      foreach (DataRow row in Content.Rows)
69        DeregisterDataRowEvents(row);
70      Content.VisualPropertiesChanged -= new EventHandler(Content_VisualPropertiesChanged);
71      Content.Rows.ItemsAdded -= new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsAdded);
72      Content.Rows.ItemsRemoved -= new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsRemoved);
73      Content.Rows.ItemsReplaced -= new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsReplaced);
74      Content.Rows.CollectionReset -= new CollectionItemsChangedEventHandler<DataRow>(Rows_CollectionReset);
75      base.DeregisterContentEvents();
76    }
77
78    /// <summary>
79    /// Adds eventhandlers to the underlying <see cref="Variable"/>.
80    /// </summary>
81    /// <remarks>Calls <see cref="ViewBase.AddItemEvents"/> of base class <see cref="ViewBase"/>.</remarks>
82    protected override void RegisterContentEvents() {
83      base.RegisterContentEvents();
84      Content.VisualPropertiesChanged += new EventHandler(Content_VisualPropertiesChanged);
85      Content.Rows.ItemsAdded += new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsAdded);
86      Content.Rows.ItemsRemoved += new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsRemoved);
87      Content.Rows.ItemsReplaced += new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsReplaced);
88      Content.Rows.CollectionReset += new CollectionItemsChangedEventHandler<DataRow>(Rows_CollectionReset);
89      foreach (DataRow row in Content.Rows)
90        RegisterDataRowEvents(row);
91    }
92
93    /// <summary>
94    /// Automatically called for every existing data row and whenever a data row is added
95    /// to the data table. Do not call this method directly.
96    /// </summary>
97    /// <param name="row">The DataRow that was added.</param>
98    protected virtual void RegisterDataRowEvents(DataRow row) {
99      row.NameChanged += new EventHandler(Row_NameChanged);
100      row.VisualPropertiesChanged += new EventHandler(Row_VisualPropertiesChanged);
101      valuesRowsTable.Add(row.Values, row);
102      row.Values.ItemsAdded += new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsAdded);
103      row.Values.ItemsRemoved += new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsRemoved);
104      row.Values.ItemsReplaced += new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsReplaced);
105      row.Values.ItemsMoved += new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsMoved);
106      row.Values.CollectionReset += new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_CollectionReset);
107    }
108
109    /// <summary>
110    /// Automatically called for every data row that is removed from the DataTable. Do
111    /// not directly call this method.
112    /// </summary>
113    /// <param name="row">The DataRow that was removed.</param>
114    protected virtual void DeregisterDataRowEvents(DataRow row) {
115      row.Values.ItemsAdded -= new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsAdded);
116      row.Values.ItemsRemoved -= new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsRemoved);
117      row.Values.ItemsReplaced -= new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsReplaced);
118      row.Values.ItemsMoved -= new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsMoved);
119      row.Values.CollectionReset -= new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_CollectionReset);
120      valuesRowsTable.Remove(row.Values);
121      row.VisualPropertiesChanged -= new EventHandler(Row_VisualPropertiesChanged);
122      row.NameChanged -= new EventHandler(Row_NameChanged);
123    }
124    #endregion
125
126    protected override void OnContentChanged() {
127      base.OnContentChanged();
128      invisibleSeries.Clear();
129      chart.Titles[0].Text = string.Empty;
130      chart.ChartAreas[0].AxisX.Title = string.Empty;
131      chart.ChartAreas[0].AxisY.Title = string.Empty;
132      chart.ChartAreas[0].AxisY2.Title = string.Empty;
133      chart.Series.Clear();
134      if (Content != null) {
135        chart.Titles[0].Text = Content.Name;
136        foreach (DataRow row in Content.Rows)
137          AddDataRow(row);
138        ConfigureChartArea(chart.ChartAreas[0]);
139        RecalculateAxesScale(chart.ChartAreas[0]);
140      }
141    }
142
143    protected override void SetEnabledStateOfControls() {
144      base.SetEnabledStateOfControls();
145      chart.Enabled = Content != null;
146    }
147
148    /// <summary>
149    /// Add the DataRow as a series to the chart.
150    /// </summary>
151    /// <param name="row">DataRow to add as series to the chart.</param>
152    protected virtual void AddDataRow(DataRow row) {
153      Series series = new Series(row.Name);
154      ConfigureSeries(series, row);
155      FillSeriesWithRowValues(series, row);
156
157      chart.Series.Add(series);
158      ConfigureChartArea(chart.ChartAreas[0]);
159      RecalculateAxesScale(chart.ChartAreas[0]);
160      UpdateYCursorInterval();
161    }
162
163    private void ConfigureSeries(Series series, DataRow row) {
164      RemoveCustomPropertyIfExists(series, "PointWidth");
165      series.BorderWidth = 1;
166      series.BorderDashStyle = ChartDashStyle.Solid;
167      series.BorderColor = Color.Empty;
168
169      if (row.VisualProperties.Color != Color.Empty)
170        series.Color = row.VisualProperties.Color;
171      else series.Color = Color.Empty;
172
173      switch (row.VisualProperties.ChartType) {
174        case DataRowVisualProperties.DataRowChartType.Line:
175          series.ChartType = SeriesChartType.FastLine;
176          series.BorderWidth = row.VisualProperties.LineWidth;
177          series.BorderDashStyle = ConvertLineStyle(row.VisualProperties.LineStyle);
178          break;
179        case DataRowVisualProperties.DataRowChartType.Bars:
180          // Bar is incompatible with anything but Bar and StackedBar*
181          if (!chart.Series.Any(x => x.ChartType != SeriesChartType.Bar && x.ChartType != SeriesChartType.StackedBar && x.ChartType != SeriesChartType.StackedBar100))
182            series.ChartType = SeriesChartType.Bar;
183          else {
184            series.ChartType = SeriesChartType.FastPoint; //default
185            row.VisualProperties.ChartType = DataRowVisualProperties.DataRowChartType.Points;
186          }
187          break;
188        case DataRowVisualProperties.DataRowChartType.Columns:
189          series.ChartType = SeriesChartType.Column;
190          break;
191        case DataRowVisualProperties.DataRowChartType.Points:
192          series.ChartType = SeriesChartType.FastPoint;
193          break;
194        case DataRowVisualProperties.DataRowChartType.Histogram:
195          series.ChartType = SeriesChartType.Column;
196          series.SetCustomProperty("PointWidth", "1");
197          if (!series.Color.IsEmpty && series.Color.GetBrightness() < 0.25)
198            series.BorderColor = Color.White;
199          else series.BorderColor = Color.Black;
200          break;
201        default:
202          series.ChartType = SeriesChartType.FastPoint;
203          break;
204      }
205      series.YAxisType = row.VisualProperties.SecondYAxis ? AxisType.Secondary : AxisType.Primary;
206      series.XAxisType = row.VisualProperties.SecondXAxis ? AxisType.Secondary : AxisType.Primary;
207      series.ToolTip = row.Name + " X = #INDEX, Y = #VAL";
208    }
209
210    private void ConfigureChartArea(ChartArea area) {
211      if (Content.VisualProperties.TitleFont != null) chart.Titles[0].Font = Content.VisualProperties.TitleFont;
212      if (!Content.VisualProperties.TitleColor.IsEmpty) chart.Titles[0].ForeColor = Content.VisualProperties.TitleColor;
213
214      if (Content.VisualProperties.AxisTitleFont != null) area.AxisX.TitleFont = Content.VisualProperties.AxisTitleFont;
215      if (!Content.VisualProperties.AxisTitleColor.IsEmpty) area.AxisX.TitleForeColor = Content.VisualProperties.AxisTitleColor;
216      area.AxisX.Title = Content.VisualProperties.XAxisTitle;
217
218      if (Content.VisualProperties.AxisTitleFont != null) area.AxisX2.TitleFont = Content.VisualProperties.AxisTitleFont;
219      if (!Content.VisualProperties.AxisTitleColor.IsEmpty) area.AxisX2.TitleForeColor = Content.VisualProperties.AxisTitleColor;
220      area.AxisX2.Title = Content.VisualProperties.SecondXAxisTitle;
221
222      if (Content.VisualProperties.AxisTitleFont != null) area.AxisY.TitleFont = Content.VisualProperties.AxisTitleFont;
223      if (!Content.VisualProperties.AxisTitleColor.IsEmpty) area.AxisY.TitleForeColor = Content.VisualProperties.AxisTitleColor;
224      area.AxisY.Title = Content.VisualProperties.YAxisTitle;
225
226      if (Content.VisualProperties.AxisTitleFont != null) area.AxisY2.TitleFont = Content.VisualProperties.AxisTitleFont;
227      if (!Content.VisualProperties.AxisTitleColor.IsEmpty) area.AxisY2.TitleForeColor = Content.VisualProperties.AxisTitleColor;
228      area.AxisY2.Title = Content.VisualProperties.SecondYAxisTitle;
229    }
230
231    private void RecalculateAxesScale(ChartArea area) {
232      // Reset the axes bounds so that RecalculateAxesScale() will assign new bounds
233      foreach (Axis a in area.Axes) {
234        a.Minimum = double.NaN;
235        a.Maximum = double.NaN;
236      }
237      area.RecalculateAxesScale();
238
239      if (!Content.VisualProperties.XAxisMinimumAuto && !double.IsNaN(Content.VisualProperties.XAxisMinimumFixedValue)) area.AxisX.Minimum = Content.VisualProperties.XAxisMinimumFixedValue;
240      #region Workaround for bug in RecalculateAxesScale() that would assign -1
241      else {
242        if (area.AxisX.IsStartedFromZero
243        && chart.Series.Where(x => x.XAxisType == AxisType.Primary).Any()
244        && chart.Series.Where(x => x.XAxisType == AxisType.Primary).SelectMany(x => x.Points).Any()) {
245          double minX = chart.Series.Where(x => x.XAxisType == AxisType.Primary).SelectMany(x => x.Points).Select(x => x.XValue).Min();
246          if (minX >= 0 && area.AxisX.Minimum < 0) area.AxisX.Minimum = 0;
247        }
248      }
249      #endregion
250      if (!Content.VisualProperties.XAxisMaximumAuto && !double.IsNaN(Content.VisualProperties.XAxisMaximumFixedValue)) area.AxisX.Maximum = Content.VisualProperties.XAxisMaximumFixedValue;
251      if (!Content.VisualProperties.SecondXAxisMinimumAuto && !double.IsNaN(Content.VisualProperties.SecondXAxisMinimumFixedValue)) area.AxisX2.Minimum = Content.VisualProperties.SecondXAxisMinimumFixedValue;
252      #region Workaround for bug in RecalculateAxesScale() that would assign -1
253      else {
254        if (area.AxisX2.IsStartedFromZero
255         && chart.Series.Where(x => x.XAxisType == AxisType.Secondary).Any()
256         && chart.Series.Where(x => x.XAxisType == AxisType.Secondary).SelectMany(x => x.Points).Any()) {
257          double minX2 = chart.Series.Where(x => x.XAxisType == AxisType.Secondary).SelectMany(x => x.Points).Select(x => x.XValue).Min();
258          if (minX2 >= 0 && area.AxisX2.Minimum < 0) area.AxisX2.Minimum = 0;
259        }
260      }
261      #endregion
262      if (!Content.VisualProperties.SecondXAxisMaximumAuto && !double.IsNaN(Content.VisualProperties.SecondXAxisMaximumFixedValue)) area.AxisX2.Maximum = Content.VisualProperties.SecondXAxisMaximumFixedValue;
263      if (!Content.VisualProperties.YAxisMinimumAuto && !double.IsNaN(Content.VisualProperties.YAxisMinimumFixedValue)) area.AxisY.Minimum = Content.VisualProperties.YAxisMinimumFixedValue;
264      if (!Content.VisualProperties.YAxisMaximumAuto && !double.IsNaN(Content.VisualProperties.YAxisMaximumFixedValue)) area.AxisY.Maximum = Content.VisualProperties.YAxisMaximumFixedValue;
265      if (!Content.VisualProperties.SecondYAxisMinimumAuto && !double.IsNaN(Content.VisualProperties.SecondYAxisMinimumFixedValue)) area.AxisY2.Minimum = Content.VisualProperties.SecondYAxisMinimumFixedValue;
266      if (!Content.VisualProperties.SecondYAxisMaximumAuto && !double.IsNaN(Content.VisualProperties.SecondYAxisMaximumFixedValue)) area.AxisY2.Maximum = Content.VisualProperties.SecondYAxisMaximumFixedValue;
267      if (area.AxisX.Minimum > area.AxisX.Maximum) area.AxisX.Maximum = area.AxisX.Minimum + 1;
268      if (area.AxisX2.Minimum > area.AxisX2.Maximum) area.AxisX2.Maximum = area.AxisX2.Minimum + 1;
269      if (area.AxisY.Minimum > area.AxisY.Maximum) area.AxisY.Maximum = area.AxisY.Minimum + 1;
270      if (area.AxisY2.Minimum > area.AxisY2.Maximum) area.AxisY2.Maximum = area.AxisY2.Minimum + 1;
271    }
272
273    /// <summary>
274    /// Set the Y Cursor interval to visible points of enabled series.
275    /// </summary>
276    protected virtual void UpdateYCursorInterval() {
277      double interestingValuesRange = (
278        from series in chart.Series
279        where series.Enabled
280        let values = (from point in series.Points
281                      where !point.IsEmpty
282                      select point.YValues[0]).DefaultIfEmpty(1.0)
283        let range = values.Max() - values.Min()
284        where range > 0.0
285        select range
286        ).DefaultIfEmpty(1.0).Min();
287
288      double digits = (int)Math.Log10(interestingValuesRange) - 3;
289      double yZoomInterval = Math.Pow(10, digits);
290      this.chart.ChartAreas[0].CursorY.Interval = yZoomInterval;
291    }
292
293
294    /// <summary>
295    /// Remove the corresponding series for a certain DataRow.
296    /// </summary>
297    /// <param name="row">DataRow which series should be removed.</param>
298    protected virtual void RemoveDataRow(DataRow row) {
299      Series series = chart.Series[row.Name];
300      chart.Series.Remove(series);
301      if (invisibleSeries.Contains(series))
302        invisibleSeries.Remove(series);
303      RecalculateAxesScale(chart.ChartAreas[0]);
304    }
305
306    #region Event Handlers
307    #region Content Event Handlers
308    protected override void Content_NameChanged(object sender, EventArgs e) {
309      if (InvokeRequired)
310        Invoke(new EventHandler(Content_NameChanged), sender, e);
311      else {
312        chart.Titles[0].Text = Content.Name;
313        base.Content_NameChanged(sender, e);
314      }
315    }
316    private void Content_VisualPropertiesChanged(object sender, EventArgs e) {
317      if (InvokeRequired)
318        Invoke(new EventHandler(Content_VisualPropertiesChanged), sender, e);
319      else {
320        ConfigureChartArea(chart.ChartAreas[0]);
321        RecalculateAxesScale(chart.ChartAreas[0]); // axes min/max could have changed
322      }
323    }
324    #endregion
325    #region Rows Event Handlers
326    private void Rows_ItemsAdded(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
327      if (InvokeRequired)
328        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsAdded), sender, e);
329      else {
330        foreach (DataRow row in e.Items) {
331          AddDataRow(row);
332          RegisterDataRowEvents(row);
333        }
334      }
335    }
336    private void Rows_ItemsRemoved(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
337      if (InvokeRequired)
338        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsRemoved), sender, e);
339      else {
340        foreach (DataRow row in e.Items) {
341          DeregisterDataRowEvents(row);
342          RemoveDataRow(row);
343        }
344      }
345    }
346    private void Rows_ItemsReplaced(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
347      if (InvokeRequired)
348        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsReplaced), sender, e);
349      else {
350        foreach (DataRow row in e.OldItems) {
351          DeregisterDataRowEvents(row);
352          RemoveDataRow(row);
353        }
354        foreach (DataRow row in e.Items) {
355          AddDataRow(row);
356          RegisterDataRowEvents(row);
357        }
358      }
359    }
360    private void Rows_CollectionReset(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
361      if (InvokeRequired)
362        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_CollectionReset), sender, e);
363      else {
364        foreach (DataRow row in e.OldItems) {
365          DeregisterDataRowEvents(row);
366          RemoveDataRow(row);
367        }
368        foreach (DataRow row in e.Items) {
369          AddDataRow(row);
370          RegisterDataRowEvents(row);
371        }
372      }
373    }
374    #endregion
375    #region Row Event Handlers
376    private void Row_VisualPropertiesChanged(object sender, EventArgs e) {
377      if (InvokeRequired)
378        Invoke(new EventHandler(Row_VisualPropertiesChanged), sender, e);
379      else {
380        DataRow row = (DataRow)sender;
381        Series series = chart.Series[row.Name];
382        series.Points.Clear();
383        ConfigureSeries(series, row);
384        FillSeriesWithRowValues(series, row);
385        RecalculateAxesScale(chart.ChartAreas[0]);
386      }
387    }
388    private void Row_NameChanged(object sender, EventArgs e) {
389      if (InvokeRequired)
390        Invoke(new EventHandler(Row_NameChanged), sender, e);
391      else {
392        DataRow row = (DataRow)sender;
393        chart.Series[row.Name].Name = row.Name;
394      }
395    }
396    #endregion
397    #region Values Event Handlers
398    private void Values_ItemsAdded(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
399      if (InvokeRequired)
400        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsAdded), sender, e);
401      else {
402        DataRow row = null;
403        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
404        if (row != null) {
405          Series rowSeries = chart.Series[row.Name];
406          if (!invisibleSeries.Contains(rowSeries)) {
407            rowSeries.Points.Clear();
408            FillSeriesWithRowValues(rowSeries, row);
409            RecalculateAxesScale(chart.ChartAreas[0]);
410            UpdateYCursorInterval();
411          }
412        }
413      }
414    }
415    private void Values_ItemsRemoved(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
416      if (InvokeRequired)
417        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsRemoved), sender, e);
418      else {
419        DataRow row = null;
420        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
421        if (row != null) {
422          Series rowSeries = chart.Series[row.Name];
423          if (!invisibleSeries.Contains(rowSeries)) {
424            rowSeries.Points.Clear();
425            FillSeriesWithRowValues(rowSeries, row);
426            RecalculateAxesScale(chart.ChartAreas[0]);
427            UpdateYCursorInterval();
428          }
429        }
430      }
431    }
432    private void Values_ItemsReplaced(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
433      if (InvokeRequired)
434        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsReplaced), sender, e);
435      else {
436        DataRow row = null;
437        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
438        if (row != null) {
439          Series rowSeries = chart.Series[row.Name];
440          if (!invisibleSeries.Contains(rowSeries)) {
441            if (row.VisualProperties.ChartType == DataRowVisualProperties.DataRowChartType.Histogram) {
442              rowSeries.Points.Clear();
443              FillSeriesWithRowValues(rowSeries, row);
444            } else {
445              foreach (IndexedItem<double> item in e.Items) {
446                if (IsInvalidValue(item.Value))
447                  rowSeries.Points[item.Index].IsEmpty = true;
448                else {
449                  rowSeries.Points[item.Index].YValues = new double[] { item.Value };
450                  rowSeries.Points[item.Index].IsEmpty = false;
451                }
452              }
453            }
454            RecalculateAxesScale(chart.ChartAreas[0]);
455            UpdateYCursorInterval();
456          }
457        }
458      }
459    }
460    private void Values_ItemsMoved(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
461      if (InvokeRequired)
462        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsMoved), sender, e);
463      else {
464        DataRow row = null;
465        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
466        if (row != null) {
467          Series rowSeries = chart.Series[row.Name];
468          if (!invisibleSeries.Contains(rowSeries)) {
469            rowSeries.Points.Clear();
470            FillSeriesWithRowValues(rowSeries, row);
471            RecalculateAxesScale(chart.ChartAreas[0]);
472            UpdateYCursorInterval();
473          }
474        }
475      }
476    }
477
478    private void Values_CollectionReset(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
479      if (InvokeRequired)
480        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_CollectionReset), sender, e);
481      else {
482        DataRow row = null;
483        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
484        if (row != null) {
485          Series rowSeries = chart.Series[row.Name];
486          if (!invisibleSeries.Contains(rowSeries)) {
487            rowSeries.Points.Clear();
488            FillSeriesWithRowValues(rowSeries, row);
489            RecalculateAxesScale(chart.ChartAreas[0]);
490            UpdateYCursorInterval();
491          }
492        }
493      }
494    }
495    #endregion
496    #endregion
497
498    #region Chart Event Handlers
499    private void chart_MouseDown(object sender, MouseEventArgs e) {
500      HitTestResult result = chart.HitTest(e.X, e.Y);
501      if (result.ChartElementType == ChartElementType.LegendItem) {
502        ToggleSeriesVisible(result.Series);
503      }
504    }
505    private void chart_MouseMove(object sender, MouseEventArgs e) {
506      HitTestResult result = chart.HitTest(e.X, e.Y);
507      if (result.ChartElementType == ChartElementType.LegendItem)
508        this.Cursor = Cursors.Hand;
509      else
510        this.Cursor = Cursors.Default;
511    }
512    private void chart_CustomizeLegend(object sender, CustomizeLegendEventArgs e) {
513      foreach (LegendItem legendItem in e.LegendItems) {
514        var series = chart.Series[legendItem.SeriesName];
515        if (series != null) {
516          bool seriesIsInvisible = invisibleSeries.Contains(series);
517          foreach (LegendCell cell in legendItem.Cells) {
518            cell.ForeColor = seriesIsInvisible ? Color.Gray : Color.Black;
519          }
520        }
521      }
522    }
523    private void chart_PropertiesClicked(object sender, EventArgs e) {
524      DataTableVisualPropertiesDialog dialog = new DataTableVisualPropertiesDialog(Content);
525      dialog.ShowDialog();
526    }
527    #endregion
528
529    private void ToggleSeriesVisible(Series series) {
530      if (!invisibleSeries.Contains(series)) {
531        series.Points.Clear();
532        invisibleSeries.Add(series);
533      } else {
534        invisibleSeries.Remove(series);
535        if (Content != null) {
536
537          var row = (from r in Content.Rows
538                     where r.Name == series.Name
539                     select r).Single();
540          FillSeriesWithRowValues(series, row);
541          this.chart.Legends[series.Legend].ForeColor = Color.Black;
542          RecalculateAxesScale(chart.ChartAreas[0]);
543          UpdateYCursorInterval();
544        }
545      }
546    }
547
548    private void FillSeriesWithRowValues(Series series, DataRow row) {
549      switch (row.VisualProperties.ChartType) {
550        case DataRowVisualProperties.DataRowChartType.Histogram:
551          CalculateHistogram(series, row);
552          break;
553        default: {
554            for (int i = 0; i < row.Values.Count; i++) {
555              var value = row.Values[i];
556              DataPoint point = new DataPoint();
557              point.XValue = row.VisualProperties.StartIndexZero ? i : i + 1;
558              if (IsInvalidValue(value))
559                point.IsEmpty = true;
560              else
561                point.YValues = new double[] { value };
562              series.Points.Add(point);
563            }
564          }
565          break;
566      }
567    }
568
569    protected virtual void CalculateHistogram(Series series, DataRow row) {
570      series.Points.Clear();
571      if (!row.Values.Any()) return;
572      int bins = row.VisualProperties.Bins;
573
574      double minValue = row.Values.Min();
575      double maxValue = row.Values.Max();
576      double intervalWidth = (maxValue - minValue) / bins;
577      if (intervalWidth <= 0) return;
578
579      if (!row.VisualProperties.ExactBins) {
580        intervalWidth = HumanRoundRange(intervalWidth);
581        minValue = Math.Floor(minValue / intervalWidth) * intervalWidth;
582        maxValue = Math.Ceiling(maxValue / intervalWidth) * intervalWidth;
583      }
584
585      double current = minValue, intervalCenter = intervalWidth / 2.0;
586      int frequency = 0;
587      foreach (double v in row.Values.Where(x => !IsInvalidValue(x)).OrderBy(x => x)) {
588        while (v > current + intervalWidth) {
589          series.Points.AddXY(current + intervalCenter, frequency);
590          current += intervalWidth;
591          frequency = 0;
592        }
593        frequency++;
594      }
595      series.Points.AddXY(current + intervalCenter, frequency);
596    }
597
598    #region Helpers
599    protected void RemoveCustomPropertyIfExists(Series series, string property) {
600      if (series.IsCustomPropertySet(property)) series.DeleteCustomProperty(property);
601    }
602
603    private double HumanRoundRange(double range) {
604      double base10 = Math.Pow(10.0, Math.Floor(Math.Log10(range)));
605      double rounding = range / base10;
606      if (rounding <= 1.5) rounding = 1;
607      else if (rounding <= 2.25) rounding = 2;
608      else if (rounding <= 3.75) rounding = 2.5;
609      else if (rounding <= 7.5) rounding = 5;
610      else rounding = 10;
611      return rounding * base10;
612    }
613
614    private double HumanRoundMax(double max) {
615      double base10;
616      if (max > 0) base10 = Math.Pow(10.0, Math.Floor(Math.Log10(max)));
617      else base10 = Math.Pow(10.0, Math.Ceiling(Math.Log10(-max)));
618      double rounding = (max > 0) ? base10 : -base10;
619      while (rounding < max) rounding += base10;
620      return rounding;
621    }
622
623    private ChartDashStyle ConvertLineStyle(DataRowVisualProperties.DataRowLineStyle dataRowLineStyle) {
624      switch (dataRowLineStyle) {
625        case DataRowVisualProperties.DataRowLineStyle.Dash:
626          return ChartDashStyle.Dash;
627        case DataRowVisualProperties.DataRowLineStyle.DashDot:
628          return ChartDashStyle.DashDot;
629        case DataRowVisualProperties.DataRowLineStyle.DashDotDot:
630          return ChartDashStyle.DashDotDot;
631        case DataRowVisualProperties.DataRowLineStyle.Dot:
632          return ChartDashStyle.Dot;
633        case DataRowVisualProperties.DataRowLineStyle.NotSet:
634          return ChartDashStyle.NotSet;
635        case DataRowVisualProperties.DataRowLineStyle.Solid:
636          return ChartDashStyle.Solid;
637        default:
638          return ChartDashStyle.NotSet;
639      }
640    }
641
642    /// <summary>
643    /// Determines whether a double value can be displayed (converted to Decimal and not an NaN).
644    /// </summary>
645    /// <param name="x">The number to check.</param>
646    /// <returns><code>true</code> if the value can be safely shwon in the chart,
647    /// <code>false</code> otherwise.</returns>
648    protected static bool IsInvalidValue(double x) {
649      return double.IsNaN(x) || x < (double)decimal.MinValue || x > (double)decimal.MaxValue;
650    }
651    #endregion
652  }
653}
Note: See TracBrowser for help on using the repository browser.