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没有继承或根本不起作用。如何解决呢?
这真是个好问题。我创建了一个示例测试应用程序来检查这一点,我遇到了完全相同的问题。在调试模式下,我检查了 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,否则它不会起作用!
对我来说它有效,但是解决方案并不优雅,所以我希望看到更好的尝试来解决这个问题。
| 归档时间: |
|
| 查看次数: |
693 次 |
| 最近记录: |