继承WPF样式的触发器无效

6

我有一个能够通过动画淡出控件的样式:

<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个回答

5

这是一个非常好的问题。我创建了一个示例测试应用程序来检查这个问题,我遇到了完全相同的问题。在调试模式下,我检查了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,然后复制触发器。一个重要的事情是,在应用样式之后必须设置AddInheritedTriggers,否则它不起作用! 对我来说,它有效,但解决方案并不优雅,所以我希望看到更好的解决此问题的尝试。

1
谢谢!这证明了这一点,但实际上,这种解决方案几乎没有比直接复制整个样式并进行必要的更改更优雅...我仍然想知道这是否是WPF中的一个错误或“特性”,以及继承触发器的“正确”方法是什么... - Zeus
我也只是复制了。显然,这些样式不像OO继承那样被继承。可能这在社区中文档不好或者没有被完全理解。 - Gerard

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接