3

我为 Storyboard 创建了一个附加的依赖属性,目的是让我能够在 Storyboard Completed 事件触发时调用 ViewModel 上的方法:

public static class StoryboardExtensions
{
    public static ICommand GetCompletedCommand(DependencyObject target)
    {
        return (ICommand)target.GetValue(CompletedCommandProperty);
    }

    public static void SetCompletedCommand(DependencyObject target, ICommand value)
    {
        target.SetValue(CompletedCommandProperty, value);
    }

    public static readonly DependencyProperty CompletedCommandProperty =
        DependencyProperty.RegisterAttached(
            "CompletedCommand",
            typeof(ICommand),
            typeof(StoryboardExtensions),
            new FrameworkPropertyMetadata(null, OnCompletedCommandChanged));

    static void OnCompletedCommandChanged(DependencyObject target, DependencyPropertyChangedEventArgs e)
    {
        Storyboard storyboard = target as Storyboard;
        if (storyboard == null) throw new InvalidOperationException("This behavior can be attached to Storyboard item only.");
        storyboard.Completed += new EventHandler(OnStoryboardCompleted);
    }

    static void OnStoryboardCompleted(object sender, EventArgs e)
    {                        
        Storyboard item = ... // snip
        ICommand command = GetCompletedCommand(item);
        command.Execute(null);
    }
}

然后我尝试在 XAML 中使用它,使用绑定语法:

<Grid>
    <Grid.Resources>
        <Storyboard x:Key="myStoryboard" my:StoryboardExtensions.CompletedCommand="{Binding AnimationCompleted}">
            <DoubleAnimation Storyboard.TargetProperty="Opacity" From="1" To="0" Duration="0:0:5" />
        </Storyboard>

        <Style x:Key="myStyle" TargetType="{x:Type Label}">
            <Style.Triggers>
                <DataTrigger 
                 Binding="{Binding Path=QuestionState}" Value="Correct">
                    <DataTrigger.EnterActions>
                        <BeginStoryboard Storyboard="{StaticResource myStoryboard}" />
                    </DataTrigger.EnterActions>
                </DataTrigger>
            </Style.Triggers>
        </Style>

    </Grid.Resources>
    <Label x:Name="labelHello" Grid.Row="0" Style="{StaticResource myStyle}">Hello</Label>
</Grid>

这失败了,但有以下例外:

System.Windows.Markup.XamlParseException 发生消息 =“无法将属性“Style”中的值转换为“System.Windows.Style”类型的对象。无法冻结此 Storyboard 时间线树以跨线程使用。对象“labelHello”中的错误标记文件“TestWpfApp;组件/window1.xaml”

有没有办法让绑定语法与情节提要的附加 ICommand 属性一起使用?

4

3 回答 3

1

这是设计使然。如果您有一个可冻结的对象放入样式中,则样式将被冻结以允许跨线程访问。但是您的绑定本质上是一个表达式,这意味着它不能被冻结,因为数据绑定是单线程的。

如果您需要这样做,请将触发器放在样式之外的框架元素下,而不是样式中。您可以在 Grid.Triggers 部分执行此操作。这确实有点糟糕,因为您的样式不再完整并且您必须复制触发器,但它是 WPF 中的“设计”功能。

MSDN 社交论坛上的完整答案在这里

于 2009-09-18T12:43:22.253 回答
0

您可以创建一个新的 Freezable 派生类以将情节提要作为 shim 启动。将该 shim 对象上的属性绑定到情节提要名称。这样,您就不必复制触发器或将它们存储在样式之外。

于 2009-09-18T12:49:39.447 回答
0

为了解决这个问题,我创建了一堆附加属性,称为 Storyboard Helpers(源代码在这里)。我放弃了尝试将它们附加到情节提要本身,现在附加到任何(任意)框架元素以在情节提要完成时在我的 ViewModel 上调用 ICommand,以及绑定到我的 ViewModel 上的特定事件以启动情节提要. 第三个附加属性指定我们正在处理的故事板:

<FrameworkElement
   my:StoryboardHelpers.Storyboard="{StaticResource rightAnswerAnimation}"
   my:StoryboardHelpers.Completed="{Binding CompletedCommand}"
   my:StoryboardHelpers.BeginEvent="{Binding StartCorrectAnswer}" />
于 2009-10-06T14:51:22.723 回答