Free cookie consent management tool by TermsFeed Policy Generator

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

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

#1465

  • Fixed some bugs
  • Fixed out-of-sync issue in DataRowVisualPropertiesControl
  • Added border in histogram (white for very dark colors)
  • Added serializer for System.Drawing.Font
  • Added option to choose the font of the title as well as the axis of the chart
File size: 25.8 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      }
140    }
141
142    protected override void SetEnabledStateOfControls() {
143      base.SetEnabledStateOfControls();
144      chart.Enabled = Content != null;
145    }
146
147    /// <summary>
148    /// Add the DataRow as a series to the chart.
149    /// </summary>
150    /// <param name="row">DataRow to add as series to the chart.</param>
151    protected virtual void AddDataRow(DataRow row) {
152      Series series = new Series(row.Name);
153      ConfigureSeries(series, row);
154      FillSeriesWithRowValues(series, row);
155
156      chart.Series.Add(series);
157      chart.ChartAreas[0].RecalculateAxesScale();
158      ConfigureChartArea(chart.ChartAreas[0]);
159      UpdateYCursorInterval();
160    }
161
162    private void ConfigureSeries(Series series, DataRow row) {
163      RemoveCustomPropertyIfExists(series, "PointWidth");
164      series.BorderWidth = 1;
165      series.BorderDashStyle = ChartDashStyle.Solid;
166      series.BorderColor = Color.Empty;
167
168      if (row.VisualProperties.Color != Color.Empty)
169        series.Color = row.VisualProperties.Color;
170      else series.Color = Color.Empty;
171
172      switch (row.VisualProperties.ChartType) {
173        case DataRowVisualProperties.DataRowChartType.Line:
174          series.ChartType = SeriesChartType.FastLine;
175          series.BorderWidth = row.VisualProperties.LineWidth;
176          series.BorderDashStyle = ConvertLineStyle(row.VisualProperties.LineStyle);
177          break;
178        case DataRowVisualProperties.DataRowChartType.Bars:
179          // Bar is incompatible with anything but Bar and StackedBar*
180          if (!chart.Series.Any(x => x.ChartType != SeriesChartType.Bar && x.ChartType != SeriesChartType.StackedBar && x.ChartType != SeriesChartType.StackedBar100))
181            series.ChartType = SeriesChartType.Bar;
182          else {
183            series.ChartType = SeriesChartType.FastPoint; //default
184            row.VisualProperties.ChartType = DataRowVisualProperties.DataRowChartType.Points;
185          }
186          break;
187        case DataRowVisualProperties.DataRowChartType.Columns:
188          series.ChartType = SeriesChartType.Column;
189          break;
190        case DataRowVisualProperties.DataRowChartType.Points:
191          series.ChartType = SeriesChartType.FastPoint;
192          break;
193        case DataRowVisualProperties.DataRowChartType.Histogram:
194          series.ChartType = SeriesChartType.Column;
195          series.SetCustomProperty("PointWidth", "1");
196          if (!series.Color.IsEmpty && series.Color.GetBrightness() < 0.25)
197            series.BorderColor = Color.White;
198          else series.BorderColor = Color.Black;
199          break;
200        default:
201          series.ChartType = SeriesChartType.FastPoint;
202          break;
203      }
204      series.YAxisType = row.VisualProperties.SecondYAxis ? AxisType.Secondary : AxisType.Primary;
205      series.XAxisType = row.VisualProperties.SecondXAxis ? AxisType.Secondary : AxisType.Primary;
206      series.ToolTip = row.Name + " X = #INDEX, Y = #VAL";
207    }
208
209    private void ConfigureChartArea(ChartArea area) {
210      if (Content.VisualProperties.TitleFont != null)
211        chart.Titles[0].Font = Content.VisualProperties.TitleFont;
212      if (!Content.VisualProperties.TitleColor.IsEmpty)
213        chart.Titles[0].ForeColor = Content.VisualProperties.TitleColor;
214
215      if (Content.VisualProperties.AxisTitleFont != null)
216        area.AxisX.TitleFont = Content.VisualProperties.AxisTitleFont;
217      if (!Content.VisualProperties.AxisTitleColor.IsEmpty)
218        area.AxisX.TitleForeColor = Content.VisualProperties.AxisTitleColor;
219      area.AxisX.Title = Content.VisualProperties.XAxisTitle;
220
221      if (Content.VisualProperties.AxisTitleFont != null)
222        area.AxisX2.TitleFont = Content.VisualProperties.AxisTitleFont;
223      if (!Content.VisualProperties.AxisTitleColor.IsEmpty)
224        area.AxisX2.TitleForeColor = Content.VisualProperties.AxisTitleColor;
225      area.AxisX2.Title = Content.VisualProperties.SecondXAxisTitle;
226
227      if (Content.VisualProperties.AxisTitleFont != null)
228        area.AxisY.TitleFont = Content.VisualProperties.AxisTitleFont;
229      if (!Content.VisualProperties.AxisTitleColor.IsEmpty)
230        area.AxisY.TitleForeColor = Content.VisualProperties.AxisTitleColor;
231      area.AxisY.Title = Content.VisualProperties.YAxisTitle;
232
233      if (Content.VisualProperties.AxisTitleFont != null)
234        area.AxisY2.TitleFont = Content.VisualProperties.AxisTitleFont;
235      if (!Content.VisualProperties.AxisTitleColor.IsEmpty)
236        area.AxisY2.TitleForeColor = Content.VisualProperties.AxisTitleColor;
237      area.AxisY2.Title = Content.VisualProperties.SecondYAxisTitle;
238
239      area.RecalculateAxesScale();
240      area.AxisX.Minimum = Content.VisualProperties.XAxisMinimumFixedValue;
241      area.AxisX.Maximum = Content.VisualProperties.XAxisMaximumFixedValue;
242      area.AxisX2.Minimum = Content.VisualProperties.SecondXAxisMinimumFixedValue;
243      area.AxisX2.Maximum = Content.VisualProperties.SecondXAxisMaximumFixedValue;
244      area.AxisY.Minimum = Content.VisualProperties.YAxisMinimumFixedValue;
245      area.AxisY.Maximum = Content.VisualProperties.YAxisMaximumFixedValue;
246      area.AxisY2.Minimum = Content.VisualProperties.SecondYAxisMinimumFixedValue;
247      area.AxisY2.Maximum = Content.VisualProperties.SecondYAxisMaximumFixedValue;
248    }
249
250    /// <summary>
251    /// Set the Y Cursor interval to visible points of enabled series.
252    /// </summary>
253    protected virtual void UpdateYCursorInterval() {
254      double interestingValuesRange = (
255        from series in chart.Series
256        where series.Enabled
257        let values = (from point in series.Points
258                      where !point.IsEmpty
259                      select point.YValues[0]).DefaultIfEmpty(1.0)
260        let range = values.Max() - values.Min()
261        where range > 0.0
262        select range
263        ).DefaultIfEmpty(1.0).Min();
264
265      double digits = (int)Math.Log10(interestingValuesRange) - 3;
266      double yZoomInterval = Math.Pow(10, digits);
267      this.chart.ChartAreas[0].CursorY.Interval = yZoomInterval;
268    }
269
270
271    /// <summary>
272    /// Remove the corresponding series for a certain DataRow.
273    /// </summary>
274    /// <param name="row">DataRow which series should be removed.</param>
275    protected virtual void RemoveDataRow(DataRow row) {
276      Series series = chart.Series[row.Name];
277      chart.Series.Remove(series);
278      if (invisibleSeries.Contains(series))
279        invisibleSeries.Remove(series);
280      chart.ChartAreas[0].RecalculateAxesScale();
281    }
282
283    #region Event Handlers
284    #region Content Event Handlers
285    protected override void Content_NameChanged(object sender, EventArgs e) {
286      if (InvokeRequired)
287        Invoke(new EventHandler(Content_NameChanged), sender, e);
288      else {
289        chart.Titles[0].Text = Content.Name;
290        base.Content_NameChanged(sender, e);
291      }
292    }
293    private void Content_VisualPropertiesChanged(object sender, EventArgs e) {
294      if (InvokeRequired)
295        Invoke(new EventHandler(Content_VisualPropertiesChanged), sender, e);
296      else {
297        ConfigureChartArea(chart.ChartAreas[0]);
298      }
299    }
300    #endregion
301    #region Rows Event Handlers
302    private void Rows_ItemsAdded(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
303      if (InvokeRequired)
304        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsAdded), sender, e);
305      else {
306        foreach (DataRow row in e.Items) {
307          AddDataRow(row);
308          RegisterDataRowEvents(row);
309        }
310      }
311    }
312    private void Rows_ItemsRemoved(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
313      if (InvokeRequired)
314        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsRemoved), sender, e);
315      else {
316        foreach (DataRow row in e.Items) {
317          DeregisterDataRowEvents(row);
318          RemoveDataRow(row);
319        }
320      }
321    }
322    private void Rows_ItemsReplaced(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
323      if (InvokeRequired)
324        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_ItemsReplaced), sender, e);
325      else {
326        foreach (DataRow row in e.OldItems) {
327          DeregisterDataRowEvents(row);
328          RemoveDataRow(row);
329        }
330        foreach (DataRow row in e.Items) {
331          AddDataRow(row);
332          RegisterDataRowEvents(row);
333        }
334      }
335    }
336    private void Rows_CollectionReset(object sender, CollectionItemsChangedEventArgs<DataRow> e) {
337      if (InvokeRequired)
338        Invoke(new CollectionItemsChangedEventHandler<DataRow>(Rows_CollectionReset), sender, e);
339      else {
340        foreach (DataRow row in e.OldItems) {
341          DeregisterDataRowEvents(row);
342          RemoveDataRow(row);
343        }
344        foreach (DataRow row in e.Items) {
345          AddDataRow(row);
346          RegisterDataRowEvents(row);
347        }
348      }
349    }
350    #endregion
351    #region Row Event Handlers
352    private void Row_VisualPropertiesChanged(object sender, EventArgs e) {
353      if (InvokeRequired)
354        Invoke(new EventHandler(Row_VisualPropertiesChanged), sender, e);
355      else {
356        DataRow row = (DataRow)sender;
357        Series series = chart.Series[row.Name];
358        series.Points.Clear();
359        ConfigureSeries(series, row);
360        FillSeriesWithRowValues(series, row);
361        chart.ChartAreas[0].RecalculateAxesScale();
362      }
363    }
364    private void Row_NameChanged(object sender, EventArgs e) {
365      if (InvokeRequired)
366        Invoke(new EventHandler(Row_NameChanged), sender, e);
367      else {
368        DataRow row = (DataRow)sender;
369        chart.Series[row.Name].Name = row.Name;
370      }
371    }
372    #endregion
373    #region Values Event Handlers
374    private void Values_ItemsAdded(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
375      if (InvokeRequired)
376        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsAdded), sender, e);
377      else {
378        DataRow row = null;
379        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
380        if (row != null) {
381          Series rowSeries = chart.Series[row.Name];
382          if (!invisibleSeries.Contains(rowSeries)) {
383            rowSeries.Points.Clear();
384            FillSeriesWithRowValues(rowSeries, row);
385            UpdateYCursorInterval();
386          }
387        }
388      }
389    }
390    private void Values_ItemsRemoved(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
391      if (InvokeRequired)
392        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsRemoved), sender, e);
393      else {
394        DataRow row = null;
395        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
396        if (row != null) {
397          Series rowSeries = chart.Series[row.Name];
398          if (!invisibleSeries.Contains(rowSeries)) {
399            rowSeries.Points.Clear();
400            FillSeriesWithRowValues(rowSeries, row);
401            UpdateYCursorInterval();
402          }
403        }
404      }
405    }
406    private void Values_ItemsReplaced(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
407      if (InvokeRequired)
408        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsReplaced), sender, e);
409      else {
410        DataRow row = null;
411        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
412        if (row != null) {
413          Series rowSeries = chart.Series[row.Name];
414          if (!invisibleSeries.Contains(rowSeries)) {
415            if (row.VisualProperties.ChartType == DataRowVisualProperties.DataRowChartType.Histogram) {
416              rowSeries.Points.Clear();
417              FillSeriesWithRowValues(rowSeries, row);
418            } else {
419              foreach (IndexedItem<double> item in e.Items) {
420                if (IsInvalidValue(item.Value))
421                  rowSeries.Points[item.Index].IsEmpty = true;
422                else {
423                  rowSeries.Points[item.Index].YValues = new double[] { item.Value };
424                  rowSeries.Points[item.Index].IsEmpty = false;
425                }
426              }
427            }
428            UpdateYCursorInterval();
429          }
430        }
431      }
432    }
433    private void Values_ItemsMoved(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
434      if (InvokeRequired)
435        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_ItemsMoved), sender, e);
436      else {
437        DataRow row = null;
438        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
439        if (row != null) {
440          Series rowSeries = chart.Series[row.Name];
441          if (!invisibleSeries.Contains(rowSeries)) {
442            rowSeries.Points.Clear();
443            FillSeriesWithRowValues(rowSeries, row);
444            UpdateYCursorInterval();
445          }
446        }
447      }
448    }
449
450    private void Values_CollectionReset(object sender, CollectionItemsChangedEventArgs<IndexedItem<double>> e) {
451      if (InvokeRequired)
452        Invoke(new CollectionItemsChangedEventHandler<IndexedItem<double>>(Values_CollectionReset), sender, e);
453      else {
454        DataRow row = null;
455        valuesRowsTable.TryGetValue((IObservableList<double>)sender, out row);
456        if (row != null) {
457          Series rowSeries = chart.Series[row.Name];
458          if (!invisibleSeries.Contains(rowSeries)) {
459            rowSeries.Points.Clear();
460            FillSeriesWithRowValues(rowSeries, row);
461            UpdateYCursorInterval();
462          }
463        }
464      }
465    }
466    #endregion
467    #endregion
468
469    #region Chart Event Handlers
470    private void chart_MouseDown(object sender, MouseEventArgs e) {
471      HitTestResult result = chart.HitTest(e.X, e.Y);
472      if (result.ChartElementType == ChartElementType.LegendItem) {
473        ToggleSeriesVisible(result.Series);
474      }
475    }
476    private void chart_MouseMove(object sender, MouseEventArgs e) {
477      HitTestResult result = chart.HitTest(e.X, e.Y);
478      if (result.ChartElementType == ChartElementType.LegendItem)
479        this.Cursor = Cursors.Hand;
480      else
481        this.Cursor = Cursors.Default;
482    }
483    private void chart_CustomizeLegend(object sender, CustomizeLegendEventArgs e) {
484      foreach (LegendItem legendItem in e.LegendItems) {
485        var series = chart.Series[legendItem.SeriesName];
486        if (series != null) {
487          bool seriesIsInvisible = invisibleSeries.Contains(series);
488          foreach (LegendCell cell in legendItem.Cells) {
489            cell.ForeColor = seriesIsInvisible ? Color.Gray : Color.Black;
490          }
491        }
492      }
493    }
494    private void chart_PropertiesClicked(object sender, EventArgs e) {
495      DataTableVisualPropertiesDialog dialog = new DataTableVisualPropertiesDialog(Content);
496      dialog.ShowDialog();
497    }
498    #endregion
499
500    private void ToggleSeriesVisible(Series series) {
501      if (!invisibleSeries.Contains(series)) {
502        series.Points.Clear();
503        invisibleSeries.Add(series);
504      } else {
505        invisibleSeries.Remove(series);
506        if (Content != null) {
507
508          var row = (from r in Content.Rows
509                     where r.Name == series.Name
510                     select r).Single();
511          FillSeriesWithRowValues(series, row);
512          this.chart.Legends[series.Legend].ForeColor = Color.Black;
513          UpdateYCursorInterval();
514        }
515      }
516    }
517
518    private void FillSeriesWithRowValues(Series series, DataRow row) {
519      switch (row.VisualProperties.ChartType) {
520        case DataRowVisualProperties.DataRowChartType.Histogram:
521          CalculateHistogram(series, row);
522          break;
523        default: {
524            for (int i = 0; i < row.Values.Count; i++) {
525              var value = row.Values[i];
526              DataPoint point = new DataPoint();
527              point.XValue = row.VisualProperties.StartIndexZero ? i : i + 1;
528              if (IsInvalidValue(value))
529                point.IsEmpty = true;
530              else
531                point.YValues = new double[] { value };
532              series.Points.Add(point);
533            }
534          }
535          break;
536      }
537    }
538
539    protected virtual void CalculateHistogram(Series series, DataRow row) {
540      series.Points.Clear();
541      if (!row.Values.Any()) return;
542      int bins = row.VisualProperties.Bins;
543
544      double minValue = row.Values.Min();
545      double maxValue = row.Values.Max();
546      double intervalWidth = (maxValue - minValue) / bins;
547      if (intervalWidth <= 0) return;
548
549      if (!row.VisualProperties.ExactBins) {
550        intervalWidth = HumanRoundRange(intervalWidth);
551        minValue = Math.Floor(minValue / intervalWidth) * intervalWidth;
552        maxValue = Math.Ceiling(maxValue / intervalWidth) * intervalWidth;
553      }
554
555      double current = minValue, intervalCenter = intervalWidth / 2.0;
556      int frequency = 0;
557      foreach (double v in row.Values.Where(x => !IsInvalidValue(x)).OrderBy(x => x)) {
558        while (v > current + intervalWidth) {
559          series.Points.AddXY(current + intervalCenter, frequency);
560          current += intervalWidth;
561          frequency = 0;
562        }
563        frequency++;
564      }
565      series.Points.AddXY(current + intervalCenter, frequency);
566    }
567
568    #region Helpers
569    protected void RemoveCustomPropertyIfExists(Series series, string property) {
570      if (series.IsCustomPropertySet(property)) series.DeleteCustomProperty(property);
571    }
572
573    private double HumanRoundRange(double range) {
574      double base10 = Math.Pow(10.0, Math.Floor(Math.Log10(range)));
575      double rounding = range / base10;
576      if (rounding <= 1.5) rounding = 1;
577      else if (rounding <= 2.25) rounding = 2;
578      else if (rounding <= 3.75) rounding = 2.5;
579      else if (rounding <= 7.5) rounding = 5;
580      else rounding = 10;
581      return rounding * base10;
582    }
583
584    private ChartDashStyle ConvertLineStyle(DataRowVisualProperties.DataRowLineStyle dataRowLineStyle) {
585      switch (dataRowLineStyle) {
586        case DataRowVisualProperties.DataRowLineStyle.Dash:
587          return ChartDashStyle.Dash;
588        case DataRowVisualProperties.DataRowLineStyle.DashDot:
589          return ChartDashStyle.DashDot;
590        case DataRowVisualProperties.DataRowLineStyle.DashDotDot:
591          return ChartDashStyle.DashDotDot;
592        case DataRowVisualProperties.DataRowLineStyle.Dot:
593          return ChartDashStyle.Dot;
594        case DataRowVisualProperties.DataRowLineStyle.NotSet:
595          return ChartDashStyle.NotSet;
596        case DataRowVisualProperties.DataRowLineStyle.Solid:
597          return ChartDashStyle.Solid;
598        default:
599          return ChartDashStyle.NotSet;
600      }
601    }
602
603    /// <summary>
604    /// Determines whether a double value can be displayed (converted to Decimal and not an NaN).
605    /// </summary>
606    /// <param name="x">The number to check.</param>
607    /// <returns><code>true</code> if the value can be safely shwon in the chart,
608    /// <code>false</code> otherwise.</returns>
609    protected static bool IsInvalidValue(double x) {
610      return double.IsNaN(x) || x < (double)decimal.MinValue || x > (double)decimal.MaxValue;
611    }
612    #endregion
613  }
614}
Note: See TracBrowser for help on using the repository browser.