5

我想知道,有没有办法设置自定义DataGrid选择颜色,何时DataGrid或包含的窗口DataGrid变为非活动状态?

例如,这里是DataGridand ListBox,显示相同的数据。两个控件都有一个选定项。最初,DataGrid有输入焦点:

在此处输入图像描述

一切看起来都不错 - 中的选定项目ListBox是灰色的。然后,让我们将焦点转移到ListBox

在此处输入图像描述

现在的行为DataGrid不正确 - 选择颜色没有改变。
我知道SystemColors.HighlightBrushKeySystemColors.ControlBrushKey。此 XAML 被放置在窗口的资源中:

    <SolidColorBrush x:Key="{x:Static SystemColors.HighlightBrushKey}" Color="BlueViolet"/>
    <SolidColorBrush x:Key="{x:Static SystemColors.ControlBrushKey}" Color="DarkGray"/>

但看起来DataGrid忽略了第二个 - SystemColors.ControlBrushKey,我想DataGrid表现得像任何其他控件(ListBox, ComboBox, ListView)。

我可以用触发器实现类似的东西:

<Style TargetType="{x:Type DataGridCell}">
    <Style.Triggers>
        <MultiTrigger>
            <MultiTrigger.Conditions>
                <Condition Property="IsFocused" Value="False"/>
                <Condition Property="IsSelected" Value="True"/>
            </MultiTrigger.Conditions>
            <Setter Property="Background" Value="DarkGray"/>
        </MultiTrigger>
    </Style.Triggers>
</Style>

在此处输入图像描述

但是这个解决方案是不完整的。首先,它是灰色选中的,但未聚焦的单元格,甚至网格选择单元是FullRow. 第二件事 - 当应用程序窗口失去焦点时触发器不会触发。

有什么建议么?

更新

此错误已在 .NET 4.5 中修复,因此不再存在。

4

2 回答 2

2

我找到了解决方案,但它看起来并不优雅。

基本问题是:

  • DataGrid.IsFocused是永久false的,因为焦点有具体的单元格,而不是网格本身。
  • 无法确定单元格样式,网格中是否有任何焦点单元格。您只能测试IsFocused当前单元格。
  • 数据网格不会对父窗口的停用做出反应。

确定数据网格是否具有焦点的唯一方法是检查DataGrid.CurrentCell属性。不幸的是,它是一个结构,你不能创建一个触发器来检查这个属性{x:Null}

为了解决这些问题,我需要两个附加属性。
首先,它们旨在确定网格中是否有任何聚焦单元格。结果一定是bool,源是DataGridCellInfo,所以,首先,转换器一定要写成:

[ValueConversion(typeof(DataGridCellInfo), typeof(bool))]
public sealed class DataGridCellInfoToBooleanConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        if (value == null || value.GetType() != typeof(DataGridCellInfo) || targetType != typeof(bool))
            return DependencyProperty.UnsetValue;

        // IsValid will be false, if there's no focused cell.
        return ((DataGridCellInfo)value).IsValid;
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        return DependencyProperty.UnsetValue;
    }
}

附带属性:

    public static bool GetHasFocusedCell(DependencyObject obj)
    {
        return (bool)obj.GetValue(HasFocusedCellProperty);
    }

    public static void SetHasFocusedCell(DependencyObject obj, bool value)
    {
        obj.SetValue(HasFocusedCellProperty, value);
    }

    public static readonly DependencyProperty HasFocusedCellProperty = DependencyProperty.RegisterAttached(
        "HasFocusedCell",
        typeof(bool), 
        typeof(FocusedCellBehavior),
        new UIPropertyMetadata(false));

当网格的父窗口变为不活动时,必须更改第二个附加属性:

    public static bool GetIsParentWindowActive(DependencyObject obj)
    {
        return (bool)obj.GetValue(IsParentWindowActiveProperty);
    }

    public static void SetIsParentWindowActive(DependencyObject obj, bool value)
    {
        obj.SetValue(IsParentWindowActiveProperty, value);
    }

    public static readonly DependencyProperty IsParentWindowActiveProperty = DependencyProperty.RegisterAttached(
        "IsParentWindowActive", 
        typeof(bool), 
        typeof(FocusedCellBehavior), 
        new UIPropertyMetadata(false));

现在,让我们在 XAML 中绑定附加属性:

        <!-- A converter to define, is there any focused cell in DataGrid -->
        <local:DataGridCellInfoToBooleanConverter x:Key="DataGridCellInfoToBooleanConverter"/>

    <DataGrid Grid.Row="0" SelectionUnit="FullRow" SelectionMode="Single"
              ItemsSource="{Binding}" 
              local:FocusedCellBehavior.HasFocusedCell="{Binding CurrentCell, RelativeSource={RelativeSource Mode=Self}, Converter={StaticResource DataGridCellInfoToBooleanConverter}}"
              local:FocusedCellBehavior.IsParentWindowActive="{Binding IsActive, RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type Window}}}"/>

接下来,我需要一个单元格样式来设置适当的背景颜色:

        <!-- A style of selected cell in DataGrid, when there's no any focused cells in DataGrid -->
        <Style TargetType="{x:Type DataGridCell}" x:Key="InactiveSelectedCellStyle">
            <Style.Triggers>
                <Trigger Property="IsSelected" Value="True">
                    <Setter Property="Background" Value="{DynamicResource {x:Static SystemColors.ControlBrushKey}}"/>
                </Trigger>
            </Style.Triggers>
        </Style>

以及触发触发器的网格样式,当附加属性将更改其值时:

        <!-- 
            A style of DataGrid, that defines a couple of triggers, which being fired 
            when helper attached properties will change their values 
        -->
        <Style TargetType="{x:Type DataGrid}">
            <Style.Triggers>
                <Trigger Property="local:FocusedCellBehavior.IsParentWindowActive" Value="False">
                    <Setter Property="CellStyle" Value="{StaticResource InactiveSelectedCellStyle}"/>
                </Trigger>
                <Trigger Property="local:FocusedCellBehavior.HasFocusedCell" Value="False">
                    <Setter Property="CellStyle" Value="{StaticResource InactiveSelectedCellStyle}"/>
                </Trigger>
            </Style.Triggers>
        </Style>

有没有更好的解决方案?

于 2012-10-03T09:53:12.627 回答
2

我通过在 .Net 4.5 中使用 DataGrid.Resources 实现了这种行为

<DataGrid.Resources>
    <SolidColorBrush x:Key="{x:Static SystemColors.InactiveSelectionHighlightBrushKey}" Color="{x:Static SystemColors.HighlightColor}"/>
    <SolidColorBrush x:Key="{x:Static SystemColors.InactiveSelectionHighlightTextBrushKey}" Color="{x:Static SystemColors.HighlightTextColor}"/>

于 2016-08-04T12:07:56.667 回答