c# - 复选框命令在最初未选中时不触发

标签 c# wpf datagrid command checkbox

我在数据网格中有一列复选框。

<DataGridTemplateColumn CanUserResize="False" Header="" Width="auto">
    <DataGridTemplateColumn.CellTemplate>
        <DataTemplate>
            <CheckBox Style="{StaticResource CheckBoxSelectTypeStyle}" IsChecked="{Binding Path=Selected}" />
        </DataTemplate>
    </DataGridTemplateColumn.CellTemplate>
</DataGridTemplateColumn>

我有一个绑定(bind)到取消选中和选中的命令。

<Style x:Key="CheckBoxSelectTypeStyle" TargetType="{x:Type CheckBox}">
    <Setter Property="HorizontalAlignment" Value="Center" />
    <Setter Property="VerticalAlignment" Value="Center" />
    <Style.Triggers>
        <Trigger Property="IsChecked" Value="True">
            <Setter Property="Command" Value="{Binding DataContext.CheckedCommand, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}}" />
            <Setter Property="CommandParameter" Value="{Binding Path=SelectedItems, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type DataGrid}}}" />
        </Trigger>
        <Trigger Property="IsChecked" Value="False">
            <Setter Property="Command" Value="{Binding DataContext.UncheckedCommand, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}}" />
            <Setter Property="CommandParameter" Value="{Binding Path=SelectedItems, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type DataGrid}}}" />
        </Trigger>
    </Style.Triggers>
</Style>

我的问题是,如果复选框一开始是未选中的,而您选中一个,则不会触发任何命令(这就是问题所在)。如果您随后取消选中相同的复选框,uncheck 命令将触发(如预期的那样)。如果您随后再次选中同一个复选框,检查命令将触发(如预期的那样)。那时该复选框的一切都会正常工作,但其他复选框仍然有同样的问题。

如果复选框以选中状态启动,它将正常工作。我的问题是当复选框以未选中状态开始时,如何使命令触发。我找不到任何不工作的理由。

对建议的回应:

我尝试为 Null 添加触发器,但它有完全相同的问题,没有任何改变。

<Style.Triggers>
    <Trigger Property="IsChecked" Value="True">
        <Setter Property="Command" Value="{Binding DataContext.CheckedCommand, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}}" />
        <Setter Property="CommandParameter" Value="{Binding Path=SelectedItems, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type DataGrid}}}" />
    </Trigger>
    <Trigger Property="IsChecked" Value="False">
        <Setter Property="Command" Value="{Binding DataContext.UncheckedCommand, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}}" />
        <Setter Property="CommandParameter" Value="{Binding Path=SelectedItems, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type DataGrid}}}" />
    </Trigger>
    <Trigger Property="IsChecked" Value="{x:Null}">
        <Setter Property="Command" Value="{Binding DataContext.UncheckedCommand, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}}" />
        <Setter Property="CommandParameter" Value="{Binding Path=SelectedItems, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type DataGrid}}}" />
    </Trigger>
</Style.Triggers>

ViewModel 代码

public ICommand CheckedCommand { get; set; }
public ICommand UncheckedCommand { get; set; }

CheckedCommand = new RelayCommand<IList>(Checked);
UncheckedCommand = new RelayCommand<IList>(Unchecked);

private void Checked(IList selectedItems)
{
    ChangedChecked(selectedItems, true);
}

private void Unchecked(IList selectedItems)
{
    ChangedChecked(selectedItems, false);
}

private void ChangedChecked(IList selectedItems, bool selected)
{
    if (selectedItems.HasValue())
        foreach (var item in selectedItems)
            if(item is SelectedTypeModel) (item as SelectedTypeModel).Selected = selected;
}

RelayCommand 实现 ICommand。正如我之前提到的,Checked 方法不会在复选框以未选中状态开始时被调用,但它会在选中、未选中、再次选中或开始时被调用选中和未选中。

我想做什么

我有一个带有一列复选框的 DataGrid。我希望能够突出显示多行,然后选中/取消选中其中一个选定行中的 CheckBox,并将所有其他行更新为相同。我还有一个用于数据网格的关键字过滤器,因此 DataGrid 绑定(bind)到 ListCollectionView。

输出信息

我启用了数据绑定(bind)的调试信息并收到此消息:

System.Windows.Data Information: 10 : Cannot retrieve value using the binding and no valid fallback value exists; using default instead. BindingExpression:Path=Selected; DataItem=null; target element is 'CheckBox' (Name=''); target property is 'IsChecked' (type 'Nullable`1')

不过,我仍然不确定如何使用这些信息来解决问题。

解决方案

我没有解决我原来的问题,我仍然不知道为什么它不能正常工作,但我使用了另一种方法,得到了相同的预期结果。这是更改后的代码:

<Style x:Key="CheckBoxSelectTypeStyle" TargetType="{x:Type CheckBox}">
    <Setter Property="Command" Value="{Binding DataContext.CheckedChangedCommand, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}}" />
    <Setter Property="CommandParameter">
        <Setter.Value>
            <MultiBinding Converter="{StaticResource SelectedItemsCheckedMultiValueConverter}">
                <Binding RelativeSource="{RelativeSource FindAncestor, AncestorType={x:Type DataGrid}}" Path="SelectedItems"/>
                <Binding RelativeSource="{RelativeSource Self}" Path="IsChecked"/>
            </MultiBinding>
        </Setter.Value>
    </Setter>
</Style>

public ICommand CheckedChangedCommand { get; set; }

CheckedChangedCommand = new RelayCommand<Tuple<IList, bool>>(CheckedChanged);

最佳答案

解决方案 我认为你可以在不需要触发器的情况下实现你的目标(我已经在小示例应用程序中测试了这个解决方案): 所以你的风格看起来像这样:

<Style x:Key="CheckBoxSelectTypeStyle" TargetType="{x:Type CheckBox}">
    <Setter Property="Command" Value="{Binding DataContext.CheckedCommand, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type UserControl}}}" />
    <Setter Property="CommandParameter" Value="{Binding Path=SelectedItems, RelativeSource={RelativeSource FindAncestor, AncestorType={x:Type DataGrid}}}" />
  ...

您的 View 模型中将只有一个命令。作为参数,该命令将接收所选项目的列表。所以只缺少一件事——用户是要选中还是取消选中复选框。目前,我可以想到一种将此信息与 SelectedItems 一起传递的可能方法 - 使用自定义转换器编写 MultiBinding,它将 selectedItems 和 IsChecked 的当前值放入类似 Tuple<..., bool> 的内容中

        <Setter Property="CommandParameter">
            <Setter.Value>
                <MultiBinding Converter="{x:Static PackTupleConverter.Instance}">
                    <Binding RelativeSource="{RelativeSource AncestorType=DataGrid}" Path="SelectedItems"/>
                    <Binding RelativeSource="{x:Static RelativeSource.Self}" Path="IsChecked"/>
                </MultiBinding>
            </Setter.Value>
        </Setter>

结束

我能够通过简单地将相同的复选框放入 ListBox 来重现您的问题。第一次完成后,我单击复选框 - 命令不会被调用

这是示例窗口 xaml:

<Window.Resources>
    <Style x:Key="CheckBoxSelectTypeStyle" TargetType="{x:Type CheckBox}">
        <Setter Property="HorizontalAlignment" Value="Center" />
        <Setter Property="VerticalAlignment" Value="Center" />
        <Style.Triggers>
            <Trigger Property="IsChecked" Value="True">
                <Setter Property="Command" Value="{Binding DataContext.Test, RelativeSource={RelativeSource AncestorType=Window}}" />
            </Trigger>
            <Trigger Property="IsChecked" Value="False">
                <Setter Property="Command" Value="{Binding DataContext.Test, RelativeSource={RelativeSource AncestorType=Window}}" />
            </Trigger>
        </Style.Triggers>
    </Style>
</Window.Resources>
<Grid>
    <ListBox>
        <ListBox.Items>
            <CheckBox Style="{StaticResource CheckBoxSelectTypeStyle}" />
        </ListBox.Items>
    </ListBox>

</Grid>

后面是:

public partial class MainWindow : Window
{
    public MainWindow()
    {
        InitializeComponent();
        DataContext = this;
        Test = new DelegateCommand(TestCommand);
    }

    public ICommand Test { get; set; }

    private void TestCommand()
    {
    }
}

这是我在输出中发现的关于绑定(bind)的内容:

System.Windows.Data Information: 10 : Cannot retrieve value using the binding and no valid fallback value exists; using default instead. BindingExpression:Path=DataContext.Test; DataItem=null; target element is 'CheckBox' (Name=''); target property is 'Command' (type 'ICommand')

关于c# - 复选框命令在最初未选中时不触发,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/14985085/

相关文章:

c# - 显示数据表的数据

c# - WPF 与网络摄像头集成

WPF 功能区 : ToggleButton groups?

wpf - WPF Datagrid 中的排序和过滤选项?

c# - 为什么我可以对不可变的 `struct` 执行此操作?

针对 SQLite 查询结果的 C# IF 语句

C# WPF 在具有 AllowTransparency ="true"的窗口上显示 WebBrowser 不显示

c# - 如何使特定行的 WPF 数据网格单元格只读?

datagrid - 如何在dojox.grid.DataGrid中启用浏览器上下文菜单?

c# - 无法使用 Amazon SES .NET SDK 发送电子邮件