WPF – DependencyProperty ignoring setter with side-effects on change
我有一个 WPF 用户控件,它是另外两个控件的package器,根据情况只显示其中一个。它拥有一个 ItemsSource 属性,该属性为两个底层控件设置 ItemsSource。我想这样做,以便可以将此属性绑定到 .xaml 文件。
我创建了一个 DependencyProperty,并更改了我的 getter 和我的 setter 以使用它。但是,当我调试代码时,我可以看到 setter 永远不会被调用。我可以看到依赖属性正在改变它的值,但它没有设置底层控件的属性。
我怎样才能让底层控件在依赖属性更改时设置其属性?
1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
public partial class AccountSelector : UserControl
{ public static readonly DependencyProperty ItemsSourceProperty = DependencyProperty.Register( “ItemsSource”, typeof(IEnumerable), typeof(AccountSelector)); public IEnumerable ItemsSource { get { return (IEnumerable)GetValue(ItemsSourceProperty); } set { if (UseComboBox) AccCombo.ItemsSource = value; else AccComplete.ItemsSource = value; SetValue(ItemsSourceProperty, value); } } } |
- 是否设置了属性的值但可能没有提升到 UI?此外,您将属性声明为只读,这将阻止分配给它,除非在同一类的构造函数或声明中。有关只读的更多信息,请参见此处。
您必须将 propertyChangedCallback 传递给您的 UIPropertyMetadata,如下所示:
1
2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 |
public static readonly DependencyProperty ItemsSourceProperty = DependencyProperty.Register(
“ItemsSource”, typeof(IEnumerable), typeof(AccountSelector), new UIPropertyMetadata((d, e) => { if (e.NewValue == null) return; var s = d as AccountSelector; if (list == null || s == null) return; |
- 这似乎成功了。谢谢。
- 这里显示的设置器是错误的。它必须调用 SetValue(ItemsSourceProperty, value) 而没有别的。有关详细信息,请参阅 XAML 加载和依赖属性。
- 所以 setter 上的逻辑必须移到回调中?
- @RenanGemignani 正是。不知道你怎么能接受这个答案。
- 我刚刚测试了它,它似乎工作。
- 还有两个笔记。检查 e.NewValue 和 list 是否为空是多余的。您只能检查 list。然后,动态转换 d as AccountSelector 在这里是错误的,因为您希望在 d 中有一个 AccountSelector 实例,而没有别的。其他一切都是错误,应该引发 InvalidCastException。所以最好写var s = (AccountSelector)d。
- 最后,没有必要使用 UIPropertyMetadata。你也可以使用普通的PropertyMetadata。
来源:https://www.codenong.com/29392205/