5

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

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

我有一个绑定到取消选中和检查的命令。

<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>

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

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

对建议的回应:

我试图添加一个触发器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>

视图模型代码

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 绑定到 ListCollectionView。

输出信息

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

System.Windows.Data 信息:10:无法使用绑定检索值,并且不存在有效的备用值;改用默认值。BindingExpression:路径=选定;数据项=空;目标元素是'CheckBox'(名称='');目标属性是“IsChecked”(类型“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);
4

1 回答 1

1

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

<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}}}" />
  ...

您的视图模型中只有一个命令。作为参数,该命令将接收所选项目的列表。所以只缺少一件事——用户是要选中还是取消选中复选框。目前,我可以想到一种与 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()
    {
    }
}

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

System.Windows.Data 信息:10:无法使用绑定检索值,并且不存在有效的备用值;改用默认值。BindingExpression:Path=DataContext.Test; 数据项=空;目标元素是'CheckBox'(名称='');目标属性是“命令”(类型“ICommand”)

于 2013-02-20T20:00:24.133 回答