WPF MVVM 将用户控件绑定到主窗口视图模型

Posted

技术标签:

【中文标题】WPF MVVM 将用户控件绑定到主窗口视图模型【英文标题】:WPF MVVM Bind User Control to Main Window View Model 【发布时间】:2013-07-22 05:46:56 【问题描述】:

我不久前在这里WPF MVVM User Control 提出了类似的问题。我得到了一些答案,但它们离我很远,所以我想我并不清楚我想做什么......

我正在使用 MVVM 开发 WPF 应用程序。该应用程序是使用基于组件的方法构建的,因此我定义了一些将在整个应用程序中使用的用户控件。例如,我有一个地址控件。我想在整个应用程序的多个地方使用它。这是一个例子。见这里:

http://sdrv.ms/1aD775H

带有绿色边框的部分是地址控件。该控件有自己的视图模型。

当我将它放在窗口或其他控件上时,我需要告诉它要为其加载地址的客户的 PK。所以我创建了一个客户 ID DependencyProperty:

public partial class AddressView : UserControl

    public AddressView()
    
        InitializeComponent();
    

    public static DependencyProperty CustomerIdProperty = DependencyProperty.Register("CustomerId", typeof(int), typeof(AddressView),
        new UIPropertyMetadata(0, AddressView.CustomerIdPropertyChangedCallback, AddressView.CustomerIdCoerce, true));


    public int CustomerId
    
        // THESE NEVER FIRE
        get  return (int)GetValue(CustomerIdProperty); 
        set  SetValue(CustomerIdProperty, value); 
    

    private static void CustomerIdPropertyChangedCallback(DependencyObject d, DependencyPropertyChangedEventArgs args)
    
        // THIS NEVER FIRES
        AddressView instance = (AddressView)d;
        instance.CustomerId = (int)args.NewValue;
    
    enter code here
    private static object CustomerIdCoerce(DependencyObject d, object value)
    
        return value;   // <=== THIS FIRES ON STARTUP
    

然后在 MainWindowView 我有:

<vw:AddressView Grid.Row="1"
                Grid.Column="0"
                x:Name="AddressList"
                CustomerId="Binding ElementName=TheMainWindow, Path=SelectedCustomer.Id, Mode=TwoWay"/>

注意我在用户控件的 CS 中的 cmets。 Coerce 在启动时触发。回调永远不会触发,CustomerId getter 或 setter 也不会触发。

我想要发生的事情看起来很简单,我就是做不到......

选择客户时,应将 CustomerId 传递给 Address UserControl。然后在 VM 中的地址 UserControl 应该处理获取和保存数据。

所以,再次提出两个问题:

1) 有人知道出了什么问题吗?

2) UserControl DP如何将PK发送到ViewModel?

如果有人感兴趣,我的示例项目在这里:http://sdrv.ms/136bj91

谢谢

【问题讨论】:

try adding UpdateSourceTrigger=PropertyChanged to your binding, when the selection changes your control won't get updated otherwise. 你能发布你的usercontrol xaml页面的代码吗.. WPF MVVM User Control 的可能副本 【参考方案1】:

在这种情况下,您的 CustomerId getter 和 setter 永远不会触发。它们只是作为辅助方法存在,以防想要从您的代码后面访问CustomerIdProperty 属性。

您的CustomerIdPropertyChangedCallback 方法不会触发,因为您的绑定表达式不正确。您需要绑定到 DataContextMainWindow 而不是窗口本身:

...
CustomerId="Binding ElementName=TheMainWindow, Path=DataContext.SelectedCustomer.Id"
...

此外,请确保在绑定到 ComboBox 的属性发生更改时调用 INotifyPropertyChanged PropertyChanged 事件。

【讨论】:

【参考方案2】:

试试这个:

CustomerId="Binding RelativeSource=RelativeSource FindAncestor, 
AncestorType=x:Type Window, Path=DataContext.YourSelectedItem.TheProperty"

我不知道如何管理您在窗口中选择的项目,所以请相应地更改yourSelectedItem.TheProperty

【讨论】:

好的,这有效 - 但只有一次。回调方法仅在第一次选择客户时触发。 你能在代码中粘贴你的 SelectedItem 的实现吗 当然,我创建了一个小型沙盒项目来学习这一点。在这里sdrv.ms/13fM5EY。

以上是关于WPF MVVM 将用户控件绑定到主窗口视图模型的主要内容,如果未能解决你的问题,请参考以下文章

使用 MVVM 和视图模型通信的 WPF 窗口模式对话框

WPF MVVM绑定窗口图标到视图模型中的属性

WPF(MVVM):从 Viewmodel 关闭视图?

在 WPF MVVM 中添加多个用户控件的最佳控件?

在 UserControl WPF MVVM caliburn 内的 UserControl 之间切换

如何在 viewmodel 中访问 mvvm 模型中的控件?