根据背景颜色动态更改前景色

时间:2013-08-30 21:44:38

标签: wpf button colors triggers styles

问题域:在我的WPF应用程序中,我根据它们包含的数据动态更改了许多UI控件的背景,如Button或ListItems。加载控件时或根据收到的使用操作/数据更改背景。

问题陈述:如果背景太暗(绿色/蓝色),我想将前景设置为白色,否则为黑色。

约束:我有一个很大的应用程序,性能是一个主要问题。这就是为什么我对使用转换器犹豫不决并且正在寻找一些基于xaml / style触发器的解决方案,因为这只是一个基于条件的问题。

已尝试解决方案:为了简单起见,我正在解释我为简单的wpf按钮尝试的内容:

<UserControl.Resources>
<Style x:Key="NoChromeButton" TargetType="{x:Type Button}">
        <Setter Property="Background" Value="{Binding Background}"/>
        <Setter Property="OverridesDefaultStyle" Value="True"/>
        <Setter Property="BorderThickness" Value="1"/>
        <Setter Property="BorderBrush" Value="White"/>
        <Setter Property="HorizontalContentAlignment" Value="Center"/>
        <Setter Property="VerticalContentAlignment" Value="Center"/>
        <Setter Property="Padding" Value="1"/>
        <Setter Property="Template">
            <Setter.Value>
                <ControlTemplate TargetType="{x:Type Button}">
                    <Grid x:Name="Chrome" 
                          Background="{TemplateBinding Background}" 
                          SnapsToDevicePixels="true"
                          HorizontalAlignment="Stretch">
                    <TextBlock 
                        Text="{Binding RelativeSource={RelativeSource TemplatedParent},Path=Content}"
                        HorizontalAlignment="{TemplateBinding HorizontalContentAlignment}" 
                                          Margin="{TemplateBinding Padding}" 
                                          Background="{Binding RelativeSource={RelativeSource TemplatedParent},Path=Background}"
                                          SnapsToDevicePixels="{TemplateBinding SnapsToDevicePixels}" 
                                          VerticalAlignment="{TemplateBinding VerticalContentAlignment}"
                        Style="{StaticResource MyTextBlockStyle}"/>
                    </Grid>
                    <ControlTemplate.Triggers>
                        <Trigger Property="IsMouseOver" Value="True">
                            <Setter Property="Background" Value="Black"/>
                        </Trigger>
                        <Trigger Property="Background" Value="White">
                            <Setter Property="Foreground" Value="Aqua"/>
                        </Trigger>
                        <Trigger Property="Background" Value="Transparent">
                            <Setter Property="Foreground" Value="BlueViolet"/>
                        </Trigger>
                        <Trigger Property="Background" Value="Green">
                            <Setter Property="Foreground" Value="Blue"/>
                        </Trigger>
                        <Trigger Property="Background" Value="Yellow">
                            <Setter Property="Foreground" Value="Red"/>
                        </Trigger>
                        <Trigger Property="Background" Value="Red">
                            <Setter Property="Foreground" Value="Yellow"/>
                        </Trigger>
                        <Trigger Property="Background" Value="Black">
                            <Setter Property="Foreground" Value="DarkSeaGreen"/>
                        </Trigger>
                    </ControlTemplate.Triggers>
                </ControlTemplate>
            </Setter.Value>
        </Setter>

        <Style x:key="MyTextBlockStyle" TargetType="TextBlock">
    <Setter Property="FontSize" Value="12"/>
    <Setter Property="FontStyle" Value="Italic"/>
    <Style.Triggers>
        <Trigger Property="Background" Value="White">
            <Setter Property="Foreground" Value="Aqua"/>
        </Trigger>
        <Trigger Property="Background" Value="Transparent">
            <Setter Property="Foreground" Value="BlueViolet"/>
        </Trigger>
        <Trigger Property="Background" Value="Green">
            <Setter Property="Foreground" Value="Blue"/>
        </Trigger>
        <Trigger Property="Background" Value="Yellow">
            <Setter Property="Foreground" Value="Red"/>
        </Trigger>
        <Trigger Property="Background" Value="Red">
            <Setter Property="Foreground" Value="Yellow"/>
        </Trigger>
        <Trigger Property="Background" Value="Black">
            <Setter Property="Foreground" Value="DarkSeaGreen"/>
        </Trigger>
</Style>
</UserControl.Resources>

在XAML中创建按钮时:

  <Button Content="{Binding Name}" Style="{StaticResource NoChromeButton}"/>

另外,我想指出以上风格的几个方面:

  1. 如果我在名为Chrome的Grid中使用了ContentPresenter而不是TextBlock,那么ContentPresenter上没有设置background属性,当我窥探(http://snoopwpf.codeplex.com/)UI时,我发现ContentPresenter有TextBlock,其背景是始终设置为默认值,因此没有样式触发器应用于TextBlock。此外,此TextBlock的背景值源为Default。
  2. 另一方面,当我在名为Chrome的Grid中直接使用TextBlock时,我可以将其背景显式设置为Grid的Background,后者设置为Button的Background。 Snooping揭示了现在TextBlock的Background ValueSource是ParentTemplate。

    1. 按钮在显示其内容时选择MyTextBlockStyle。

    2. Button或TextBlock的样式触发器从未被触发,除非我将鼠标悬停在按钮上,将按钮的背景更改为Black并将此值传播到TextBlock背景,将TextBlock的前景颜色更改为DarkSeaGreen。

      < / LI>
    3. 此外,在应用程序运行时更改snoop实用程序中按钮的背景,会触发样式触发器。

    4. 问题:

      1. 为什么没有Style触发器适用于Background属性,而它们适用于IsMouseOver属性?

      2. 我做错了什么?

      3. 任何解决方案?

1 个答案:

答案 0 :(得分:1)

我找到了解决问题的方法。 TextBlock不是从Control派生的。任何控件在UI上显示的任何文本都在内部使用TextBlock来表示文本内容。如果使用ResourceDictionary中的以下内容设置TextBlock样式:

<Style TargetType="TextBlock">
    <Setter Property="FontFamily" Value="Arial" />
    <Setter Property="FontSize" Value="12" />
    <Setter Property="FontStyle" Value="Normal" />
</Style>

任何表示文本的控件都将具有此样式(因为没有为此样式分配键,这意味着默认情况下所有TextBlock都会获取它),除非控件的模板覆盖TextBlock的默认样式,可以按如下方式执行:

<Button Grid.Column="1" Style="{StaticResource NoChromeButton}">
    <TextBlock Style="{x:Null}" Text="abc" FontFamily="Segoe UI Symbol"/>
</Button>

这个简单的设置解决了我们在动态前景色变化方面遇到的大多数问题。