【问题标题】:Bind multiple views to the same data context (WPF-MVVM)将多个视图绑定到同一个数据上下文 (WPF-MVVM)
【发布时间】:2017-06-20 10:10:43
【问题描述】:

我的问题可能很愚蠢,但我对 WPF 和 MVVM 还是很陌生。

那么我们来看下面的例子:

我有一个通讯录应用程序,它在 MainWindow-View 中使用 ObservableCollection<Contact> 显示所有联系人。

当我按下“添加联系人”按钮时,会打开一个新视图,其中包含供用户输入新联系人的文本字段和一个保存按钮。但是,当我保存条目时,它会保存到新的集合中,因为我不在同一个数据上下文中(两者都使用相同的 ViewModel 作为 DataContext)。

我的问题是我如何与两个视图共享ObservableCollection<Contact>,以便两个视图都可以编辑它并使用相同的集合(即在视图中保存新联系人并且主视图中的列表也会更新)?

几个代码sn-ps:

private Contact contact;
private ObservableCollection<Contact> _contacts;

public ContactManager()
{
    Contacts = new ObservableCollection<Contact>();
    contact = new Contact();
}

public ObservableCollection<Contact> Contacts
{
    get { return _contacts; }
    set
    {
        _contacts = value;
        RaisePropertyChangedEvent("Contacts");
    }
}

public String Anrede
{
    get { return contact.Anrede; }
    set
    {
        contact.Anrede = value;
        RaisePropertyChangedEvent("Anrede");
    }
}

public String Vorname
{
    get { return contact.Vorname; }
    set
    {
        contact.Vorname = value;
        RaisePropertyChangedEvent("Vorname");
    }
}

public String Nachname
{
    get { return contact.Nachname; }
    set
    {
        contact.Nachname = value;
        RaisePropertyChangedEvent("Nachname");
    }
}

public String Adresse
{
    get { return contact.Adresse; }
    set
    {
        contact.Adresse = value;
        RaisePropertyChangedEvent("Adresse");
    }
}

public String Telefonnummer
{
    get { return contact.Telefonnummer; }
    set
    {
        contact.Telefonnummer = value;
        RaisePropertyChangedEvent("Telefonnummer");
    }
}


public ICommand CreateTestContactCommand
{
    get { return new DelegateCommand(CreateTestContact); }
}

public ICommand AddNewContactCommand
{
    get { return new DelegateCommand(AddNewContact); }
}

private void CreateTestContact()
{
    var testContact = new Contact
    {
        Anrede = "Herr",
        Vorname = "Max",
        Nachname = "Mustermann",
        Adresse = "Mustermannstraße 13",
        Telefonnummer = "123456789"
    };

    Contacts.Add(testContact);
}

private void AddNewContact()
{
    Contacts.Add(contact);
}

和 XAML:

主窗口:

<Window
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:local="clr-namespace:ContactBook_Test"
    xmlns:ViewModels="clr-namespace:ContactBook_Test.ViewModels" x:Class="ContactBook_Test.MainWindow"
    mc:Ignorable="d"
    Title="MainWindow" Height="488" Width="533" ResizeMode="NoResize">
<Window.DataContext>
    <ViewModels:ContactManager/>
</Window.DataContext>
<Grid>
    <Grid.DataContext>
        <ViewModels:ContactManager/>
    </Grid.DataContext>
    <Label Content="Kontaktbuch" HorizontalContentAlignment="Center" HorizontalAlignment="Left" Margin="10,10,0,0" VerticalAlignment="Top" FontSize="36" FontFamily="Calibri" Width="498"/>
    <DataGrid ItemsSource="{Binding Contacts}" AutoGenerateColumns="True"  CanUserAddRows="false" HorizontalAlignment="Left" Height="327" Margin="10,69,0,0" VerticalAlignment="Top" Width="498"/>
    <Button Command="{Binding CreateTestContactCommand}" Content="Testkontakt hinzufügen" HorizontalAlignment="Left" Margin="198,415,0,0" VerticalAlignment="Top" Width="128"/>
    <Button Content="Neuer Kontakt" HorizontalAlignment="Left" Margin="16,31,0,0" VerticalAlignment="Top" Width="87" Click="Button_Click"/>

</Grid>

添加联系人:

<Window
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:ViewModels="clr-namespace:ContactBook_Test.ViewModels" x:Class="ContactBook_Test.Views.AddContact"
    mc:Ignorable="d"
    Title="AddContact" Height="300" Width="300">
<Window.DataContext>
    <ViewModels:ContactManager/>
</Window.DataContext>
<Grid>
    <Grid.DataContext>
        <ViewModels:ContactManager/>
    </Grid.DataContext>
    <TextBox HorizontalAlignment="Left" Height="23" Margin="84,46,0,0" TextWrapping="Wrap" Text="{Binding Anrede}" VerticalAlignment="Top" Width="200" />
    <TextBox HorizontalAlignment="Left" Height="23" Margin="84,85,0,0" TextWrapping="Wrap" Text="{Binding Vorname}" VerticalAlignment="Top" Width="200"/>
    <TextBox HorizontalAlignment="Left" Height="23" Margin="84,123,0,0" TextWrapping="Wrap" Text="{Binding Nachname}" VerticalAlignment="Top" Width="200"/>
    <TextBox HorizontalAlignment="Left" Height="23" Margin="84,163,0,0" TextWrapping="Wrap" Text="{Binding Adresse}" VerticalAlignment="Top" Width="200"/>
    <TextBox HorizontalAlignment="Left" Height="23" Margin="84,201,0,0" TextWrapping="Wrap" Text="{Binding Telefonnummer}" VerticalAlignment="Top" Width="200"/>
    <Button Command ="{Binding AddNewContactCommand}" Content="Speichern" HorizontalAlignment="Left" Margin="107,240,0,0" VerticalAlignment="Top" Width="75"/>
    <Label Content="Anrede" HorizontalAlignment="Left" Margin="12,43,0,0" VerticalAlignment="Top"/>
    <Label Content="Name" HorizontalAlignment="Left" Margin="12,82,0,0" VerticalAlignment="Top"/>
    <Label Content="Nachname" HorizontalAlignment="Left" Margin="12,120,0,0" VerticalAlignment="Top"/>
    <Label Content="Adresse" HorizontalAlignment="Left" Margin="12,159,0,0" VerticalAlignment="Top"/>
    <Label Content="Telefon" HorizontalAlignment="Left" Margin="12,198,0,0" VerticalAlignment="Top"/>

</Grid>

【问题讨论】:

  • 也许发布一些代码来帮助我们帮助你。但是,如果您已将两个视图的数据上下文设置为相同的视图模型,只需将 ICommand 添加到“Add-Contact”按钮,然后在该命令中将新条目添加到 ObservableCollection
  • @DanieleSartori 用代码更新了问题
  • 通常当您创建新的Window 时,您只需使用var newWnd = new AddContact(); newWnd.DataContext = this.DataContext;

标签: c# wpf mvvm


【解决方案1】:

您应该为两个视图使用相同的视图模型实例。从两个视图中删除它:

<Grid.DataContext>
    <ViewModels:ContactManager/>
</Grid.DataContext>

Grid 将从父窗口继承 DataContext。到目前为止,您正在每个视图中创建视图模型的两个实例。

也将其从AddContact 视图中删除:

<Window.DataContext>
    <ViewModels:ContactManager/>
</Window.DataContext>

...并在打开窗口时设置其DataContext

AddContact addContact = new AddContact();
addContact.DataContext = this.DataContext;
addContact.Show():

【讨论】:

    【解决方案2】:

    有一些方法可以解决这个问题:

    简单的方法

    当您在主窗口 Button_Click 中创建 AddContact 时,inyect 或分配 ViewModel。例如:

    AddContact(this.DataContext);
    

    new AddContact(){DataContext=this.DataContext);
    

    活动方式

    在 MVVM 架构中,您可以通过事件在 ViewModel 之间共享信息,请参阅Sharin variables between different ViewModels

    Inyect 导航数据

    通常用于导航到页面的方法允许发送数据, 例如NavigationService.Navigate

    希望对你有帮助。

    【讨论】:

      猜你喜欢
      • 2010-12-31
      • 2019-02-05
      • 2011-03-11
      • 2010-10-15
      • 2013-01-23
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      相关资源
      最近更新 更多