// 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 : Collection, 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 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; } } } }