Free cookie consent management tool by TermsFeed Policy Generator

source: trunk/HeuristicLab.Data/3.3/StringMatrix.cs @ 16713

Last change on this file since 16713 was 16565, checked in by gkronber, 6 years ago

#2520: merged changes from PersistenceOverhaul branch (r16451:16564) into trunk

File size: 12.0 KB
RevLine 
[2694]1#region License Information
2/* HeuristicLab
[16565]3 * Copyright (C) 2002-2019 Heuristic and Evolutionary Algorithms Laboratory (HEAL)
[2694]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
22using System;
23using System.Collections;
[3308]24using System.Collections.Generic;
[3306]25using System.Drawing;
[3308]26using System.Linq;
[2694]27using System.Text;
28using HeuristicLab.Common;
29using HeuristicLab.Core;
[16565]30using HEAL.Attic;
[2694]31
32namespace HeuristicLab.Data {
[3048]33  [Item("StringMatrix", "Represents a matrix of strings.")]
[16565]34  [StorableType("0BFE5727-D2CF-418C-94BE-A8A0BBA195D8")]
[3430]35  public class StringMatrix : Item, IEnumerable<string>, IStringConvertibleMatrix {
[9433]36    private const int maximumToStringLength = 100;
37
[7201]38    public static new Image StaticItemImage {
[5287]39      get { return HeuristicLab.Common.Resources.VSImageLibrary.Class; }
[3306]40    }
41
[2694]42    [Storable]
[3054]43    protected string[,] matrix;
[2694]44
[3308]45    [Storable]
[3430]46    protected List<string> columnNames;
47    public virtual IEnumerable<string> ColumnNames {
[3308]48      get { return this.columnNames; }
49      set {
[3430]50        if (ReadOnly) throw new NotSupportedException("ColumnNames cannot be set. StringMatrix is read-only.");
[3308]51        if (value == null || value.Count() == 0)
52          columnNames = new List<string>();
53        else if (value.Count() != Columns)
[3310]54          throw new ArgumentException("A column name must be specified for each column .");
[3308]55        else
56          columnNames = new List<string>(value);
[5150]57        OnColumnNamesChanged();
[3308]58      }
59    }
[3310]60    [Storable]
[3430]61    protected List<string> rowNames;
62    public virtual IEnumerable<string> RowNames {
[3310]63      get { return this.rowNames; }
64      set {
[3430]65        if (ReadOnly) throw new NotSupportedException("RowNames cannot be set. StringMatrix is read-only.");
[3310]66        if (value == null || value.Count() == 0)
67          rowNames = new List<string>();
68        else if (value.Count() != Rows)
69          throw new ArgumentException("A row name must be specified for each row.");
70        else
71          rowNames = new List<string>(value);
[5150]72        OnRowNamesChanged();
[3310]73      }
74    }
[3320]75    [Storable]
[3430]76    protected bool sortableView;
77    public virtual bool SortableView {
[3320]78      get { return sortableView; }
79      set {
[3430]80        if (ReadOnly) throw new NotSupportedException("SortableView cannot be set. StringMatrix is read-only.");
[3320]81        if (value != sortableView) {
82          sortableView = value;
83          OnSortableViewChanged();
84        }
85      }
86    }
[3308]87
[3054]88    public virtual int Rows {
89      get { return matrix.GetLength(0); }
90      protected set {
[3430]91        if (ReadOnly) throw new NotSupportedException("Rows cannot be set. StringMatrix is read-only.");
[2694]92        if (value != Rows) {
[3054]93          string[,] newMatrix = new string[value, Columns];
94          Array.Copy(matrix, newMatrix, Math.Min(value * Columns, matrix.Length));
95          matrix = newMatrix;
[3320]96          while (rowNames.Count > value)
97            rowNames.RemoveAt(rowNames.Count - 1);
98          while (rowNames.Count < value)
99            rowNames.Add("Row " + rowNames.Count);
[5150]100          OnRowsChanged();
101          OnRowNamesChanged();
[2694]102          OnReset();
103        }
104      }
105    }
[3054]106    public virtual int Columns {
107      get { return matrix.GetLength(1); }
108      protected set {
[3430]109        if (ReadOnly) throw new NotSupportedException("Columns cannot be set. StringMatrix is read-only.");
[2694]110        if (value != Columns) {
[3054]111          string[,] newMatrix = new string[Rows, value];
[2694]112          for (int i = 0; i < Rows; i++)
[3054]113            Array.Copy(matrix, i * Columns, newMatrix, i * value, Math.Min(value, Columns));
114          matrix = newMatrix;
[3320]115          while (columnNames.Count > value)
116            columnNames.RemoveAt(columnNames.Count - 1);
117          while (columnNames.Count < value)
118            columnNames.Add("Column " + columnNames.Count);
[5150]119          OnColumnsChanged();
120          OnColumnNamesChanged();
[2694]121          OnReset();
122        }
123      }
124    }
[3054]125    public virtual string this[int rowIndex, int columnIndex] {
126      get { return matrix[rowIndex, columnIndex]; }
[2694]127      set {
[3430]128        if (ReadOnly) throw new NotSupportedException("Item cannot be set. StringMatrix is read-only.");
[3054]129        if (value != matrix[rowIndex, columnIndex]) {
130          if ((value != null) || (matrix[rowIndex, columnIndex] != string.Empty)) {
131            matrix[rowIndex, columnIndex] = value != null ? value : string.Empty;
[2694]132            OnItemChanged(rowIndex, columnIndex);
133          }
134        }
135      }
136    }
137
[3430]138    [Storable]
139    protected bool readOnly;
140    public virtual bool ReadOnly {
141      get { return readOnly; }
142    }
143
[4722]144    [StorableConstructor]
[16565]145    protected StringMatrix(StorableConstructorFlag _) : base(_) { }
[4722]146    protected StringMatrix(StringMatrix original, Cloner cloner)
147      : base(original, cloner) {
148      this.matrix = (string[,])original.matrix.Clone();
149      this.columnNames = new List<string>(original.columnNames);
150      this.rowNames = new List<string>(original.rowNames);
151      this.sortableView = original.sortableView;
152      this.readOnly = original.readOnly;
153    }
[3048]154    public StringMatrix() {
[3054]155      matrix = new string[0, 0];
[3308]156      columnNames = new List<string>();
[3310]157      rowNames = new List<string>();
[3320]158      sortableView = false;
[3430]159      readOnly = false;
[2694]160    }
[3048]161    public StringMatrix(int rows, int columns) {
[3054]162      matrix = new string[rows, columns];
163      for (int i = 0; i < matrix.GetLength(0); i++) {
164        for (int j = 0; j < matrix.GetLength(1); j++)
165          matrix[i, j] = string.Empty;
[2694]166      }
[3308]167      columnNames = new List<string>();
[3310]168      rowNames = new List<string>();
[3320]169      sortableView = false;
[3430]170      readOnly = false;
[2694]171    }
[3308]172    protected StringMatrix(int rows, int columns, IEnumerable<string> columnNames)
173      : this(rows, columns) {
174      ColumnNames = columnNames;
175    }
[3310]176    protected StringMatrix(int rows, int columns, IEnumerable<string> columnNames, IEnumerable<string> rowNames)
[5150]177      : this(rows, columns, columnNames) {
[3310]178      RowNames = rowNames;
179    }
[3048]180    public StringMatrix(string[,] elements) {
[2694]181      if (elements == null) throw new ArgumentNullException();
[3054]182      matrix = new string[elements.GetLength(0), elements.GetLength(1)];
183      for (int i = 0; i < matrix.GetLength(0); i++) {
184        for (int j = 0; j < matrix.GetLength(1); j++)
185          matrix[i, j] = elements[i, j] == null ? string.Empty : elements[i, j];
[2694]186      }
[3308]187      columnNames = new List<string>();
[3310]188      rowNames = new List<string>();
[3320]189      sortableView = false;
[3430]190      readOnly = false;
[2694]191    }
[3308]192    protected StringMatrix(string[,] elements, IEnumerable<string> columnNames)
193      : this(elements) {
194      ColumnNames = columnNames;
195    }
[5150]196    protected StringMatrix(string[,] elements, IEnumerable<string> columnNames, IEnumerable<string> rowNames)
197      : this(elements, columnNames) {
[3310]198      RowNames = rowNames;
199    }
[2694]200
201    public override IDeepCloneable Clone(Cloner cloner) {
[4722]202      return new StringMatrix(this, cloner);
[2694]203    }
204
[3430]205    public virtual StringMatrix AsReadOnly() {
206      StringMatrix readOnlyStringMatrix = (StringMatrix)this.Clone();
207      readOnlyStringMatrix.readOnly = true;
208      return readOnlyStringMatrix;
209    }
210
[16280]211    public string[,] CloneAsMatrix() {
212      return (string[,])matrix.Clone();
213    }
214
215    public virtual IEnumerable<string> GetRow(int row) {
216      for (var col = 0; col < Columns; col++) {
217        yield return matrix[row, col];
218      }
219    }
220
221    public virtual IEnumerable<string> GetColumn(int col) {
222      for (var row = 0; row < Rows; row++) {
223        yield return matrix[row, col];
224      }
225    }
226
[2694]227    public override string ToString() {
[9433]228      if (matrix.Length == 0) return "[]";
229
[2694]230      StringBuilder sb = new StringBuilder();
231      sb.Append("[");
[9433]232      for (int i = 0; i < Rows; i++) {
233        sb.Append("[").Append(matrix[i, 0]);
234        for (int j = 1; j < Columns; j++)
235          sb.Append(";").Append(matrix[i, j]);
236        sb.Append("]");
237
238        if (sb.Length > maximumToStringLength) {
239          sb.Append("[...]");
240          break;
[2694]241        }
242      }
243      sb.Append("]");
[9433]244
[2694]245      return sb.ToString();
246    }
247
[3430]248    public virtual IEnumerator<string> GetEnumerator() {
249      return matrix.Cast<string>().GetEnumerator();
[2694]250    }
251
[3430]252    IEnumerator IEnumerable.GetEnumerator() {
253      return GetEnumerator();
254    }
255
[3054]256    protected virtual bool Validate(string value, out string errorMessage) {
[2694]257      if (value == null) {
258        errorMessage = "Invalid Value (string must not be null)";
259        return false;
260      } else {
261        errorMessage = string.Empty;
262        return true;
263      }
264    }
[3054]265    protected virtual string GetValue(int rowIndex, int columIndex) {
[2694]266      return this[rowIndex, columIndex];
267    }
[3054]268    protected virtual bool SetValue(string value, int rowIndex, int columnIndex) {
[2694]269      if (value != null) {
270        this[rowIndex, columnIndex] = value;
271        return true;
272      } else {
273        return false;
274      }
275    }
[3054]276
[5150]277    #region events
278    public event EventHandler ColumnsChanged;
279    protected virtual void OnColumnsChanged() {
280      EventHandler handler = ColumnsChanged;
281      if (handler != null)
282        handler(this, EventArgs.Empty);
283    }
284    public event EventHandler RowsChanged;
285    protected virtual void OnRowsChanged() {
286      EventHandler handler = RowsChanged;
287      if (handler != null)
288        handler(this, EventArgs.Empty);
289    }
[3320]290    public event EventHandler ColumnNamesChanged;
291    protected virtual void OnColumnNamesChanged() {
292      EventHandler handler = ColumnNamesChanged;
293      if (handler != null)
294        handler(this, EventArgs.Empty);
295    }
296    public event EventHandler RowNamesChanged;
297    protected virtual void OnRowNamesChanged() {
298      EventHandler handler = RowNamesChanged;
299      if (handler != null)
300        handler(this, EventArgs.Empty);
301    }
302    public event EventHandler SortableViewChanged;
303    protected virtual void OnSortableViewChanged() {
304      EventHandler handler = SortableViewChanged;
305      if (handler != null)
306        handler(this, EventArgs.Empty);
307    }
[2973]308    public event EventHandler<EventArgs<int, int>> ItemChanged;
[3054]309    protected virtual void OnItemChanged(int rowIndex, int columnIndex) {
[2694]310      if (ItemChanged != null)
311        ItemChanged(this, new EventArgs<int, int>(rowIndex, columnIndex));
[9433]312
313      //approximation to avoid firing of unnecessary ToStringChangedEvents
314      //columnIndex is not used, because always full rows are returned in the ToString method
315      if (rowIndex * Columns < maximumToStringLength)
316        OnToStringChanged();
[2694]317    }
[2973]318    public event EventHandler Reset;
[3054]319    protected virtual void OnReset() {
[2694]320      if (Reset != null)
321        Reset(this, EventArgs.Empty);
[2932]322      OnToStringChanged();
[2694]323    }
[5150]324    #endregion
[3054]325
326    #region IStringConvertibleMatrix Members
327    int IStringConvertibleMatrix.Rows {
328      get { return Rows; }
329      set { Rows = value; }
330    }
331    int IStringConvertibleMatrix.Columns {
332      get { return Columns; }
333      set { Columns = value; }
334    }
335    bool IStringConvertibleMatrix.Validate(string value, out string errorMessage) {
336      return Validate(value, out errorMessage);
337    }
338    string IStringConvertibleMatrix.GetValue(int rowIndex, int columIndex) {
339      return GetValue(rowIndex, columIndex);
340    }
341    bool IStringConvertibleMatrix.SetValue(string value, int rowIndex, int columnIndex) {
342      return SetValue(value, rowIndex, columnIndex);
343    }
[2694]344    #endregion
345  }
346}
Note: See TracBrowser for help on using the repository browser.