#region License Information /* HeuristicLab * Copyright (C) 2002-2012 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.Linq; using System.Windows.Forms; using System.Windows.Forms.DataVisualization.Charting; using HeuristicLab.Common; using HeuristicLab.MainForm; using HeuristicLab.MainForm.WindowsForms; namespace HeuristicLab.Problems.DataAnalysis.Views { [View("Classification Threshold")] [Content(typeof(IDiscriminantFunctionClassificationSolution), false)] public sealed partial class DiscriminantFunctionClassificationSolutionThresholdView : DataAnalysisSolutionEvaluationView { private const double TrainingAxisValue = 0.0; private const double TestAxisValue = 10.0; private const double TrainingTestBorder = (TestAxisValue - TrainingAxisValue) / 2; private const string TrainingLabelText = "Training Samples"; private const string TestLabelText = "Test Samples"; private const double ClassNameLeftOffset = 10.0; private const double ClassNameBottomOffset = 5.0; public new IDiscriminantFunctionClassificationSolution Content { get { return (IDiscriminantFunctionClassificationSolution)base.Content; } set { base.Content = value; } } private Dictionary classValueSeriesMapping; private Random random; private bool updateInProgress, updateThresholds; public DiscriminantFunctionClassificationSolutionThresholdView() : base() { InitializeComponent(); classValueSeriesMapping = new Dictionary(); random = new Random(); updateInProgress = false; this.chart.CustomizeAllChartAreas(); this.chart.ChartAreas[0].CursorX.IsUserSelectionEnabled = true; this.chart.ChartAreas[0].AxisX.ScaleView.Zoomable = true; this.chart.ChartAreas[0].AxisX.Minimum = TrainingAxisValue - TrainingTestBorder; this.chart.ChartAreas[0].AxisX.Maximum = TestAxisValue + TrainingTestBorder; AddCustomLabelToAxis(this.chart.ChartAreas[0].AxisX); this.chart.ChartAreas[0].AxisY.Title = "Estimated Values"; this.chart.ChartAreas[0].CursorY.IsUserSelectionEnabled = true; this.chart.ChartAreas[0].AxisX.ScaleView.Zoomable = true; } private void AddCustomLabelToAxis(Axis axis) { CustomLabel trainingLabel = new CustomLabel(); trainingLabel.Text = TrainingLabelText; trainingLabel.FromPosition = TrainingAxisValue - TrainingTestBorder; trainingLabel.ToPosition = TrainingAxisValue + TrainingTestBorder; axis.CustomLabels.Add(trainingLabel); CustomLabel testLabel = new CustomLabel(); testLabel.Text = TestLabelText; testLabel.FromPosition = TestAxisValue - TrainingTestBorder; testLabel.ToPosition = TestAxisValue + TrainingTestBorder; axis.CustomLabels.Add(testLabel); } protected override void RegisterContentEvents() { base.RegisterContentEvents(); Content.ModelChanged += new EventHandler(Content_ModelChanged); Content.ProblemDataChanged += new EventHandler(Content_ProblemDataChanged); } protected override void DeregisterContentEvents() { base.DeregisterContentEvents(); Content.ModelChanged -= new EventHandler(Content_ModelChanged); Content.ProblemDataChanged -= new EventHandler(Content_ProblemDataChanged); } private void Content_ProblemDataChanged(object sender, EventArgs e) { UpdateChart(); } private void Content_ModelChanged(object sender, EventArgs e) { Content.Model.ThresholdsChanged += new EventHandler(Model_ThresholdsChanged); UpdateChart(); } private void Model_ThresholdsChanged(object sender, EventArgs e) { AddThresholds(); } protected override void OnContentChanged() { base.OnContentChanged(); UpdateChart(); } private void UpdateChart() { if (InvokeRequired) Invoke((Action)UpdateChart); else if (!updateInProgress) { updateInProgress = true; chart.Series.Clear(); classValueSeriesMapping.Clear(); if (Content != null) { IEnumerator classNameEnumerator = Content.ProblemData.ClassNames.GetEnumerator(); IEnumerator classValueEnumerator = Content.ProblemData.ClassValues.OrderBy(x => x).GetEnumerator(); while (classNameEnumerator.MoveNext() && classValueEnumerator.MoveNext()) { Series series = new Series(classNameEnumerator.Current); series.ChartType = SeriesChartType.FastPoint; series.Tag = classValueEnumerator.Current; chart.Series.Add(series); classValueSeriesMapping.Add(classValueEnumerator.Current, series); FillSeriesWithDataPoints(series); } updateThresholds = true; } chart.ChartAreas[0].RecalculateAxesScale(); updateInProgress = false; } } private void FillSeriesWithDataPoints(Series series) { List estimatedValues = Content.EstimatedValues.ToList(); var targetValues = Content.ProblemData.Dataset.GetDoubleValues(Content.ProblemData.TargetVariable).ToList(); foreach (int row in Content.ProblemData.TrainingIndices) { double estimatedValue = estimatedValues[row]; double targetValue = targetValues[row]; if (targetValue.IsAlmost((double)series.Tag)) { double jitterValue = random.NextDouble() * 2.0 - 1.0; DataPoint point = new DataPoint(); point.XValue = TrainingAxisValue + 0.01 * jitterValue * JitterTrackBar.Value * (TrainingTestBorder * 0.9); point.YValues[0] = estimatedValue; point.Tag = new KeyValuePair(TrainingAxisValue, jitterValue); series.Points.Add(point); } } foreach (int row in Content.ProblemData.TestIndices) { double estimatedValue = estimatedValues[row]; double targetValue = targetValues[row]; if (targetValue.IsAlmost((double)series.Tag)) { double jitterValue = random.NextDouble() * 2.0 - 1.0; DataPoint point = new DataPoint(); point.XValue = TestAxisValue + 0.01 * jitterValue * JitterTrackBar.Value * (TrainingTestBorder * 0.9); point.YValues[0] = estimatedValue; point.Tag = new KeyValuePair(TestAxisValue, jitterValue); series.Points.Add(point); } } UpdateCursorInterval(); } private void chart_PostPaint(object sender, ChartPaintEventArgs e) { if (updateThresholds) { AddThresholds(); updateThresholds = false; } } private void AddThresholds() { chart.Annotations.Clear(); int classIndex = 1; SizeF textSizeInPixel; IClassificationProblemData problemData = Content.ProblemData; var classValues = Content.Model.ClassValues.ToArray(); Graphics g = chart.CreateGraphics(); Axis y = chart.ChartAreas[0].AxisY; Axis x = chart.ChartAreas[0].AxisX; string name; foreach (double threshold in Content.Model.Thresholds) { if (!double.IsInfinity(threshold)) { HorizontalLineAnnotation annotation = new HorizontalLineAnnotation(); annotation.AllowMoving = true; annotation.AllowResizing = false; annotation.LineWidth = 2; annotation.LineColor = Color.Red; annotation.IsInfinitive = true; annotation.ClipToChartArea = chart.ChartAreas[0].Name; annotation.Tag = classIndex; //save classIndex as Tag to avoid moving the threshold accross class bounderies annotation.AxisX = chart.ChartAreas[0].AxisX; annotation.AxisY = y; annotation.Y = threshold; name = problemData.GetClassName(classValues[classIndex - 1]); TextAnnotation beneathLeft = CreateTextAnnotation(name, classIndex, x, y); beneathLeft.Y = threshold; beneathLeft.X = x.Minimum; TextAnnotation beneathRigth = CreateTextAnnotation(name, classIndex, x, y); beneathRigth.Y = threshold; textSizeInPixel = g.MeasureString(beneathRigth.Text, beneathRigth.Font); double textWidthPixelPos = x.ValueToPixelPosition(x.Maximum) - textSizeInPixel.Width - ClassNameLeftOffset; //check if position is within the position boundary beneathRigth.X = textWidthPixelPos < 0 || textWidthPixelPos > chart.Width ? double.NaN : x.PixelPositionToValue(textWidthPixelPos); name = problemData.GetClassName(classValues[classIndex]); TextAnnotation aboveLeft = CreateTextAnnotation(name, classIndex, x, y); textSizeInPixel = g.MeasureString(aboveLeft.Text, aboveLeft.Font); double textHeightPixelPos = y.ValueToPixelPosition(threshold) - textSizeInPixel.Height - ClassNameBottomOffset; //check if position is within the position boundary aboveLeft.Y = textHeightPixelPos < 0 || textHeightPixelPos > chart.Height ? double.NaN : y.PixelPositionToValue(textHeightPixelPos); aboveLeft.X = x.Minimum; TextAnnotation aboveRight = CreateTextAnnotation(name, classIndex, x, y); aboveRight.Y = aboveLeft.Y; textWidthPixelPos = x.ValueToPixelPosition(x.Maximum) - textSizeInPixel.Width - ClassNameLeftOffset; //check if position is within the position boundary aboveRight.X = textWidthPixelPos < 0 || textWidthPixelPos > chart.Width ? double.NaN : x.PixelPositionToValue(textWidthPixelPos); chart.Annotations.Add(annotation); chart.Annotations.Add(beneathLeft); chart.Annotations.Add(aboveLeft); chart.Annotations.Add(beneathRigth); chart.Annotations.Add(aboveRight); classIndex++; } } } private TextAnnotation CreateTextAnnotation(string name, int classIndex, Axis x, Axis y) { TextAnnotation annotation = new TextAnnotation(); annotation.Text = name; annotation.AllowMoving = true; annotation.AllowResizing = false; annotation.AllowSelecting = false; annotation.ClipToChartArea = chart.ChartAreas[0].Name; annotation.Tag = classIndex; annotation.AxisX = chart.ChartAreas[0].AxisX; annotation.AxisY = y; return annotation; } private void JitterTrackBar_ValueChanged(object sender, EventArgs e) { foreach (Series series in chart.Series) { foreach (DataPoint point in series.Points) { double value = ((KeyValuePair)point.Tag).Key; double jitterValue = ((KeyValuePair)point.Tag).Value; ; point.XValue = value + 0.01 * jitterValue * JitterTrackBar.Value * (TrainingTestBorder * 0.9); } } } private void chart_CustomizeLegend(object sender, CustomizeLegendEventArgs e) { foreach (LegendItem legendItem in e.LegendItems) { var series = chart.Series[legendItem.SeriesName]; if (series != null) { bool seriesIsInvisible = series.Points.Count == 0; foreach (LegendCell cell in legendItem.Cells) cell.ForeColor = seriesIsInvisible ? Color.Gray : Color.Black; } } } private void chart_MouseMove(object sender, MouseEventArgs e) { HitTestResult result = chart.HitTest(e.X, e.Y); if (result.ChartElementType == ChartElementType.LegendItem) this.Cursor = Cursors.Hand; else this.Cursor = Cursors.Default; } private void ToggleSeries(Series series) { if (series.Points.Count == 0) FillSeriesWithDataPoints(series); else series.Points.Clear(); } private void chart_MouseDown(object sender, MouseEventArgs e) { HitTestResult result = chart.HitTest(e.X, e.Y); if (result.ChartElementType == ChartElementType.LegendItem) { if (result.Series != null) ToggleSeries(result.Series); } } private void chart_AnnotationPositionChanging(object sender, AnnotationPositionChangingEventArgs e) { int classIndex = (int)e.Annotation.Tag; double[] thresholds = Content.Model.Thresholds.ToArray(); thresholds[classIndex] = e.NewLocationY; Array.Sort(thresholds); Content.Model.SetThresholdsAndClassValues(thresholds, Content.Model.ClassValues); } private void UpdateCursorInterval() { Series series = chart.Series[0]; double[] xValues = (from point in series.Points where !point.IsEmpty select point.XValue) .DefaultIfEmpty(1.0) .ToArray(); double[] yValues = (from point in series.Points where !point.IsEmpty select point.YValues[0]) .DefaultIfEmpty(1.0) .ToArray(); double xRange = xValues.Max() - xValues.Min(); double yRange = yValues.Max() - yValues.Min(); if (xRange.IsAlmost(0.0)) xRange = 1.0; if (yRange.IsAlmost(0.0)) yRange = 1.0; double xDigits = (int)Math.Log10(xRange) - 3; double yDigits = (int)Math.Log10(yRange) - 3; double xZoomInterval = Math.Pow(10, xDigits); double yZoomInterval = Math.Pow(10, yDigits); this.chart.ChartAreas[0].CursorX.Interval = xZoomInterval; this.chart.ChartAreas[0].CursorY.Interval = yZoomInterval; } } }