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

Zeu*_*eus 5 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>
Run Code Online (Sandbox Code Playgroud)

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

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

<StackPanel Style="{StaticResource ExpireFadeStyle}">
    ...etc
Run Code Online (Sandbox Code Playgroud)

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

<Style x:Key="ExpireTextFadeStyle" BasedOn="{StaticResource ExpireFadeStyle}"/>
Run Code Online (Sandbox Code Playgroud)

...然后在相同控件上以相同方式使用派生样式,则不起作用。(当然是要对它进行一些更改,尤其是FinalVal,但它必须首先在平凡的情况下起作用)。

继承的样式本身似乎确实起作用:如果在其中添加一些样式Setter,则会看到其效果。似乎Triggers没有继承或根本不起作用。如何解决呢?

bak*_*a12 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);
    }
}
Run Code Online (Sandbox Code Playgroud)

xaml 如下所示:

<local:ExpirdedControl IsExpired="{Binding IsExpired}" Style="{StaticResource InheritedStyle}" x:Name="ExpiredName" local:StyleTriggersInheritanceHelper.AddInheritedTriggers="True"/>
Run Code Online (Sandbox Code Playgroud)

请注意,将 AddInheritedTriggers 附加属性设置为 true,然后复制触发器。很重要的一点是你必须在应用Style后设置AddInheritedTriggers,否则它不会起作用!

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