这里我有一个简单的混合行为。它有一个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();
}
}