DataGridRow的背景属性是错误的

时间:2013-02-27 20:27:49

标签: wpf xaml datagrid triggers styles

我有一个使用DataGrid控件的WPF应用程序,并被警车警察使用。我使用合并的词典来实现日夜“模式”,当您在两者之间切换程序时,调色板会发生变化。该应用程序从我公司制作的特殊传感器收集数据并显示给官员。

有问题的DataGrid表现得很奇怪。程序首次启动时,它最初为空。收集数据后,行将添加到DataGrid。启动程序时,它最初处于日间模式。问题是第一行的背景不会更改为控件的夜间模式颜色。它保持白色,这是白天模式的颜色。如果您在白天模式和夜间模式之间来回切换,它会保持白色。

这与添加到DataGrid之后的任何行不同,后者具有正确的颜色并在颜色之间来回切换。

这是我在App.xaml中为DataGridRow类定义的样式:

<Application x:Class="MyApplication.App"
     . . .>

    <Application.Resources>
        <ResourceDictionary>
            <ResourceDictionary.MergedDictionaries>
                <ResourceDictionary Source="pack://application:,,,/MyApplication;component/DayTime.xaml" />

                <ResourceDictionary>
                    . . .
                    <Style TargetType="{x:Type DataGridRow}">
                    <Setter Property="BorderBrush"     Value="{DynamicResource DataBorder}" />
                    <Setter Property="Background"      Value="{DynamicResource DataBackground}" />
                    <Setter Property="Foreground"      Value="{DynamicResource DataForeground}" />
                <Style.Triggers>
                    <Trigger Property="IsFocused" Value="True">
                        <Setter Property="Background"  Value="{DynamicResource DataBackground}" />
                        <Setter Property="BorderBrush" Value="{DynamicResource DataBorderFocused}" />
                        <Setter Property="Foreground"  Value="{DynamicResource DataForeground}" />
                    </Trigger>
                    <Trigger Property="IsKeyboardFocused" Value="True">
                        <Setter Property="Background"  Value="{DynamicResource DataBackground}" />
                        <Setter Property="BorderBrush" Value="{DynamicResource DataBorderFocused}" />
                        <Setter Property="Foreground"  Value="{DynamicResource DataForeground}" />
                    </Trigger>
                    <Trigger Property="IsSelected" Value="True">
                        <Setter Property="Background"  Value="{DynamicResource DataBackgroundSelected}" />
                        <Setter Property="BorderBrush" Value="{DynamicResource DataBorderSelected}" />
                        <Setter Property="Foreground"  Value="{DynamicResource DataForegroundSelected}" />
                    </Trigger>
                </Style.Triggers>
            </Style>
            . . .
                </ResourceDictionary>
            </ResourceDictionary>
        </ResourceDictionary>
    </Application.Resources>
</Application>

当我在程序上运行Snoop并向下钻取到有问题的DataGridRow时,Background属性的值为白色(#FFFFFFFF),并且它的Value Source设置为“DefaultStyle”。这似乎不是我定义的风格,因为当我切换到白天模式时它不会改变。回到白色。我认为这是微软定义的默认样式,它根本不使用我的风格。但只有在第一行插入DataGrid时,如果它最初是空的。

对于后续行,“值源”列将显示“ParentTemplate”。这必须是我的风格,因为您切换夜间模式时背景颜色会正确更改。

如何解决此问题,以便DataGrid中的每一行都正确?

编辑:

为了完整性,这里是DataGrid控件使用的样式,以防它有用。

<Style TargetType="{x:Type DataGrid}">
    <Setter Property="Background"                    Value="{DynamicResource DataBackground}" />
    <Setter Property="Foreground"                    Value="{DynamicResource TextForeground}" />
    <Setter Property="BorderBrush"                   Value="{DynamicResource DataBorder}" />
    <Setter Property="BorderThickness"               Value="1" />
    <Setter Property="RowDetailsVisibilityMode"      Value="VisibleWhenSelected" />
    <Setter Property="ScrollViewer.CanContentScroll" Value="true" />
    <Setter Property="ScrollViewer.PanningMode"      Value="Both" />
    <Setter Property="Stylus.IsFlicksEnabled"        Value="False" />
    <Setter Property="Template">
        <Setter.Value>
            <ControlTemplate TargetType="{x:Type DataGrid}">
                <Border BorderBrush="{TemplateBinding BorderBrush}"
                        BorderThickness="{TemplateBinding BorderThickness}"
                        Background="{TemplateBinding Background}"
                        Padding="{TemplateBinding Padding}"
                        SnapsToDevicePixels="True">
                    <ScrollViewer x:Name="DG_ScrollViewer" Focusable="false">
                        <ScrollViewer.Template>
                            <ControlTemplate TargetType="{x:Type ScrollViewer}">
                                <Grid>
                                    <Grid.ColumnDefinitions>
                                        <ColumnDefinition Width="Auto" />
                                        <ColumnDefinition Width="*" />
                                        <ColumnDefinition Width="Auto" />
                                    </Grid.ColumnDefinitions>
                                    <Grid.RowDefinitions>
                                        <RowDefinition Height="Auto" />
                                        <RowDefinition Height="*" />
                                        <RowDefinition Height="Auto" />
                                    </Grid.RowDefinitions>
                                    <Button Command="{x:Static DataGrid.SelectAllCommand}"
                                            Focusable="false"
                                            Style="{DynamicResource {ComponentResourceKey ResourceId=DataGridSelectAllButtonStyle, TypeInTargetAssembly={x:Type DataGrid}}}"
                                            Visibility="{Binding HeadersVisibility, ConverterParameter={x:Static DataGridHeadersVisibility.All}, Converter={x:Static DataGrid.HeadersVisibilityConverter}, RelativeSource={RelativeSource AncestorType={x:Type DataGrid}}}"
                                            Width="{Binding CellsPanelHorizontalOffset, RelativeSource={RelativeSource AncestorType={x:Type DataGrid}}}" />
                                    <DataGridColumnHeadersPresenter Grid.Column="1"
                                                                    x:Name="PART_ColumnHeadersPresenter"
                                                                    Visibility="{Binding HeadersVisibility, ConverterParameter={x:Static DataGridHeadersVisibility.Column}, Converter={x:Static DataGrid.HeadersVisibilityConverter}, RelativeSource={RelativeSource AncestorType={x:Type DataGrid}}}" />
                                    <ScrollContentPresenter x:Name="PART_ScrollContentPresenter"
                                                            CanContentScroll="{TemplateBinding CanContentScroll}"
                                                            Grid.ColumnSpan="2"
                                                            Grid.Row="1" />
                                    <ScrollBar x:Name="PART_VerticalScrollBar"
                                               Grid.Column="2"
                                               Maximum="{TemplateBinding ScrollableHeight}"
                                               Orientation="Vertical"
                                               Grid.Row="1"
                                               Visibility="{TemplateBinding ComputedVerticalScrollBarVisibility}"
                                               Value="{Binding VerticalOffset, Mode=OneWay, RelativeSource={RelativeSource TemplatedParent}}"
                                               ViewportSize="{TemplateBinding ViewportHeight}"
                                               MinWidth="45" Width="50" />
                                    <Grid Grid.Column="1"
                                          Grid.Row="2">
                                        <Grid.ColumnDefinitions>
                                            <ColumnDefinition Width="{Binding NonFrozenColumnsViewportHorizontalOffset, RelativeSource={RelativeSource AncestorType={x:Type DataGrid}}}" />
                                            <ColumnDefinition Width="*" />
                                        </Grid.ColumnDefinitions>
                                        <ScrollBar x:Name="PART_HorizontalScrollBar"
                                                   Grid.Column="1"
                                                   Maximum="{TemplateBinding ScrollableWidth}"
                                                   Orientation="Horizontal"
                                                   Visibility="{TemplateBinding ComputedHorizontalScrollBarVisibility}"
                                                   Value="{Binding HorizontalOffset, Mode=OneWay, RelativeSource={RelativeSource TemplatedParent}}"
                                                   ViewportSize="{TemplateBinding ViewportWidth}" />
                                    </Grid>
                                </Grid>
                            </ControlTemplate>
                        </ScrollViewer.Template>
                        <ItemsPresenter SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}" />
                    </ScrollViewer>
                </Border>
            </ControlTemplate>
        </Setter.Value>
    </Setter>
    <Style.Triggers>
        <Trigger Property="IsGrouping" Value="true">
            <Setter Property="ScrollViewer.CanContentScroll" Value="false" />
        </Trigger>
    </Style.Triggers>
</Style>

编辑:

作为一个expermient,我将这个成员变量添加到窗口的代码隐藏中,问题是:

private static Style dataGridRowStyle = null;

然后我将此代码添加到窗口的构造函数中:

if ( dataGridRowStyle == null ) {
    dataGridRowStyle = FindResource( typeof( DataGridRow ) ) as Style;
    MyGrid.RowStyle = dataGridRowStyle;
}

通过这样做,我看到添加到DataGrid的每一行都具有原始默认样式。当我将上述代码移动到Loaded事件处理程序时,这也会出现问题。

接下来,我删除了上面的代码,并在app.xaml文件中的Style定义中添加了x:Key属性。然后我将此属性添加到DataGrid控件的定义中:

RowStyle={DynamicResource MyDataGridRowStyle}

现在每行都有我的风格。这很好,但我认为仅使用TargetType属性声明我的样式就足以让它应用于所有行。为什么不呢?

1 个答案:

答案 0 :(得分:1)

经过多次头痛,心痛和网络搜索后,我终于弄清楚我的程序中发生了什么。

我的程序使用合并的词典来实现日夜模式。事实证明,合并的词典是问题的原因,正如in this tutorial所述。

修复方法是在根词典中放置默认样式。实际上,我的代码在标签内的资源字典中包含了所有模板。我将它们移动到一个级别,现在WPF在第一行和每一行找到我的默认模板。