1336 lines
40 KiB
C#
1336 lines
40 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Collections;
|
|
using System.Diagnostics;
|
|
using System.Runtime.InteropServices;
|
|
using System.ComponentModel;
|
|
|
|
namespace DevComponents.DotNetBar.Primitives
|
|
{
|
|
/// <summary>
|
|
/// Represents custom collection with INotifyPropertyChanged and INotifyCollectionChanged interface support.
|
|
/// </summary>
|
|
/// <typeparam name="T"></typeparam>
|
|
[Serializable, DebuggerDisplay("Count = {Count}"), ComVisible(false)]
|
|
public class CustomCollection<T> : IList<T>, IList, INotifyPropertyChanged, INotifyCollectionChanged
|
|
{
|
|
#region Events
|
|
/// <summary>
|
|
/// Occurs when property value has changed.
|
|
/// </summary>
|
|
public event PropertyChangedEventHandler PropertyChanged;
|
|
#endregion
|
|
|
|
#region Private variables
|
|
|
|
private SimpleMonitor _monitor;
|
|
private object _SyncRoot;
|
|
private readonly IList<T> _Items;
|
|
private bool _FloatLastItem;
|
|
|
|
#endregion
|
|
|
|
#region Constructors
|
|
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
public CustomCollection()
|
|
{
|
|
_monitor = new SimpleMonitor();
|
|
_Items = new List<T>();
|
|
}
|
|
|
|
public void Dispose()
|
|
{
|
|
_monitor.Dispose();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
public CustomCollection(int capacity)
|
|
{
|
|
_monitor = new SimpleMonitor();
|
|
_Items = new List<T>(capacity);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="list">List to initialize collection with.</param>
|
|
public CustomCollection(IList<T> list)
|
|
{
|
|
if (list == null)
|
|
throw new ArgumentNullException("list");
|
|
|
|
_monitor = new SimpleMonitor();
|
|
_Items = list;
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Add
|
|
|
|
/// <summary>
|
|
/// Add item to collection.
|
|
/// </summary>
|
|
/// <param name="item">Item to add.</param>
|
|
public void Add(T item)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
InsertItem(_Items.Count, item);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Sort
|
|
//
|
|
// Summary:
|
|
// Sorts the elements in the entire collection using
|
|
// the default comparer.
|
|
//
|
|
// Exceptions:
|
|
// System.InvalidOperationException:
|
|
// The default comparer System.Collections.Generic.Comparer<T>.Default cannot
|
|
// find an implementation of the System.IComparable<T> generic interface or
|
|
// the System.IComparable interface for type T.
|
|
public void Sort()
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
if(!(_Items is List<T>))
|
|
throw new NotSupportedException("collection must be default List collection");
|
|
((List<T>)_Items).Sort();
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Reset, null, -1);
|
|
}
|
|
|
|
//
|
|
// Summary:
|
|
// Reverse sorts the elements in the entire collection.
|
|
//
|
|
public void SortReverse()
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
if (!(_Items is List<T>))
|
|
throw new NotSupportedException("collection must be default List collection");
|
|
((List<T>)_Items).Sort();
|
|
((List<T>)_Items).Reverse();
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Reset, null, -1);
|
|
}
|
|
|
|
//
|
|
// Summary:
|
|
// Sorts the elements in the entire collection using
|
|
// the specified System.Comparison<T>.
|
|
//
|
|
// Parameters:
|
|
// comparison:
|
|
// The System.Comparison<T> to use when comparing elements.
|
|
//
|
|
// Exceptions:
|
|
// System.ArgumentNullException:
|
|
// comparison is null.
|
|
//
|
|
// System.ArgumentException:
|
|
// The implementation of comparison caused an error during the sort. For example,
|
|
// comparison might not return 0 when comparing an item with itself.
|
|
public void Sort(Comparison<T> comparison)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
if (!(_Items is List<T>))
|
|
throw new NotSupportedException("collection must be default List collection");
|
|
((List<T>)_Items).Sort(comparison);
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Reset, null, -1);
|
|
}
|
|
//
|
|
// Summary:
|
|
// Sorts the elements in the entire System.Collections.Generic.List<T> using
|
|
// the specified comparer.
|
|
//
|
|
// Parameters:
|
|
// comparer:
|
|
// The System.Collections.Generic.IComparer<T> implementation to use when comparing
|
|
// elements, or null to use the default comparer System.Collections.Generic.Comparer<T>.Default.
|
|
//
|
|
// Exceptions:
|
|
// System.InvalidOperationException:
|
|
// comparer is null, and the default comparer System.Collections.Generic.Comparer<T>.Default
|
|
// cannot find implementation of the System.IComparable<T> generic interface
|
|
// or the System.IComparable interface for type T.
|
|
//
|
|
// System.ArgumentException:
|
|
// The implementation of comparer caused an error during the sort. For example,
|
|
// comparer might not return 0 when comparing an item with itself.
|
|
public void Sort(IComparer<T> comparer)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
if (!(_Items is List<T>))
|
|
throw new NotSupportedException("collection must be default List collection");
|
|
((List<T>)_Items).Sort(comparer);
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Reset, null, -1);
|
|
}
|
|
//
|
|
// Summary:
|
|
// Sorts the elements in a range of elements in System.Collections.Generic.List<T>
|
|
// using the specified comparer.
|
|
//
|
|
// Parameters:
|
|
// index:
|
|
// The zero-based starting index of the range to sort.
|
|
//
|
|
// count:
|
|
// The length of the range to sort.
|
|
//
|
|
// comparer:
|
|
// The System.Collections.Generic.IComparer<T> implementation to use when comparing
|
|
// elements, or null to use the default comparer System.Collections.Generic.Comparer<T>.Default.
|
|
//
|
|
// Exceptions:
|
|
// System.ArgumentOutOfRangeException:
|
|
// index is less than 0.-or-count is less than 0.
|
|
//
|
|
// System.ArgumentException:
|
|
// index and count do not specify a valid range in the System.Collections.Generic.List<T>.-or-The
|
|
// implementation of comparer caused an error during the sort. For example,
|
|
// comparer might not return 0 when comparing an item with itself.
|
|
//
|
|
// System.InvalidOperationException:
|
|
// comparer is null, and the default comparer System.Collections.Generic.Comparer<T>.Default
|
|
// cannot find implementation of the System.IComparable<T> generic interface
|
|
// or the System.IComparable interface for type T.
|
|
public void Sort(int index, int count, IComparer<T> comparer)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
if (!(_Items is List<T>))
|
|
throw new NotSupportedException("collection must be default List collection");
|
|
((List<T>)_Items).Sort(index, count, comparer);
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Reset, null, -1);
|
|
}
|
|
#endregion
|
|
|
|
#region Clear
|
|
|
|
/// <summary>
|
|
/// Remove all items from collection.
|
|
/// </summary>
|
|
public void Clear()
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
ClearItems();
|
|
}
|
|
|
|
/// <summary>
|
|
/// Remove all items from collection.
|
|
/// </summary>
|
|
protected virtual void ClearItems()
|
|
{
|
|
CheckReentrancy();
|
|
|
|
T item = default(T);
|
|
|
|
if (_FloatLastItem == true)
|
|
{
|
|
if (_Items.Count > 0)
|
|
item = _Items[_Items.Count - 1];
|
|
}
|
|
|
|
_Items.Clear();
|
|
|
|
OnPropertyChanged(new PropertyChangedEventArgs(CountString));
|
|
OnPropertyChanged(new PropertyChangedEventArgs(ItemString));
|
|
|
|
OnCollectionReset();
|
|
|
|
if (_FloatLastItem == true)
|
|
_Items.Add(item);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Contains
|
|
|
|
/// <summary>
|
|
/// Checks whether collection contains item.
|
|
/// </summary>
|
|
/// <param name="item">Item to look for.</param>
|
|
/// <returns>true if item is in collection.</returns>
|
|
public bool Contains(T item)
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
return (_Items.Contains(item));
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region CopyTo
|
|
|
|
/// <summary>
|
|
/// Copy collection to array.
|
|
/// </summary>
|
|
/// <param name="array">Array to copy to.</param>
|
|
/// <param name="index">Index to copy from.</param>
|
|
public void CopyTo(T[] array, int index)
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
_Items.CopyTo(array, index);
|
|
}
|
|
/// <summary>
|
|
/// Copy collection to array.
|
|
/// </summary>
|
|
/// <param name="array">Array to copy to.</param>
|
|
public void CopyTo(T[] array)
|
|
{
|
|
CopyTo(array, 0);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Count
|
|
|
|
/// <summary>
|
|
/// Returns number of items in collection.
|
|
/// </summary>
|
|
public int Count
|
|
{
|
|
get
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
return (_Items.Count);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region CountString
|
|
|
|
private string CountString
|
|
{
|
|
get { return ("Count"); }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region FloatLastItem
|
|
|
|
internal bool FloatLastItem
|
|
{
|
|
get { return (_FloatLastItem); }
|
|
set { _FloatLastItem = value; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region GetEnumerator
|
|
|
|
/// <summary>
|
|
/// Gets enumerator for collection.
|
|
/// </summary>
|
|
/// <returns>Enumerator.</returns>
|
|
public IEnumerator<T> GetEnumerator()
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
return (_Items.GetEnumerator());
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Indexer[int]
|
|
|
|
/// <summary>
|
|
/// Returns item at index.
|
|
/// </summary>
|
|
/// <param name="index">Index of item.</param>
|
|
/// <returns>Item at index.</returns>
|
|
public T this[int index]
|
|
{
|
|
get
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
return (_Items[index]);
|
|
}
|
|
|
|
set
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
if ((index < 0) || (index >= _Items.Count))
|
|
throw new ArgumentOutOfRangeException();
|
|
|
|
SetItem(index, value);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IndexOf
|
|
|
|
/// <summary>
|
|
/// Returns index of an item.
|
|
/// </summary>
|
|
/// <param name="item">Reference to item.</param>
|
|
/// <returns>Index of item.</returns>
|
|
public int IndexOf(T item)
|
|
{
|
|
OnCollectionReadAccess();
|
|
return (_Items.IndexOf(item));
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Insert
|
|
|
|
/// <summary>
|
|
/// Insert item at specified location.
|
|
/// </summary>
|
|
/// <param name="index">Index to insert item in.</param>
|
|
/// <param name="item">Item to insert.</param>
|
|
public void Insert(int index, T item)
|
|
{
|
|
if ((index < 0) || (index > this._Items.Count))
|
|
throw new ArgumentOutOfRangeException("index");
|
|
|
|
InsertItem(index, item);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region InsertItem
|
|
|
|
/// <summary>
|
|
/// Inserts item.
|
|
/// </summary>
|
|
/// <param name="index">Index to insert item at.</param>
|
|
/// <param name="item">Reference to item.</param>
|
|
protected virtual void InsertItem(int index, T item)
|
|
{
|
|
CheckReentrancy();
|
|
|
|
if (_FloatLastItem == true)
|
|
{
|
|
if (_Items.Count > 0 && index == _Items.Count)
|
|
index--;
|
|
}
|
|
|
|
_Items.Insert(index, item);
|
|
|
|
OnPropertyChanged(new PropertyChangedEventArgs(CountString));
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Add, item, index);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IsCompatibleObject
|
|
|
|
private static bool IsCompatibleObject(object value)
|
|
{
|
|
return (value is T) || (value == null && !typeof(T).IsValueType);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ItemString
|
|
|
|
private string ItemString
|
|
{
|
|
get { return ("Item[]"); }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Remove
|
|
|
|
/// <summary>
|
|
/// Removes item from collection.
|
|
/// </summary>
|
|
/// <param name="item">Item to remove.</param>
|
|
/// <returns>true if item was removed.</returns>
|
|
public bool Remove(T item)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
int index = _Items.IndexOf(item);
|
|
|
|
if (index < 0)
|
|
return (false);
|
|
|
|
RemoveItem(index);
|
|
|
|
return (true);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region RemoveAt
|
|
|
|
/// <summary>
|
|
/// Remove item at specified location.
|
|
/// </summary>
|
|
/// <param name="index">Index of item to remove.</param>
|
|
public void RemoveAt(int index)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
if ((index < 0) || (index >= _Items.Count))
|
|
throw new ArgumentOutOfRangeException();
|
|
|
|
RemoveItem(index);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region RemoveItem
|
|
|
|
/// <summary>
|
|
/// Remove item at specified location.
|
|
/// </summary>
|
|
/// <param name="index">Index of item to remove.</param>
|
|
protected virtual void RemoveItem(int index)
|
|
{
|
|
CheckReentrancy();
|
|
|
|
object item = _Items[index];
|
|
_Items.RemoveAt(index);
|
|
|
|
OnPropertyChanged(new PropertyChangedEventArgs(CountString));
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Remove, item, index);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region SetItem
|
|
|
|
/// <summary>
|
|
/// Set item on location.
|
|
/// </summary>
|
|
/// <param name="index">Index</param>
|
|
/// <param name="item">Item to assign.</param>
|
|
protected virtual void SetItem(int index, T item)
|
|
{
|
|
CheckReentrancy();
|
|
|
|
T oldItem = _Items[index];
|
|
_Items[index] = item;
|
|
|
|
OnPropertyChanged(new PropertyChangedEventArgs(CountString));
|
|
OnPropertyChanged(new PropertyChangedEventArgs(ItemString));
|
|
|
|
OnCollectionChanged(NotifyCollectionChangedAction.Replace, oldItem, item, index);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IList support
|
|
|
|
#region Add
|
|
|
|
int IList.Add(object value)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
VerifyValueType(value);
|
|
|
|
Add((T)value);
|
|
|
|
return (Count - 1);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Contains
|
|
|
|
bool IList.Contains(object value)
|
|
{
|
|
return (IsCompatibleObject(value) && Contains((T)value));
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region GetItemsDirect
|
|
|
|
/// <summary>
|
|
/// Returns items directly without checks.
|
|
/// </summary>
|
|
/// <returns>List of items.</returns>
|
|
protected IList<T> GetItemsDirect()
|
|
{
|
|
return (_Items);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Indexer[int]
|
|
|
|
object IList.this[int index]
|
|
{
|
|
get
|
|
{
|
|
OnCollectionReadAccess();
|
|
return this._Items[index];
|
|
}
|
|
set
|
|
{
|
|
VerifyValueType(value);
|
|
this[index] = (T)value;
|
|
}
|
|
}
|
|
|
|
#region IndexOf
|
|
|
|
int IList.IndexOf(object value)
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
return (IsCompatibleObject(value) ? IndexOf((T)value) : -1);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Insert
|
|
|
|
void IList.Insert(int index, object value)
|
|
{
|
|
if (this._Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
VerifyValueType(value);
|
|
|
|
Insert(index, (T)value);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IsFixedSize
|
|
|
|
bool IList.IsFixedSize
|
|
{
|
|
get
|
|
{
|
|
IList items = _Items as IList;
|
|
|
|
return ((items != null) && items.IsFixedSize);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IsReadOnly
|
|
|
|
bool IList.IsReadOnly
|
|
{
|
|
get { return (_Items.IsReadOnly); }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Items
|
|
|
|
/// <summary>
|
|
/// Returns the IList interface for items in collection.
|
|
/// </summary>
|
|
protected IList<T> Items
|
|
{
|
|
get
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
return (_Items);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region Remove
|
|
|
|
void IList.Remove(object value)
|
|
{
|
|
if (_Items.IsReadOnly)
|
|
throw new NotSupportedException("collection is read-only");
|
|
|
|
if (IsCompatibleObject(value))
|
|
Remove((T)value);
|
|
}
|
|
|
|
#endregion
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
#region VerifyValueType
|
|
|
|
private static void VerifyValueType(object value)
|
|
{
|
|
if (!IsCompatibleObject(value))
|
|
throw new ArgumentException("value is of wrong type");
|
|
}
|
|
|
|
#endregion
|
|
|
|
#endregion
|
|
|
|
#region IEnumerable.GetEnumerator
|
|
|
|
IEnumerator IEnumerable.GetEnumerator()
|
|
{
|
|
OnCollectionReadAccess();
|
|
|
|
return (_Items.GetEnumerator());
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ICollection support
|
|
|
|
#region CopyTo
|
|
|
|
void ICollection.CopyTo(Array array, int index)
|
|
{
|
|
CheckReentrancy();
|
|
OnCollectionReadAccess();
|
|
|
|
if (array == null)
|
|
throw new ArgumentNullException("array");
|
|
|
|
if (array.Rank != 1)
|
|
throw new ArgumentException("Argument array.Rank multi-dimensional not supported");
|
|
|
|
if (array.GetLowerBound(0) != 0)
|
|
throw new ArgumentException("Argument array non zero lower bound not supported");
|
|
|
|
if (index < 0)
|
|
throw new ArgumentOutOfRangeException("index", "index must be non-negative number");
|
|
|
|
if ((array.Length - index) < this.Count)
|
|
throw new ArgumentException("array too small");
|
|
|
|
T[] localArray = array as T[];
|
|
|
|
if (localArray != null)
|
|
{
|
|
_Items.CopyTo(localArray, index);
|
|
}
|
|
else
|
|
{
|
|
Type elementType = array.GetType().GetElementType();
|
|
Type c = typeof(T);
|
|
|
|
if (elementType == null ||
|
|
(!elementType.IsAssignableFrom(c) && !c.IsAssignableFrom(elementType)))
|
|
{
|
|
throw new ArgumentException("Argument array of invalid type");
|
|
}
|
|
|
|
object[] objArray = array as object[];
|
|
|
|
if (objArray == null)
|
|
throw new ArgumentException("Argument array invalid type");
|
|
|
|
int count = this._Items.Count;
|
|
|
|
try
|
|
{
|
|
for (int i = 0; i < count; i++)
|
|
objArray[index++] = this._Items[i];
|
|
}
|
|
catch (ArrayTypeMismatchException)
|
|
{
|
|
throw new ArgumentException("Argument array invalid type");
|
|
}
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IsSynchronized
|
|
|
|
bool ICollection.IsSynchronized
|
|
{
|
|
get { return false; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region IsReadOnly
|
|
|
|
bool ICollection<T>.IsReadOnly
|
|
{
|
|
get { return this._Items.IsReadOnly; }
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region SyncRoot
|
|
|
|
object ICollection.SyncRoot
|
|
{
|
|
get
|
|
{
|
|
if (_SyncRoot == null)
|
|
{
|
|
ICollection items = _Items as ICollection;
|
|
|
|
if (items != null)
|
|
_SyncRoot = items.SyncRoot;
|
|
else
|
|
System.Threading.Interlocked.CompareExchange(ref _SyncRoot, new object(), null);
|
|
}
|
|
|
|
return (_SyncRoot);
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#endregion
|
|
|
|
#region OnCollectionReadAccess
|
|
|
|
/// <summary>
|
|
/// Occurs when collection is read.
|
|
/// </summary>
|
|
protected virtual void OnCollectionReadAccess()
|
|
{
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region OnPropertyChanged
|
|
|
|
/// <summary>
|
|
/// Occurs when collection property has changed.
|
|
/// </summary>
|
|
/// <param name="e">Event arguments.</param>
|
|
protected virtual void OnPropertyChanged(PropertyChangedEventArgs e)
|
|
{
|
|
if (PropertyChanged != null)
|
|
PropertyChanged(this, e);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region BlockReentrancy
|
|
|
|
/// <summary>
|
|
/// Blocks the collection reentrancy.
|
|
/// </summary>
|
|
/// <returns>IDisposable to end re-entrancy</returns>
|
|
protected IDisposable BlockReentrancy()
|
|
{
|
|
_monitor.Enter();
|
|
|
|
return (_monitor);
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region CheckReentrancy
|
|
|
|
/// <summary>
|
|
/// Checks whether call creates reentrancy.
|
|
/// </summary>
|
|
protected void CheckReentrancy()
|
|
{
|
|
if ((_monitor.Busy && (CollectionChanged != null)) &&
|
|
(CollectionChanged.GetInvocationList().Length > 1))
|
|
{
|
|
throw new InvalidOperationException("CustomCollectionReentrancyNotAllowed");
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region SimpleMonitor
|
|
|
|
[Serializable]
|
|
private class SimpleMonitor : IDisposable
|
|
{
|
|
// Fields
|
|
private int _BusyCount;
|
|
|
|
// Methods
|
|
public void Dispose()
|
|
{
|
|
_BusyCount--;
|
|
}
|
|
|
|
public void Enter()
|
|
{
|
|
_BusyCount++;
|
|
}
|
|
|
|
// Properties
|
|
public bool Busy
|
|
{
|
|
get { return (_BusyCount > 0); }
|
|
}
|
|
}
|
|
#endregion
|
|
|
|
#region INotifyCollectionChanged Members
|
|
|
|
/// <summary>
|
|
/// Occurs when collection has changed.
|
|
/// </summary>
|
|
public event NotifyCollectionChangedEventHandler CollectionChanged;
|
|
|
|
private void OnCollectionChanged(NotifyCollectionChangedAction action, object item, int index)
|
|
{
|
|
this.OnCollectionChanged(new NotifyCollectionChangedEventArgs(action, item, index));
|
|
}
|
|
|
|
private void OnCollectionChanged(NotifyCollectionChangedAction action, object oldItem, object newItem, int index)
|
|
{
|
|
this.OnCollectionChanged(new NotifyCollectionChangedEventArgs(action, newItem, oldItem, index));
|
|
}
|
|
|
|
private void OnCollectionReset()
|
|
{
|
|
this.OnCollectionChanged(new NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction.Reset));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Called when collection has changed.
|
|
/// </summary>
|
|
/// <param name="e">Event arguments.</param>
|
|
protected virtual void OnCollectionChanged(NotifyCollectionChangedEventArgs e)
|
|
{
|
|
if (this.CollectionChanged != null)
|
|
{
|
|
using (this.BlockReentrancy())
|
|
{
|
|
this.CollectionChanged(this, e);
|
|
}
|
|
}
|
|
}
|
|
#endregion
|
|
}
|
|
|
|
#region INotifyCollectionChanged
|
|
/// <summary>
|
|
/// Represents collection changed notification interface.
|
|
/// </summary>
|
|
public interface INotifyCollectionChanged
|
|
{
|
|
/// <summary>
|
|
/// Occurs when collection changed.
|
|
/// </summary>
|
|
event NotifyCollectionChangedEventHandler CollectionChanged;
|
|
}
|
|
/// <summary>
|
|
/// Defines change actions.
|
|
/// </summary>
|
|
public enum NotifyCollectionChangedAction
|
|
{
|
|
/// <summary>
|
|
/// Items were added.
|
|
/// </summary>
|
|
Add,
|
|
/// <summary>
|
|
/// Items were removed.
|
|
/// </summary>
|
|
Remove,
|
|
/// <summary>
|
|
/// Items were replaced.
|
|
/// </summary>
|
|
Replace,
|
|
/// <summary>
|
|
/// Items were moved.
|
|
/// </summary>
|
|
Move,
|
|
/// <summary>
|
|
/// Collection was reset.
|
|
/// </summary>
|
|
Reset
|
|
}
|
|
/// <summary>
|
|
/// Defines delegate for collection notification events.
|
|
/// </summary>
|
|
/// <param name="sender">Event sender.</param>
|
|
/// <param name="e">Event arguments.</param>
|
|
public delegate void NotifyCollectionChangedEventHandler(object sender, NotifyCollectionChangedEventArgs e);
|
|
/// <summary>
|
|
/// Defines collection change notification event arguments.
|
|
/// </summary>
|
|
public class NotifyCollectionChangedEventArgs : EventArgs
|
|
{
|
|
#region Private Vars
|
|
|
|
private NotifyCollectionChangedAction _Action;
|
|
private IList _NewItems;
|
|
private int _NewStartingIndex;
|
|
private IList _OldItems;
|
|
private int _OldStartingIndex;
|
|
|
|
#endregion
|
|
|
|
/// <summary>
|
|
/// Create new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (action != NotifyCollectionChangedAction.Reset)
|
|
{
|
|
throw new ArgumentException("WrongActionForCtor");
|
|
}
|
|
this.InitializeAdd(action, null, -1);
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Specifies action.</param>
|
|
/// <param name="changedItems">List of changed items.</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, IList changedItems)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (((action != NotifyCollectionChangedAction.Add) && (action != NotifyCollectionChangedAction.Remove)) && (action != NotifyCollectionChangedAction.Reset))
|
|
{
|
|
throw new ArgumentException("MustBeResetAddOrRemoveActionForCtor");
|
|
}
|
|
if (action == NotifyCollectionChangedAction.Reset)
|
|
{
|
|
if (changedItems != null)
|
|
{
|
|
throw new ArgumentException("ResetActionRequiresNullItem");
|
|
}
|
|
this.InitializeAdd(action, null, -1);
|
|
}
|
|
else
|
|
{
|
|
if (changedItems == null)
|
|
{
|
|
throw new ArgumentNullException("changedItems");
|
|
}
|
|
this.InitializeAddOrRemove(action, changedItems, -1);
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Specifies action.</param>
|
|
/// <param name="changedItem">Item that was changed.</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, object changedItem)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (((action != NotifyCollectionChangedAction.Add) && (action != NotifyCollectionChangedAction.Remove)) && (action != NotifyCollectionChangedAction.Reset))
|
|
{
|
|
throw new ArgumentException("MustBeResetAddOrRemoveActionForCtor");
|
|
}
|
|
if (action == NotifyCollectionChangedAction.Reset)
|
|
{
|
|
if (changedItem != null)
|
|
{
|
|
throw new ArgumentException("ResetActionRequiresNullItem");
|
|
}
|
|
this.InitializeAdd(action, null, -1);
|
|
}
|
|
else
|
|
{
|
|
this.InitializeAddOrRemove(action, new object[] { changedItem }, -1);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action.</param>
|
|
/// <param name="newItems">New items in collection.</param>
|
|
/// <param name="oldItems">Old items in collection.</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, IList newItems, IList oldItems)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (action != NotifyCollectionChangedAction.Replace)
|
|
{
|
|
throw new ArgumentException("WrongActionForCtor");
|
|
}
|
|
if (newItems == null)
|
|
{
|
|
throw new ArgumentNullException("newItems");
|
|
}
|
|
if (oldItems == null)
|
|
{
|
|
throw new ArgumentNullException("oldItems");
|
|
}
|
|
this.InitializeMoveOrReplace(action, newItems, oldItems, -1, -1);
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action.</param>
|
|
/// <param name="changedItems">List of changed items.</param>
|
|
/// <param name="startingIndex">Starting index of change.</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, IList changedItems, int startingIndex)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (((action != NotifyCollectionChangedAction.Add) && (action != NotifyCollectionChangedAction.Remove)) && (action != NotifyCollectionChangedAction.Reset))
|
|
{
|
|
throw new ArgumentException("MustBeResetAddOrRemoveActionForCtor");
|
|
}
|
|
if (action == NotifyCollectionChangedAction.Reset)
|
|
{
|
|
if (changedItems != null)
|
|
{
|
|
throw new ArgumentException("ResetActionRequiresNullItem");
|
|
}
|
|
if (startingIndex != -1)
|
|
{
|
|
throw new ArgumentException("ResetActionRequiresIndexMinus1");
|
|
}
|
|
this.InitializeAdd(action, null, -1);
|
|
}
|
|
else
|
|
{
|
|
if (changedItems == null)
|
|
{
|
|
throw new ArgumentNullException("changedItems");
|
|
}
|
|
if (startingIndex < -1)
|
|
{
|
|
throw new ArgumentException("IndexCannotBeNegative");
|
|
}
|
|
this.InitializeAddOrRemove(action, changedItems, startingIndex);
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action</param>
|
|
/// <param name="changedItem">Changed item</param>
|
|
/// <param name="index">Index of change</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, object changedItem, int index)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (((action != NotifyCollectionChangedAction.Add) && (action != NotifyCollectionChangedAction.Remove)) && (action != NotifyCollectionChangedAction.Reset))
|
|
{
|
|
throw new ArgumentException("MustBeResetAddOrRemoveActionForCtor");
|
|
}
|
|
if (action == NotifyCollectionChangedAction.Reset)
|
|
{
|
|
if (changedItem != null)
|
|
{
|
|
throw new ArgumentException("ResetActionRequiresNullItem");
|
|
}
|
|
if (index != -1)
|
|
{
|
|
throw new ArgumentException("ResetActionRequiresIndexMinus1");
|
|
}
|
|
this.InitializeAdd(action, null, -1);
|
|
}
|
|
else
|
|
{
|
|
this.InitializeAddOrRemove(action, new object[] { changedItem }, index);
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action</param>
|
|
/// <param name="newItem">New item</param>
|
|
/// <param name="oldItem">Old item</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, object newItem, object oldItem)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (action != NotifyCollectionChangedAction.Replace)
|
|
{
|
|
throw new ArgumentException("WrongActionForCtor");
|
|
}
|
|
this.InitializeMoveOrReplace(action, new object[] { newItem }, new object[] { oldItem }, -1, -1);
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action</param>
|
|
/// <param name="newItems">New items.</param>
|
|
/// <param name="oldItems">Removed items.</param>
|
|
/// <param name="startingIndex">Starting index of change.</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, IList newItems, IList oldItems, int startingIndex)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (action != NotifyCollectionChangedAction.Replace)
|
|
{
|
|
throw new ArgumentException("WrongActionForCtor");
|
|
}
|
|
if (newItems == null)
|
|
{
|
|
throw new ArgumentNullException("newItems");
|
|
}
|
|
if (oldItems == null)
|
|
{
|
|
throw new ArgumentNullException("oldItems");
|
|
}
|
|
this.InitializeMoveOrReplace(action, newItems, oldItems, startingIndex, startingIndex);
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action</param>
|
|
/// <param name="changedItems">Changed items</param>
|
|
/// <param name="index">New index</param>
|
|
/// <param name="oldIndex">Old index</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, IList changedItems, int index, int oldIndex)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (action != NotifyCollectionChangedAction.Move)
|
|
{
|
|
throw new ArgumentException("WrongActionForCtor");
|
|
}
|
|
if (index < 0)
|
|
{
|
|
throw new ArgumentException("IndexCannotBeNegative");
|
|
}
|
|
this.InitializeMoveOrReplace(action, changedItems, changedItems, index, oldIndex);
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action</param>
|
|
/// <param name="changedItem">Changed item</param>
|
|
/// <param name="index">New index</param>
|
|
/// <param name="oldIndex">Old index</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, object changedItem, int index, int oldIndex)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (action != NotifyCollectionChangedAction.Move)
|
|
{
|
|
throw new ArgumentException("WrongActionForCtor");
|
|
}
|
|
if (index < 0)
|
|
{
|
|
throw new ArgumentException("IndexCannotBeNegative");
|
|
}
|
|
object[] newItems = new object[] { changedItem };
|
|
this.InitializeMoveOrReplace(action, newItems, newItems, index, oldIndex);
|
|
}
|
|
/// <summary>
|
|
/// Creates new instance of object.
|
|
/// </summary>
|
|
/// <param name="action">Action.</param>
|
|
/// <param name="newItem">New item</param>
|
|
/// <param name="oldItem">Old item</param>
|
|
/// <param name="index">New index</param>
|
|
public NotifyCollectionChangedEventArgs(NotifyCollectionChangedAction action, object newItem, object oldItem, int index)
|
|
{
|
|
this._NewStartingIndex = -1;
|
|
this._OldStartingIndex = -1;
|
|
if (action != NotifyCollectionChangedAction.Replace)
|
|
{
|
|
throw new ArgumentException("WrongActionForCtor");
|
|
}
|
|
this.InitializeMoveOrReplace(action, new object[] { newItem }, new object[] { oldItem }, index, index);
|
|
}
|
|
|
|
private void InitializeAdd(NotifyCollectionChangedAction action, IList newItems, int newStartingIndex)
|
|
{
|
|
this._Action = action;
|
|
this._NewItems = (newItems == null) ? null : ArrayList.ReadOnly(newItems);
|
|
this._NewStartingIndex = newStartingIndex;
|
|
}
|
|
|
|
private void InitializeAddOrRemove(NotifyCollectionChangedAction action, IList changedItems, int startingIndex)
|
|
{
|
|
if (action == NotifyCollectionChangedAction.Add)
|
|
{
|
|
this.InitializeAdd(action, changedItems, startingIndex);
|
|
}
|
|
else if (action == NotifyCollectionChangedAction.Remove)
|
|
{
|
|
this.InitializeRemove(action, changedItems, startingIndex);
|
|
}
|
|
}
|
|
|
|
private void InitializeMoveOrReplace(NotifyCollectionChangedAction action, IList newItems, IList oldItems, int startingIndex, int oldStartingIndex)
|
|
{
|
|
this.InitializeAdd(action, newItems, startingIndex);
|
|
this.InitializeRemove(action, oldItems, oldStartingIndex);
|
|
}
|
|
|
|
private void InitializeRemove(NotifyCollectionChangedAction action, IList oldItems, int oldStartingIndex)
|
|
{
|
|
this._Action = action;
|
|
this._OldItems = (oldItems == null) ? null : ArrayList.ReadOnly(oldItems);
|
|
this._OldStartingIndex = oldStartingIndex;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the type of the collection change action.
|
|
/// </summary>
|
|
public NotifyCollectionChangedAction Action
|
|
{
|
|
get
|
|
{
|
|
return this._Action;
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Gets list of newly added items.
|
|
/// </summary>
|
|
public IList NewItems
|
|
{
|
|
get
|
|
{
|
|
return this._NewItems;
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Gets new starting index.
|
|
/// </summary>
|
|
public int NewStartingIndex
|
|
{
|
|
get
|
|
{
|
|
return this._NewStartingIndex;
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Gets list of removed items.
|
|
/// </summary>
|
|
public IList OldItems
|
|
{
|
|
get
|
|
{
|
|
return this._OldItems;
|
|
}
|
|
}
|
|
/// <summary>
|
|
/// Old starting index.
|
|
/// </summary>
|
|
public int OldStartingIndex
|
|
{
|
|
get
|
|
{
|
|
return this._OldStartingIndex;
|
|
}
|
|
}
|
|
}
|
|
#endregion
|
|
|
|
} |