单击DataGrid行中的复选框时触发PropertyChanged事件

时间:2018-02-27 20:42:35

标签: c# wpf mvvm

我有一个如下所示的DataGrid:

<DataGrid Name="Users" ItemsSource="{Binding Users, Mode=TwoWay}" 
          AutoGenerateColumns="False" Grid.ColumnSpan="2" CanUserAddRows="False">
    <DataGrid.Columns>
        <DataGridTemplateColumn>
            <DataGridTemplateColumn.Header>
                <CheckBox Name="SelectAllUsers" 
                          Checked="SelectAllUsers_Checked" 
                          Unchecked="SelectAllUsers_Unchecked"/>
            </DataGridTemplateColumn.Header>
            <DataGridTemplateColumn.CellTemplate>
                <DataTemplate>
                    <CheckBox x:Name="SelectUser" 
                              IsChecked="{Binding IsSelected, Mode=TwoWay}" 
                              Checked="SelectUser_Checked" 
                              Unchecked="SelectUser_Unchecked"/>
                </DataTemplate>
            </DataGridTemplateColumn.CellTemplate>
        </DataGridTemplateColumn>
        <DataGridTextColumn x:Name="UserName" 
                            Binding="{Binding UserName}" 
                            Header="Name" IsReadOnly="True"/>
    </DataGrid.Columns>
</DataGrid>

它目前是数据绑定到ViewModel中的BindingList

class SecurityDialogViewModel
{
    public BindingList<User> Users { get; set; }

User是一个实施INotifyPropertyChanged的DTO(Fody.PropertyChanged):

[AddINotifyPropertyChangedInterface]
class User
{
    public int UserId { get; set; }
    public string UserName { get; set; }
    public bool IsSelected { get; set; }
}

SelectAllUsers_Checked在ViewModel中有一些代码将所有复选框设置为true,然后根据Users BindingList集合中的复选框选择调用ViewModel中的方法执行一些魔法(现在全部都是选中):

private void SelectAllUsers_Checked(object sender, RoutedEventArgs e)
{
    SelectAll(Users, true);
    (DataContext as MyViewModel).PerformMagic();
}

private void SelectAll(DataGrid dataGrid, bool value)
{
    foreach (dynamic d in dataGrid.ItemsSource)
    {
        d.IsSelected = value;
    }
}

这一切都很好。但是,检查单个用户不会。基础绑定列表显然未经修改:

private void SelectUser_Checked(object sender, RoutedEventArgs e)
{
    (DataContext as MyViewModel).PerformMagic(); // Doesn't execute
}

我以为我知道为什么。在您离开记录之前,BindingList不会触发PropertyChanged事件。但是取消选中另一行上的另一个复选框(从而移出原始记录并据称触发PropertyChanged事件)也没有效果。

我真正需要的是一个集合,它会在点击其中一个DataGrid行复选框时触发PropertyChanged事件。

我该怎么做呢?我可以继承ObservableCollection并捕获其中一个事件吗?或者我需要自己写吗?也许我可以简单地将事件挂钩到现在没有被观察到的现有ObservableCollection上?

1 个答案:

答案 0 :(得分:2)

嗯,解决方案比我想象的要简单得多:

private void SelectUser_Checked(object sender, RoutedEventArgs e)
{
    var binding = (sender as CheckBox).GetBindingExpression(CheckBox.IsCheckedProperty);
    binding.UpdateSource(); // Triggers PropertyChanged Notification

    (DataContext as MyViewModel).PerformMagic();    
}

这也有效:

<CheckBox 
      x:Name="SelectUser" 
      IsChecked="{Binding IsSelected, UpdateSourceTrigger=PropertyChanged}" />

<强>注意事项

BindableList有自己的问题。

  • 您无法从DataGrid对其进行排序,这意味着您必须subclass it if you want this capability

  • 在某些情况下会导致内存泄漏;具体而言,BindingList未实现INotifyCollectionChanged,如果绑定到控件的ItemsSource属性,将导致内存泄漏。

最终我决定使用ObservableCollection并在代码隐藏中执行事件处理,而不是尝试依赖数据绑定事件。事实证明它更简单,更容易掌握,不需要任何特殊的集合子类,我得到了我想要的精细控制。

进一步阅读
The UpdateSourceTrigger property
Sortable BindingList Implementation
Another Sortable BindingList Implementation

相关问题