WPF:如何为内容更改创建路由事件?

时间:2016-04-03 10:37:34

标签: c# wpf xaml eventtrigger routed-events

我有一个框架。我用这一行切换页面:

FrameName.Content = new PageName();

我希望在页面发生变化时开始使用故事板,我希望在XAML中进行,而不是在代码隐藏中。我尝试了以下代码:

<Frame.Triggers>
    <EventTrigger RoutedEvent="ContentChanged">
        <BeginStoryboard Storyboard="{StaticResource storyboardName}" />
    </EventTrigger>
</Frame.Triggers>

在搜索了一下后,我意识到没有那种性质的内置路由事件。第一个答案here表明

  

最动态的方法是简单地派生自己的标签控件,该控件提供ContentChanged事件。

我试图在这个答案中实现代码:

using System.Windows;
using System.Windows.Controls;

namespace ContentChangedTest
{
    class MyFrame : Frame
    {
        public event DependencyPropertyChangedEventHandler ContentChanged;

        static MyFrame()
        {
            ContentProperty.OverrideMetadata(typeof(MyFrame), new FrameworkPropertyMetadata(new PropertyChangedCallback(OnContentChanged)));
        }

        private static void OnContentChanged(DependencyObject dp, DependencyPropertyChangedEventArgs e)
        {
            MyFrame frame = dp as MyFrame;

            if (frame.ContentChanged != null)
            {
                DependencyPropertyChangedEventArgs args = new DependencyPropertyChangedEventArgs(ContentProperty, e.OldValue, e.NewValue);
                frame.ContentChanged(frame, args);
            }
        }
    }
}

在XAML中我使用它看起来像这样:

<local:MyFrame ContentChanged="MyFrame_ContentChanged" />

问题是我最终需要在代码隐藏中创建一个事件处理程序MyFrame_ContentChanged。有没有办法在纯XAML中执行此操作?例如 - 我可以将ContentChanged依赖项属性转换为某种路由事件吗?

1 个答案:

答案 0 :(得分:3)

为了使用EventTriggers的事件,他们应该被路由事件。路由事件的定义方式与依赖属性类似。这是一个关于如何入门的快速教程:How to: Create a Custom Routed Event

以下是从ContentControl派生的类的示例,该类定义了ContentChanged事件:

public class MyContentControl : ContentControl
{
    public static readonly RoutedEvent ContentChangedEvent 
        = EventManager.RegisterRoutedEvent(
            "ContentChanged",
            RoutingStrategy.Bubble,
            typeof(RoutedEventHandler), 
            typeof(MyContentControl));

    public event RoutedEventHandler ContentChanged
    {
        add { AddHandler(ContentChangedEvent, value); }
        remove { RemoveHandler(ContentChangedEvent, value); }
    }

    protected override void OnContentChanged(object oldContent, object newContent)
    {
        base.OnContentChanged(oldContent, newContent);
        RaiseEvent(new RoutedEventArgs(ContentChangedEvent, this));
    }
}

我还不确定原因,但是在测试此行时Style内有效,但在控件的触发器集合中直接使用时抛出异常:

<EventTrigger RoutedEvent="ContentChanged">...</EventTrigger>

为了使它在这种情况下工作,我必须指定一个完全限定的事件路径:

<EventTrigger RoutedEvent="local:MyContentControl.ContentChanged">...</EventTrigger>