带触发器的继承WPF样式不起作用

时间:2016-08-04 02:13:21

标签: wpf triggers wpf-style

我有一个用动画淡化控件的风格:

<Style x:Key="ExpireFadeStyle">
    <Style.Resources>
        <!--Change this in a derived style if required-->
        <sys:Double x:Key="FinalVal">0.25</sys:Double>
    </Style.Resources>
    <Style.Triggers>
        <DataTrigger Binding="{Binding IsExpired}" Value="True">
            <DataTrigger.EnterActions>
                <BeginStoryboard x:Name="ExpireAnimation">
                    <Storyboard>
                        <DoubleAnimation Storyboard.TargetProperty="Opacity"
                                From="1" To="{StaticResource FinalVal}" Duration="0:0:3" />
                    </Storyboard>
                </BeginStoryboard>
            </DataTrigger.EnterActions>
            <DataTrigger.ExitActions>
                <StopStoryboard BeginStoryboardName="ExpireAnimation" />
            </DataTrigger.ExitActions>
        </DataTrigger>
    </Style.Triggers>
</Style>

IsExpired属性始终存在于应该使用样式的控件的DataContext中。)

当我在控件中直接使用此样式时,一切正常:

<StackPanel Style="{StaticResource ExpireFadeStyle}">
    ...etc

但是当我从这种风格中衍生出来时,就像

一样简单
<Style x:Key="ExpireTextFadeStyle" BasedOn="{StaticResource ExpireFadeStyle}"/>

...然后在同一控件上使用派生样式的方式相同,它不起作用。 (当然,目的是改变它,特别是FinalVal,但它必须先处理一个小问题。)

继承的样式本身似乎有用:如果我向它添加一些Setter,我会看到它的效果。似乎Triggers不是继承的,或者根本不起作用。如何解决它?

1 个答案:

答案 0 :(得分:4)

这真是一个很好的问题。我创建了一个示例测试应用程序来检查这一点,我遇到了完全相同的问题。在调试模式下,我检查了Style属性的值,它是正确的(正确设置了BasedOn,并且BasedOn.Triggers包含你的触发器)。然而,触发器在需要时不会启动。

我找到了一个解决方案,使这个工作,但它不是一个完美的... 我定义了一个附加的依赖项属性,它在设置为true时复制触发器。

public class StyleTriggersInheritanceHelper
{
    public static readonly DependencyProperty AddInheritedTriggersProperty =
        DependencyProperty.RegisterAttached("AddInheritedTriggers", typeof(bool), typeof(FrameworkElement), new FrameworkPropertyMetadata(new PropertyChangedCallback(OnAddInheritedTriggers)));

    private static void OnAddInheritedTriggers(DependencyObject sender, DependencyPropertyChangedEventArgs e)
    {
        var b = e.NewValue as bool?;
        if (b.HasValue && b.Value)
        {
            FrameworkElement element = sender as FrameworkElement;
            if(element != null)
            {
                Style style = element.Style;
                if(style != null)
                {
                    Style baseStyle = element.Style.BasedOn;
                    Style newStyle = new Style() { BasedOn = style };
                    if(baseStyle != null)
                    {
                        foreach (var tr in style.Triggers)
                            newStyle.Triggers.Add(tr);
                        foreach (var tr in baseStyle.Triggers)
                            newStyle.Triggers.Add(tr);
                    }
                    element.Style = newStyle;
                }
            }
        } 
    }

    public static bool GetAddInheritedTriggers(DependencyObject obj)
    {
        return (bool)obj.GetValue(AddInheritedTriggersProperty);
    }

    public static void SetAddInheritedTriggers(DependencyObject obj, bool value)
    {
        obj.SetValue(AddInheritedTriggersProperty, value);
    }
}

xaml看起来如下:

<local:ExpirdedControl IsExpired="{Binding IsExpired}" Style="{StaticResource InheritedStyle}" x:Name="ExpiredName" local:StyleTriggersInheritanceHelper.AddInheritedTriggers="True"/>

请注意,AddInheritedTriggers附加属性设置为true,然后复制触发器。一个重要的事情是你必须在应用Style之后设置AddInheritedTriggers,否则它将无效!

对我来说它有效,但解决方案并不优雅,所以我希望看到更好的解决这个问题的尝试。