// *
// * Copyright (C) 2008 Roger Alsing : http://www.RogerAlsing.com
// *
// * This library is free software; you can redistribute it and/or modify it
// * under the terms of the GNU Lesser General Public License 2.1 or later, as
// * published by the Free Software Foundation. See the included license.txt
// * or http://www.gnu.org/copyleft/lesser.html for details.
// *
// *
using System;
using System.Collections;
using T = Alsing.SourceCode.UndoBlock;
namespace Alsing.SourceCode
{
///
///
///
public sealed class UndoBlockCollection : ICollection, IList, IEnumerable,
ICloneable
{
private const int DefaultMinimumCapacity = 16;
private UndoBlock[] m_array = new UndoBlock[DefaultMinimumCapacity];
private int m_count;
private int m_version;
///
///
///
public string Name = "UndoAction";
// Construction
///
///
///
public UndoBlockCollection() {}
///
///
///
///
public UndoBlockCollection(UndoBlockCollection collection)
{
AddRange(collection);
}
///
///
///
///
public UndoBlockCollection(UndoBlock[] array)
{
AddRange(array);
}
///
///
///
public UndoBlock this[int index]
{
get
{
ValidateIndex(index); // throws
return m_array[index];
}
set
{
ValidateIndex(index); // throws
++m_version;
m_array[index] = value;
}
}
///
///
///
public int Capacity
{
get { return m_array.Length; }
set
{
if (value < m_count)
value = m_count;
if (value < DefaultMinimumCapacity)
value = DefaultMinimumCapacity;
if (m_array.Length == value)
return;
++m_version;
var temp = new UndoBlock[value];
// for (int i=0; i < m_count; ++i) temp[i] = m_array[i];
Array.Copy(m_array, 0, temp, 0, m_count);
m_array = temp;
}
}
#region ICloneable Members
object ICloneable.Clone()
{
return (Clone());
}
#endregion
// Operations (type-safe ICollection)
#region ICollection Members
///
///
///
public int Count
{
get { return m_count; }
}
bool ICollection.IsSynchronized
{
get { return m_array.IsSynchronized; }
}
object ICollection.SyncRoot
{
get { return m_array.SyncRoot; }
}
void ICollection.CopyTo(Array array, int start)
{
CopyTo((UndoBlock[]) array, start);
}
IEnumerator IEnumerable.GetEnumerator()
{
return (GetEnumerator());
}
#endregion
#region IList Members
///
///
///
public void Clear()
{
++m_version;
m_array = new UndoBlock[DefaultMinimumCapacity];
m_count = 0;
}
///
///
///
///
public void RemoveAt(int index)
{
ValidateIndex(index); // throws
++m_version;
m_count--;
// for (int i=index; i < m_count; ++i) m_array[i] = m_array[i+1];
Array.Copy(m_array, index + 1, m_array, index, m_count - index);
if (NeedsTrimming())
Trim();
}
bool IList.IsFixedSize
{
get { return false; }
}
bool IList.IsReadOnly
{
get { return false; }
}
object IList.this[int index]
{
get { return this[index]; }
set { this[index] = (UndoBlock) value; }
}
int IList.Add(object item)
{
return Add((UndoBlock) item);
}
/* redundant w/ type-safe method
void IList.Clear()
{
this.Clear();
}
*/
bool IList.Contains(object item)
{
return Contains((UndoBlock) item);
}
int IList.IndexOf(object item)
{
return IndexOf((UndoBlock) item);
}
void IList.Insert(int position, object item)
{
Insert(position, (UndoBlock) item);
}
void IList.Remove(object item)
{
Remove((UndoBlock) item);
}
#endregion
///
///
///
///
public void CopyTo(UndoBlock[] array)
{
CopyTo(array, 0);
}
///
///
///
///
///
public void CopyTo(UndoBlock[] array, int start)
{
if (m_count > array.GetUpperBound(0) + 1 - start)
throw new ArgumentException(
"Destination array was not long enough.");
// for (int i=0; i < m_count; ++i) array[start+i] = m_array[i];
Array.Copy(m_array, 0, array, start, m_count);
}
// Operations (type-safe IList)
///
///
///
///
///
public int Add(UndoBlock item)
{
if (NeedsGrowth())
Grow();
++m_version;
m_array[m_count] = item;
return m_count++;
}
///
///
///
///
///
public bool Contains(UndoBlock item)
{
return ((IndexOf(item) == - 1) ? false : true);
}
///
///
///
///
///
public int IndexOf(UndoBlock item)
{
for (int i = 0; i < m_count; ++i)
if (m_array[i] == (item))
return i;
return - 1;
}
///
///
///
///
///
public void Insert(int position, UndoBlock item)
{
ValidateIndex(position, true); // throws
if (NeedsGrowth())
Grow();
++m_version;
// for (int i=m_count; i > position; --i) m_array[i] = m_array[i-1];
Array.Copy(m_array, position, m_array, position + 1, m_count - position);
m_array[position] = item;
m_count++;
}
///
///
///
///
public void Remove(UndoBlock item)
{
int index = IndexOf(item);
if (index < 0)
throw new ArgumentException(
"Cannot remove the specified item because it was not found in the specified Collection.");
RemoveAt(index);
}
// Operations (type-safe IEnumerable)
///
///
///
///
public Enumerator GetEnumerator()
{
return new Enumerator(this);
}
// Operations (type-safe ICloneable)
///
///
///
///
public UndoBlockCollection Clone()
{
var tc = new UndoBlockCollection();
tc.AddRange(this);
tc.Capacity = m_array.Length;
tc.m_version = m_version;
return tc;
}
// Public helpers (just to mimic some nice features of ArrayList)
///
///
///
///
public void AddRange(UndoBlockCollection collection)
{
// for (int i=0; i < collection.Count; ++i) Add(collection[i]);
++m_version;
Capacity += collection.Count;
Array.Copy(collection.m_array, 0, m_array, m_count,
collection.m_count);
m_count += collection.Count;
}
///
///
///
///
public void AddRange(UndoBlock[] array)
{
// for (int i=0; i < array.Length; ++i) Add(array[i]);
++m_version;
Capacity += array.Length;
Array.Copy(array, 0, m_array, m_count, array.Length);
m_count += array.Length;
}
// Implementation (helpers)
private void ValidateIndex(int index)
{
ValidateIndex(index, false);
}
private void ValidateIndex(int index, bool allowEqualEnd)
{
int max = (allowEqualEnd) ? (m_count) : (m_count - 1);
if (index < 0 || index > max)
throw new ArgumentOutOfRangeException(
"Index was out of range. Must be non-negative and less than the size of the collection.", index,
"Specified argument was out of the range of valid values.");
}
private bool NeedsGrowth()
{
return (m_count >= Capacity);
}
private void Grow()
{
if (NeedsGrowth())
Capacity = m_count*2;
}
private bool NeedsTrimming()
{
return (m_count <= Capacity/2);
}
private void Trim()
{
if (NeedsTrimming())
Capacity = m_count;
}
// Implementation (ICollection)
/* redundant w/ type-safe method
int ICollection.Count
{
get
{ return m_count; }
}
*/
// Nested enumerator class
#region Nested type: Enumerator
///
///
///
public class Enumerator : IEnumerator
{
private readonly UndoBlockCollection m_collection;
private readonly int m_version;
private int m_index;
// Construction
public Enumerator(UndoBlockCollection tc)
{
m_collection = tc;
m_index = - 1;
m_version = tc.m_version;
}
// Operations (type-safe IEnumerator)
///
///
///
public UndoBlock Current
{
get { return m_collection[m_index]; }
}
#region IEnumerator Members
///
///
///
///
public bool MoveNext()
{
if (m_version != m_collection.m_version)
throw new InvalidOperationException(
"Collection was modified; enumeration operation may not execute.");
++m_index;
return (m_index < m_collection.Count) ? true : false;
}
///
///
///
public void Reset()
{
if (m_version != m_collection.m_version)
throw new InvalidOperationException(
"Collection was modified; enumeration operation may not execute.");
m_index = - 1;
}
// Implementation (IEnumerator)
object IEnumerator.Current
{
get { return (Current); }
}
#endregion
/* redundant w/ type-safe method
bool IEnumerator.MoveNext()
{
return this.MoveNext();
}
*/
/* redundant w/ type-safe method
void IEnumerator.Reset()
{
this.Reset();
}
*/
}
#endregion
}
}