Loads of times I have been asked in forums, how do I make a GridViewColumn communicate with another in WPF. A common use case is that you have a checkbox in Column1 and you want that when the checkbox is checked something happens in Columnx.
The first obvious answer is make a property in your business object that can reflect that change. Yet some times this is strictly UI and you don’t want to clutter your business object with UI code.
One solution could be to use the power of DataBinding. Lets start by explaining in very high level, how a GridView works. So a GridView is simply a Logical entity. To make it sound more familiar we can say that a GridViewColumn is like Grid ColumnDefinition / RowDefinition. It’s an entity that will effect how the parent (in our case the ListView) will look. A GridView will construct a GridViewHeaderRowPresenter and a GridViewRowPresenter that will construct the header of the ListView and the ListView rows. Because the GridView does this, we can do a trick with databinding that will enable us to make Inter-Column communication possible.
Basically the idea is to bind the CheckBox’s IsChecked property to the Tag property of it’s GridViewRowPresenter with a OneWayToSource binding. This will make sure that when the user check/unchecks the Checkbox the IsChecked state is stored in the GridViewRowPresenter. something like this
1: <GridViewColumn Header="checkbox here" Width="100">
2: <GridViewColumn.CellTemplate>
3: <DataTemplate>
4: <CheckBox IsChecked="{Binding RelativeSource={RelativeSource AncestorType={x:Type GridViewRowPresenter}}, Path=Tag, Mode=OneWayToSource}"/>
5: </DataTemplate>
6: </GridViewColumn.CellTemplate>
7: </GridViewColumn>
Now that we have that in place other columns can easily become aware of the state changes by doing a DataTrigger on the GridViewRowPresenter like so
1: <GridViewColumn Header="text here" Width="100">
2: <GridViewColumn.CellTemplate>
3: <DataTemplate>
4: <TextBlock Text="{Binding}" Foreground="Red" x:Name="text"/>
5: <DataTemplate.Triggers>
6: <DataTrigger Binding="{Binding RelativeSource={RelativeSource AncestorType={x:Type GridViewRowPresenter}}, Path=Tag}" Value="True">
7: <Setter TargetName="text" Property="Foreground" Value="Green"/>
8: </DataTrigger>
9: </DataTemplate.Triggers>
10: </DataTemplate>
11: </GridViewColumn.CellTemplate>
12: </GridViewColumn>
and that’s basically it…
Thank you WPF!!!! We love you 😀