[2] | 1 | #region License Information
|
---|
| 2 | /* HeuristicLab
|
---|
| 3 | * Copyright (C) 2002-2008 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
|
---|
| 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 |
|
---|
| 22 | using System;
|
---|
| 23 | using System.Collections.Generic;
|
---|
| 24 | using System.Text;
|
---|
| 25 | using System.Xml;
|
---|
| 26 |
|
---|
| 27 | namespace HeuristicLab.Core {
|
---|
[776] | 28 | /// <summary>
|
---|
| 29 | /// The base class for all operators.
|
---|
| 30 | /// </summary>
|
---|
[2] | 31 | public abstract class OperatorBase : ConstrainedItemBase, IOperator {
|
---|
| 32 | private string myName;
|
---|
[776] | 33 | /// <summary>
|
---|
| 34 | /// Gets or sets the name of the operator.
|
---|
| 35 | /// </summary>
|
---|
| 36 | /// <remarks>Calls <see cref="OnNameChanged"/> in the setter.</remarks>
|
---|
[2] | 37 | public string Name {
|
---|
| 38 | get { return myName; }
|
---|
| 39 | set {
|
---|
| 40 | if (myName != value) {
|
---|
| 41 | myName = value;
|
---|
| 42 | OnNameChanged();
|
---|
| 43 | }
|
---|
| 44 | }
|
---|
| 45 | }
|
---|
[776] | 46 | /// <summary>
|
---|
| 47 | /// Gets the description of the current operator.
|
---|
| 48 | /// </summary>
|
---|
| 49 | /// <remarks>Returns "No operator description available" if the method is not overriden.</remarks>
|
---|
[2] | 50 | public virtual string Description {
|
---|
| 51 | get { return "No operator description available."; }
|
---|
| 52 | }
|
---|
[776] | 53 | /// <summary>
|
---|
| 54 | /// Flag whether the current instance has been canceled.
|
---|
| 55 | /// </summary>
|
---|
[76] | 56 | protected bool myCanceled;
|
---|
[776] | 57 | /// <inheritdoc/>
|
---|
[2] | 58 | public bool Canceled {
|
---|
| 59 | get { return myCanceled; }
|
---|
| 60 | }
|
---|
| 61 | private bool myBreakpoint;
|
---|
[776] | 62 | /// <inheritdoc/>
|
---|
| 63 | /// <remarks>Calls <see cref="OnBreakpointChanged"/> in the setter.</remarks>
|
---|
[2] | 64 | public bool Breakpoint {
|
---|
| 65 | get { return myBreakpoint; }
|
---|
| 66 | set {
|
---|
| 67 | if (value != myBreakpoint) {
|
---|
| 68 | myBreakpoint = value;
|
---|
| 69 | OnBreakpointChanged();
|
---|
| 70 | }
|
---|
| 71 | }
|
---|
| 72 | }
|
---|
| 73 |
|
---|
| 74 | private List<IOperator> mySubOperators;
|
---|
[776] | 75 | /// <summary>
|
---|
| 76 | /// Gets a list of all suboperators.
|
---|
| 77 | /// <note type="caution"> Returns the suboperators read-only!</note>
|
---|
| 78 | /// </summary>
|
---|
[2] | 79 | public virtual IList<IOperator> SubOperators {
|
---|
| 80 | get { return mySubOperators.AsReadOnly(); }
|
---|
| 81 | }
|
---|
| 82 | private Dictionary<string, IVariableInfo> myVariableInfos;
|
---|
[776] | 83 | /// <inheritdoc/>
|
---|
[2] | 84 | public virtual ICollection<IVariableInfo> VariableInfos {
|
---|
| 85 | get { return myVariableInfos.Values; }
|
---|
| 86 | }
|
---|
| 87 | private Dictionary<string, IVariable> myVariables;
|
---|
[776] | 88 | /// <inheritdoc/>
|
---|
[2] | 89 | public virtual ICollection<IVariable> Variables {
|
---|
| 90 | get { return myVariables.Values; }
|
---|
| 91 | }
|
---|
| 92 |
|
---|
[776] | 93 | /// <summary>
|
---|
| 94 | /// Initializes a new instance of <see cref="OperatorBase"/> setting the breakpoint flag and
|
---|
| 95 | /// the canceled flag to <c>false</c> and the name of the operator to the type name.
|
---|
| 96 | /// </summary>
|
---|
[2] | 97 | protected OperatorBase() {
|
---|
| 98 | myName = this.GetType().Name;
|
---|
| 99 | myCanceled = false;
|
---|
| 100 | myBreakpoint = false;
|
---|
| 101 | mySubOperators = new List<IOperator>();
|
---|
| 102 | myVariableInfos = new Dictionary<string, IVariableInfo>();
|
---|
| 103 | myVariables = new Dictionary<string, IVariable>();
|
---|
| 104 | }
|
---|
| 105 |
|
---|
[776] | 106 | /// <summary>
|
---|
| 107 | /// Clones the current instance (deep clone).
|
---|
| 108 | /// </summary>
|
---|
| 109 | /// <remarks>Clones also sub operators, variables and variable infos.</remarks>
|
---|
| 110 | /// <param name="clonedObjects">Dictionary of all already cloned objects. (Needed to avoid cycles.)</param>
|
---|
| 111 | /// <returns>The cloned object as <see cref="OperatorBase"/>.</returns>
|
---|
[2] | 112 | public override object Clone(IDictionary<Guid, object> clonedObjects) {
|
---|
| 113 | OperatorBase clone = (OperatorBase)base.Clone(clonedObjects);
|
---|
| 114 | clone.myName = Name;
|
---|
| 115 | clone.mySubOperators.Clear();
|
---|
| 116 | for (int i = 0; i < SubOperators.Count; i++)
|
---|
| 117 | clone.AddSubOperator((IOperator)Auxiliary.Clone(SubOperators[i], clonedObjects));
|
---|
| 118 | clone.myVariableInfos.Clear();
|
---|
| 119 | foreach (IVariableInfo variableInfo in myVariableInfos.Values)
|
---|
| 120 | clone.AddVariableInfo((IVariableInfo)Auxiliary.Clone(variableInfo, clonedObjects));
|
---|
| 121 | clone.myVariables.Clear();
|
---|
| 122 | foreach (IVariable variable in myVariables.Values)
|
---|
| 123 | clone.AddVariable((IVariable)Auxiliary.Clone(variable, clonedObjects));
|
---|
| 124 | return clone;
|
---|
| 125 | }
|
---|
| 126 |
|
---|
[776] | 127 | /// <summary>
|
---|
| 128 | /// Creates a new instance of <see cref="OperatorBaseView"/> to represent the current operator
|
---|
| 129 | /// visually.
|
---|
| 130 | /// </summary>
|
---|
| 131 | /// <returns>The created view as <see cref="OperatorBaseView"/>.</returns>
|
---|
[2] | 132 | public override IView CreateView() {
|
---|
| 133 | return new OperatorBaseView(this);
|
---|
| 134 | }
|
---|
| 135 |
|
---|
| 136 | #region SubOperator Methods
|
---|
[776] | 137 | /// <inheritdoc cref="HeuristicLab.Core.IOperator.AddSubOperator(HeuristicLab.Core.IOperator)"/>
|
---|
| 138 | /// <param name="subOperator">The sub operator to add.</param>
|
---|
| 139 | /// <remarks>Calls <see cref="OnSubOperatorAdded"/>.</remarks>
|
---|
[2] | 140 | public virtual void AddSubOperator(IOperator subOperator) {
|
---|
| 141 | mySubOperators.Add(subOperator);
|
---|
| 142 | OnSubOperatorAdded(subOperator, mySubOperators.Count - 1);
|
---|
| 143 | }
|
---|
[776] | 144 | /// <inheritdoc cref="IOperator.TryAddSubOperator(HeuristicLab.Core.IOperator)"/>
|
---|
| 145 | /// <param name="subOperator">The sub operator to add.</param>
|
---|
| 146 | /// <remarks>Calls <see cref="OnSubOperatorAdded"/>.</remarks>
|
---|
[2] | 147 | public virtual bool TryAddSubOperator(IOperator subOperator) {
|
---|
| 148 | mySubOperators.Add(subOperator);
|
---|
| 149 | if (IsValid()) {
|
---|
| 150 | OnSubOperatorAdded(subOperator, mySubOperators.Count - 1);
|
---|
| 151 | return true;
|
---|
| 152 | } else {
|
---|
| 153 | mySubOperators.RemoveAt(mySubOperators.Count - 1);
|
---|
| 154 | return false;
|
---|
| 155 | }
|
---|
| 156 | }
|
---|
[776] | 157 | /// <inheritdoc cref="HeuristicLab.Core.IOperator.TryAddSubOperator(HeuristicLab.Core.IOperator,
|
---|
| 158 | /// out System.Collections.Generic.ICollection<HeuristicLab.Core.IConstraint>)"/>
|
---|
| 159 | /// <param name="subOperator">The sub operator to add.</param>
|
---|
| 160 | /// <remarks>Calls <see cref="OnSubOperatorAdded"/>.</remarks>
|
---|
[2] | 161 | public virtual bool TryAddSubOperator(IOperator subOperator, out ICollection<IConstraint> violatedConstraints) {
|
---|
| 162 | mySubOperators.Add(subOperator);
|
---|
| 163 | if (IsValid(out violatedConstraints)) {
|
---|
| 164 | OnSubOperatorAdded(subOperator, mySubOperators.Count - 1);
|
---|
| 165 | return true;
|
---|
| 166 | } else {
|
---|
| 167 | mySubOperators.RemoveAt(mySubOperators.Count - 1);
|
---|
| 168 | return false;
|
---|
| 169 | }
|
---|
| 170 | }
|
---|
[776] | 171 | /// <inheritdoc cref="HeuristicLab.Core.IOperator.AddSubOperator(HeuristicLab.Core.IOperator, int)"/>
|
---|
| 172 | /// <param name="subOperator">The sub operator to add.</param>
|
---|
| 173 | /// <remarks>Calls <see cref="OnSubOperatorAdded"/>.</remarks>
|
---|
[2] | 174 | public virtual void AddSubOperator(IOperator subOperator, int index) {
|
---|
| 175 | mySubOperators.Insert(index, subOperator);
|
---|
| 176 | OnSubOperatorAdded(subOperator, index);
|
---|
| 177 | }
|
---|
[776] | 178 | /// <inheritdoc cref="IOperator.TryAddSubOperator(HeuristicLab.Core.IOperator, int)"/>
|
---|
| 179 | /// <param name="subOperator">The sub operator to add.</param>
|
---|
| 180 | /// <remarks>Calls <see cref="OnSubOperatorAdded"/>.</remarks>
|
---|
[2] | 181 | public virtual bool TryAddSubOperator(IOperator subOperator, int index) {
|
---|
| 182 | mySubOperators.Insert(index, subOperator);
|
---|
| 183 | if (IsValid()) {
|
---|
| 184 | OnSubOperatorAdded(subOperator, index);
|
---|
| 185 | return true;
|
---|
| 186 | } else {
|
---|
| 187 | mySubOperators.RemoveAt(index);
|
---|
| 188 | return false;
|
---|
| 189 | }
|
---|
| 190 | }
|
---|
[776] | 191 | /// <inheritdoc cref="IOperator.TryAddSubOperator(HeuristicLab.Core.IOperator, int, out
|
---|
| 192 | /// System.Collections.Generic.ICollection<HeuristicLab.Core.IConstraint>)"/>
|
---|
| 193 | /// <param name="subOperator">The sub operator to add.</param>
|
---|
| 194 | /// <remarks>Calls <see cref="OnSubOperatorAdded"/>.</remarks>
|
---|
[2] | 195 | public virtual bool TryAddSubOperator(IOperator subOperator, int index, out ICollection<IConstraint> violatedConstraints) {
|
---|
| 196 | mySubOperators.Insert(index, subOperator);
|
---|
| 197 | if (IsValid(out violatedConstraints)) {
|
---|
| 198 | OnSubOperatorAdded(subOperator, index);
|
---|
| 199 | return true;
|
---|
| 200 | } else {
|
---|
| 201 | mySubOperators.RemoveAt(index);
|
---|
| 202 | return false;
|
---|
| 203 | }
|
---|
| 204 | }
|
---|
[776] | 205 | /// <inheritdoc/>
|
---|
| 206 | /// <remarks>Calls <see cref="OnSubOperatorRemoved"/>.</remarks>
|
---|
[2] | 207 | public virtual void RemoveSubOperator(int index) {
|
---|
| 208 | IOperator op = mySubOperators[index];
|
---|
| 209 | mySubOperators.RemoveAt(index);
|
---|
| 210 | OnSubOperatorRemoved(op, index);
|
---|
| 211 | }
|
---|
[776] | 212 | /// <inheritdoc/>
|
---|
| 213 | /// <remarks>Calls <see cref="OnSubOperatorRemoved"/>.</remarks>
|
---|
[2] | 214 | public virtual bool TryRemoveSubOperator(int index) {
|
---|
| 215 | IOperator op = mySubOperators[index];
|
---|
| 216 | mySubOperators.RemoveAt(index);
|
---|
| 217 | if (IsValid()) {
|
---|
| 218 | OnSubOperatorRemoved(op, index);
|
---|
| 219 | return true;
|
---|
| 220 | } else {
|
---|
| 221 | mySubOperators.Insert(index, op);
|
---|
| 222 | return false;
|
---|
| 223 | }
|
---|
| 224 | }
|
---|
[776] | 225 | /// <inheritdoc/>
|
---|
| 226 | /// <remarks>Calls <see cref="OnSubOperatorRemoved"/>.</remarks>
|
---|
[2] | 227 | public virtual bool TryRemoveSubOperator(int index, out ICollection<IConstraint> violatedConstraints) {
|
---|
| 228 | IOperator op = mySubOperators[index];
|
---|
| 229 | mySubOperators.RemoveAt(index);
|
---|
| 230 | if (IsValid(out violatedConstraints)) {
|
---|
| 231 | OnSubOperatorRemoved(op, index);
|
---|
| 232 | return true;
|
---|
| 233 | } else {
|
---|
| 234 | mySubOperators.Insert(index, op);
|
---|
| 235 | return false;
|
---|
| 236 | }
|
---|
| 237 | }
|
---|
| 238 | #endregion
|
---|
| 239 |
|
---|
| 240 | #region VariableInfo Methods
|
---|
[776] | 241 | /// <inheritdoc/>
|
---|
[2] | 242 | public virtual IVariableInfo GetVariableInfo(string formalName) {
|
---|
| 243 | IVariableInfo info;
|
---|
| 244 | if (myVariableInfos.TryGetValue(formalName, out info))
|
---|
| 245 | return info;
|
---|
| 246 | else
|
---|
| 247 | return null;
|
---|
| 248 | }
|
---|
[776] | 249 | /// <inheritdoc/>
|
---|
| 250 | /// <remarks>Calls <see cref="OnVariableInfoAdded"/>.</remarks>
|
---|
[2] | 251 | public virtual void AddVariableInfo(IVariableInfo variableInfo) {
|
---|
| 252 | myVariableInfos.Add(variableInfo.FormalName, variableInfo);
|
---|
| 253 | OnVariableInfoAdded(variableInfo);
|
---|
| 254 | }
|
---|
[776] | 255 | /// <inheritdoc/>
|
---|
| 256 | /// <remarks>Calls <see cref="OnVariableInfoAdded"/>.</remarks>
|
---|
[2] | 257 | public virtual bool TryAddVariableInfo(IVariableInfo variableInfo) {
|
---|
| 258 | myVariableInfos.Add(variableInfo.FormalName, variableInfo);
|
---|
| 259 | if (IsValid()) {
|
---|
| 260 | OnVariableInfoAdded(variableInfo);
|
---|
| 261 | return true;
|
---|
| 262 | } else {
|
---|
| 263 | myVariableInfos.Remove(variableInfo.FormalName);
|
---|
| 264 | return false;
|
---|
| 265 | }
|
---|
| 266 | }
|
---|
[776] | 267 | /// <inheritdoc/>
|
---|
| 268 | /// <remarks>Calls <see cref="OnVariableInfoAdded"/>.</remarks>
|
---|
[2] | 269 | public virtual bool TryAddVariableInfo(IVariableInfo variableInfo, out ICollection<IConstraint> violatedConstraints) {
|
---|
| 270 | myVariableInfos.Add(variableInfo.FormalName, variableInfo);
|
---|
| 271 | if (IsValid(out violatedConstraints)) {
|
---|
| 272 | OnVariableInfoAdded(variableInfo);
|
---|
| 273 | return true;
|
---|
| 274 | } else {
|
---|
| 275 | myVariableInfos.Remove(variableInfo.FormalName);
|
---|
| 276 | return false;
|
---|
| 277 | }
|
---|
| 278 | }
|
---|
[776] | 279 | /// <inheritdoc/>
|
---|
| 280 | /// <remarks>Calls <see cref="OnVariableInfoRemoved"/>.</remarks>
|
---|
[2] | 281 | public virtual void RemoveVariableInfo(string formalName) {
|
---|
| 282 | IVariableInfo variableInfo;
|
---|
| 283 | if (myVariableInfos.TryGetValue(formalName, out variableInfo)) {
|
---|
| 284 | myVariableInfos.Remove(formalName);
|
---|
| 285 | OnVariableInfoRemoved(variableInfo);
|
---|
| 286 | }
|
---|
| 287 | }
|
---|
[776] | 288 | /// <inheritdoc/>
|
---|
| 289 | /// <remarks>Calls <see cref="OnVariableInfoRemoved"/>.</remarks>
|
---|
[2] | 290 | public virtual bool TryRemoveVariableInfo(string formalName) {
|
---|
| 291 | IVariableInfo variableInfo;
|
---|
| 292 | if (myVariableInfos.TryGetValue(formalName, out variableInfo)) {
|
---|
| 293 | myVariableInfos.Remove(formalName);
|
---|
| 294 | if (IsValid()) {
|
---|
| 295 | OnVariableInfoRemoved(variableInfo);
|
---|
| 296 | return true;
|
---|
| 297 | } else {
|
---|
| 298 | myVariableInfos.Add(formalName, variableInfo);
|
---|
| 299 | return false;
|
---|
| 300 | }
|
---|
| 301 | }
|
---|
| 302 | return true;
|
---|
| 303 | }
|
---|
[776] | 304 | /// <inheritdoc/>
|
---|
| 305 | /// <remarks>Calls <see cref="OnVariableInfoRemoved"/>.</remarks>
|
---|
[2] | 306 | public virtual bool TryRemoveVariableInfo(string formalName, out ICollection<IConstraint> violatedConstraints) {
|
---|
| 307 | IVariableInfo variableInfo;
|
---|
| 308 | if (myVariableInfos.TryGetValue(formalName, out variableInfo)) {
|
---|
| 309 | myVariableInfos.Remove(formalName);
|
---|
| 310 | if (IsValid(out violatedConstraints)) {
|
---|
| 311 | OnVariableInfoRemoved(variableInfo);
|
---|
| 312 | return true;
|
---|
| 313 | } else {
|
---|
| 314 | myVariableInfos.Add(formalName, variableInfo);
|
---|
| 315 | return false;
|
---|
| 316 | }
|
---|
| 317 | }
|
---|
| 318 | violatedConstraints = new List<IConstraint>();
|
---|
| 319 | return true;
|
---|
| 320 | }
|
---|
| 321 | #endregion
|
---|
| 322 |
|
---|
| 323 | #region Variable Methods
|
---|
[776] | 324 | /// <inheritdoc/>
|
---|
[2] | 325 | public virtual IVariable GetVariable(string name) {
|
---|
| 326 | IVariable variable;
|
---|
| 327 | if (myVariables.TryGetValue(name, out variable))
|
---|
| 328 | return variable;
|
---|
| 329 | else
|
---|
| 330 | return null;
|
---|
| 331 | }
|
---|
[776] | 332 | /// <inheritdoc/>
|
---|
| 333 | /// <remarks>Calls <see cref="OnVariableAdded"/> and adds <c>NameChanging</c> and <c>NameChanged</c>
|
---|
| 334 | /// event handlers.</remarks>
|
---|
[2] | 335 | public virtual void AddVariable(IVariable variable) {
|
---|
| 336 | myVariables.Add(variable.Name, variable);
|
---|
| 337 | variable.NameChanging += new EventHandler<NameChangingEventArgs>(Variable_NameChanging);
|
---|
| 338 | variable.NameChanged += new EventHandler(Variable_NameChanged);
|
---|
| 339 | OnVariableAdded(variable);
|
---|
| 340 | }
|
---|
[776] | 341 | /// <inheritdoc/>
|
---|
| 342 | /// <remarks>Calls <see cref="OnVariableAdded"/> and adds <c>NameChanging</c> and <c>NameChanged</c>
|
---|
| 343 | /// event handlers.</remarks>
|
---|
[2] | 344 | public virtual bool TryAddVariable(IVariable variable) {
|
---|
| 345 | myVariables.Add(variable.Name, variable);
|
---|
| 346 | if (IsValid()) {
|
---|
| 347 | variable.NameChanging += new EventHandler<NameChangingEventArgs>(Variable_NameChanging);
|
---|
| 348 | variable.NameChanged += new EventHandler(Variable_NameChanged);
|
---|
| 349 | OnVariableAdded(variable);
|
---|
| 350 | return true;
|
---|
| 351 | } else {
|
---|
| 352 | myVariableInfos.Remove(variable.Name);
|
---|
| 353 | return false;
|
---|
| 354 | }
|
---|
| 355 | }
|
---|
[776] | 356 | /// <inheritdoc/>
|
---|
| 357 | /// <remarks>Calls <see cref="OnVariableAdded"/> and adds <c>NameChanging</c> and <c>NameChanged</c>
|
---|
| 358 | /// event handlers.</remarks>
|
---|
[2] | 359 | public virtual bool TryAddVariable(IVariable variable, out ICollection<IConstraint> violatedConstraints) {
|
---|
| 360 | myVariables.Add(variable.Name, variable);
|
---|
| 361 | if (IsValid(out violatedConstraints)) {
|
---|
| 362 | variable.NameChanging += new EventHandler<NameChangingEventArgs>(Variable_NameChanging);
|
---|
| 363 | variable.NameChanged += new EventHandler(Variable_NameChanged);
|
---|
| 364 | OnVariableAdded(variable);
|
---|
| 365 | return true;
|
---|
| 366 | } else {
|
---|
| 367 | myVariableInfos.Remove(variable.Name);
|
---|
| 368 | return false;
|
---|
| 369 | }
|
---|
| 370 | }
|
---|
[776] | 371 | /// <inheritdoc/>
|
---|
| 372 | /// <remarks>Calls <see cref="OnVariableRemoved"/> and removes <c>NameChanging</c> and <c>NameChanged</c>
|
---|
| 373 | /// event handlers.</remarks>
|
---|
[2] | 374 | public virtual void RemoveVariable(string name) {
|
---|
| 375 | IVariable variable;
|
---|
| 376 | if (myVariables.TryGetValue(name, out variable)) {
|
---|
| 377 | variable.NameChanging -= new EventHandler<NameChangingEventArgs>(Variable_NameChanging);
|
---|
| 378 | variable.NameChanged -= new EventHandler(Variable_NameChanged);
|
---|
| 379 | myVariables.Remove(name);
|
---|
| 380 | OnVariableRemoved(variable);
|
---|
| 381 | }
|
---|
| 382 | }
|
---|
[776] | 383 | /// <inheritdoc/>
|
---|
| 384 | /// <remarks>Calls <see cref="OnVariableRemoved"/> and removes <c>NameChanging</c> and <c>NameChanged</c>
|
---|
| 385 | /// event handlers.</remarks>
|
---|
[2] | 386 | public virtual bool TryRemoveVariable(string name) {
|
---|
| 387 | IVariable variable;
|
---|
| 388 | if (myVariables.TryGetValue(name, out variable)) {
|
---|
| 389 | myVariables.Remove(name);
|
---|
| 390 | if (IsValid()) {
|
---|
| 391 | variable.NameChanging -= new EventHandler<NameChangingEventArgs>(Variable_NameChanging);
|
---|
| 392 | variable.NameChanged -= new EventHandler(Variable_NameChanged);
|
---|
| 393 | OnVariableRemoved(variable);
|
---|
| 394 | return true;
|
---|
| 395 | } else {
|
---|
| 396 | myVariables.Add(name, variable);
|
---|
| 397 | return false;
|
---|
| 398 | }
|
---|
| 399 | }
|
---|
| 400 | return true;
|
---|
| 401 | }
|
---|
[776] | 402 | /// <inheritdoc/>
|
---|
| 403 | /// <remarks>Calls <see cref="OnVariableRemoved"/> and removes <c>NameChanging</c> and <c>NameChanged</c>
|
---|
| 404 | /// event handlers.</remarks>
|
---|
[2] | 405 | public virtual bool TryRemoveVariable(string name, out ICollection<IConstraint> violatedConstraints) {
|
---|
| 406 | IVariable variable;
|
---|
| 407 | if (myVariables.TryGetValue(name, out variable)) {
|
---|
| 408 | myVariables.Remove(name);
|
---|
| 409 | if (IsValid(out violatedConstraints)) {
|
---|
| 410 | variable.NameChanging -= new EventHandler<NameChangingEventArgs>(Variable_NameChanging);
|
---|
| 411 | variable.NameChanged -= new EventHandler(Variable_NameChanged);
|
---|
| 412 | OnVariableRemoved(variable);
|
---|
| 413 | return true;
|
---|
| 414 | } else {
|
---|
| 415 | myVariables.Add(name, variable);
|
---|
| 416 | return false;
|
---|
| 417 | }
|
---|
| 418 | }
|
---|
| 419 | violatedConstraints = new List<IConstraint>();
|
---|
| 420 | return true;
|
---|
| 421 | }
|
---|
| 422 | private void Variable_NameChanging(object sender, NameChangingEventArgs e) {
|
---|
| 423 | e.Cancel = myVariables.ContainsKey(e.Name);
|
---|
| 424 | }
|
---|
| 425 | private void Variable_NameChanged(object sender, EventArgs e) {
|
---|
| 426 | IVariable variable = (IVariable)sender;
|
---|
| 427 | string oldName = null;
|
---|
| 428 | foreach (KeyValuePair<string, IVariable> element in myVariables) {
|
---|
| 429 | if (element.Value == variable)
|
---|
| 430 | oldName = element.Key;
|
---|
| 431 | }
|
---|
| 432 | myVariables.Remove(oldName);
|
---|
| 433 | myVariables.Add(variable.Name, variable);
|
---|
| 434 | }
|
---|
[776] | 435 | /// <inheritdoc cref="IOperator.GetVariableValue<T>(string, HeuristicLab.Core.IScope, bool)"/>
|
---|
| 436 | /// <remarks>Calls <see cref="GetVariableValue<T>(string, HeuristicLab.Core.IScope, bool, bool)"/>
|
---|
| 437 | /// with <c>throwOnError</c> set to <c>false</c>.</remarks>
|
---|
[2] | 438 | public T GetVariableValue<T>(string formalName, IScope scope, bool recursiveLookup) where T : class, IItem {
|
---|
| 439 | return GetVariableValue<T>(formalName, scope, recursiveLookup, true);
|
---|
| 440 | }
|
---|
[776] | 441 | /// <inheritdoc cref="IOperator.GetVariableValue<T>(string, HeuristicLab.Core.IScope, bool, bool)"/>
|
---|
| 442 | /// <remarks>Calls
|
---|
| 443 | /// <see cref="GetVariableValue(string, HeuristicLab.Core.IScope, bool, bool)"/>.</remarks>
|
---|
[2] | 444 | public T GetVariableValue<T>(string formalName, IScope scope, bool recursiveLookup, bool throwOnError) where T : class, IItem {
|
---|
| 445 | return (T)GetVariableValue(formalName, scope, recursiveLookup, throwOnError);
|
---|
| 446 | }
|
---|
[776] | 447 | /// <inheritdoc cref="IOperator.GetVariableValue(string, HeuristicLab.Core.IScope, bool)"/>
|
---|
| 448 | /// <remarks>Calls <see cref="GetVariableValue(string, HeuristicLab.Core.IScope, bool, bool)"/>
|
---|
| 449 | /// with <c>throwOnError</c> set to <c>false</c>.</remarks>
|
---|
[2] | 450 | public IItem GetVariableValue(string formalName, IScope scope, bool recursiveLookup) {
|
---|
| 451 | return GetVariableValue(formalName, scope, recursiveLookup, true);
|
---|
| 452 | }
|
---|
[776] | 453 | /// <inheritdoc cref="IOperator.GetVariableValue(string, HeuristicLab.Core.IScope, bool, bool)"/>
|
---|
[2] | 454 | public virtual IItem GetVariableValue(string formalName, IScope scope, bool recursiveLookup, bool throwOnError) {
|
---|
| 455 | IVariableInfo info = GetVariableInfo(formalName);
|
---|
| 456 | if (info.Local) {
|
---|
| 457 | IVariable variable;
|
---|
| 458 | if (myVariables.TryGetValue(info.ActualName, out variable))
|
---|
| 459 | return variable.Value;
|
---|
| 460 | else {
|
---|
| 461 | if (throwOnError)
|
---|
| 462 | throw new ArgumentException("Variable " + info.ActualName + " not found");
|
---|
| 463 | else
|
---|
| 464 | return null;
|
---|
| 465 | }
|
---|
| 466 | } else {
|
---|
[61] | 467 | return scope.GetVariableValue(formalName, recursiveLookup, throwOnError);
|
---|
[2] | 468 | }
|
---|
| 469 | }
|
---|
| 470 | #endregion
|
---|
[776] | 471 | /// <inheritdoc/>
|
---|
[2] | 472 | public virtual IOperation Execute(IScope scope) {
|
---|
| 473 | myCanceled = false;
|
---|
[61] | 474 |
|
---|
| 475 | foreach (IVariableInfo variableInfo in VariableInfos)
|
---|
| 476 | scope.AddAlias(variableInfo.FormalName, variableInfo.ActualName);
|
---|
| 477 |
|
---|
[2] | 478 | IOperation next = Apply(scope);
|
---|
[76] | 479 |
|
---|
| 480 | foreach (IVariableInfo variableInfo in VariableInfos)
|
---|
| 481 | scope.RemoveAlias(variableInfo.FormalName);
|
---|
| 482 |
|
---|
[2] | 483 | OnExecuted();
|
---|
| 484 | return next;
|
---|
| 485 | }
|
---|
[776] | 486 | /// <inheritdoc/>
|
---|
| 487 | /// <remarks>Sets property <see cref="Canceled"/> to <c>true</c>.</remarks>
|
---|
[2] | 488 | public virtual void Abort() {
|
---|
| 489 | myCanceled = true;
|
---|
| 490 | }
|
---|
[776] | 491 | /// <summary>
|
---|
| 492 | /// Performs the current operator on the specified <paramref name="scope"/>.
|
---|
| 493 | /// </summary>
|
---|
| 494 | /// <param name="scope">The scope where to execute the operator</param>
|
---|
| 495 | /// <returns><c>null</c>.</returns>
|
---|
[2] | 496 | public virtual IOperation Apply(IScope scope) {
|
---|
| 497 | return null;
|
---|
| 498 | }
|
---|
[776] | 499 | /// <inheritdoc/>
|
---|
[2] | 500 | public event EventHandler NameChanged;
|
---|
[776] | 501 | /// <summary>
|
---|
| 502 | /// Fires a new <c>NameChanged</c> event.
|
---|
| 503 | /// </summary>
|
---|
[2] | 504 | protected virtual void OnNameChanged() {
|
---|
| 505 | if (NameChanged != null) {
|
---|
| 506 | NameChanged(this, new EventArgs());
|
---|
| 507 | }
|
---|
| 508 | }
|
---|
[776] | 509 | /// <inheritdoc/>
|
---|
[2] | 510 | public event EventHandler BreakpointChanged;
|
---|
[776] | 511 | /// <summary>
|
---|
| 512 | /// Fires a new <c>BreakpointChanged</c> event.
|
---|
| 513 | /// </summary>
|
---|
[2] | 514 | protected virtual void OnBreakpointChanged() {
|
---|
| 515 | if (BreakpointChanged != null) {
|
---|
| 516 | BreakpointChanged(this, new EventArgs());
|
---|
| 517 | }
|
---|
| 518 | }
|
---|
[776] | 519 | /// <inheritdoc/>
|
---|
[2] | 520 | public event EventHandler<OperatorIndexEventArgs> SubOperatorAdded;
|
---|
[776] | 521 | /// <summary>
|
---|
| 522 | /// Fires a new <c>SubOperatorAdded</c> event.
|
---|
| 523 | /// </summary>
|
---|
| 524 | /// <param name="subOperator">The sub operator that has been added.</param>
|
---|
| 525 | /// <param name="index">The position where the operator has been added.</param>
|
---|
[2] | 526 | protected virtual void OnSubOperatorAdded(IOperator subOperator, int index) {
|
---|
| 527 | if (SubOperatorAdded != null)
|
---|
| 528 | SubOperatorAdded(this, new OperatorIndexEventArgs(subOperator, index));
|
---|
| 529 | }
|
---|
[776] | 530 | /// <inheritdoc/>
|
---|
[2] | 531 | public event EventHandler<OperatorIndexEventArgs> SubOperatorRemoved;
|
---|
[776] | 532 | /// <summary>
|
---|
| 533 | /// Fires a new <c>SubOperatorRemoved</c> event.
|
---|
| 534 | /// </summary>
|
---|
| 535 | /// <param name="subOperator">The sub operator that has been removed.</param>
|
---|
| 536 | /// <param name="index">The position where the operator has been removed.</param>
|
---|
[2] | 537 | protected virtual void OnSubOperatorRemoved(IOperator subOperator, int index) {
|
---|
| 538 | if (SubOperatorRemoved != null)
|
---|
| 539 | SubOperatorRemoved(this, new OperatorIndexEventArgs(subOperator, index));
|
---|
| 540 | }
|
---|
[776] | 541 | /// <inheritdoc/>
|
---|
[2] | 542 | public event EventHandler<VariableInfoEventArgs> VariableInfoAdded;
|
---|
[776] | 543 | /// <summary>
|
---|
| 544 | /// Fires a new <c>VariableInfoAdded</c> event.
|
---|
| 545 | /// </summary>
|
---|
| 546 | /// <param name="variableInfo">The variable info that has been added.</param>
|
---|
[2] | 547 | protected virtual void OnVariableInfoAdded(IVariableInfo variableInfo) {
|
---|
| 548 | if (VariableInfoAdded != null)
|
---|
| 549 | VariableInfoAdded(this, new VariableInfoEventArgs(variableInfo));
|
---|
| 550 | }
|
---|
[776] | 551 | /// <inheritdoc/>
|
---|
[2] | 552 | public event EventHandler<VariableInfoEventArgs> VariableInfoRemoved;
|
---|
[776] | 553 | /// <summary>
|
---|
| 554 | /// Fires a new <c>VariableInfoRemoved</c> event.
|
---|
| 555 | /// </summary>
|
---|
| 556 | /// <param name="variableInfo">The variable info that has been removed.</param>
|
---|
[2] | 557 | protected virtual void OnVariableInfoRemoved(IVariableInfo variableInfo) {
|
---|
| 558 | if (VariableInfoRemoved != null)
|
---|
| 559 | VariableInfoRemoved(this, new VariableInfoEventArgs(variableInfo));
|
---|
| 560 | }
|
---|
[776] | 561 | /// <inheritdoc/>
|
---|
[2] | 562 | public event EventHandler<VariableEventArgs> VariableAdded;
|
---|
[776] | 563 | /// <summary>
|
---|
| 564 | /// Fires a new <c>VariableAdded</c> event.
|
---|
| 565 | /// </summary>
|
---|
| 566 | /// <param name="variable">The variable that has been added.</param>
|
---|
[2] | 567 | protected virtual void OnVariableAdded(IVariable variable) {
|
---|
| 568 | if (VariableAdded != null)
|
---|
| 569 | VariableAdded(this, new VariableEventArgs(variable));
|
---|
| 570 | }
|
---|
[776] | 571 | /// <inheritdoc/>
|
---|
[2] | 572 | public event EventHandler<VariableEventArgs> VariableRemoved;
|
---|
[776] | 573 | /// <summary>
|
---|
| 574 | /// Fires a new <c>VariableRemoved</c> event.
|
---|
| 575 | /// </summary>
|
---|
| 576 | /// <param name="variable">The variable that has been removed</param>
|
---|
[2] | 577 | protected virtual void OnVariableRemoved(IVariable variable) {
|
---|
| 578 | if (VariableRemoved != null)
|
---|
| 579 | VariableRemoved(this, new VariableEventArgs(variable));
|
---|
| 580 | }
|
---|
[776] | 581 | /// <inheritdoc/>
|
---|
[2] | 582 | public event EventHandler Executed;
|
---|
[776] | 583 | /// <summary>
|
---|
| 584 | /// Fires a new <c>Executed</c> event.
|
---|
| 585 | /// </summary>
|
---|
[2] | 586 | protected virtual void OnExecuted() {
|
---|
| 587 | if (Executed != null) {
|
---|
| 588 | Executed(this, new EventArgs());
|
---|
| 589 | }
|
---|
| 590 | }
|
---|
| 591 |
|
---|
| 592 | #region Persistence Methods
|
---|
[776] | 593 | /// <summary>
|
---|
| 594 | /// Saves the current instance as <see cref="XmlNode"/> in the specified <paramref name="document"/>.
|
---|
| 595 | /// </summary>
|
---|
| 596 | /// <remarks>
|
---|
| 597 | /// Calls <see cref="ConstrainedItemBase.GetXmlNode"/> of base class <see cref="ConstrainedItemBase"/>.
|
---|
| 598 | /// <br/>A quick overview how the single elements of the current instance are saved:
|
---|
| 599 | /// <list type="bullet">
|
---|
| 600 | /// <item>
|
---|
| 601 | /// <term>Name: </term>
|
---|
| 602 | /// <description>Saved as an <see cref="XmlAttribute"/> with the name <c>Name</c>.</description>
|
---|
| 603 | /// </item>
|
---|
| 604 | /// <item>
|
---|
| 605 | /// <term>Breakpoint: </term>
|
---|
| 606 | /// <description>Is only saved if it set to <c>true</c>.
|
---|
| 607 | /// Saved as an <see cref="XmlAttribute"/> with the name <c>Breakpoint</c>.</description>
|
---|
| 608 | /// </item>
|
---|
| 609 | /// <item>
|
---|
| 610 | /// <term>Sub operators: </term>
|
---|
| 611 | /// <description>Saved as child node with tag name <c>SubOperators</c>. All sub operators are themselves
|
---|
| 612 | /// saved as child nodes.</description>
|
---|
| 613 | /// </item>
|
---|
| 614 | /// <item>
|
---|
| 615 | /// <term>Variable infos: </term>
|
---|
| 616 | /// <description>Saved as child node with tag name <c>VariableInfos</c>. All variable infos are themselves
|
---|
| 617 | /// saved as child nodes.</description>
|
---|
| 618 | /// </item>
|
---|
| 619 | /// <item>
|
---|
| 620 | /// <term>Variables: </term>
|
---|
| 621 | /// <description>Saved as child node with tag name <c>Variables</c>. All variables are themselves
|
---|
| 622 | /// saved as child nodes.</description>
|
---|
| 623 | /// </item>
|
---|
| 624 | /// </list>
|
---|
| 625 | /// </remarks>
|
---|
| 626 | /// <param name="name">The (tag)name of the <see cref="XmlNode"/>.</param>
|
---|
| 627 | /// <param name="document">The <see cref="XmlDocument"/> where to save the data.</param>
|
---|
| 628 | /// <param name="persistedObjects">The dictionary of all already persisted objects. (Needed to avoid cycles.)</param>
|
---|
| 629 | /// <returns>The saved <see cref="XmlNode"/>.</returns>
|
---|
[2] | 630 | public override XmlNode GetXmlNode(string name, XmlDocument document, IDictionary<Guid,IStorable> persistedObjects) {
|
---|
| 631 | XmlNode node = base.GetXmlNode(name, document, persistedObjects);
|
---|
| 632 | XmlAttribute nameAttribute = document.CreateAttribute("Name");
|
---|
| 633 | nameAttribute.Value = Name;
|
---|
| 634 | node.Attributes.Append(nameAttribute);
|
---|
| 635 | if (Breakpoint) {
|
---|
| 636 | XmlAttribute breakpointAttribute = document.CreateAttribute("Breakpoint");
|
---|
| 637 | breakpointAttribute.Value = Breakpoint.ToString();
|
---|
| 638 | node.Attributes.Append(breakpointAttribute);
|
---|
| 639 | }
|
---|
| 640 | XmlNode subOperatorsNode = document.CreateNode(XmlNodeType.Element, "SubOperators", null);
|
---|
| 641 | for (int i = 0; i < SubOperators.Count; i++)
|
---|
| 642 | subOperatorsNode.AppendChild(PersistenceManager.Persist(SubOperators[i], document, persistedObjects));
|
---|
| 643 | node.AppendChild(subOperatorsNode);
|
---|
| 644 | XmlNode infosNode = document.CreateNode(XmlNodeType.Element, "VariableInfos", null);
|
---|
| 645 | foreach (IVariableInfo info in myVariableInfos.Values)
|
---|
| 646 | infosNode.AppendChild(PersistenceManager.Persist(info, document, persistedObjects));
|
---|
| 647 | node.AppendChild(infosNode);
|
---|
| 648 | XmlNode variablesNode = document.CreateNode(XmlNodeType.Element, "Variables", null);
|
---|
| 649 | foreach (IVariable variable in myVariables.Values)
|
---|
| 650 | variablesNode.AppendChild(PersistenceManager.Persist(variable, document, persistedObjects));
|
---|
| 651 | node.AppendChild(variablesNode);
|
---|
| 652 | return node;
|
---|
| 653 | }
|
---|
[776] | 654 | /// <summary>
|
---|
| 655 | /// Loads the persisted operation from the specified <paramref name="node"/>.
|
---|
| 656 | /// </summary>
|
---|
| 657 | /// <remarks>Calls <see cref="ConstrainedItemBase.Populate"/> of base class
|
---|
| 658 | /// <see cref="ConstrainedItemBase"/>.
|
---|
| 659 | /// For informations how the different elements must be saved please see <see cref="GetXmlNode"/>.</remarks>
|
---|
| 660 | /// <param name="node">The <see cref="XmlNode"/> where the operation is saved.</param>
|
---|
| 661 | /// <param name="restoredObjects">A dictionary of all already restored objects. (Needed to avoid cycles.)</param>
|
---|
[2] | 662 | public override void Populate(XmlNode node, IDictionary<Guid,IStorable> restoredObjects) {
|
---|
| 663 | base.Populate(node, restoredObjects);
|
---|
| 664 | myName = node.Attributes["Name"].Value;
|
---|
| 665 | if (node.Attributes["Breakpoint"] != null)
|
---|
| 666 | myBreakpoint = bool.Parse(node.Attributes["Breakpoint"].Value);
|
---|
| 667 | XmlNode subOperatorsNode = node.SelectSingleNode("SubOperators");
|
---|
| 668 | for (int i = 0; i < subOperatorsNode.ChildNodes.Count; i++)
|
---|
| 669 | AddSubOperator((IOperator)PersistenceManager.Restore(subOperatorsNode.ChildNodes[i], restoredObjects));
|
---|
| 670 | XmlNode infosNode = node.SelectSingleNode("VariableInfos");
|
---|
| 671 | myVariableInfos.Clear();
|
---|
| 672 | foreach (XmlNode infoNode in infosNode.ChildNodes)
|
---|
| 673 | AddVariableInfo((IVariableInfo)PersistenceManager.Restore(infoNode, restoredObjects));
|
---|
| 674 | XmlNode variablesNode = node.SelectSingleNode("Variables");
|
---|
| 675 | myVariables.Clear();
|
---|
| 676 | foreach (XmlNode variableNode in variablesNode.ChildNodes)
|
---|
| 677 | AddVariable((IVariable)PersistenceManager.Restore(variableNode, restoredObjects));
|
---|
| 678 | }
|
---|
| 679 | #endregion
|
---|
| 680 | }
|
---|
| 681 | }
|
---|