#region License Information /* HeuristicLab * Copyright (C) 2002-2010 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.Text; using System.Xml; using HeuristicLab.Common; using HeuristicLab.Core; using HeuristicLab.Persistence.Default.CompositeSerializers.Storable; namespace HeuristicLab.Parameters { /// /// A parameter whose value is retrieved from the scope. /// [Item("LookupParameter", "A parameter whose value is retrieved from or written to a scope.")] public class LookupParameter : Parameter, ILookupParameter where T : class, IItem { [Storable] private string actualName; public string ActualName { get { return actualName; } set { if (value == null) throw new ArgumentNullException(); if (!actualName.Equals(value)) { actualName = value; OnActualNameChanged(); } } } public new T ActualValue { get { return (T)GetActualValue(); } set { SetActualValue(value); } } public LookupParameter() : base("Anonymous", typeof(T)) { actualName = Name; } public LookupParameter(string name) : base(name, typeof(T)) { actualName = Name; } public LookupParameter(string name, string description) : base(name, description, typeof(T)) { actualName = Name; } public override IDeepCloneable Clone(Cloner cloner) { LookupParameter clone = (LookupParameter)base.Clone(cloner); clone.actualName = actualName; return clone; } public override string ToString() { return string.Format("{0}: {1} ({2})", Name, ActualName, DataType.Name); } private IValueParameter GetParameter(out string name) { IValueParameter valueParam = this as IValueParameter; ILookupParameter lookupParam = this as ILookupParameter; ExecutionContext current = ExecutionContext; name = Name; while ((valueParam != null) && (lookupParam != null)) { if ((valueParam != null) && (valueParam.Value != null)) return valueParam; if (lookupParam != null) name = lookupParam.ActualName; current = current.Parent; while ((current != null) && !current.Operator.Parameters.ContainsKey(name)) current = current.Parent; if (current != null) { valueParam = current.Operator.Parameters[name] as IValueParameter; lookupParam = current.Operator.Parameters[name] as ILookupParameter; if ((valueParam == null) && (lookupParam == null)) throw new InvalidOperationException( string.Format("Parameter look-up chain broken. Parameter \"{0}\" is not an \"{1}\" or an \"{2}\".", name, typeof(IValueParameter).GetPrettyName(), typeof(ILookupParameter).GetPrettyName()) ); } else { valueParam = null; lookupParam = null; } } return null; } private IVariable LookupVariable(string name) { IScope scope = ExecutionContext.Scope; while ((scope != null) && !scope.Variables.ContainsKey(name)) scope = scope.Parent; return scope != null ? scope.Variables[actualName] : null; } protected override IItem GetActualValue() { string name; // try to get local value from context stack IValueParameter param = GetParameter(out name); if (param != null) return param.Value; else { // try to get variable from scope IVariable var = LookupVariable(name); if (var != null) { T value = var.Value as T; if (value == null) throw new InvalidOperationException( string.Format("Type mismatch. Variable \"{0}\" does not contain a \"{1}\".", name, typeof(T).GetPrettyName()) ); return value; } } return null; } protected override void SetActualValue(IItem value) { T val = value as T; if (val == null) throw new InvalidOperationException( string.Format("Type mismatch. Value is not a \"{0}\".", typeof(T).GetPrettyName()) ); // try to get local value from context stack string name; IValueParameter param = GetParameter(out name); if (param != null) param.Value = val; else { // try to get variable from scope IVariable var = LookupVariable(name); if (var != null) var.Value = val; else ExecutionContext.Scope.Variables.Add(new Variable(name, value)); } } public event EventHandler ActualNameChanged; private void OnActualNameChanged() { if (ActualNameChanged != null) ActualNameChanged(this, EventArgs.Empty); OnChanged(); } public static string TranslateName(string name, ExecutionContext context) { string currentName = name; ExecutionContext currentContext = context; IParameter param; ILookupParameter lookupParam; while (currentContext != null) { currentContext.Operator.Parameters.TryGetValue(currentName, out param); if (param != null) { lookupParam = param as ILookupParameter; if (lookupParam == null) throw new InvalidOperationException( string.Format("Parameter look-up chain broken. Parameter \"{0}\" is not an \"{1}\".", currentName, typeof(ILookupParameter).GetPrettyName()) ); currentName = lookupParam.ActualName; } currentContext = currentContext.Parent; } return currentName; } } }