使用IBindingList在WinForms中进行数据绑定在空列表中失败
我有一个特殊的问题,实现我自己的集合应该支持IBindingList
。
我有一个特定数据类( DataItem
)的集合类( DataCollection
)。 该集合实现接口IBindingList
, IList
, IList
, DataItem
实现INotifyPropertyChanged
(并具有数据绑定的公共属性)。
当我尝试通过设置网格的DataSource
属性将集合绑定到DataGridView
, 如果集合在绑定时不为空,则它可以正常工作。 否则,如果集合为空,则在从集合中添加或删除行(即DataItems
)时,网格会注意到,但单元格保持为空。 与此问题相关的是,在AutoGenerateColumns=true
情况下,网格无法识别数据类的公共成员,并且无法生成列。
我也尝试过,使用BindingList
绑定我的DataItems
。 在这种情况下,即使列表在设置DataSource
为空,网格也能正常工作。 另一方面,如果我使用BindingList
(但与内容相同的DataItems
),行为与我的DataCollection
一样错误。 我想问题是,如果在绑定时列表为空,则数据绑定无法正确检测DataItem
类型,并且当最终项目添加到集合时,它也无法在以后恢复。
重要的是,如果集合在绑定时不为空,则它可以工作。
请注意,指定列时会发生相同的错误:
this.dataGridView.ReadOnly = true; this.dataGridView.AutoGenerateColumns = false; DataGridViewTextBoxColumn column; column = new DataGridViewTextBoxColumn(); column.DataPropertyName = "Id"; column.HeaderText = "Id"; this.dataGridView.Columns.Add(column); column = new DataGridViewTextBoxColumn(); column.DataPropertyName = "UserName"; column.HeaderText = "UserName"; this.dataGridView.Columns.Add(column); this.dataGridView.DataSource = myList;
我也尝试在我的IBindingList
的AllowNew
上返回true
。 这没有可观察到的影响。
还失败的是以下内容:
var bindingSource = new BindingSource(); bindingSource.DataSource = myList; this.dataGridView.DataSource = bindingSource;
问题是,我怎样才能告诉绑定机制识别我的DataItems
?
(谢谢)
更新1:
我做了一个小测试项目,显示了这个问题:
public partial class Form1: Form { public Form1() { InitializeComponent(); } class DataItem: INotifyPropertyChanged { private int _id; public int Id { get { return _id; } set { if (value != _id) { _id = value; OnPropertyChanged("Id"); } } } private string _userName; public string UserName { get { return _userName; } set { if (value != _userName) { _userName = value; OnPropertyChanged("UserName"); } } } private void OnPropertyChanged(string propertyName) { var handler = PropertyChanged; if (handler != null) { handler(this, new PropertyChangedEventArgs(propertyName)); } } public event PropertyChangedEventHandler PropertyChanged; } /// Make a list of type DataItem or object... //BindingList list = new BindingList() { BindingList list = new BindingList() { //new DataItem() { // Id = 1, // UserName = "testuser" //} }; private void Form1_Load(object sender, EventArgs e) { DataGridView dataGridView = new System.Windows.Forms.DataGridView(); dataGridView.Size = new Size(this.Width-20, this.Height-30); dataGridView.AutoGenerateColumns = true; DataGridViewTextBoxColumn column = new DataGridViewTextBoxColumn(); column.DataPropertyName = "Id"; column.HeaderText = "Id"; dataGridView.Columns.Add(column); this.Controls.Add(dataGridView); dataGridView.DataSource = list; list.Add( new DataItem() { Id = 3, UserName = "admin" } ); // Make some modifications on the data... (new System.Threading.Thread( state => { System.Threading.Thread.CurrentThread.IsBackground = true; System.Threading.Thread.Sleep(2000); this.Invoke( (Action)( () => { list.Add(new DataItem() { Id = 2, UserName = "guest" }); } ) ); System.Threading.Thread.Sleep(2000); this.Invoke( (Action)( () => { DataItem user = (list.First( obj => ((DataItem)obj).Id == 3 )) as DataItem; user.UserName = "Administrator"; } ) ); })).Start(); } }
如果列表的类型是BindingList
则它可以正常工作。 如果类型为BindingList
则仅在初始化DataSource
时列表不为空时才有效。
数据绑定将首先查看列表项以尝试获取其属性,但是对于空列表,它将从列表项的Type
中获取其所有信息。 如果使用空BindingList
,数据绑定无法发现任何属性的原因是该object
没有可绑定属性。
要完全确保DataCollection
类正确支持绑定(即使为空),请实现ITypedList
接口。 它包含方法GetItemProperties()
,它允许您显式声明哪些属性是可绑定的。 在此方法中,您可以使用以下命令返回DataItem
上的属性:
return TypeDescriptor.GetProperties(typeof(DataItem));
这样,即使集合为空,数据绑定也会知道要显示的属性。