Free cookie consent management tool by TermsFeed Policy Generator

source: trunk/sources/HeuristicLab.Problems.DataAnalysis.Views/3.4/Regression/RegressionSolutionResidualHistogram.cs @ 10878

Last change on this file since 10878 was 9456, checked in by swagner, 12 years ago

Updated copyright year and added some missing license headers (#1889)

File size: 10.0 KB
RevLine 
[7186]1#region License Information
2/* HeuristicLab
[9456]3 * Copyright (C) 2002-2013 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
[7186]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;
[7255]24using System.Drawing;
[7186]25using System.Linq;
[7255]26using System.Windows.Forms;
[7186]27using System.Windows.Forms.DataVisualization.Charting;
28using HeuristicLab.MainForm;
[7255]29using HeuristicLab.MainForm.WindowsForms;
[7186]30
[7255]31namespace HeuristicLab.Problems.DataAnalysis.Views {
[7186]32  [View("Residual Histogram")]
33  [Content(typeof(IRegressionSolution))]
34  public partial class RegressionSolutionResidualHistogram : DataAnalysisSolutionEvaluationView {
[7485]35
36    #region variables
[7255]37    protected const string ALL_SAMPLES = "All samples";
38    protected const string TRAINING_SAMPLES = "Training samples";
39    protected const string TEST_SAMPLES = "Test samples";
[7485]40    /// <summary>
41    /// approximate amount of bins
42    /// </summary>
[7255]43    protected const double bins = 25;
[7485]44    #endregion
[7255]45
[7186]46    public new IRegressionSolution Content {
47      get { return (IRegressionSolution)base.Content; }
48      set { base.Content = value; }
49    }
50
[7255]51    public RegressionSolutionResidualHistogram()
52      : base() {
[7186]53      InitializeComponent();
[8104]54      foreach (string series in new List<String>() { ALL_SAMPLES, TRAINING_SAMPLES, TEST_SAMPLES }) {
[7485]55        chart.Series.Add(series);
56        chart.Series[series].LegendText = series;
57        chart.Series[series].ChartType = SeriesChartType.Column;
58        chart.Series[series]["PointWidth"] = "0.9";
59        chart.Series[series].BorderWidth = 1;
60        chart.Series[series].BorderDashStyle = ChartDashStyle.Solid;
61        chart.Series[series].BorderColor = Color.Black;
62        chart.Series[series].ToolTip = series + " Y = #VALY from #CUSTOMPROPERTY(from) to #CUSTOMPROPERTY(to)";
[7255]63      }
[7186]64      //configure axis
[7485]65      chart.CustomizeAllChartAreas();
66      chart.ChartAreas[0].AxisX.Title = "Residuals";
67      chart.ChartAreas[0].CursorX.IsUserSelectionEnabled = true;
68      chart.ChartAreas[0].AxisX.ScaleView.Zoomable = true;
69      chart.ChartAreas[0].CursorX.Interval = 1;
70      chart.ChartAreas[0].CursorY.Interval = 1;
71      chart.ChartAreas[0].AxisY.Title = "Relative Frequency";
72      chart.ChartAreas[0].CursorY.IsUserSelectionEnabled = true;
73      chart.ChartAreas[0].AxisY.ScaleView.Zoomable = true;
74      chart.ChartAreas[0].AxisY.IsStartedFromZero = true;
[7186]75    }
76
77    private void RedrawChart() {
[8104]78      foreach (Series series in chart.Series) {
79        series.Points.Clear();
[7255]80      }
[7186]81      if (Content != null) {
[8173]82        List<double> residuals = CalculateResiduals(Content);
[7186]83
[8173]84        double max = 0.0;
[8104]85        foreach (Series series in chart.Series) {
[8173]86          CalculateFrequencies(residuals, series);
87          double seriesMax = series.Points.Select(p => p.YValues.First()).Max();
88          max = max < seriesMax ? seriesMax : max;
[7186]89        }
90
[8173]91        // ALL_SAMPLES has to be calculated to know its highest frequency, but it is not shown in the beginning
[8176]92        chart.Series.First(s => s.Name.Equals(ALL_SAMPLES)).Points.Clear();
[8173]93
94        double roundedMax, intervalWidth;
95        CalculateResidualParameters(residuals, out roundedMax, out intervalWidth);
96
[7255]97        ChartArea chartArea = chart.ChartAreas[0];
[7503]98        chartArea.AxisX.Minimum = -roundedMax - intervalWidth;
99        chartArea.AxisX.Maximum = roundedMax + intervalWidth;
100        // get the highest frequency of a residual of any series
[8173]101        chartArea.AxisY.Maximum = max;
[7503]102        if (chartArea.AxisY.Maximum < 0.1) {
103          chartArea.AxisY.Interval = 0.01;
104          chartArea.AxisY.Maximum = Math.Ceiling(chartArea.AxisY.Maximum * 100) / 100;
105        } else {
106          chartArea.AxisY.Interval = 0.1;
107          chartArea.AxisY.Maximum = Math.Ceiling(chartArea.AxisY.Maximum * 10) / 10;
108        }
[7255]109        chartArea.AxisX.Interval = intervalWidth;
[7503]110        int curBins = (int)Math.Round((roundedMax * 2) / intervalWidth);
[7255]111        //shifts the x axis label so that zero is in the middle
112        if (curBins % 2 == 0)
113          chartArea.AxisX.IntervalOffset = intervalWidth;
114        else
115          chartArea.AxisX.IntervalOffset = intervalWidth / 2;
116      }
117    }
118
[8173]119    private List<double> CalculateResiduals(IRegressionSolution solution) {
[8104]120      List<double> residuals = new List<double>();
[7255]121
[8173]122      IRegressionProblemData problemdata = solution.ProblemData;
[7255]123      List<double> targetValues = problemdata.Dataset.GetDoubleValues(Content.ProblemData.TargetVariable).ToList();
[8173]124      List<double> estimatedValues = solution.EstimatedValues.ToList();
[7255]125
[8173]126      for (int i = 0; i < solution.ProblemData.Dataset.Rows; i++) {
[7255]127        double residual = estimatedValues[i] - targetValues[i];
[8104]128        residuals.Add(residual);
[7255]129      }
130      return residuals;
131    }
132
[8173]133    private void CalculateFrequencies(List<double> residualValues, Series series) {
134      double roundedMax, intervalWidth;
135      CalculateResidualParameters(residualValues, out roundedMax, out intervalWidth);
136
[8104]137      IEnumerable<double> relevantResiduals = residualValues;
138      IRegressionProblemData problemdata = Content.ProblemData;
[8173]139      if (series.Name.Equals(TRAINING_SAMPLES)) {
[8104]140        relevantResiduals = residualValues.Skip(problemdata.TrainingPartition.Start).Take(problemdata.TrainingPartition.Size);
[8173]141      } else if (series.Name.Equals(TEST_SAMPLES)) {
[8104]142        relevantResiduals = residualValues.Skip(problemdata.TestPartition.Start).Take(problemdata.TestPartition.Size);
143      }
144
[7255]145      double intervalCenter = intervalWidth / 2.0;
[8104]146      double sampleCount = relevantResiduals.Count();
[8173]147      double current = -roundedMax;
148      DataPointCollection seriesPoints = series.Points;
[7255]149
150      for (int i = 0; i <= bins; i++) {
[8104]151        IEnumerable<double> help = relevantResiduals.Where(x => x >= (current - intervalCenter) && x < (current + intervalCenter));
[8173]152        seriesPoints.AddXY(current, help.Count() / sampleCount);
153        seriesPoints[seriesPoints.Count - 1]["from"] = (current - intervalCenter).ToString();
154        seriesPoints[seriesPoints.Count - 1]["to"] = (current + intervalCenter).ToString();
[7255]155        current += intervalWidth;
156      }
157    }
158
[8104]159    private void ToggleSeriesData(Series series) {
160      if (series.Points.Count > 0) {  //checks if series is shown
161        if (chart.Series.Any(s => s != series && s.Points.Count > 0)) {
162          series.Points.Clear();
163        }
164      } else if (Content != null) {
[8173]165        List<double> residuals = CalculateResiduals(Content);
166        CalculateFrequencies(residuals, series);
[8104]167        chart.Legends[series.Legend].ForeColor = Color.Black;
168        chart.Refresh();
169      }
170    }
171
[8173]172    private static void CalculateResidualParameters(List<double> residuals, out double roundedMax, out double intervalWidth) {
173      double realMax = Math.Max(Math.Abs(residuals.Min()), Math.Abs(residuals.Max()));
174      roundedMax = HumanRoundMax(realMax);
175      intervalWidth = (roundedMax * 2.0) / bins;
176      intervalWidth = HumanRoundMax(intervalWidth);
177      // sets roundedMax to a value, so that zero will be in the middle of the x axis
178      double help = realMax / intervalWidth;
179      help = help % 1 < 0.5 ? (int)help : (int)help + 1;
180      roundedMax = help * intervalWidth;
181    }
182
183    private static double HumanRoundMax(double max) {
[7255]184      double base10;
185      if (max > 0) base10 = Math.Pow(10.0, Math.Floor(Math.Log10(max)));
186      else base10 = Math.Pow(10.0, Math.Ceiling(Math.Log10(-max)));
187      double rounding = (max > 0) ? base10 : -base10;
188      while (rounding < max) rounding += base10;
189      return rounding;
190    }
191
[7186]192    #region events
193    protected override void RegisterContentEvents() {
194      base.RegisterContentEvents();
195      Content.ModelChanged += new EventHandler(Content_ModelChanged);
196      Content.ProblemDataChanged += new EventHandler(Content_ProblemDataChanged);
197    }
198    protected override void DeregisterContentEvents() {
199      base.DeregisterContentEvents();
200      Content.ModelChanged -= new EventHandler(Content_ModelChanged);
201      Content.ProblemDataChanged -= new EventHandler(Content_ProblemDataChanged);
202    }
203
204    protected override void OnContentChanged() {
205      base.OnContentChanged();
206      RedrawChart();
207    }
208    private void Content_ProblemDataChanged(object sender, EventArgs e) {
209      RedrawChart();
210    }
211    private void Content_ModelChanged(object sender, EventArgs e) {
212      RedrawChart();
213    }
[7255]214    private void chart_MouseDown(object sender, MouseEventArgs e) {
215      HitTestResult result = chart.HitTest(e.X, e.Y);
216      if (result.ChartElementType == ChartElementType.LegendItem) {
217        ToggleSeriesData(result.Series);
218      }
219    }
220    private void chart_MouseMove(object sender, MouseEventArgs e) {
221      HitTestResult result = chart.HitTest(e.X, e.Y);
222      if (result.ChartElementType == ChartElementType.LegendItem)
223        Cursor = Cursors.Hand;
224      else
225        Cursor = Cursors.Default;
226    }
227    private void chart_CustomizeLegend(object sender, CustomizeLegendEventArgs e) {
228      if (chart.Series.Count != 3) return;
[7485]229      e.LegendItems[0].Cells[1].ForeColor = chart.Series[ALL_SAMPLES].Points.Count == 0 ? Color.Gray : Color.Black;
230      e.LegendItems[1].Cells[1].ForeColor = chart.Series[TRAINING_SAMPLES].Points.Count == 0 ? Color.Gray : Color.Black;
231      e.LegendItems[2].Cells[1].ForeColor = chart.Series[TEST_SAMPLES].Points.Count == 0 ? Color.Gray : Color.Black;
[7255]232    }
[7186]233    #endregion
234  }
235}
Note: See TracBrowser for help on using the repository browser.