#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 } }