2

这里我有一个简单的混合行为。它有一个DependencyProperty我想在OnAttached. 但我不能,因为行为似乎在初始化之前就已经附加了!

用法:

<Button>
    <e:Interaction.Behaviors>
        <local:TestBehavior Test1="{Binding ValueOnViewModel}"/>
    </e:Interaction.Behaviors>
<Button>

定义:

class TestBehavior : Behavior<FrameworkObject>
{
    public static readonly DependencyProperty Test1Property
        = DependencyProperty.Register("Test1", typeof(int), typeof(TestBehavior),
        new PropertyMetadata(OnTest1Property_Changed));

    private static void OnTest1Property_Changed(DependencyObject sender,
        DependencyPropertyChangedEventArgs args)
    {
        // This gets called _after_ OnAttached!
    }

    public int Test1
    {
        get { return (int)GetValue(Test1Property); }
        set { SetValue(Test1Property, value); }
    }

    protected override void OnAttached()
    {
        base.OnAttached();

        // No matter what Test1Property is bound to in XAML 'a' will be default
        // value because as this point the Behavior is attached, but not
        // initialized!
        int a = Test1;
    }
}

这让我觉得很奇怪。在这个简单的情况下,我可以通过在 OnTest1Property_Changed 而不是 OnAttached 中执行初始化来解决这个问题(尽管在静态上下文而不是实例上下文中会带来轻微的不便)。

但是,如果我有一个具有多个属性的相当简单的 Behavior 怎么办?在行为的某些用法中,可能会显式设置所有 DP,而在其他情况下,可能会在使用未显式设置的 DP 的默认值时仅设置一些 DP。我可以处理行为定义的所有 DP 的更改事件,但我无法知道客户端在 XAML 中明确设置了哪些 DP,因此我无法知道在初始化完成之前必须发生更改通知.

那么在那种不平凡的情况下,我怎么可能知道行为的初始化何时完成?这似乎是一个明显的弱点,我只能假设我错过了一些明显的东西。

[更新]

使用 Sorskoot 的建议,我敲开了这个简单的基类,我可以将其用作我的行为的基础。

public class BehaviorBase<T> : Behavior<T> where T : FrameworkElement
{
    protected override void OnAttached()
    {
        base.OnAttached();
        AssociatedObject.Loaded += AssociatedObject_Loaded;
    }

    protected override void OnDetaching()
    {
        base.OnDetaching();
        AssociatedObject.Loaded -= AssociatedObject_Loaded;
    }

    protected virtual void OnLoaded()
    {

    }

    private void AssociatedObject_Loaded(object sender, RoutedEventArgs e)
    {
        OnLoaded();
    }
}
4

1 回答 1

2

我建议LoadedOnAttached方法中附加到事件并在那里进行初始化。

 protected override void OnAttached()
 {
      base.OnAttached();

      base.AssociatedObject.Loaded += onloaded;
 }

 private void onloaded(object sender, RoutedEventArgs e)
 {
      int a = Test1;
      // Test1 should contain a value here. 
 }
于 2013-05-07T20:23:16.797 回答