Free cookie consent management tool by TermsFeed Policy Generator

source: branches/2457_ExpertSystem/HeuristicLab.OptimizationExpertSystem/3.3/Views/KnowledgeCenterAllinOneView.cs @ 17203

Last change on this file since 17203 was 16958, checked in by abeham, 6 years ago

#2457: adapted to trunk

File size: 25.6 KB
RevLine 
[8955]1#region License Information
2/* HeuristicLab
[13713]3 * Copyright (C) 2002-2016 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
[8955]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
[16958]22using System;
23using System.Collections.Generic;
24using System.ComponentModel;
25using System.Linq;
26using System.Text.RegularExpressions;
27using System.Windows.Forms;
28using System.Windows.Forms.DataVisualization.Charting;
[13663]29using HeuristicLab.Analysis;
30using HeuristicLab.Analysis.QualityAnalysis;
[13713]31using HeuristicLab.Clients.OKB.RunCreation;
[13663]32using HeuristicLab.Common;
[13649]33using HeuristicLab.Common.Resources;
[13713]34using HeuristicLab.Core;
[13649]35using HeuristicLab.Core.Views;
[13663]36using HeuristicLab.Data;
[13713]37using HeuristicLab.Data.Views;
[13649]38using HeuristicLab.MainForm;
[13722]39using HeuristicLab.MainForm.WindowsForms;
[13649]40using HeuristicLab.Optimization;
41using HeuristicLab.Optimization.Views;
[13667]42using HeuristicLab.OptimizationExpertSystem.Common;
[8955]43
[13667]44namespace HeuristicLab.OptimizationExpertSystem {
[13718]45  [View("Knowledge Center (all-in-one view)")]
[13722]46  [Content(typeof(KnowledgeCenter), IsDefaultView = true)]
47  public partial class KnowledgeCenterAllinOneView : AsynchronousContentView {
[13713]48    private EnumValueView<SeedingStrategyTypes> seedingStrategyView;
49    private CheckedItemListView<IScope> seedingSolutionsView;
[8955]50    protected virtual bool SuppressEvents { get; set; }
[13485]51    private bool okbDownloadInProgress;
[8955]52
[13722]53    public new KnowledgeCenter Content {
54      get { return (KnowledgeCenter)base.Content; }
[8955]55      set { base.Content = value; }
56    }
57
[13722]58    public KnowledgeCenterAllinOneView() {
[8955]59      InitializeComponent();
[13551]60      // brings progress panel to front (it is not visible by default, but obstructs other elements in designer)
61      this.Controls.SetChildIndex(this.progressPanel, 0);
62      algorithmStartButton.Text = string.Empty;
63      algorithmStartButton.Image = VSImageLibrary.Play;
[13668]64      algorithmCloneButton.Text = string.Empty;
65      algorithmCloneButton.Image = VSImageLibrary.Clone;
[13475]66      refreshMapButton.Text = string.Empty;
67      refreshMapButton.Image = VSImageLibrary.Refresh;
[13713]68      seedingStrategyView = new EnumValueView<SeedingStrategyTypes>() {
69        Dock = DockStyle.Fill
70      };
71      seedingSolutionsView = new CheckedItemListView<IScope>() {
72        Dock = DockStyle.Fill
73      };
[13718]74      seedingStrategyPanel.Controls.Add(seedingStrategyView);
75      solutionSeedingTabPage.Controls.Add(seedingSolutionsView);
[8955]76    }
77
[13713]78    #region Event Registration
[8955]79    protected override void DeregisterContentEvents() {
[13774]80      Content.AlgorithmInstances.CollectionReset -= SuggestedInstancesOnChanged;
81      Content.AlgorithmInstances.ItemsAdded -= SuggestedInstancesOnChanged;
82      Content.AlgorithmInstances.ItemsMoved -= SuggestedInstancesOnChanged;
83      Content.AlgorithmInstances.ItemsRemoved -= SuggestedInstancesOnChanged;
84      Content.AlgorithmInstances.ItemsReplaced -= SuggestedInstancesOnChanged;
[13718]85      Content.SolutionSeedingPool.CollectionReset -= SeedingPoolOnChanged;
86      Content.SolutionSeedingPool.ItemsAdded -= SeedingPoolOnChanged;
87      Content.SolutionSeedingPool.ItemsRemoved -= SeedingPoolOnChanged;
88      Content.SolutionSeedingPool.ItemsReplaced -= SeedingPoolOnChanged;
89      Content.SolutionSeedingPool.CheckedItemsChanged -= SeedingPoolOnChanged;
[13713]90      DeregisterProblemEvents(Content.Problem);
[8955]91      base.DeregisterContentEvents();
92    }
[13713]93
[8955]94    protected override void RegisterContentEvents() {
95      base.RegisterContentEvents();
[13774]96      Content.AlgorithmInstances.CollectionReset += SuggestedInstancesOnChanged;
97      Content.AlgorithmInstances.ItemsAdded += SuggestedInstancesOnChanged;
98      Content.AlgorithmInstances.ItemsMoved += SuggestedInstancesOnChanged;
99      Content.AlgorithmInstances.ItemsRemoved += SuggestedInstancesOnChanged;
100      Content.AlgorithmInstances.ItemsReplaced += SuggestedInstancesOnChanged;
[13718]101      Content.SolutionSeedingPool.CollectionReset += SeedingPoolOnChanged;
102      Content.SolutionSeedingPool.ItemsAdded += SeedingPoolOnChanged;
103      Content.SolutionSeedingPool.ItemsRemoved += SeedingPoolOnChanged;
104      Content.SolutionSeedingPool.ItemsReplaced += SeedingPoolOnChanged;
105      Content.SolutionSeedingPool.CheckedItemsChanged += SeedingPoolOnChanged;
[13713]106      RegisterProblemEvents(Content.Problem);
[8955]107    }
108
[13713]109    private void DeregisterProblemEvents(OKBProblem problem) {
110      if (problem == null) return;
111      problem.Solutions.ItemsAdded -= SolutionsOnChanged;
112      problem.Solutions.ItemsRemoved -= SolutionsOnChanged;
113      problem.Solutions.ItemsReplaced -= SolutionsOnChanged;
114      problem.Solutions.CollectionReset -= SolutionsOnChanged;
115      problem.ProblemChanged -= ContentOnProblemChanged;
116    }
117
118    private void RegisterProblemEvents(OKBProblem problem) {
119      if (problem == null) return;
120      problem.Solutions.ItemsAdded += SolutionsOnChanged;
121      problem.Solutions.ItemsRemoved += SolutionsOnChanged;
122      problem.Solutions.ItemsReplaced += SolutionsOnChanged;
123      problem.Solutions.CollectionReset += SolutionsOnChanged;
124      problem.ProblemChanged += ContentOnProblemChanged;
125    }
126    #endregion
127
[8955]128    protected override void OnContentChanged() {
129      base.OnContentChanged();
130      SuppressEvents = true;
[13485]131      okbDownloadInProgress = false;
[8955]132      try {
133        if (Content == null) {
[12804]134          maxEvaluationsTextBox.Text = String.Empty;
[12847]135          problemViewHost.Content = null;
[13713]136          solverParametersView.Content = null;
137          solverResultsView.Content = null;
[8955]138          runsView.Content = null;
[13551]139          kbViewHost.Content = null;
[13475]140          problemInstancesView.Content = null;
[13713]141          seedingStrategyView.Content = null;
142          seedingSolutionsView.Content = null;
[8955]143        } else {
[12804]144          maxEvaluationsTextBox.Text = Content.MaximumEvaluations.ToString();
[12847]145          problemViewHost.Content = Content.Problem;
[13722]146          runsView.Content = Content.InstanceRuns;
[13551]147          kbViewHost.ViewType = typeof(RunCollectionRLDView);
148          kbViewHost.Content = Content.KnowledgeBase;
[13475]149          problemInstancesView.Content = Content.ProblemInstances;
[13722]150          solverResultsView.Content = null;
[13713]151          seedingStrategyView.Content = Content.SeedingStrategy;
152          seedingSolutionsView.Content = Content.SolutionSeedingPool;
[8955]153        }
154      } finally { SuppressEvents = false; }
[12860]155      UpdateSuggestedInstancesCombobox();
[13713]156      UpdateSimilarityCalculators();
[13663]157      UpdateNamesComboboxes();
[8955]158    }
[8956]159
[8955]160    protected override void SetEnabledStateOfControls() {
161      base.SetEnabledStateOfControls();
[12804]162      maxEvaluationsTextBox.Enabled = Content != null && !ReadOnly && !Locked;
[13551]163      problemViewHost.Enabled = Content != null && !ReadOnly && !Locked && !okbDownloadInProgress;
164      suggestedInstancesComboBox.Enabled = Content != null && !ReadOnly && !Locked && !okbDownloadInProgress;
165      algorithmStartButton.Enabled = Content != null && !ReadOnly && !Locked && suggestedInstancesComboBox.SelectedIndex >= 0;
[13668]166      algorithmCloneButton.Enabled = Content != null && !ReadOnly && !Locked && suggestedInstancesComboBox.SelectedIndex >= 0;
[8955]167      runsView.Enabled = Content != null;
[13551]168      kbViewHost.Enabled = Content != null && !okbDownloadInProgress;
169      problemInstancesView.Enabled = Content != null && !okbDownloadInProgress;
[13475]170      refreshMapButton.Enabled = Content != null;
[13551]171      okbDownloadButton.Enabled = Content != null && Content.Problem != null && Content.Problem.ProblemId >= 0 && !ReadOnly && !Locked && !okbDownloadInProgress;
[8955]172    }
173
174    #region Event Handlers
175    #region Content events
[13551]176    private void ContentOnProblemChanged(object sender, EventArgs eventArgs) {
177      UpdateSuggestedInstancesCombobox();
[13663]178      UpdateSimilarityCalculators();
[13551]179      SetEnabledStateOfControls();
180    }
181
[12847]182    private void SuggestedInstancesOnChanged(object sender, EventArgs e) {
[12860]183      UpdateSuggestedInstancesCombobox();
[12847]184    }
[13663]185
[13718]186    private void SeedingPoolOnChanged(object sender, EventArgs e) {
187      UpdateSolutionVisualization();
188    }
189
[13713]190    private void SolutionsOnChanged(object sender, EventArgs e) {
[13663]191      UpdateNamesComboboxes();
[13713]192      UpdateSolutionVisualization();
[13663]193    }
[8955]194    #endregion
195
196    #region Control events
[13561]197    private void MaxEvaluationsTextBoxOnValidating(object sender, CancelEventArgs e) {
[8961]198      if (SuppressEvents) return;
[12804]199      if (InvokeRequired) {
[13561]200        Invoke((Action<object, CancelEventArgs>)MaxEvaluationsTextBoxOnValidating, sender, e);
[12804]201        return;
202      }
203      int value;
204      if (!int.TryParse(maxEvaluationsTextBox.Text, out value)) {
205        e.Cancel = !maxEvaluationsTextBox.ReadOnly && maxEvaluationsTextBox.Enabled;
[13718]206        //errorProvider.SetError(maxEvaluationsTextBox, "Please enter a valid integer number.");
[12804]207      } else {
[13722]208        Content.MaximumEvaluations.Value = value;
[12804]209        e.Cancel = false;
[13718]210        //errorProvider.SetError(maxEvaluationsTextBox, null);
[12804]211      }
212    }
[13475]213
[13561]214    private void RefreshMapButtonOnClick(object sender, EventArgs e) {
215      UpdateProjectionComboBox();
[13475]216    }
[13561]217
218    private void OkbDownloadButtonOnClick(object sender, EventArgs e) {
[13569]219      if (Content.Problem.ProblemId == -1) {
[13551]220        MessageBox.Show("Please select a problem instance first.");
221        return;
222      }
[13485]223      var progress = new Progress();
224      progress.ProgressStateChanged += OkbDownloadProgressOnStateChanged;
225      Content.UpdateKnowledgeBaseAsync(progress);
[16958]226      Progress.ShowOnControl(progressPanel, progress);
[13485]227      progressPanel.Visible = true;
228      SetEnabledStateOfControls();
229    }
[13475]230
[13713]231    private void OkbDownloadProgressOnStateChanged(object sender, EventArgs e) {
232      if (InvokeRequired) { Invoke((Action<object, EventArgs>)OkbDownloadProgressOnStateChanged, sender, e); return; }
[13485]233      var progress = (IProgress)sender;
234      okbDownloadInProgress = progress.ProgressState == ProgressState.Started;
235      SetEnabledStateOfControls();
236      if (!okbDownloadInProgress) {
237        progressPanel.Visible = false;
238        progress.ProgressStateChanged -= OkbDownloadProgressOnStateChanged;
[13475]239      }
240    }
[13551]241
[13722]242    private async void AlgorithmStartButtonOnClick(object sender, EventArgs e) {
243      if (suggestedInstancesComboBox.SelectedIndex >= 0) {
244        solverResultsView.Content = await Content.StartAlgorithmAsync(suggestedInstancesComboBox.SelectedIndex);
245      }
[13551]246    }
[13561]247
[13668]248    private void AlgorithmCloneButtonOnClick(object sender, EventArgs e) {
249      if (suggestedInstancesComboBox.SelectedIndex >= 0)
[13774]250        MainFormManager.MainForm.ShowContent((IAlgorithm)Content.AlgorithmInstances[suggestedInstancesComboBox.SelectedIndex].Clone());
[13668]251    }
252
[13561]253    private void ProjectionComboBoxOnSelectedIndexChanged(object sender, EventArgs e) {
[13713]254      if (InvokeRequired) { Invoke((Action<object, EventArgs>)ProjectionComboBoxOnSelectedIndexChanged, sender, e); return; }
[13561]255      if (projectionComboBox.SelectedIndex < 0) return;
256      var projection = (string)projectionComboBox.SelectedItem;
257      var instancesSeries = instanceMapChart.Series["InstancesSeries"];
258      var currentInstanceSeries = instanceMapChart.Series["CurrentInstanceSeries"];
259
260      instancesSeries.Points.Clear();
261      currentInstanceSeries.Points.Clear();
262
263      foreach (var run in Content.ProblemInstances) {
264        var xKey = "Projection." + projection + ".X";
265        var yKey = "Projection." + projection + ".Y";
266        if (!run.Results.ContainsKey(xKey) || !run.Results.ContainsKey(yKey)
267          || !(run.Results[xKey] is Data.DoubleValue) || !(run.Results[yKey] is Data.DoubleValue)) continue;
268        var x = ((Data.DoubleValue)run.Results[xKey]).Value;
269        var y = ((Data.DoubleValue)run.Results[yKey]).Value;
270        var dataPoint = new DataPoint(x, y) {
271          Label = run.Name
272        };
[13751]273        if (!Content.IsCurrentInstance(run)) instancesSeries.Points.Add(dataPoint);
274        else currentInstanceSeries.Points.Add(dataPoint);
[13561]275      }
276    }
[13649]277
[13713]278    private void SuggestedInstancesComboBoxOnSelectedIndexChanged(object sender, EventArgs e) {
279      if (InvokeRequired) { Invoke((Action<object, EventArgs>)SuggestedInstancesComboBoxOnSelectedIndexChanged, sender, e); return; }
280      if (suggestedInstancesComboBox.SelectedIndex >= 0) {
[13774]281        var alg = Content.AlgorithmInstances[suggestedInstancesComboBox.SelectedIndex];
[13718]282        solverParametersView.Content = alg.Parameters;
283        var engineAlg = alg as EngineAlgorithm;
284        if (engineAlg != null) operatorGraphViewHost.Content = engineAlg.OperatorGraph;
285        else operatorGraphViewHost.Content = new Data.StringValue("Algorithm is not modeled as an operator graph.");
286      } else {
287        solverParametersView.Content = null;
288        operatorGraphViewHost.Content = null;
289      }
[13713]290      SetEnabledStateOfControls();
291    }
292
293    private void SimilarityComboBoxOnSelectedIndexChanged(object sender, EventArgs e) {
294      if (InvokeRequired) { Invoke((Action<object, EventArgs>)SimilarityComboBoxOnSelectedIndexChanged, sender, e); return; }
295      var calculator = (ISolutionSimilarityCalculator)similarityComboBox.SelectedItem;
296      if (calculator != null) {
297        calculator.SolutionVariableName = (string)solutionNameComboBox.SelectedItem;
298        calculator.QualityVariableName = Content.Problem.Problem.Evaluator.QualityParameter.ActualName;
299      }
300      UpdateSolutionDiversityAnalysis(calculator);
[13718]301      UpdateSolutionFdcAnalysis(calculator, fdcBetweenBestCheckBox.Checked);
[13713]302      UpdateSolutionLengthScaleAnalysis(calculator);
303      UpdateSolutionNetworkAnalysis(calculator);
304    }
[13718]305
306    private void FdcBetweenBestCheckBoxOnCheckedChanged(object sender, EventArgs e) {
307      if (InvokeRequired) { Invoke((Action<object, EventArgs>)FdcBetweenBestCheckBoxOnCheckedChanged, sender, e); return; }
308      UpdateSolutionFdcAnalysis((ISolutionSimilarityCalculator)similarityComboBox.SelectedItem, fdcBetweenBestCheckBox.Checked);
309    }
310
311    private void SolutionsNetworkChartOnMouseClick(object sender, MouseEventArgs e) {
312      var result = solutionsNetworkChart.HitTest(e.X, e.Y);
313      if (result.ChartElementType == ChartElementType.DataPoint) {
314        var point = (DataPoint)result.Object;
315        var solutionScope = point.Tag as IScope;
316        if (solutionScope == null || !Content.SolutionSeedingPool.Contains(solutionScope)) return;
317        Content.SolutionSeedingPool.SetItemCheckedState(solutionScope, !Content.SolutionSeedingPool.ItemChecked(solutionScope));
318      }
319    }
[13713]320    #endregion
321    #endregion
322
323    #region Control Configuration
324    private void UpdateSuggestedInstancesCombobox() {
325      var prevSelection = (IAlgorithm)suggestedInstancesComboBox.SelectedItem;
326      var prevNewIndex = -1;
327      suggestedInstancesComboBox.Items.Clear();
328      if (Content == null) return;
329
[13774]330      for (var i = 0; i < Content.AlgorithmInstances.Count; i++) {
331        suggestedInstancesComboBox.Items.Add(Content.AlgorithmInstances[i]);
332        if (prevSelection == null || Content.AlgorithmInstances[i].Name == prevSelection.Name)
[13713]333          prevNewIndex = prevSelection == null ? 0 : i;
334      }
335      if (prevNewIndex >= 0) {
336        suggestedInstancesComboBox.SelectedIndex = prevNewIndex;
337      }
338    }
339
340    private void UpdateSimilarityCalculators() {
341      var selected = (ISolutionSimilarityCalculator)(similarityComboBox.SelectedIndex >= 0 ? similarityComboBox.SelectedItem : null);
342      similarityComboBox.Items.Clear();
343
344      if (Content == null || Content.Problem == null) return;
345
346      foreach (var calc in Content.Problem.Operators.OfType<ISolutionSimilarityCalculator>()) {
347        similarityComboBox.Items.Add(calc);
348        if (selected != null && calc.ItemName == selected.ItemName) similarityComboBox.SelectedItem = calc;
349      }
350      if (selected == null && similarityComboBox.Items.Count > 0)
351        similarityComboBox.SelectedIndex = 0;
352    }
353
354    private void UpdateNamesComboboxes() {
355      var selectedSolutionName = solutionNameComboBox.SelectedIndex >= 0 ? (string)solutionNameComboBox.SelectedItem : string.Empty;
356
357      solutionNameComboBox.Items.Clear();
358      if (Content == null) return;
359      var solutionNames = Content.Problem.Solutions.Select(x => x.Solution).OfType<IScope>().SelectMany(x => x.Variables);
360
361      foreach (var sn in solutionNames.GroupBy(x => x.Name).OrderBy(x => x.Key)) {
362        solutionNameComboBox.Items.Add(sn.Key);
363        // either it was previously selected, or the variable value is defined in the HeuristicLab.Encodings sub-namespace
364        if (sn.Key == selectedSolutionName || (string.IsNullOrEmpty(selectedSolutionName) && sn.All(x => x.Value != null && x.Value.GetType().FullName.StartsWith("HeuristicLab.Encodings."))))
365          solutionNameComboBox.SelectedItem = sn.Key;
366      }
367    }
368   
369    private void UpdateProjectionComboBox() {
370      projectionComboBox.Items.Clear();
371      var projStrings = Content.ProblemInstances
372        .SelectMany(x => x.Results.Where(y => Regex.IsMatch(y.Key, "^Projection[.].*[.][XY]$")))
373        .Select(x => Regex.Match(x.Key, "Projection[.](?<g>.*)[.][XY]").Groups["g"].Value)
374        .Distinct();
375      foreach (var str in projStrings) {
376        projectionComboBox.Items.Add(str);
377      }
378    }
379    #endregion
380
381    #region Visualization
382    #region Solution Visualization
383    public void UpdateSolutionVisualization() {
384      if (InvokeRequired) { Invoke((Action)UpdateSolutionVisualization); return;  }
385      var qualityName = Content.Problem.Problem.Evaluator.QualityParameter.ActualName;
386      UpdateSolutionQualityAnalysis(qualityName);
387      if (similarityComboBox.SelectedIndex >= 0) {
388        var calculator = (ISolutionSimilarityCalculator)similarityComboBox.SelectedItem;
389        UpdateSolutionDiversityAnalysis(calculator);
[13718]390        UpdateSolutionFdcAnalysis(calculator, fdcBetweenBestCheckBox.Checked);
[13713]391        UpdateSolutionLengthScaleAnalysis(calculator);
392        UpdateSolutionNetworkAnalysis(calculator);
393      } else {
394        solutionsDiversityViewHost.Content = null;
395        solutionsFdcViewHost.Content = null;
396        solutionsLengthScaleViewHost.Content = null;
397        solutionsNetworkChart.Series.First().Points.Clear();
398      }
399    }
[13663]400    private void UpdateSolutionQualityAnalysis(string qualityName) {
401      var dt = solutionsQualityViewHost.Content as DataTable;
402      if (dt == null) {
403        dt = QualityDistributionAnalyzer.PrepareTable(qualityName);
[13713]404        dt.VisualProperties.Title = "Quality Distribution";
[13663]405        solutionsQualityViewHost.Content = dt;
406      }
[13718]407      QualityDistributionAnalyzer.UpdateTable(dt, GetSolutionScopes().Select(x => GetQuality(x, qualityName) ?? double.NaN).Where(x => !double.IsNaN(x)));
[13663]408    }
409
410    private void UpdateSolutionDiversityAnalysis(ISolutionSimilarityCalculator calculator) {
411      try {
412        solutionsDiversityViewHost.Content = null;
[13718]413        var solutionScopes = GetSolutionScopes();
414        var similarities = new double[solutionScopes.Count, solutionScopes.Count];
415        for (var i = 0; i < solutionScopes.Count; i++) {
416          for (var j = 0; j < solutionScopes.Count; j++)
417            similarities[i, j] = calculator.CalculateSolutionSimilarity(solutionScopes[i], solutionScopes[j]);
[13663]418        }
419        var hm = new HeatMap(similarities, "Solution Similarities", 0.0, 1.0);
420        solutionsDiversityViewHost.Content = hm;
421      } catch { }
422    }
423
[13718]424    private void UpdateSolutionFdcAnalysis(ISolutionSimilarityCalculator calculator, bool distanceToBest) {
[13663]425      try {
426        solutionsFdcViewHost.Content = null;
[13718]427        var solutionScopes = GetSolutionScopes();
[13663]428        var points = new List<Point2D<double>>();
[13718]429        if (distanceToBest) {
430          var maximization = ((IValueParameter<Data.BoolValue>)Content.Problem.MaximizationParameter).Value.Value;
431          var bestSolutions = (maximization ? solutionScopes.MaxItems(x => GetQuality(x, calculator.QualityVariableName) ?? double.NegativeInfinity)
432                                            : solutionScopes.MinItems(x => GetQuality(x, calculator.QualityVariableName) ?? double.PositiveInfinity)).ToList();
433          foreach (var solScope in solutionScopes.Except(bestSolutions)) {
434            var maxSimilarity = bestSolutions.Max(x => calculator.CalculateSolutionSimilarity(solScope, x));
435            var qDiff = (GetQuality(solScope, calculator.QualityVariableName) ?? double.NaN)
436                      - (GetQuality(bestSolutions[0], calculator.QualityVariableName) ?? double.NaN);
437            points.Add(new Point2D<double>(Math.Abs(qDiff), 1.0 - maxSimilarity));
[13663]438          }
[13718]439        } else {
440          for (int i = 0; i < solutionScopes.Count; i++) {
441            for (int j = 0; j < solutionScopes.Count; j++) {
442              if (i == j) continue;
443              var qDiff = (GetQuality(solutionScopes[i], calculator.QualityVariableName) ?? double.NaN)
444                          - (GetQuality(solutionScopes[j], calculator.QualityVariableName) ?? double.NaN);
445              if (double.IsNaN(qDiff)) continue;
446              points.Add(new Point2D<double>(Math.Abs(qDiff), 1.0 - calculator.CalculateSolutionSimilarity(solutionScopes[i], solutionScopes[j])));
447            }
448          }
[13663]449        }
450        var splot = new ScatterPlot("Fitness-Distance", "");
[13713]451        splot.VisualProperties.XAxisTitle = "Absolute Fitness Difference";
452        splot.VisualProperties.XAxisMinimumFixedValue = 0.0;
[13718]453        splot.VisualProperties.XAxisMinimumAuto = false;
[13713]454        splot.VisualProperties.YAxisTitle = "Solution Distance";
455        splot.VisualProperties.YAxisMinimumFixedValue = 0.0;
[13718]456        splot.VisualProperties.YAxisMinimumAuto = false;
[13713]457        splot.VisualProperties.YAxisMaximumFixedValue = 1.0;
[13718]458        splot.VisualProperties.YAxisMaximumAuto = false;
[13663]459        var row = new ScatterPlotDataRow("Fdc", "", points);
460        row.VisualProperties.PointSize = 7;
461        splot.Rows.Add(row);
462        solutionsFdcViewHost.Content = splot;
463      } catch { }
464    }
465
466    private void UpdateSolutionLengthScaleAnalysis(ISolutionSimilarityCalculator calculator) {
467      try {
468        solutionsLengthScaleViewHost.Content = null;
469        var dt = solutionsLengthScaleViewHost.Content as DataTable;
470        if (dt == null) {
471          dt = QualityDistributionAnalyzer.PrepareTable("Length Scale");
472          solutionsLengthScaleViewHost.Content = dt;
473        }
474        QualityDistributionAnalyzer.UpdateTable(dt, CalculateLengthScale(calculator));
[13718]475      } catch {
476        solutionsLengthScaleViewHost.Content = null;
477      }
[13663]478    }
479
480    private IEnumerable<double> CalculateLengthScale(ISolutionSimilarityCalculator calculator) {
[13718]481      var solutionScopes = GetSolutionScopes();
482      for (var i = 0; i < solutionScopes.Count; i++) {
483        for (var j = 0; j < solutionScopes.Count; j++) {
[13663]484          if (i == j) continue;
[13718]485          var sim = calculator.CalculateSolutionSimilarity(solutionScopes[i], solutionScopes[j]);
[13663]486          if (sim.IsAlmost(0)) continue;
[13718]487          var qDiff = (GetQuality(solutionScopes[i], calculator.QualityVariableName) ?? double.NaN)
488                    - (GetQuality(solutionScopes[j], calculator.QualityVariableName) ?? double.NaN);
489          if (!double.IsNaN(qDiff)) yield return Math.Abs(qDiff) / sim;
[13663]490        }
491      }
492    }
493
[13668]494    private void UpdateSolutionNetworkAnalysis(ISolutionSimilarityCalculator calculator) {
[13718]495      var series = solutionsNetworkChart.Series["SolutionSeries"];
496      var seedingSeries = solutionsNetworkChart.Series["SeedingSolutionSeries"];
[13668]497      try {
498        series.Points.Clear();
[13718]499        seedingSeries.Points.Clear();
500        var solutionScopes = GetSolutionScopes();
501        var dissimilarities = new DoubleMatrix(solutionScopes.Count, solutionScopes.Count);
502        for (var i = 0; i < solutionScopes.Count; i++) {
503          for (var j = 0; j < solutionScopes.Count; j++) {
[13668]504            if (i == j) continue;
[13718]505            dissimilarities[i, j] = 1.0 - calculator.CalculateSolutionSimilarity(solutionScopes[i], solutionScopes[j]);
[13668]506          }
507        }
508        var coords = MultidimensionalScaling.KruskalShepard(dissimilarities);
509        for (var i = 0; i < coords.Rows; i++) {
[13718]510          var quality = GetQuality(solutionScopes[i], calculator.QualityVariableName) ?? double.NaN;
511          var dataPoint = new DataPoint() {
[13713]512            Name = (i + 1).ToString(),
[13668]513            XValue = coords[i, 0],
[13718]514            YValues = new[] {coords[i, 1], quality},
515            Label = i + ": " + quality,
516            Tag = solutionScopes[i]
517          };
518          if (Content.SolutionSeedingPool.Contains(solutionScopes[i]) && Content.SolutionSeedingPool.ItemChecked(solutionScopes[i]))
519            seedingSeries.Points.Add(dataPoint);
520          else series.Points.Add(dataPoint);
[13668]521        }
[13718]522      } catch {
523        // problems in calculating the similarity
524        series.Points.Clear();
525        seedingSeries.Points.Clear();
526      }
[13668]527    }
[13713]528    #endregion
529    #endregion
[13718]530
531    private List<IScope> GetSolutionScopes() {
532      return Content.Problem.Solutions.Select(x => x.Solution).OfType<IScope>().ToList();
533    }
534
535    private double? GetQuality(IScope scope, string qualityName) {
536      IVariable v;
537      if (!scope.Variables.TryGetValue(qualityName, out v)) return null;
538      var dval = v.Value as Data.DoubleValue;
539      if (dval == null) return null;
540      return dval.Value;
541    }
[8955]542  }
543}
Note: See TracBrowser for help on using the repository browser.