#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.Collections.Generic; using System.ComponentModel; using System.Linq; using HeuristicLab.Collections; using HeuristicLab.Common; using HeuristicLab.Core; using HEAL.Attic; namespace HeuristicLab.Analysis { [Item("IndexedDataRow", "A data row that contains a series of points.")] [StorableType("0B0BB900-4C30-4485-82C2-C9E633110685")] public class IndexedDataRow : NamedItem, IDataRow { private DataRowVisualProperties visualProperties; public DataRowVisualProperties VisualProperties { get { return visualProperties; } set { if (visualProperties != value) { if (value == null) throw new ArgumentNullException("VisualProperties"); if (visualProperties != null) visualProperties.PropertyChanged -= new PropertyChangedEventHandler(VisualProperties_PropertyChanged); visualProperties = value; visualProperties.PropertyChanged += new PropertyChangedEventHandler(VisualProperties_PropertyChanged); OnVisualPropertiesChanged(); } } } private ObservableList> values; public ObservableList> Values { get { return values; } } #region Persistence Properties [Storable(Name = "visualProperties")] private DataRowVisualProperties StorableVisualProperties { get { return visualProperties; } set { visualProperties = value; } } // BackwardsCompatibility3.3 #region Backwards compatible code, remove with 3.4 // tuples are stored inefficiently [Storable(Name = "values", AllowOneWay = true)] private IEnumerable> StorableValues { set { values = new ObservableList>(value); } } #endregion private T[] storableX; [Storable(Name = "x")] private T[] StorableX { get { return Values.Select(x => x.Item1).ToArray(); } set { storableX = value; } } private double[] storableY; [Storable(Name = "y")] private double[] StorableY { get { return Values.Select(x => x.Item2).ToArray(); } set { storableY = value; } } #endregion [StorableConstructor] protected IndexedDataRow(StorableConstructorFlag _) : base(_) { } protected IndexedDataRow(IndexedDataRow original, Cloner cloner) : base(original, cloner) { values = new ObservableList>(original.values.Select(x => Tuple.Create(x.Item1, x.Item2)).ToList()); VisualProperties = cloner.Clone(original.visualProperties); } public IndexedDataRow() { values = new ObservableList>(); VisualProperties = new DataRowVisualProperties(); } public IndexedDataRow(string name) : base(name) { values = new ObservableList>(); VisualProperties = new DataRowVisualProperties(name); } public IndexedDataRow(string name, string description) : base(name, description) { values = new ObservableList>(); VisualProperties = new DataRowVisualProperties(name); } public IndexedDataRow(string name, string description, IEnumerable> values) : base(name, description) { this.values = new ObservableList>(values); VisualProperties = new DataRowVisualProperties(name); } public override IDeepCloneable Clone(Cloner cloner) { return new IndexedDataRow(this, cloner); } public event EventHandler VisualPropertiesChanged; protected virtual void OnVisualPropertiesChanged() { EventHandler handler = VisualPropertiesChanged; if (handler != null) handler(this, EventArgs.Empty); } private void VisualProperties_PropertyChanged(object sender, PropertyChangedEventArgs e) { OnVisualPropertiesChanged(); } protected override void OnNameChanged() { base.OnNameChanged(); VisualProperties.DisplayName = Name; } [StorableHook(HookType.AfterDeserialization)] private void AfterDeserialization() { if (storableX != null && storableY != null) { values = new ObservableList>(storableX.Zip(storableY, (x, y) => Tuple.Create(x, y))); storableX = null; storableY = null; } else if (values == null) throw new InvalidOperationException("Deserialization problem with IndexedDataRow."); } } }