#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;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using HeuristicLab.Common;
using HeuristicLab.Persistence;
namespace HeuristicLab.Collections {
[StorableType("a5adee08-dd7e-4a1c-a694-0d231d2e7a20")]
[Serializable]
public class ObservableArray : IObservableArray {
[Storable]
protected T[] array;
#region Properties
public int Length {
get { return array.Length; }
}
int ICollection.Count {
get { return array.Length; }
}
bool ICollection.IsReadOnly {
get { return array.IsReadOnly; }
}
public T this[int index] {
get {
return array[index];
}
set {
T item = array[index];
if (!((item == null) && (value == null)) && ((item == null) || (!item.Equals(value)))) {
array[index] = value;
OnItemsReplaced(new IndexedItem[] { new IndexedItem(index, value) }, new IndexedItem[] { new IndexedItem(index, item) });
OnPropertyChanged("Item[]");
}
}
}
#endregion
#region Constructors
public ObservableArray() {
array = new T[0];
}
public ObservableArray(int length) {
array = new T[length];
}
public ObservableArray(T[] array) {
this.array = (T[])array.Clone();
}
public ObservableArray(IEnumerable collection) {
array = collection.ToArray();
}
[StorableConstructor]
protected ObservableArray(StorableConstructorFlag deserializing) { }
#endregion
#region Access
public bool Contains(T item) {
return IndexOf(item) != -1;
}
public int IndexOf(T item) {
return Array.IndexOf(array, item);
}
public int IndexOf(T item, int startIndex) {
return Array.IndexOf(array, item, startIndex);
}
public int IndexOf(T item, int startIndex, int count) {
return Array.IndexOf(array, item, startIndex, count);
}
public int LastIndexOf(T item) {
return Array.LastIndexOf(array, item);
}
public int LastIndexOf(T item, int startIndex) {
return Array.LastIndexOf(array, item, startIndex);
}
public int LastIndexOf(T item, int startIndex, int count) {
return Array.LastIndexOf(array, item, startIndex, count);
}
public int BinarySearch(T item) {
return Array.BinarySearch(array, item);
}
public int BinarySearch(T item, IComparer comparer) {
return Array.BinarySearch(array, item, comparer);
}
public int BinarySearch(int index, int count, T item) {
return Array.BinarySearch(array, index, count, item);
}
public int BinarySearch(int index, int count, T item, IComparer comparer) {
return Array.BinarySearch(array, index, count, item, comparer);
}
public bool Exists(Predicate match) {
return Array.Exists(array, match);
}
public T Find(Predicate match) {
return Array.Find(array, match);
}
public T[] FindAll(Predicate match) {
return Array.FindAll(array, match);
}
public T FindLast(Predicate match) {
return Array.FindLast(array, match);
}
public int FindIndex(Predicate match) {
return Array.FindIndex(array, match);
}
public int FindIndex(int startIndex, Predicate match) {
return Array.FindIndex(array, startIndex, match);
}
public int FindIndex(int startIndex, int count, Predicate match) {
return Array.FindIndex(array, startIndex, count, match);
}
public int FindLastIndex(Predicate match) {
return Array.FindLastIndex(array, match);
}
public int FindLastIndex(int startIndex, Predicate match) {
return Array.FindLastIndex(array, startIndex, match);
}
public int FindLastIndex(int startIndex, int count, Predicate match) {
return Array.FindLastIndex(array, startIndex, count, match);
}
#endregion
#region Manipulation
void ICollection.Add(T item) {
throw new NotSupportedException();
}
void IList.Insert(int index, T item) {
throw new NotSupportedException();
}
bool ICollection.Remove(T item) {
throw new NotSupportedException();
}
void IList.RemoveAt(int index) {
throw new NotSupportedException();
}
public void Clear(int index, int length) {
if (length > 0) {
IndexedItem[] oldItems = GetIndexedItems(index, length);
Array.Clear(array, index, length);
OnPropertyChanged("Item[]");
OnItemsReplaced(GetIndexedItems(index, length), oldItems);
}
}
void ICollection.Clear() {
Clear(0, array.Length);
}
public void Resize(int newSize) {
if (newSize != array.Length) {
IndexedItem[] oldItems = GetIndexedItems();
Array.Resize(ref array, newSize);
OnPropertyChanged("Length");
OnPropertyChanged("Item[]");
OnCollectionReset(GetIndexedItems(), oldItems);
}
}
public void Reverse() {
if (array.Length > 1) {
IndexedItem[] oldItems = GetIndexedItems();
Array.Reverse(array);
OnPropertyChanged("Item[]");
OnItemsMoved(GetIndexedItems(), oldItems);
}
}
public void Reverse(int index, int length) {
if (length > 1) {
IndexedItem[] oldItems = GetIndexedItems(index, length);
Array.Reverse(array, index, length);
OnPropertyChanged("Item[]");
OnItemsMoved(GetIndexedItems(index, length), oldItems);
}
}
public void Sort() {
if (array.Length > 1) {
IndexedItem[] oldItems = GetIndexedItems();
array.StableSort();
OnPropertyChanged("Item[]");
OnItemsMoved(GetIndexedItems(), oldItems);
}
}
public void Sort(Comparison comparison) {
if (array.Length > 1) {
IndexedItem[] oldItems = GetIndexedItems();
array.StableSort(comparison);
OnPropertyChanged("Item[]");
OnItemsMoved(GetIndexedItems(), oldItems);
}
}
public void Sort(IComparer comparer) {
if (array.Length > 1) {
IndexedItem[] oldItems = GetIndexedItems();
array.StableSort(comparer);
OnPropertyChanged("Item[]");
OnItemsMoved(GetIndexedItems(), oldItems);
}
}
public void Sort(int index, int length) {
if (length > 1) {
IndexedItem[] oldItems = GetIndexedItems(index, length);
array.StableSort(index, length);
OnPropertyChanged("Item[]");
OnItemsMoved(GetIndexedItems(index, length), oldItems);
}
}
public void Sort(int index, int length, IComparer comparer) {
if (length > 1) {
IndexedItem[] oldItems = GetIndexedItems(index, length);
array.StableSort(index, length, comparer);
OnPropertyChanged("Item[]");
OnItemsMoved(GetIndexedItems(index, length), oldItems);
}
}
#endregion
#region Conversion
public ReadOnlyObservableArray AsReadOnly() {
return new ReadOnlyObservableArray(this);
}
public void CopyTo(T[] array) {
Array.Copy(this.array, array, this.array.Length);
}
public void CopyTo(T[] array, int arrayIndex) {
Array.Copy(this.array, 0, array, arrayIndex, this.array.Length);
}
public void CopyTo(int index, T[] array, int arrayIndex, int count) {
Array.Copy(this.array, index, array, arrayIndex, count);
}
public TOutput[] ConvertAll(Converter converter) {
return Array.ConvertAll(array, converter);
}
#endregion
#region Processing
public void ForEach(Action action) {
Array.ForEach(array, action);
}
public bool TrueForAll(Predicate match) {
return Array.TrueForAll(array, match);
}
#endregion
#region Enumeration
public IEnumerator GetEnumerator() {
foreach (object o in ((IEnumerable)this))
yield return (T)o;
}
IEnumerator IEnumerable.GetEnumerator() {
return array.GetEnumerator();
}
#endregion
#region Events
[field: NonSerialized]
public event CollectionItemsChangedEventHandler> ItemsReplaced;
protected virtual void OnItemsReplaced(IEnumerable> items, IEnumerable> oldItems) {
CollectionItemsChangedEventHandler> handler = ItemsReplaced;
if (handler != null) handler(this, new CollectionItemsChangedEventArgs>(items, oldItems));
}
[field: NonSerialized]
public event CollectionItemsChangedEventHandler> ItemsMoved;
protected virtual void OnItemsMoved(IEnumerable> items, IEnumerable> oldItems) {
CollectionItemsChangedEventHandler> handler = ItemsMoved;
if (handler != null) handler(this, new CollectionItemsChangedEventArgs>(items, oldItems));
}
[field: NonSerialized]
public event CollectionItemsChangedEventHandler> CollectionReset;
protected virtual void OnCollectionReset(IEnumerable> items, IEnumerable> oldItems) {
CollectionItemsChangedEventHandler> handler = CollectionReset;
if (handler != null) handler(this, new CollectionItemsChangedEventArgs>(items, oldItems));
}
[field: NonSerialized]
public event PropertyChangedEventHandler PropertyChanged;
protected virtual void OnPropertyChanged(string propertyName) {
PropertyChangedEventHandler handler = PropertyChanged;
if (handler != null) handler(this, new PropertyChangedEventArgs(propertyName));
}
#endregion
#region Private helpers
private IndexedItem[] GetIndexedItems() {
IndexedItem[] items = new IndexedItem[array.Length];
for (int i = 0; i < array.Length; i++)
items[i] = new IndexedItem(i, array[i]);
return items;
}
private IndexedItem[] GetIndexedItems(int index, int count) {
IndexedItem[] items = new IndexedItem[count];
for (int i = 0; i < count; i++)
items[i] = new IndexedItem(index + i, array[index + i]);
return items;
}
#endregion
}
}