#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.Linq;
using HeuristicLab.Collections;
using HeuristicLab.Common;
using HeuristicLab.Core;
using HeuristicLab.Persistence;
namespace HeuristicLab.DebugEngine {
[StorableType("049f12b3-6bd2-4c47-bbbd-b05083d34d0a")]
public class OperatorTrace : ObservableList, IContent, IDeepCloneable {
#region fields
[Storable]
protected Dictionary parents;
[Storable]
protected bool isEnabled;
#endregion
#region events
public event EventHandler IsEnabledChanged;
protected virtual void OnIsEnabledChanged() {
EventHandler handler = IsEnabledChanged;
if (handler != null)
handler(this, EventArgs.Empty);
}
#endregion
#region Constructors & Cloning
public OperatorTrace() {
parents = new Dictionary();
}
public OperatorTrace(int capacity)
: base(capacity) {
parents = new Dictionary();
}
public OperatorTrace(IEnumerable collection)
: base(collection) {
parents = new Dictionary();
}
[StorableConstructor]
protected OperatorTrace(StorableConstructorFlag deserializing) : base(deserializing) { }
protected OperatorTrace(OperatorTrace original, Cloner cloner) {
cloner.RegisterClonedObject(original, this);
AddRange(original.Select(op => cloner.Clone(op)));
parents = original.parents.ToDictionary(kvp => cloner.Clone(kvp.Key), kvp => cloner.Clone(kvp.Value));
}
public object Clone() {
return Clone(new Cloner());
}
public virtual IDeepCloneable Clone(Cloner cloner) {
return new OperatorTrace(this, cloner);
}
#endregion
#region Additional List Modifiers
public virtual void ReplaceAll(IEnumerable operators) {
var oldList = list;
list = new List(operators);
if (oldList.Count != list.Count)
OnPropertyChanged("Count");
OnPropertyChanged("Item[]");
OnCollectionReset(
list.Select((op, i) => new IndexedItem(i, op)),
oldList.Select((op, i) => new IndexedItem(i, op)));
}
#endregion
#region Parent Tracing
public virtual void RegisterParenthood(IAtomicOperation parent, IOperation children) {
if (!isEnabled)
return;
OperationCollection operations = children as OperationCollection;
if (operations != null)
foreach (var op in operations)
RegisterParenthood(parent, op);
IAtomicOperation atomicOperation = children as IAtomicOperation;
if (atomicOperation != null && atomicOperation.Operator != null && !parents.ContainsKey(atomicOperation))
parents[atomicOperation] = parent;
}
public virtual void Reset() {
Clear();
parents.Clear();
}
public virtual void Regenerate(IAtomicOperation operation) {
if (!isEnabled) {
Reset();
return;
}
if (operation == null)
return;
Stack trace = new Stack();
while (operation != null) {
trace.Push(operation.Operator);
IAtomicOperation parent = null;
parents.TryGetValue(operation, out parent);
operation = parent;
}
ReplaceAll(trace);
}
public bool IsEnabled {
get { return isEnabled; }
set {
if (isEnabled == value)
return;
isEnabled = value;
if (!isEnabled)
Reset();
OnIsEnabledChanged();
}
}
#endregion
}
}