387 lines
9.4 KiB
C#
387 lines
9.4 KiB
C#
|
// Permission is hereby granted, free of charge, to any person obtaining
|
||
|
// a copy of this software and associated documentation files (the
|
||
|
// "Software"), to deal in the Software without restriction, including
|
||
|
// without limitation the rights to use, copy, modify, merge, publish,
|
||
|
// distribute, sublicense, and/or sell copies of the Software, and to
|
||
|
// permit persons to whom the Software is furnished to do so, subject to
|
||
|
// the following conditions:
|
||
|
//
|
||
|
// The above copyright notice and this permission notice shall be
|
||
|
// included in all copies or substantial portions of the Software.
|
||
|
//
|
||
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
||
|
// EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
||
|
// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
||
|
// NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
||
|
// LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
||
|
// OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
||
|
// WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
||
|
//
|
||
|
// Copyright (c) 2007 Novell, Inc.
|
||
|
//
|
||
|
|
||
|
using System;
|
||
|
using System.Collections;
|
||
|
using System.Collections.Generic;
|
||
|
using System.Collections.ObjectModel;
|
||
|
using System.Reflection;
|
||
|
|
||
|
namespace System.ComponentModel {
|
||
|
[SerializableAttribute]
|
||
|
public class BindingList<T> : Collection<T>,
|
||
|
IBindingList, IList, ICollection,
|
||
|
IEnumerable, ICancelAddNew, IRaiseItemChangedEvents
|
||
|
{
|
||
|
bool allow_edit = true;
|
||
|
bool allow_remove = true;
|
||
|
bool allow_new;
|
||
|
bool allow_new_set;
|
||
|
|
||
|
bool raise_list_changed_events = true;
|
||
|
|
||
|
bool type_has_default_ctor;
|
||
|
bool type_raises_item_changed_events;
|
||
|
|
||
|
bool add_pending;
|
||
|
int pending_add_index;
|
||
|
|
||
|
void CheckType ()
|
||
|
{
|
||
|
ConstructorInfo ci = typeof (T).GetConstructor (Type.EmptyTypes);
|
||
|
type_has_default_ctor = (ci != null);
|
||
|
type_raises_item_changed_events = typeof (INotifyPropertyChanged).IsAssignableFrom (typeof (T));
|
||
|
}
|
||
|
|
||
|
public BindingList (IList<T> list) : base(list)
|
||
|
{
|
||
|
CheckType ();
|
||
|
}
|
||
|
|
||
|
public BindingList () : base ()
|
||
|
{
|
||
|
CheckType ();
|
||
|
}
|
||
|
|
||
|
public bool AllowEdit {
|
||
|
get { return allow_edit; }
|
||
|
set {
|
||
|
if (allow_edit != value) {
|
||
|
allow_edit = value;
|
||
|
|
||
|
if (raise_list_changed_events)
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.Reset, -1 /* XXX */));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public bool AllowNew {
|
||
|
get {
|
||
|
/* if the user explicitly it, return that value */
|
||
|
if (allow_new_set)
|
||
|
return allow_new;
|
||
|
|
||
|
/* if the list type has a default constructor we allow new */
|
||
|
if (type_has_default_ctor)
|
||
|
return true;
|
||
|
|
||
|
/* if the user adds a delegate, we return true even if
|
||
|
the type doesn't have a default ctor */
|
||
|
if (AddingNew != null)
|
||
|
return true;
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
set {
|
||
|
// this funky check (using AllowNew
|
||
|
// instead of allow_new allows us to
|
||
|
// keep the logic for the 3 cases in
|
||
|
// one place (the getter) instead of
|
||
|
// spreading them around the file (in
|
||
|
// the ctor, in the AddingNew add
|
||
|
// handler, etc.
|
||
|
if (AllowNew != value) {
|
||
|
allow_new_set = true;
|
||
|
|
||
|
allow_new = value;
|
||
|
|
||
|
if (raise_list_changed_events)
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.Reset, -1 /* XXX */));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public bool AllowRemove {
|
||
|
get { return allow_remove; }
|
||
|
set {
|
||
|
if (allow_remove != value) {
|
||
|
allow_remove = value;
|
||
|
|
||
|
if (raise_list_changed_events)
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.Reset, -1 /* XXX */));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected virtual bool IsSortedCore {
|
||
|
get { return false; }
|
||
|
}
|
||
|
|
||
|
public bool RaiseListChangedEvents {
|
||
|
get { return raise_list_changed_events; }
|
||
|
set { raise_list_changed_events = value; }
|
||
|
}
|
||
|
|
||
|
protected virtual ListSortDirection SortDirectionCore {
|
||
|
get { return ListSortDirection.Ascending; }
|
||
|
}
|
||
|
|
||
|
protected virtual PropertyDescriptor SortPropertyCore {
|
||
|
get { return null; }
|
||
|
}
|
||
|
|
||
|
protected virtual bool SupportsChangeNotificationCore {
|
||
|
get { return true; }
|
||
|
}
|
||
|
|
||
|
protected virtual bool SupportsSearchingCore {
|
||
|
get { return false; }
|
||
|
}
|
||
|
|
||
|
protected virtual bool SupportsSortingCore {
|
||
|
get { return false; }
|
||
|
}
|
||
|
|
||
|
public event AddingNewEventHandler AddingNew;
|
||
|
public event ListChangedEventHandler ListChanged;
|
||
|
|
||
|
public T AddNew ()
|
||
|
{
|
||
|
return (T)AddNewCore ();
|
||
|
}
|
||
|
|
||
|
protected virtual object AddNewCore ()
|
||
|
{
|
||
|
if (!AllowNew)
|
||
|
throw new InvalidOperationException ();
|
||
|
|
||
|
AddingNewEventArgs args = new AddingNewEventArgs ();
|
||
|
|
||
|
OnAddingNew (args);
|
||
|
|
||
|
T new_obj = (T)args.NewObject;
|
||
|
if (new_obj == null) {
|
||
|
if (!type_has_default_ctor)
|
||
|
throw new InvalidOperationException ();
|
||
|
|
||
|
new_obj = (T)Activator.CreateInstance (typeof (T));
|
||
|
}
|
||
|
|
||
|
Add (new_obj);
|
||
|
pending_add_index = IndexOf (new_obj);
|
||
|
add_pending = true;
|
||
|
|
||
|
return new_obj;
|
||
|
}
|
||
|
|
||
|
protected virtual void ApplySortCore (PropertyDescriptor prop, ListSortDirection direction)
|
||
|
{
|
||
|
throw new NotSupportedException ();
|
||
|
}
|
||
|
|
||
|
public virtual void CancelNew (int itemIndex)
|
||
|
{
|
||
|
if (!add_pending)
|
||
|
return;
|
||
|
|
||
|
if (itemIndex != pending_add_index)
|
||
|
return;
|
||
|
|
||
|
add_pending = false;
|
||
|
|
||
|
base.RemoveItem (itemIndex);
|
||
|
|
||
|
if (raise_list_changed_events)
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.ItemDeleted, itemIndex));
|
||
|
}
|
||
|
|
||
|
protected override void ClearItems ()
|
||
|
{
|
||
|
EndNew (pending_add_index);
|
||
|
if (type_raises_item_changed_events) {
|
||
|
foreach ( T item in base.Items ) {
|
||
|
(item as INotifyPropertyChanged).PropertyChanged -= Item_PropertyChanged;
|
||
|
}
|
||
|
}
|
||
|
base.ClearItems ();
|
||
|
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.Reset, -1));
|
||
|
}
|
||
|
|
||
|
public virtual void EndNew (int itemIndex)
|
||
|
{
|
||
|
if (!add_pending)
|
||
|
return;
|
||
|
|
||
|
if (itemIndex != pending_add_index)
|
||
|
return;
|
||
|
|
||
|
add_pending = false;
|
||
|
}
|
||
|
|
||
|
protected virtual int FindCore (PropertyDescriptor prop, object key)
|
||
|
{
|
||
|
throw new NotSupportedException ();
|
||
|
}
|
||
|
|
||
|
protected override void InsertItem (int index, T item)
|
||
|
{
|
||
|
EndNew (pending_add_index);
|
||
|
|
||
|
base.InsertItem (index, item);
|
||
|
|
||
|
if (raise_list_changed_events)
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.ItemAdded, index));
|
||
|
|
||
|
if (item != null && type_raises_item_changed_events)
|
||
|
(item as INotifyPropertyChanged).PropertyChanged += Item_PropertyChanged;
|
||
|
}
|
||
|
|
||
|
void Item_PropertyChanged (object item, PropertyChangedEventArgs args)
|
||
|
{
|
||
|
var property_info = item.GetType ().GetProperty (args.PropertyName);
|
||
|
|
||
|
if (property_info != null) {
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.ItemChanged, base.IndexOf ((T) item),
|
||
|
new ReflectionPropertyDescriptor (property_info)) );
|
||
|
} else {
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.ItemChanged, base.IndexOf ((T) item)) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
protected virtual void OnAddingNew (AddingNewEventArgs e)
|
||
|
{
|
||
|
if (AddingNew != null)
|
||
|
AddingNew (this, e);
|
||
|
}
|
||
|
|
||
|
protected virtual void OnListChanged (ListChangedEventArgs e)
|
||
|
{
|
||
|
if (ListChanged != null)
|
||
|
ListChanged (this, e);
|
||
|
}
|
||
|
|
||
|
protected override void RemoveItem (int index)
|
||
|
{
|
||
|
if (!AllowRemove)
|
||
|
throw new NotSupportedException ();
|
||
|
|
||
|
EndNew (pending_add_index);
|
||
|
if (type_raises_item_changed_events) {
|
||
|
(base[index] as INotifyPropertyChanged).PropertyChanged -= Item_PropertyChanged;
|
||
|
}
|
||
|
base.RemoveItem (index);
|
||
|
|
||
|
if (raise_list_changed_events)
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.ItemDeleted, index));
|
||
|
}
|
||
|
|
||
|
protected virtual void RemoveSortCore ()
|
||
|
{
|
||
|
throw new NotSupportedException ();
|
||
|
}
|
||
|
|
||
|
public void ResetBindings ()
|
||
|
{
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.Reset, -1));
|
||
|
}
|
||
|
|
||
|
public void ResetItem (int position)
|
||
|
{
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.ItemChanged, position));
|
||
|
}
|
||
|
|
||
|
protected override void SetItem (int index, T item)
|
||
|
{
|
||
|
if (type_raises_item_changed_events) {
|
||
|
(base[index] as INotifyPropertyChanged).PropertyChanged -= Item_PropertyChanged;
|
||
|
(item as INotifyPropertyChanged).PropertyChanged += Item_PropertyChanged;
|
||
|
}
|
||
|
base.SetItem (index, item);
|
||
|
|
||
|
OnListChanged (new ListChangedEventArgs (ListChangedType.ItemChanged, index));
|
||
|
}
|
||
|
|
||
|
void IBindingList.AddIndex (PropertyDescriptor index)
|
||
|
{
|
||
|
/* no implementation by default */
|
||
|
}
|
||
|
|
||
|
object IBindingList.AddNew ()
|
||
|
{
|
||
|
return AddNew ();
|
||
|
}
|
||
|
|
||
|
void IBindingList.ApplySort (PropertyDescriptor property, ListSortDirection direction)
|
||
|
{
|
||
|
ApplySortCore (property, direction);
|
||
|
}
|
||
|
|
||
|
int IBindingList.Find (PropertyDescriptor property, object key)
|
||
|
{
|
||
|
return FindCore (property, key);
|
||
|
}
|
||
|
|
||
|
void IBindingList.RemoveIndex (PropertyDescriptor property)
|
||
|
{
|
||
|
/* no implementation by default */
|
||
|
}
|
||
|
|
||
|
void IBindingList.RemoveSort ()
|
||
|
{
|
||
|
RemoveSortCore ();
|
||
|
}
|
||
|
|
||
|
bool IBindingList.IsSorted {
|
||
|
get { return IsSortedCore; }
|
||
|
}
|
||
|
|
||
|
ListSortDirection IBindingList.SortDirection {
|
||
|
get { return SortDirectionCore; }
|
||
|
}
|
||
|
|
||
|
PropertyDescriptor IBindingList.SortProperty {
|
||
|
get { return SortPropertyCore; }
|
||
|
}
|
||
|
|
||
|
bool IBindingList.AllowEdit {
|
||
|
get { return AllowEdit; }
|
||
|
}
|
||
|
|
||
|
bool IBindingList.AllowNew {
|
||
|
get { return AllowNew; }
|
||
|
}
|
||
|
|
||
|
bool IBindingList.AllowRemove {
|
||
|
get { return AllowRemove; }
|
||
|
}
|
||
|
|
||
|
bool IBindingList.SupportsChangeNotification {
|
||
|
get { return SupportsChangeNotificationCore; }
|
||
|
}
|
||
|
|
||
|
bool IBindingList.SupportsSearching {
|
||
|
get { return SupportsSearchingCore; }
|
||
|
}
|
||
|
|
||
|
bool IBindingList.SupportsSorting {
|
||
|
get { return SupportsSortingCore; }
|
||
|
}
|
||
|
|
||
|
bool IRaiseItemChangedEvents.RaisesItemChangedEvents {
|
||
|
get { return type_raises_item_changed_events; }
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
|