#region License Information
/* HeuristicLab
* Copyright (C) 2002-2016 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.ComponentModel;
using System.Drawing;
using System.Globalization;
using System.Linq;
using System.Windows.Forms;
using System.Windows.Forms.DataVisualization.Charting;
using HeuristicLab.Analysis;
using HeuristicLab.Collections;
using HeuristicLab.Core.Views;
using HeuristicLab.Data;
using HeuristicLab.MainForm;
using HeuristicLab.MainForm.WindowsForms;
namespace HeuristicLab.Optimization.Views {
[View("Run-length Distribution View")]
[Content(typeof(RunCollection), false)]
public partial class RunCollectionRLDView : ItemView {
private List invisibleTargetSeries;
private const string AllRuns = "All Runs";
private static readonly Color[] colors = new[] {
Color.FromArgb(0x40, 0x6A, 0xB7),
Color.FromArgb(0xB1, 0x6D, 0x01),
Color.FromArgb(0x4E, 0x8A, 0x06),
Color.FromArgb(0x75, 0x50, 0x7B),
Color.FromArgb(0x72, 0x9F, 0xCF),
Color.FromArgb(0xA4, 0x00, 0x00),
Color.FromArgb(0xAD, 0x7F, 0xA8),
Color.FromArgb(0x29, 0x50, 0xCF),
Color.FromArgb(0x90, 0xB0, 0x60),
Color.FromArgb(0xF5, 0x89, 0x30),
Color.FromArgb(0x55, 0x57, 0x53),
Color.FromArgb(0xEF, 0x59, 0x59),
Color.FromArgb(0xED, 0xD4, 0x30),
Color.FromArgb(0x63, 0xC2, 0x16),
};
private static readonly ChartDashStyle[] lineStyles = new[] {
ChartDashStyle.Solid,
ChartDashStyle.Dash,
ChartDashStyle.DashDot,
ChartDashStyle.Dot
};
private static readonly DataRowVisualProperties.DataRowLineStyle[] hlLineStyles = new[] {
DataRowVisualProperties.DataRowLineStyle.Solid,
DataRowVisualProperties.DataRowLineStyle.Dash,
DataRowVisualProperties.DataRowLineStyle.DashDot,
DataRowVisualProperties.DataRowLineStyle.Dot
};
public new RunCollection Content {
get { return (RunCollection)base.Content; }
set { base.Content = value; }
}
private double[] targets;
private double[] budgets;
private bool suppressUpdates;
private readonly IndexedDataTable byCostDataTable;
public IndexedDataTable ByCostDataTable {
get { return byCostDataTable; }
}
public RunCollectionRLDView() {
InitializeComponent();
invisibleTargetSeries = new List();
targetChart.CustomizeAllChartAreas();
targetChart.ChartAreas[0].CursorX.Interval = 1;
targetChart.SuppressExceptions = true;
byCostDataTable = new IndexedDataTable("ECDF by Cost", "A data table containing the ECDF of each of a number of groups.") {
VisualProperties = {
YAxisTitle = "Proportion of unused budgets",
YAxisMinimumFixedValue = 0,
YAxisMinimumAuto = false,
YAxisMaximumFixedValue = 1,
YAxisMaximumAuto = false
}
};
byCostViewHost.Content = byCostDataTable;
suppressUpdates = false;
}
#region Content events
protected override void RegisterContentEvents() {
base.RegisterContentEvents();
Content.ItemsAdded += Content_ItemsAdded;
Content.ItemsRemoved += Content_ItemsRemoved;
Content.CollectionReset += Content_CollectionReset;
Content.UpdateOfRunsInProgressChanged += Content_UpdateOfRunsInProgressChanged;
Content.OptimizerNameChanged += Content_AlgorithmNameChanged;
}
protected override void DeregisterContentEvents() {
Content.ItemsAdded -= Content_ItemsAdded;
Content.ItemsRemoved -= Content_ItemsRemoved;
Content.CollectionReset -= Content_CollectionReset;
Content.UpdateOfRunsInProgressChanged -= Content_UpdateOfRunsInProgressChanged;
Content.OptimizerNameChanged -= Content_AlgorithmNameChanged;
base.DeregisterContentEvents();
}
private void Content_ItemsAdded(object sender, CollectionItemsChangedEventArgs e) {
if (suppressUpdates) return;
if (InvokeRequired) {
Invoke(new CollectionItemsChangedEventHandler(Content_ItemsAdded), sender, e);
return;
}
UpdateGroupAndProblemComboBox();
UpdateDataTableComboBox();
foreach (var run in e.Items)
RegisterRunEvents(run);
}
private void Content_ItemsRemoved(object sender, CollectionItemsChangedEventArgs e) {
if (suppressUpdates) return;
if (InvokeRequired) {
Invoke(new CollectionItemsChangedEventHandler(Content_ItemsRemoved), sender, e);
return;
}
UpdateGroupAndProblemComboBox();
UpdateDataTableComboBox();
foreach (var run in e.Items)
DeregisterRunEvents(run);
}
private void Content_CollectionReset(object sender, CollectionItemsChangedEventArgs e) {
if (suppressUpdates) return;
if (InvokeRequired) {
Invoke(new CollectionItemsChangedEventHandler(Content_CollectionReset), sender, e);
return;
}
UpdateGroupAndProblemComboBox();
UpdateDataTableComboBox();
foreach (var run in e.OldItems)
DeregisterRunEvents(run);
}
private void Content_AlgorithmNameChanged(object sender, EventArgs e) {
if (InvokeRequired)
Invoke(new EventHandler(Content_AlgorithmNameChanged), sender, e);
else UpdateCaption();
}
private void Content_UpdateOfRunsInProgressChanged(object sender, EventArgs e) {
if (InvokeRequired) {
Invoke(new EventHandler(Content_UpdateOfRunsInProgressChanged), sender, e);
return;
}
suppressUpdates = Content.UpdateOfRunsInProgress;
if (!suppressUpdates) {
UpdateDataTableComboBox();
UpdateGroupAndProblemComboBox();
UpdateRuns();
}
}
private void RegisterRunEvents(IRun run) {
run.PropertyChanged += run_PropertyChanged;
}
private void DeregisterRunEvents(IRun run) {
run.PropertyChanged -= run_PropertyChanged;
}
private void run_PropertyChanged(object sender, PropertyChangedEventArgs e) {
if (suppressUpdates) return;
if (InvokeRequired) {
Invoke((Action)run_PropertyChanged, sender, e);
} else {
if (e.PropertyName == "Visible")
UpdateRuns();
}
}
#endregion
protected override void OnContentChanged() {
base.OnContentChanged();
dataTableComboBox.Items.Clear();
groupComboBox.Items.Clear();
targetChart.ChartAreas[0].AxisX.IsLogarithmic = false;
targetChart.Series.Clear();
invisibleTargetSeries.Clear();
byCostDataTable.VisualProperties.XAxisLogScale = false;
byCostDataTable.Rows.Clear();
UpdateCaption();
if (Content != null) {
UpdateGroupAndProblemComboBox();
UpdateDataTableComboBox();
}
}
private void UpdateGroupAndProblemComboBox() {
var selectedGroupItem = (string)groupComboBox.SelectedItem;
var groupings = Content.ParameterNames.OrderBy(x => x).ToArray();
groupComboBox.Items.Clear();
groupComboBox.Items.Add(AllRuns);
groupComboBox.Items.AddRange(groupings);
if (selectedGroupItem != null && groupComboBox.Items.Contains(selectedGroupItem)) {
groupComboBox.SelectedItem = selectedGroupItem;
} else if (groupComboBox.Items.Count > 0) {
groupComboBox.SelectedItem = groupComboBox.Items[0];
}
var problems = new HashSet();
foreach (var run in Content) {
problems.Add(new ProblemDescription(run));
}
var problemTypesDifferent = problems.Select(x => x.ProblemType).Where(x => !string.IsNullOrEmpty(x)).Distinct().Count() > 1;
var problemNamesDifferent = problems.Select(x => x.ProblemName).Where(x => !string.IsNullOrEmpty(x)).Distinct().Count() > 1;
var evaluatorDifferent = problems.Select(x => x.Evaluator).Where(x => !string.IsNullOrEmpty(x)).Distinct().Count() > 1;
var maximizationDifferent = problems.Select(x => x.Maximization).Distinct().Count() > 1;
var allEqual = !problemTypesDifferent && !problemNamesDifferent && !evaluatorDifferent && !maximizationDifferent;
var selectedProblemItem = (ProblemDescription)problemComboBox.SelectedItem;
problemComboBox.Items.Clear();
problemComboBox.Items.Add(ProblemDescription.MatchAll);
if (selectedProblemItem == null || selectedProblemItem == ProblemDescription.MatchAll)
problemComboBox.SelectedIndex = 0;
foreach (var prob in problems.OrderBy(x => x.ToString()).ToList()) {
prob.DisplayProblemType = problemTypesDifferent;
prob.DisplayProblemName = problemNamesDifferent || allEqual;
prob.DisplayEvaluator = evaluatorDifferent;
prob.DisplayMaximization = maximizationDifferent;
problemComboBox.Items.Add(prob);
if (prob.Equals(selectedProblemItem)) problemComboBox.SelectedItem = prob;
}
SetEnabledStateOfControls();
}
private void UpdateDataTableComboBox() {
string selectedItem = (string)dataTableComboBox.SelectedItem;
dataTableComboBox.Items.Clear();
var dataTables = (from run in Content
from result in run.Results
where result.Value is IndexedDataTable
select result.Key).Distinct().ToArray();
dataTableComboBox.Items.AddRange(dataTables);
if (selectedItem != null && dataTableComboBox.Items.Contains(selectedItem)) {
dataTableComboBox.SelectedItem = selectedItem;
} else if (dataTableComboBox.Items.Count > 0) {
dataTableComboBox.SelectedItem = dataTableComboBox.Items[0];
}
}
protected override void SetEnabledStateOfControls() {
base.SetEnabledStateOfControls();
groupComboBox.Enabled = Content != null;
problemComboBox.Enabled = Content != null && problemComboBox.Items.Count > 1;
dataTableComboBox.Enabled = Content != null && dataTableComboBox.Items.Count > 1;
addTargetsAsResultButton.Enabled = Content != null && targets != null && dataTableComboBox.SelectedIndex >= 0;
addBudgetsAsResultButton.Enabled = Content != null && budgets != null && dataTableComboBox.SelectedIndex >= 0;
}
private Dictionary>>> GroupRuns() {
var groupedRuns = new Dictionary>>>();
var table = (string)dataTableComboBox.SelectedItem;
if (string.IsNullOrEmpty(table)) return groupedRuns;
var selectedGroup = (string)groupComboBox.SelectedItem;
if (string.IsNullOrEmpty(selectedGroup)) return groupedRuns;
var selectedProblem = (ProblemDescription)problemComboBox.SelectedItem;
if (selectedProblem == null) return groupedRuns;
var targetsPerProblem = CalculateBestTargetPerProblemInstance(table);
foreach (var x in (from r in Content
where (selectedGroup == AllRuns || r.Parameters.ContainsKey(selectedGroup))
&& selectedProblem.Match(r)
&& r.Results.ContainsKey(table)
&& r.Visible
group r by selectedGroup == AllRuns ? AllRuns : r.Parameters[selectedGroup].ToString() into g
select Tuple.Create(g.Key, g.ToList()))) {
var pDict = new Dictionary>>();
foreach (var y in (from r in x.Item2
let pd = new ProblemDescription(r)
group r by pd into g
select Tuple.Create(g.Key, g.ToList()))) {
pDict[y.Item1] = Tuple.Create(targetsPerProblem[y.Item1], y.Item2);
}
groupedRuns[x.Item1] = pDict;
}
return groupedRuns;
}
#region Performance analysis by (multiple) target(s)
private void UpdateResultsByTarget() {
// necessary to reset log scale -> empty chart cannot use log scaling
targetChart.ChartAreas[0].AxisX.IsLogarithmic = false;
targetChart.Series.Clear();
invisibleTargetSeries.Clear();
var table = (string)dataTableComboBox.SelectedItem;
if (string.IsNullOrEmpty(table)) return;
if (targets == null) GenerateDefaultTargets();
var groupedRuns = GroupRuns();
if (groupedRuns.Count == 0) return;
var xAxisTitles = new HashSet();
var colorCount = 0;
var lineStyleCount = 0;
// if the group contains multiple different problem instances we want to use the
// minimal maximal observed effort otherwise we run into situations where we don't
// have data for a certain problem instance anymore this is a special case when
// aggregating over multiple problem instances
var maxEfforts = new Dictionary();
double minEff = double.MaxValue, maxEff = double.MinValue;
foreach (var group in groupedRuns) {
foreach (var problem in group.Value) {
double problemSpecificMaxEff;
if (!maxEfforts.TryGetValue(problem.Key, out problemSpecificMaxEff)) {
problemSpecificMaxEff = 0;
}
var bestKnownTarget = problem.Value.Item1;
var max = problem.Key.IsMaximization();
var worstTarget = (max ? (1 - targets.Max()) : (1 + targets.Max())) * bestKnownTarget;
var bestTarget = (max ? (1 - targets.Min()) : (1 + targets.Min())) * bestKnownTarget;
foreach (var run in problem.Value.Item2) {
var row = ((IndexedDataTable)run.Results[table]).Rows.First().Values;
var a = row.FirstOrDefault(x => max ? x.Item2 >= worstTarget : x.Item2 <= worstTarget);
var b = row.FirstOrDefault(x => max ? x.Item2 >= bestTarget : x.Item2 <= bestTarget);
var firstEff = (a == default(Tuple)) ? row.Last().Item1 : a.Item1;
var lastEff = (b == default(Tuple)) ? row.Last().Item1 : b.Item1;
if (minEff > firstEff) minEff = firstEff;
if (maxEff < lastEff) maxEff = lastEff;
if (problemSpecificMaxEff < lastEff) problemSpecificMaxEff = lastEff;
}
maxEfforts[problem.Key] = problemSpecificMaxEff;
}
}
maxEff = Math.Min(maxEff, maxEfforts.Values.Min());
var minZeros = (int)Math.Floor(Math.Log10(minEff));
var maxZeros = (int)Math.Floor(Math.Log10(maxEff));
var axisMin = (decimal)Math.Pow(10, minZeros);
var axisMax = (decimal)Math.Pow(10, maxZeros);
if (!targetLogScalingCheckBox.Checked) {
var minAdd = (decimal)Math.Pow(10, minZeros - 1) * 2;
var maxAdd = (decimal)Math.Pow(10, maxZeros - 1) * 2;
while (axisMin + minAdd < (decimal)minEff) axisMin += minAdd;
while (axisMax <= (decimal)maxEff) axisMax += maxAdd;
} else axisMax = (decimal)Math.Pow(10, (int)Math.Ceiling(Math.Log10(maxEff)));
targetChart.ChartAreas[0].AxisX.Minimum = (double)axisMin;
targetChart.ChartAreas[0].AxisX.Maximum = (double)axisMax;
foreach (var group in groupedRuns) {
// hits describes the number of target hits at a certain time for a certain group
var hits = new Dictionary>();
// misses describes the number of target misses after a certain time for a certain group
// for instance when a run ends, but has not achieved all targets, misses describes
// how many targets have been left open at the point when the run ended
var misses = new Dictionary>();
var maxLength = 0.0;
var noRuns = 0;
foreach (var problem in group.Value) {
foreach (var run in problem.Value.Item2) {
var resultsTable = (IndexedDataTable)run.Results[table];
xAxisTitles.Add(resultsTable.VisualProperties.XAxisTitle);
if (aggregateTargetsCheckBox.Checked) {
var length = CalculateHitsForAllTargets(hits, misses, resultsTable.Rows.First(), problem.Key, group.Key, problem.Value.Item1, maxEff);
maxLength = Math.Max(length, maxLength);
} else {
CalculateHitsForEachTarget(hits, misses, resultsTable.Rows.First(), problem.Key, group.Key, problem.Value.Item1, maxEff);
}
noRuns++;
}
}
foreach (var list in hits) {
var row = new Series(list.Key) {
ChartType = SeriesChartType.StepLine,
BorderWidth = 2,
Color = colors[colorCount],
BorderDashStyle = lineStyles[lineStyleCount],
};
var rowShade = new Series(list.Key + "-range") {
IsVisibleInLegend = false,
ChartType = SeriesChartType.Range,
Color = Color.FromArgb(32, colors[colorCount])
};
var ecdf = 0.0;
var missedecdf = 0.0;
var iter = misses[list.Key].GetEnumerator();
var moreMisses = iter.MoveNext();
var totalTargets = noRuns;
if (aggregateTargetsCheckBox.Checked) totalTargets *= targets.Length;
var movingTargets = totalTargets;
foreach (var h in list.Value) {
while (moreMisses && iter.Current.Key <= h.Key) {
missedecdf += iter.Current.Value;
movingTargets -= iter.Current.Value;
if (row.Points.Count > 0 && row.Points.Last().XValue == iter.Current.Key)
row.Points.Last().SetValueY(ecdf / movingTargets);
else row.Points.AddXY(iter.Current.Key, ecdf / movingTargets);
if (boundShadingCheckBox.Checked) {
if (rowShade.Points.Count > 0 && rowShade.Points.Last().XValue == iter.Current.Key)
rowShade.Points.Last().SetValueY(ecdf / totalTargets, (ecdf + missedecdf) / totalTargets);
else rowShade.Points.Add(new DataPoint(iter.Current.Key, new[] { ecdf / totalTargets, (ecdf + missedecdf) / totalTargets }));
}
moreMisses = iter.MoveNext();
}
ecdf += h.Value;
if (row.Points.Count > 0 && row.Points.Last().XValue == h.Key)
row.Points.Last().SetValueY(ecdf / movingTargets);
else row.Points.AddXY(h.Key, ecdf / movingTargets);
if (missedecdf > 0 && boundShadingCheckBox.Checked) {
if (rowShade.Points.Count > 0 && rowShade.Points.Last().XValue == h.Key)
rowShade.Points.Last().SetValueY(ecdf / totalTargets, (ecdf + missedecdf) / totalTargets);
else rowShade.Points.Add(new DataPoint(h.Key, new[] { ecdf / totalTargets, (ecdf + missedecdf) / totalTargets }));
}
}
while (moreMisses) {
// if there are misses beyond the last hit we extend the shaded area
missedecdf += iter.Current.Value;
//movingTargets -= iter.Current.Value;
if (row.Points.Count > 0 && row.Points.Last().XValue == iter.Current.Key)
row.Points.Last().SetValueY(ecdf / movingTargets);
else row.Points.AddXY(iter.Current.Key, ecdf / movingTargets);
if (boundShadingCheckBox.Checked) {
if (rowShade.Points.Count > 0 && rowShade.Points.Last().XValue == iter.Current.Key)
rowShade.Points.Last().SetValueY(ecdf / totalTargets, (ecdf + missedecdf) / totalTargets);
else rowShade.Points.Add(new DataPoint(iter.Current.Key, new[] { ecdf / totalTargets, (ecdf + missedecdf) / totalTargets }));
}
moreMisses = iter.MoveNext();
}
if (maxLength > 0 && (row.Points.Count == 0 || row.Points.Last().XValue < maxLength))
row.Points.AddXY(maxLength, ecdf / movingTargets);
if (row.Points.Count > 0) {
var point = row.Points.Last();
point.Label = row.Name;
point.MarkerStyle = MarkerStyle.Cross;
point.MarkerBorderWidth = 1;
}
ConfigureSeries(row);
targetChart.Series.Add(rowShade);
targetChart.Series.Add(row);
}
colorCount = (colorCount + 1) % colors.Length;
if (colorCount == 0) lineStyleCount = (lineStyleCount + 1) % lineStyles.Length;
}
if (targets.Length == 1)
targetChart.ChartAreas[0].AxisY.Title = "Probability to be " + (targets[0] * 100) + "% worse than best";
else targetChart.ChartAreas[0].AxisY.Title = "Proportion of reached targets";
targetChart.ChartAreas[0].AxisX.Title = string.Join(" / ", xAxisTitles);
targetChart.ChartAreas[0].AxisX.IsLogarithmic = CanDisplayLogarithmic();
targetChart.ChartAreas[0].CursorY.Interval = 0.05;
UpdateErtTables(groupedRuns);
}
private void GenerateDefaultTargets() {
targets = new[] { 0.1, 0.05, 0.02, 0.01, 0 };
suppressTargetsEvents = true;
targetsTextBox.Text = string.Join("% ; ", targets.Select(x => x * 100)) + "%";
suppressTargetsEvents = false;
}
private void CalculateHitsForEachTarget(Dictionary> hits,
Dictionary> misses,
IndexedDataRow row, ProblemDescription problem,
string group, double bestTarget, double maxEffort) {
foreach (var t in targets.Select(x => Tuple.Create((problem.IsMaximization() ? (1 - x) : (1 + x)) * bestTarget, x))) {
var l = t.Item1;
var key = group + "_" + (t.Item2 * 100) + "%_" + l;
if (!hits.ContainsKey(key)) {
hits.Add(key, new SortedList());
misses.Add(key, new SortedList());
}
var hit = false;
foreach (var v in row.Values) {
if (v.Item1 > maxEffort) break;
if (problem.IsMaximization() && v.Item2 >= l || !problem.IsMaximization() && v.Item2 <= l) {
if (hits[key].ContainsKey(v.Item1))
hits[key][v.Item1]++;
else hits[key][v.Item1] = 1;
hit = true;
break;
}
}
if (!hit) {
var max = Math.Min(row.Values.Last().Item1, maxEffort);
if (misses[key].ContainsKey(max))
misses[key][max]++;
else misses[key][max] = 1;
}
}
}
private double CalculateHitsForAllTargets(Dictionary> hits,
Dictionary> misses,
IndexedDataRow row, ProblemDescription problem,
string group, double bestTarget, double maxEffort) {
var values = row.Values;
if (!hits.ContainsKey(group)) {
hits.Add(group, new SortedList());
misses.Add(group, new SortedList());
}
var i = 0;
var j = 0;
while (i < targets.Length && j < values.Count) {
var target = (problem.IsMaximization() ? (1 - targets[i]) : (1 + targets[i])) * bestTarget;
var current = values[j];
if (current.Item1 > maxEffort) break;
if (problem.IsMaximization() && current.Item2 >= target
|| !problem.IsMaximization() && current.Item2 <= target) {
if (hits[group].ContainsKey(current.Item1)) hits[group][current.Item1]++;
else hits[group][current.Item1] = 1;
i++;
} else {
j++;
}
}
if (j == values.Count) j--;
var effort = Math.Min(values[j].Item1, maxEffort);
if (i < targets.Length) {
if (misses[group].ContainsKey(effort))
misses[group][effort] += targets.Length - i;
else misses[group][effort] = targets.Length - i;
}
return effort;
}
private void UpdateErtTables(Dictionary>>> groupedRuns) {
ertTableView.Content = null;
var columns = 1 + targets.Length + 1;
var matrix = new string[groupedRuns.Count * groupedRuns.Max(x => x.Value.Count) + groupedRuns.Max(x => x.Value.Count), columns];
var rowCount = 0;
var tableName = (string)dataTableComboBox.SelectedItem;
if (string.IsNullOrEmpty(tableName)) return;
var targetsPerProblem = CalculateBestTargetPerProblemInstance(tableName);
var colNames = new string[columns];
colNames[0] = colNames[colNames.Length - 1] = string.Empty;
for (var i = 0; i < targets.Length; i++) {
colNames[i + 1] = targets[i].ToString("0.0%");
}
var problems = groupedRuns.SelectMany(x => x.Value.Keys).Distinct().ToList();
foreach (var problem in problems) {
matrix[rowCount, 0] = problem.ToString();
for (var i = 0; i < targets.Length; i++) {
matrix[rowCount, i + 1] = (targetsPerProblem[problem] * (problem.IsMaximization() ? (1 - targets[i]) : (1 + targets[i]))).ToString(CultureInfo.CurrentCulture.NumberFormat);
}
matrix[rowCount, columns - 1] = "#succ";
rowCount++;
foreach (var group in groupedRuns) {
matrix[rowCount, 0] = group.Key;
if (!group.Value.ContainsKey(problem)) {
matrix[rowCount, columns - 1] = "N/A";
rowCount++;
continue;
}
var runs = group.Value[problem].Item2;
ErtCalculationResult result = default(ErtCalculationResult);
for (var i = 0; i < targets.Length; i++) {
result = ExpectedRuntimeHelper.CalculateErt(runs, tableName, (problem.IsMaximization() ? (1 - targets[i]) : (1 + targets[i])) * group.Value[problem].Item1, problem.IsMaximization());
matrix[rowCount, i + 1] = result.ToString();
}
matrix[rowCount, columns - 1] = targets.Length > 0 ? result.SuccessfulRuns + "/" + result.TotalRuns : "-";
rowCount++;
}
}
ertTableView.Content = new StringMatrix(matrix) { ColumnNames = colNames };
ertTableView.DataGridView.AutoResizeColumns(DataGridViewAutoSizeColumnsMode.AllCells);
}
#endregion
#region Performance analysis by (multiple) budget(s)
private void UpdateResultsByCost() {
// necessary to reset log scale -> empty chart cannot use log scaling
byCostDataTable.VisualProperties.XAxisLogScale = false;
byCostDataTable.Rows.Clear();
var table = (string)dataTableComboBox.SelectedItem;
if (string.IsNullOrEmpty(table)) return;
if (budgets == null) GenerateDefaultBudgets(table);
var groupedRuns = GroupRuns();
if (groupedRuns.Count == 0) return;
var colorCount = 0;
var lineStyleCount = 0;
var targetsPerProblem = CalculateBestTargetPerProblemInstance((string)dataTableComboBox.SelectedItem);
foreach (var group in groupedRuns) {
var hits = new Dictionary>();
foreach (var problem in group.Value) {
foreach (var run in problem.Value.Item2) {
var resultsTable = (IndexedDataTable)run.Results[table];
if (eachOrAllBudgetsCheckBox.Checked) {
CalculateHitsForEachBudget(hits, resultsTable.Rows.First(), group.Value.Count, problem.Key, group.Key, problem.Value.Item2.Count, targetsPerProblem[problem.Key]);
} else {
CalculateHitsForAllBudgets(hits, resultsTable.Rows.First(), group.Value.Count, problem.Key, group.Key, problem.Value.Item2.Count, targetsPerProblem[problem.Key]);
}
}
}
foreach (var list in hits) {
var row = new IndexedDataRow(list.Key) {
VisualProperties = {
ChartType = DataRowVisualProperties.DataRowChartType.StepLine,
LineWidth = 2,
Color = colors[colorCount],
LineStyle = hlLineStyles[lineStyleCount],
StartIndexZero = false
}
};
var total = 0.0;
foreach (var h in list.Value) {
total += h.Value;
row.Values.Add(Tuple.Create(h.Key, total));
}
byCostDataTable.Rows.Add(row);
}
colorCount = (colorCount + 1) % colors.Length;
if (colorCount == 0) lineStyleCount = (lineStyleCount + 1) % lineStyles.Length;
}
byCostDataTable.VisualProperties.XAxisTitle = "Targets to Best-Known Ratio";
byCostDataTable.VisualProperties.XAxisLogScale = byCostDataTable.Rows.Count > 0 && budgetLogScalingCheckBox.Checked;
}
private void GenerateDefaultBudgets(string table) {
var runs = GroupRuns().SelectMany(x => x.Value.Values).SelectMany(x => x.Item2).ToList();
var min = runs.Select(x => ((IndexedDataTable)x.Results[table]).Rows.First().Values.Select(y => y.Item1).Min()).Min();
var max = runs.Select(x => ((IndexedDataTable)x.Results[table]).Rows.First().Values.Select(y => y.Item1).Max()).Max();
var maxMagnitude = (int)Math.Ceiling(Math.Log10(max));
var minMagnitude = (int)Math.Floor(Math.Log10(min));
if (maxMagnitude - minMagnitude >= 3) {
budgets = new double[maxMagnitude - minMagnitude];
for (var i = minMagnitude; i < maxMagnitude; i++) {
budgets[i - minMagnitude] = Math.Pow(10, i);
}
} else {
var range = max - min;
budgets = Enumerable.Range(0, 6).Select(x => min + (x / 5.0) * range).ToArray();
}
suppressBudgetsEvents = true;
budgetsTextBox.Text = string.Join(" ; ", budgets);
suppressBudgetsEvents = false;
}
private void CalculateHitsForEachBudget(Dictionary> hits, IndexedDataRow row, int groupCount, ProblemDescription problem, string groupName, int problemCount, double bestTarget) {
foreach (var b in budgets) {
var key = groupName + "-" + b;
if (!hits.ContainsKey(key)) hits.Add(key, new SortedList());
Tuple prev = null;
foreach (var v in row.Values) {
if (v.Item1 >= b) {
// the budget may be too low to achieve any target
if (prev == null && v.Item1 != b) break;
var tgt = ((prev == null || v.Item1 == b) ? v.Item2 : prev.Item2);
tgt = problem.IsMaximization() ? bestTarget / tgt : tgt / bestTarget;
if (hits[key].ContainsKey(tgt))
hits[key][tgt] += 1.0 / (groupCount * problemCount);
else hits[key][tgt] = 1.0 / (groupCount * problemCount);
break;
}
prev = v;
}
if (hits[key].Count == 0) hits.Remove(key);
}
}
private void CalculateHitsForAllBudgets(Dictionary> hits, IndexedDataRow row, int groupCount, ProblemDescription problem, string groupName, int problemCount, double bestTarget) {
var values = row.Values;
if (!hits.ContainsKey(groupName)) hits.Add(groupName, new SortedList());
var i = 0;
var j = 0;
Tuple prev = null;
while (i < budgets.Length && j < values.Count) {
var current = values[j];
if (current.Item1 >= budgets[i]) {
if (prev != null || current.Item1 == budgets[i]) {
var tgt = (prev == null || current.Item1 == budgets[i]) ? current.Item2 : prev.Item2;
tgt = problem.IsMaximization() ? bestTarget / tgt : tgt / bestTarget;
if (!hits[groupName].ContainsKey(tgt)) hits[groupName][tgt] = 0;
hits[groupName][tgt] += 1.0 / (groupCount * problemCount * budgets.Length);
}
i++;
} else {
j++;
prev = current;
}
}
var lastTgt = values.Last().Item2;
lastTgt = problem.IsMaximization() ? bestTarget / lastTgt : lastTgt / bestTarget;
if (i < budgets.Length && !hits[groupName].ContainsKey(lastTgt)) hits[groupName][lastTgt] = 0;
while (i < budgets.Length) {
hits[groupName][lastTgt] += 1.0 / (groupCount * problemCount * budgets.Length);
i++;
}
}
#endregion
private void UpdateCaption() {
Caption = Content != null ? Content.OptimizerName + " RLD View" : ViewAttribute.GetViewName(GetType());
}
private void groupComboBox_SelectedIndexChanged(object sender, EventArgs e) {
UpdateRuns();
SetEnabledStateOfControls();
}
private void problemComboBox_SelectedIndexChanged(object sender, EventArgs e) {
UpdateRuns();
SetEnabledStateOfControls();
}
private void dataTableComboBox_SelectedIndexChanged(object sender, EventArgs e) {
if (dataTableComboBox.SelectedIndex >= 0)
GenerateDefaultBudgets((string)dataTableComboBox.SelectedItem);
UpdateRuns();
SetEnabledStateOfControls();
}
private void logScalingCheckBox_CheckedChanged(object sender, EventArgs e) {
UpdateResultsByTarget();
byCostDataTable.VisualProperties.XAxisLogScale = byCostDataTable.Rows.Count > 0 && budgetLogScalingCheckBox.Checked;
}
private void boundShadingCheckBox_CheckedChanged(object sender, EventArgs e) {
UpdateResultsByTarget();
}
#region Event handlers for target analysis
private bool suppressTargetsEvents;
private void targetsTextBox_Validating(object sender, CancelEventArgs e) {
if (suppressTargetsEvents) return;
var targetStrings = targetsTextBox.Text.Split(new[] { '%', ';', '\t', ' ' }, StringSplitOptions.RemoveEmptyEntries);
var targetList = new List();
foreach (var ts in targetStrings) {
decimal t;
if (!decimal.TryParse(ts, out t)) {
errorProvider.SetError(targetsTextBox, "Not all targets can be parsed: " + ts);
e.Cancel = true;
return;
}
targetList.Add(t / 100);
}
if (targetList.Count == 0) {
errorProvider.SetError(targetsTextBox, "Give at least one target value!");
e.Cancel = true;
return;
}
e.Cancel = false;
errorProvider.SetError(targetsTextBox, null);
targets = targetList.Select(x => (double)x).OrderByDescending(x => x).ToArray();
suppressTargetsEvents = true;
try { targetsTextBox.Text = string.Join("% ; ", targets.Select(x => x * 100)) + "%"; } finally { suppressTargetsEvents = false; }
UpdateResultsByTarget();
SetEnabledStateOfControls();
}
private void aggregateTargetsCheckBox_CheckedChanged(object sender, EventArgs e) {
SuspendRepaint();
try {
UpdateResultsByTarget();
} finally { ResumeRepaint(true); }
}
private void generateTargetsButton_Click(object sender, EventArgs e) {
decimal max = 1, min = 0, count = 10;
if (targets != null) {
max = (decimal)targets.Max();
min = (decimal)targets.Min();
count = targets.Length;
} else if (Content.Count > 0 && dataTableComboBox.SelectedIndex >= 0) {
var table = (string)dataTableComboBox.SelectedItem;
max = (decimal)Content.Where(x => x.Results.ContainsKey(table)).Select(x => ((IndexedDataTable)x.Results[table]).Rows.First().Values.Max(y => y.Item2)).Max();
min = (decimal)Content.Where(x => x.Results.ContainsKey(table)).Select(x => ((IndexedDataTable)x.Results[table]).Rows.First().Values.Min(y => y.Item2)).Min();
count = 6;
}
using (var dialog = new DefineArithmeticProgressionDialog(false, min, max, (max - min) / count)) {
if (dialog.ShowDialog() == DialogResult.OK) {
if (dialog.Values.Any()) {
targets = dialog.Values.Select(x => (double)x).ToArray();
suppressTargetsEvents = true;
targetsTextBox.Text = string.Join("% ; ", targets);
suppressTargetsEvents = false;
UpdateResultsByTarget();
SetEnabledStateOfControls();
}
}
}
}
private void addTargetsAsResultButton_Click(object sender, EventArgs e) {
var table = (string)dataTableComboBox.SelectedItem;
var targetsPerProblem = CalculateBestTargetPerProblemInstance(table);
foreach (var run in Content) {
if (!run.Results.ContainsKey(table)) continue;
var resultsTable = (IndexedDataTable)run.Results[table];
var values = resultsTable.Rows.First().Values;
var i = 0;
var j = 0;
var pd = new ProblemDescription(run);
while (i < targets.Length && j < values.Count) {
var target = (pd.IsMaximization() ? (1 - targets[i]) : (1 + targets[i])) * targetsPerProblem[pd];
var current = values[j];
if (pd.IsMaximization() && current.Item2 >= target
|| !pd.IsMaximization() && current.Item2 <= target) {
run.Results[table + ".Target" + target] = new DoubleValue(current.Item1);
i++;
} else {
j++;
}
}
}
}
#endregion
#region Event handlers for cost analysis
private bool suppressBudgetsEvents;
private void budgetsTextBox_Validating(object sender, CancelEventArgs e) {
if (suppressBudgetsEvents) return;
var budgetStrings = budgetsTextBox.Text.Split(new[] { ';', '\t', ' ' }, StringSplitOptions.RemoveEmptyEntries);
var budgetList = new List();
foreach (var ts in budgetStrings) {
double b;
if (!double.TryParse(ts, out b)) {
errorProvider.SetError(budgetsTextBox, "Not all targets can be parsed: " + ts);
e.Cancel = true;
return;
}
budgetList.Add(b);
}
if (budgetList.Count == 0) {
errorProvider.SetError(budgetsTextBox, "Give at least one target value!");
e.Cancel = true;
return;
}
e.Cancel = false;
errorProvider.SetError(budgetsTextBox, null);
budgets = budgetList.ToArray();
UpdateResultsByCost();
SetEnabledStateOfControls();
}
private void eachOrAllBudgetsCheckBox_CheckedChanged(object sender, EventArgs e) {
var each = eachOrAllBudgetsCheckBox.Checked;
eachOrAllBudgetsCheckBox.Text = each ? "each" : "all";
SuspendRepaint();
try {
UpdateResultsByCost();
} finally { ResumeRepaint(true); }
}
private void generateBudgetsButton_Click(object sender, EventArgs e) {
decimal max = 1, min = 0, count = 10;
if (budgets != null) {
max = (decimal)budgets.Max();
min = (decimal)budgets.Min();
count = budgets.Length;
} else if (Content.Count > 0 && dataTableComboBox.SelectedIndex >= 0) {
var table = (string)dataTableComboBox.SelectedItem;
min = (decimal)Content.Where(x => x.Results.ContainsKey(table)).Select(x => ((IndexedDataTable)x.Results[table]).Rows.First().Values.Min(y => y.Item1)).Min();
max = (decimal)Content.Where(x => x.Results.ContainsKey(table)).Select(x => ((IndexedDataTable)x.Results[table]).Rows.First().Values.Max(y => y.Item1)).Max();
count = 6;
}
using (var dialog = new DefineArithmeticProgressionDialog(false, min, max, (max - min) / count)) {
if (dialog.ShowDialog() == DialogResult.OK) {
if (dialog.Values.Any()) {
budgets = dialog.Values.OrderBy(x => x).Select(x => (double)x).ToArray();
suppressBudgetsEvents = true;
budgetsTextBox.Text = string.Join(" ; ", budgets);
suppressBudgetsEvents = false;
UpdateResultsByCost();
SetEnabledStateOfControls();
}
}
}
}
private void addBudgetsAsResultButton_Click(object sender, EventArgs e) {
var table = (string)dataTableComboBox.SelectedItem;
var budgetStrings = budgetsTextBox.Text.Split(new[] { ';', '\t', ' ' }, StringSplitOptions.RemoveEmptyEntries);
if (budgetStrings.Length == 0) {
MessageBox.Show("Define a number of budgets.");
return;
}
var budgetList = new List();
foreach (var bs in budgetStrings) {
double v;
if (!double.TryParse(bs, out v)) {
MessageBox.Show("Budgets must be a valid number: " + bs);
return;
}
budgetList.Add(v);
}
budgetList.Sort();
foreach (var run in Content) {
if (!run.Results.ContainsKey(table)) continue;
var resultsTable = (IndexedDataTable)run.Results[table];
var values = resultsTable.Rows.First().Values;
var i = 0;
var j = 0;
Tuple prev = null;
while (i < budgetList.Count && j < values.Count) {
var current = values[j];
if (current.Item1 >= budgetList[i]) {
if (prev != null || current.Item1 == budgetList[i]) {
var tgt = (prev == null || current.Item1 == budgetList[i]) ? current.Item2 : prev.Item2;
run.Results[table + ".Cost" + budgetList[i]] = new DoubleValue(tgt);
}
i++;
} else {
j++;
prev = current;
}
}
}
}
#endregion
#region Helpers
private Dictionary CalculateBestTargetPerProblemInstance(string table) {
return (from r in Content
where r.Visible
let pd = new ProblemDescription(r)
let target = r.Parameters.ContainsKey("BestKnownQuality")
&& r.Parameters["BestKnownQuality"] is DoubleValue
? ((DoubleValue)r.Parameters["BestKnownQuality"]).Value
: ((IndexedDataTable)r.Results[table]).Rows.First().Values.Last().Item2
group target by pd into g
select new { Problem = g.Key, Target = g.Key.IsMaximization() ? g.Max() : g.Min() })
.ToDictionary(x => x.Problem, x => x.Target);
}
private void UpdateRuns() {
if (InvokeRequired) {
Invoke((Action)UpdateRuns);
return;
}
SuspendRepaint();
try {
UpdateResultsByTarget();
UpdateResultsByCost();
} finally { ResumeRepaint(true); }
}
#endregion
private void ConfigureSeries(Series series) {
series.SmartLabelStyle.Enabled = true;
series.SmartLabelStyle.AllowOutsidePlotArea = LabelOutsidePlotAreaStyle.No;
series.SmartLabelStyle.CalloutLineAnchorCapStyle = LineAnchorCapStyle.None;
series.SmartLabelStyle.CalloutLineColor = series.Color;
series.SmartLabelStyle.CalloutLineWidth = 2;
series.SmartLabelStyle.CalloutStyle = LabelCalloutStyle.Underlined;
series.SmartLabelStyle.IsOverlappedHidden = false;
series.SmartLabelStyle.MaxMovingDistance = 200;
series.ToolTip = series.LegendText + " X = #VALX, Y = #VALY";
}
private void chart_MouseDown(object sender, MouseEventArgs e) {
HitTestResult result = targetChart.HitTest(e.X, e.Y);
if (result.ChartElementType == ChartElementType.LegendItem) {
ToggleTargetChartSeriesVisible(result.Series);
}
}
private void chart_MouseMove(object sender, MouseEventArgs e) {
HitTestResult result = targetChart.HitTest(e.X, e.Y);
if (result.ChartElementType == ChartElementType.LegendItem)
this.Cursor = Cursors.Hand;
else
this.Cursor = Cursors.Default;
}
private void chart_CustomizeLegend(object sender, CustomizeLegendEventArgs e) {
foreach (LegendItem legendItem in e.LegendItems) {
var series = targetChart.Series[legendItem.SeriesName];
if (series != null) {
bool seriesIsInvisible = invisibleTargetSeries.Any(x => x.Name == series.Name);
foreach (LegendCell cell in legendItem.Cells) {
cell.ForeColor = seriesIsInvisible ? Color.Gray : Color.Black;
}
}
}
}
private void ToggleTargetChartSeriesVisible(Series series) {
var indexList = invisibleTargetSeries.FindIndex(x => x.Name == series.Name);
var indexChart = targetChart.Series.IndexOf(series);
if (targetChart.Series.Count == 1) targetChart.ChartAreas[0].AxisX.IsLogarithmic = false;
targetChart.Series.RemoveAt(indexChart);
var s = indexList >= 0 ? invisibleTargetSeries[indexList] : new Series(series.Name) {
Color = series.Color,
ChartType = series.ChartType,
BorderWidth = series.BorderWidth,
BorderDashStyle = series.BorderDashStyle
};
if (indexList < 0) {
// hide
invisibleTargetSeries.Add(series);
var shadeSeries = targetChart.Series.FirstOrDefault(x => x.Name == series.Name + "-range");
if (shadeSeries != null) {
if (targetChart.Series.Count == 1) targetChart.ChartAreas[0].AxisX.IsLogarithmic = false;
targetChart.Series.Remove(shadeSeries);
invisibleTargetSeries.Add(shadeSeries);
indexChart--;
}
} else {
// show
invisibleTargetSeries.RemoveAt(indexList);
var shadeSeries = invisibleTargetSeries.FirstOrDefault(x => x.Name == series.Name + "-range");
if (shadeSeries != null) {
invisibleTargetSeries.Remove(shadeSeries);
InsertOrAddSeries(indexChart, shadeSeries);
indexChart++;
}
}
InsertOrAddSeries(indexChart, s);
targetChart.ChartAreas[0].AxisX.IsLogarithmic = CanDisplayLogarithmic();
}
private bool CanDisplayLogarithmic() {
return targetLogScalingCheckBox.Checked
&& targetChart.Series.Count > 0 // must have a series
&& targetChart.Series.Any(x => x.Points.Count > 0) // at least one series must have points
&& targetChart.Series.All(s => s.Points.All(p => p.XValue > 0)); // all points must be positive
}
private void InsertOrAddSeries(int index, Series s) {
if (targetChart.Series.Count <= index)
targetChart.Series.Add(s);
else targetChart.Series.Insert(index, s);
}
private class ProblemDescription {
private readonly bool matchAll;
public static readonly ProblemDescription MatchAll = new ProblemDescription() {
ProblemName = "All with Best-Known"
};
private ProblemDescription() {
ProblemType = string.Empty;
ProblemName = string.Empty;
Evaluator = string.Empty;
Maximization = string.Empty;
DisplayProblemType = false;
DisplayProblemName = false;
DisplayEvaluator = false;
DisplayMaximization = false;
matchAll = true;
}
public ProblemDescription(IRun run) {
ProblemType = GetStringValueOrEmpty(run, "Problem Type");
ProblemName = GetStringValueOrEmpty(run, "Problem Name");
Evaluator = GetStringValueOrEmpty(run, "Evaluator");
Maximization = GetMaximizationValueOrEmpty(run, "Maximization");
DisplayProblemType = !string.IsNullOrEmpty(ProblemType);
DisplayProblemName = !string.IsNullOrEmpty(ProblemName);
DisplayEvaluator = !string.IsNullOrEmpty(Evaluator);
DisplayMaximization = !string.IsNullOrEmpty(Maximization);
matchAll = false;
}
public bool DisplayProblemType { get; set; }
public string ProblemType { get; set; }
public bool DisplayProblemName { get; set; }
public string ProblemName { get; set; }
public bool DisplayEvaluator { get; set; }
public string Evaluator { get; set; }
public bool DisplayMaximization { get; set; }
public string Maximization { get; set; }
public bool IsMaximization() {
return Maximization == "MAX";
}
public bool Match(IRun run) {
return matchAll ||
GetStringValueOrEmpty(run, "Problem Type") == ProblemType
&& GetStringValueOrEmpty(run, "Problem Name") == ProblemName
&& GetStringValueOrEmpty(run, "Evaluator") == Evaluator
&& GetMaximizationValueOrEmpty(run, "Maximization") == Maximization;
}
private string GetStringValueOrEmpty(IRun run, string key) {
return run.Parameters.ContainsKey(key) ? ((StringValue)run.Parameters[key]).Value : string.Empty;
}
private string GetMaximizationValueOrEmpty(IRun run, string key) {
return run.Parameters.ContainsKey(key) ? (((BoolValue)run.Parameters[key]).Value ? "MAX" : "MIN") : string.Empty;
}
public override bool Equals(object obj) {
var other = obj as ProblemDescription;
if (other == null) return false;
return ProblemType == other.ProblemType
&& ProblemName == other.ProblemName
&& Evaluator == other.Evaluator
&& Maximization == other.Maximization;
}
public override int GetHashCode() {
return ProblemType.GetHashCode() ^ ProblemName.GetHashCode() ^ Evaluator.GetHashCode() ^ Maximization.GetHashCode();
}
public override string ToString() {
return string.Join(" -- ", new[] {
(DisplayProblemType ? ProblemType : string.Empty),
(DisplayProblemName ? ProblemName : string.Empty),
(DisplayEvaluator ? Evaluator : string.Empty),
(DisplayMaximization ? Maximization : string.Empty)}.Where(x => !string.IsNullOrEmpty(x)));
}
}
}
}