20
votes

I have a DataGridView bound to a BindingList (C# Windows Forms). If I change one of the values in an item in the list it does not immediately show up in the grid. If I click on the changed cell, or minimize then maximize the window it updates properly, but I need it to happen automatically.

I had the same problem earlier, but in that situation I had to change the cell's background colour at the same time that the value changed. This caused the cell to refresh correctly.

The only way I can get it to work is...

dataGridView.DataSource = null;  
dataGridView.DataSource = myBindingList

...but I'd really like to avoid this as it makes the scrollbar pop back to the top, and means that I'd have to set my cell background colours again. Surely there's a better way. I've tried Refresh (as well as refreshing the parent), Update, and Invalidate, but they're not doing what I need.

I've seen this problem mentioned on a few message boards, but haven't seen a working answer to it yet.

5

5 Answers

24
votes

ListChanged notifications for item value changes are only raised if the list item type implements the INotifyPropertyChanged interface.

Taken from: http://msdn.microsoft.com/en-us/library/ms132742.aspx

So my first question would be: Implements your item the INotifyPropertyChanged properly?

8
votes

Your datasource should implement INotifyPropertyChanged for any change in the BindingList to be reflected in the datagridview.

class Books : INotifyPropertyChanged
{
   private int m_id;
   private string m_author;
   private string m_title;

   public int ID { get { return m_id; } set { m_id = value; NotifyPropertyChanged("ID"); } }
   public string Author { get { return m_author; } set { m_author = value; NotifyPropertyChanged("Author"); } }
   public string Title { get { return m_title; } set { m_title = value; NotifyPropertyChanged("Title"); } }


   public event PropertyChangedEventHandler PropertyChanged;

   private void NotifyPropertyChanged(string p)
   {
       if (PropertyChanged != null)
           PropertyChanged(this, new PropertyChangedEventArgs(p));
   }
}

BindingList<Books> books= new BindingList<Books>();

datagridView.DataSource = books;
6
votes

Just call myBindingList.ResetBindings() whenever your data changes!

1
votes

It sounds as your Change Object notification is not triggered / handled correctly. I personally always use the BindingSource object when binding to a dataGridView.

I would check out the DataGridView FAQ and the DataBinding FAQ and search for object change notification.

If you are using ADO.Net, don't forget the call the .Validate() and .EndEdit() methods.

0
votes
    private void refreshDataGrid()
    {
        dataGridView1.DataSource = typeof(List<>);
        dataGridView1.DataSource = myBindingList;
        dataGridView1.AutoResizeColumns();
        dataGridView1.Refresh();
    }

Then, just call for the refreshDataGrid Method anytime a change occurs to your list.