Donnerstag, 12. Januar 2012

INotifyCollectionChanged and INotifyPropertyChanged when ItemsControl.ItemsSource is databound

When an ItemsControl should update its elements when the collection that is used as ItemsSource via Binding is changed in code behind (best in the Model when using MVVM), you have to implement the INotifyCollectionChanged interface or you have to use a collection which already implements it (like ObservableCollection<>).

Also you will have to implement the INotifyPropertyChanged in the items class.
Here is a small sample how this can work:

First of all there is some base class implementing INotifyPropertyChanged, and a class (the items of the collection) which has some properties using the implementation:
  public class ViewModelBase : INotifyPropertyChanged
  {
    #region INotifyPropertyChanged
    protected void OnPropertyChanged(string propertyName)
    {
      if (PropertyChanged != null)
        PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
    }
    public event PropertyChangedEventHandler PropertyChanged;
    #endregion INotifyPropertyChanged
  }


  public class CollectionItem : ViewModelBase
  {
    private string _sSampleString;
    public string SampleString
    {
      get { return this._sSampleString; }
      set
      {
        if (this._sSampleString != value)
        {
          this._sSampleString = value;
          this.OnPropertyChanged("SampleString");
        }
      }
    }
    // ... other properties ...
  }


This way we take care, that every change made in code behind for every item will make the bindings in our ItemsControls Items (user interface, UI) refresh.

Next we have to implement a collection which uses the INotifyCollectionChanged interface (or, as said before, use some given collection which already implements it):

  public class CollectionSample : INotifyCollectionChanged, IEnumerable
  {
    private List<CollectionItem> _lstItems
      = new List<CollectionItem>();

    public void Add(CollectionItem item)
    {
      this._lstItems.Add(item);
      this.OnNotifyCollectionChanged(
        new NotifyCollectionChangedEventArgs(
          NotifyCollectionChangedAction.Add, item));
    }

    public void Remove(CollectionItem item)
    {
      this._lstItems.Remove(item);
      this.OnNotifyCollectionChanged(
        new NotifyCollectionChangedEventArgs(
          NotifyCollectionChangedAction.Remove, item));
    }

    // ... other actions for the collection ...

    public CollectionItem this[Int32 index]
    {
      get
      {
        return this._lstItems[index];
      }
    }

    #region INotifyCollectionChanged
    private void OnNotifyCollectionChanged(NotifyCollectionChangedEventArgs args)
    {
      if (this.CollectionChanged != null)
      {
        this.CollectionChanged(this, args);
      }
    }
    public event NotifyCollectionChangedEventHandler CollectionChanged;
    #endregion INotifyCollectionChanged

    #region IEnumerable
    public List<CollectionItem>.Enumerator GetEnumerator()
    {
      return this._lstItems.GetEnumerator();
    }
    IEnumerator IEnumerable.GetEnumerator()
    {
      return (IEnumerator)this.GetEnumerator();
    }
    #endregion IEnumerable
  }


Also there is an implementation of the IEnumerator interface and a this-pointer. But our main view is on the INotifyCollectionChanged interface:

For every important event that happens with the items there should be a method, which performs it. But we do not have to take care of the properties of the items itself. This is managed by implementing the INotifyPropertyChanged interface.

So let’s just make some XAML code and not very usefull code behind to see how it works:
<Window x:Class="WpfApplication1.View"
        xmlns:util="clr-namespace:WpfApplication1"
        xmlns:controls="clr-namespace:System.Windows.Controls;assembly=PresentationFramework"
        Initialized="Window_Initialized"
        Title="MainWindow" Height="400" Width="600">
  <Grid>
    <Grid.RowDefinitions>
      <RowDefinition Height="*"/>
      <RowDefinition Height="Auto"/>
    </Grid.RowDefinitions>
    <ListView Name="listview"
              Grid.Row="0">
      <ListView.ItemTemplate>
        <DataTemplate>
          <TextBox Text="{Binding SampleString}"/>
        </DataTemplate>
      </ListView.ItemTemplate>
    </ListView>
    <Button Grid.Row="1"
            Content="refresh"
            Click="Button_Click"/>
  </Grid>
</Window>


  public partial class View : Window
  {
    public View()
    {
      InitializeComponent();
    }

    private void Window_Initialized(object sender, EventArgs e)
    {
      this.listview.ItemsSource = new CollectionSample();
    }

    private void Button_Click(object sender, RoutedEventArgs e)
    {
      (this.listview.ItemsSource as CollectionSample).Add(
        new CollectionItem { SampleString = DateTime.Now.ToString() });
    }
  }
You can manipulate everything in code behind and it will directly be recognized by the UI.
And that’s it J.

Mittwoch, 11. Januar 2012

DataGrid with DataGridCheckBoxColumn and Header to select/deselect all CheckBoxes

The topic says what our target is:


Because we’d like to stick to the MVVM pattern, first of all there should be a ViewModel base class which implements the INotifyPropertyChanged-interface:


  public class ViewModelBase : INotifyPropertyChanged
  {
    #region INotifyPropertyChanged
    protected void OnPropertyChanged(string propertyName)
    {
      if (PropertyChanged != null)
        PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
    }
    public event PropertyChangedEventHandler PropertyChanged;
    #endregion INotifyPropertyChanged
  }

Next there is a basic class for each row:

  public class RowViewModelClass : ViewModelBase
  {
    private bool _bSelected = false;
    public bool Selected
    {
      get { return this._bSelected; }
      set
      {
        if (this._bSelected != value)
        {
          this._bSelected = value;
          this.OnPropertyChanged("Selected");
        }
      }
    }

    // all the other stuff for each Row...
  }

To save some work (and for simplicity) we implement an ObservableCollection of that basic class (based on http://dzaebel.net/WpfObservable.htm):


  public class RowCollection : ObservableCollection<RowViewModelClass>
  {
    protected override void InsertItem(int index, RowViewModelClass item)
    {
      base.InsertItem(index, item);
      this[index].PropertyChanged += new PropertyChangedEventHandler(RowCollection_PropertyChanged);
    }

    void RowCollection_PropertyChanged(object sender, PropertyChangedEventArgs e)
    {
      if (this.ItemPropertyChanged != null)
        this.ItemPropertyChanged(sender as RowViewModelClass, e.PropertyName);
    }

    public event Action<RowViewModelClass, string> ItemPropertyChanged;
  }



Some notes: we’d like to react to changes of property Selected of each row. That means: if all CheckBoxes of all rows are selected, and one is deselected, our header for selecting all rows should be deselected, too.
We achieve this by implementing an Action which fires/is executed every time a property of an item is changed (we ignore that this implementation is very basic…removing items for example isn’t even mentioned).

So finally we have our class with the flag to select/deselect all row-values:

  public class ViewModel : ViewModelBase
  {
    private bool _bAllSelected = false;
    public bool AllSelected
    {
      get { return this._bAllSelected; }
      set
      {
        if (this._bAllSelected != value)
        {
          this._bAllSelected = value;
          this.OnPropertyChanged("AllSelected");
        }
      }
    }

    private RowCollection _rcRows = new RowCollection();
    public RowCollection Rows
    {
      get { return this._rcRows; }
    }
  }
That’s our ViewModel. Now follows the Model:
  public class Model
  {
    private ViewModel _viewmodel;

    public void SetViewModel(ViewModel viewmodel)
    {
      this._viewmodel = viewmodel;
      this._viewmodel.PropertyChanged += new PropertyChangedEventHandler(_viewmodel_PropertyChanged);
      this._viewmodel.Rows.ItemPropertyChanged += this.Rows_ItemPropertyChanged;
    }

    void Rows_ItemPropertyChanged(RowViewModelClass item, string property)
    {
      if (property == "Selected"
        && item.Selected == false)
        this._viewmodel.AllSelected = false;
    }

    void  _viewmodel_PropertyChanged(object sender, PropertyChangedEventArgs e)
    {
      if (e.PropertyName == "AllSelected")
      {
        if (this._viewmodel.AllSelected == true)
          foreach (var item in this._viewmodel.Rows)
            item.Selected = true;
        else
        {
          bool bAllSelected = true;
          foreach (var item in this._viewmodel.Rows)
          {
            if (item.Selected == false)
              bAllSelected = false;
          }
          if (bAllSelected)
            foreach (var item in this._viewmodel.Rows)
              item.Selected = false;
        }
      }
    }
  }

Very basic…we register to all events that are important to us.
The last step: our View:

<Window x:Class="WpfApplication1.View"
        xmlns:util="clr-namespace:WpfApplication1"
        xmlns:controls="clr-namespace:System.Windows.Controls;assembly=PresentationFramework"
        Title="MainWindow" Height="400" Width="600">
  <DataGrid AlternationCount="2"
            AutoGenerateColumns="False"
            CanUserAddRows="False"
            ItemsSource="{Binding Rows}">
    <!--Column Definitions-->
    <DataGrid.Columns>
      <DataGridCheckBoxColumn Binding="{Binding Selected, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}">
        <DataGridCheckBoxColumn.Header>
          <DockPanel>
            <Border Name="border"
                    Background="#BAFFFFFF">
              <CheckBox IsChecked="{Binding AllSelected}"/>
            </Border>
          </DockPanel>
        </DataGridCheckBoxColumn.Header>
      </DataGridCheckBoxColumn>
    </DataGrid.Columns>
  </DataGrid>
</Window>

With code behind:

  public partial class View : Window
  {
    public View()
    {
      InitializeComponent();
    }

    public void SetViewModel(ViewModel viewmodel)
    {
      this.DataContext = viewmodel;
      this.border.DataContext = viewmodel;
    }
  }

Now here I have to mention two lines:

      this.DataContext = viewmodel;
      this.border.DataContext = viewmodel;

Why that? Well, when you leave line two, then your Binding in the Header won’t work. The Binding in XAML will not find any DataContext beyond the Border. Also all the stuff with RelativeSource and so on won’t work, the property DataGridCheckBoxColumn.Header seems to be a bit primitive for WPF:
public Object Header { get; set; }
GroupBox.Header for example looks like this:
[BindableAttribute(true)]
[LocalizabilityAttribute(LocalizationCategory.Label)]
public Object Header { get; set; } 
This way or the other: this is my solution for this.