#region License Information
/* HeuristicLab
* Copyright (C) 2002-2019 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.Drawing;
using HeuristicLab.Common;
using HeuristicLab.Common.Resources;
using HeuristicLab.Core;
using HeuristicLab.Parameters;
using HEAL.Attic;
namespace HeuristicLab.Optimization {
[Item("ResultParameter", "A parameter whose value is written to a result collection.")]
[StorableType("CF10EF50-82B6-4A98-82C0-3C5ECED48904")]
public sealed class ResultParameter : LookupParameter, IResultParameter where T : class, IItem {
public override Image ItemImage { get { return VSImageLibrary.Exception; } }
public override bool CanChangeDescription { get { return true; } }
[Storable]
private string resultCollectionName;
public string ResultCollectionName {
get { return resultCollectionName; }
set {
if (value == null) throw new ArgumentNullException();
if (string.IsNullOrWhiteSpace(value)) throw new ArgumentNullException();
else if (!value.Equals(resultCollectionName)) {
resultCollectionName = value;
OnResultCollectionNameChanged();
}
}
}
[Storable]
private T defaultValue;
public T DefaultValue {
get { return defaultValue; }
set {
if (value != defaultValue) {
defaultValue = value;
OnDefaultValueChanged();
}
}
}
[StorableConstructor]
private ResultParameter(StorableConstructorFlag _) : base(_) { }
private ResultParameter(ResultParameter original, Cloner cloner)
: base(original, cloner) {
resultCollectionName = original.resultCollectionName;
defaultValue = cloner.Clone(original.defaultValue);
}
public override IDeepCloneable Clone(Cloner cloner) {
return new ResultParameter(this, cloner);
}
public ResultParameter() : this("Anonymous", string.Empty, "Results") { }
public ResultParameter(string name, string description) : this(name, description, "Results") { }
public ResultParameter(string name, string description, string resultCollectionName)
: base(name, description, string.Empty) {
if (string.IsNullOrEmpty(resultCollectionName)) throw new ArgumentException("resultCollectionName");
this.resultCollectionName = resultCollectionName;
Hidden = false;
}
public ResultParameter(string name, string description, string resultCollectionName, T defaultValue)
: base(name, description, string.Empty) {
if (string.IsNullOrEmpty(resultCollectionName)) throw new ArgumentException("resultCollectionName");
if (defaultValue == null) throw new ArgumentNullException("defaultValue");
this.resultCollectionName = resultCollectionName;
this.defaultValue = defaultValue;
Hidden = false;
}
protected override IItem GetActualValue() {
ResultCollection results;
if (CachedActualValue != null) {
results = CachedActualValue as ResultCollection;
if (results == null) throw new InvalidOperationException("ResultParameter (" + ActualName + "): ResultCollection not found.");
} else {
var tmp = ResultCollectionName;
// verifyType has to be disabled, because the ResultCollection may not be identical to the generic type of the parameter
results = GetValue(ExecutionContext, ref tmp) as ResultCollection;
if (results == null) throw new InvalidOperationException("ResultParameter (" + ActualName + "): ResultCollection with name " + tmp + " not found.");
CachedActualValue = results;
}
IResult result;
if (!results.TryGetValue(ActualName, out result)) {
if (DefaultValue == null) throw new InvalidOperationException("ResultParameter (" + ActualName + "): Result not found and no default value specified.");
result = ItemDescription == Description ? new Result(ActualName, (T)DefaultValue.Clone()) : new Result(ActualName, Description, (T)DefaultValue.Clone());
results.Add(result);
}
var resultValue = result.Value as T;
if (resultValue == null)
throw new InvalidOperationException(string.Format("Type mismatch. Result \"{0}\" does not contain a \"{1}\".", ActualName, typeof(T).GetPrettyName()));
return resultValue;
}
protected override void SetActualValue(IItem value) {
ResultCollection results;
if (CachedActualValue != null) {
results = CachedActualValue as ResultCollection;
if (results == null) throw new InvalidOperationException("ResultParameter (" + ActualName + "): ResultCollection not found.");
} else {
var tmp = ResultCollectionName;
results = GetValue(ExecutionContext, ref tmp) as ResultCollection;
if (results == null) throw new InvalidOperationException("ResultParameter (" + ActualName + "): ResultCollection with name " + tmp + " not found.");
CachedActualValue = results;
}
IResult result;
if (!results.TryGetValue(ActualName, out result)) {
result = ItemDescription == Description ? new Result(ActualName, value) : new Result(ActualName, Description, value);
results.Add(result);
} else result.Value = value;
}
public event EventHandler ResultCollectionNameChanged;
private void OnResultCollectionNameChanged() {
var handler = ResultCollectionNameChanged;
if (handler != null) handler(this, EventArgs.Empty);
OnToStringChanged();
}
public event EventHandler DefaultValueChanged;
private void OnDefaultValueChanged() {
EventHandler handler = DefaultValueChanged;
if (handler != null) handler(this, EventArgs.Empty);
OnItemImageChanged();
}
}
}