0

我正在尝试实现 MVVM,所以我不知道以下内容是否正确。看起来 ViewModel 是视图的某种模型,所以视图中的关联应该显示在 ViewModel 中,这样的话,ViewModel 之间就应该有一些关联。因此,通过为 ViewModel 类型创建一些模板,应用程序似乎可以工作,这里是一些示例代码:

视图模型:

public class SomeVm : INotifyPropertyChanged
{
    public SomeVm()
    {
        SomeOtherVm = new SomeOtherVm();
    }
    public INotifyPropertyChanged SomeOtherVm { set; get; }

    private int _a;
    public int A
    {
        set { 
            _a= value;
            B = value;
        }
        get { return _a; }
    }

    private int _b;
    public int B
    {
        set 
        {
            _b = value;
            OnPropertyChanged("B");
        }
        get { return _b; }
    }

    public event PropertyChangedEventHandler PropertyChanged;

    protected virtual void OnPropertyChanged(string propertyName)
    {
        var handler = PropertyChanged;
        if (handler != null) handler(this, new PropertyChangedEventArgs(propertyName));
    }
}

public class SomeOtherVm : INotifyPropertyChanged
{
    public event PropertyChangedEventHandler PropertyChanged;

    private int _c;
    public int C
    {
        set
        {
            _c = value;
            D = value;
        }
        get { return _c; }
    }

    private int _d;
    public int D
    {
        set
        {
            _d = value;
            OnPropertyChanged("D");
        }
        get { return _d; }
    }
    protected virtual void OnPropertyChanged(string propertyName)
    {
        var handler = PropertyChanged;
        if (handler != null) handler(this, new PropertyChangedEventArgs(propertyName));
    }
}

和视图:

<Window
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:wpfApplication1="clr-namespace:WpfApplication1"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
        mc:Ignorable="d" 
        x:Class="WpfApplication1.MainWindow"
        Title="MainWindow" Height="350" Width="525">
    <Window.Resources>
        <wpfApplication1:SomeVm x:Key="SomeVm"/>
        <DataTemplate DataType="{x:Type wpfApplication1:SomeVm}">
            <StackPanel d:DesignWidth="339" d:DesignHeight="54">
                <TextBox HorizontalAlignment="Left" TextWrapping="Wrap" Text="{Binding A}" VerticalAlignment="Stretch"/>
                <TextBlock HorizontalAlignment="Left" TextWrapping="Wrap" Text="{Binding B}" VerticalAlignment="Stretch"/>
                <ContentPresenter Content="{Binding SomeOtherVm}"/>
            </StackPanel>

        </DataTemplate>
        <DataTemplate DataType="{x:Type wpfApplication1:SomeOtherVm}">
            <StackPanel d:DesignWidth="339" d:DesignHeight="54">
                <TextBox HorizontalAlignment="Left" TextWrapping="Wrap" Text="{Binding C}" VerticalAlignment="Stretch"/>
                <TextBlock HorizontalAlignment="Left" TextWrapping="Wrap" Text="{Binding D}" VerticalAlignment="Stretch"/>
            </StackPanel>
        </DataTemplate>

    </Window.Resources>
    <Grid>
        <ContentPresenter Content="{DynamicResource SomeVm}" />   
    </Grid>
</Window>

这样就可以在一些Resource Dictionaries中创建所有的视图,那么问题来了:这样使用MVVM合适吗?如果是,有什么缺点?

4

2 回答 2

1

通常 ViewModel 应该是整个视图的 DataContext ,即它应该是负责提供数据给视图以呈现自身并监听 UI 命令、事件和属性更改以与业务层(模型)交互的实体。

您实现它的方式是您将 VM 作为资源并将其设置为内容而不是 DataContext 用于呈现的一个内容,并且对于您提到的场景,它可能运行良好。但是您应该将 VM 设置为整个视图的 DataContext,以便视图中的所有元素都可以绑定到 VM 中的属性以呈现它们的状态。

在您的场景中,如果您必须在除 ContentPresenter 之外的视图中再添加一个 UI 元素,那么您将不得不再次访问您的资源 VM。

因此,如果您将 VM 实例设置为 DataContext(例如 this.DataContext = new ViewModel())并将您的 contentpresenter Content 绑定到 View 的 DataContext,例如 Content={Binding},这将更正确,如果您想扩展你的看法。这是一篇关于 mvvm 实现的不错的 msdn 文章http://msdn.microsoft.com/en-us/library/gg405484(v=pandp.40).aspx

谢谢

于 2013-09-07T18:51:50.413 回答
1

说到ViewModel嵌套,这段代码一看就对了。您在 XAML 中设置的绑定也是正确的。

关于缺点,我会避免wpfApplication1:SomeVm在窗口资源中创建。通常DataContextofWindow被设置为 a 的一个实例WindowViewModel,它又会持有对 的引用SomeVm。想象这样一个类:

public class WindowViewModel
{
    public SomeVM SomeVM{get; set;}
    public string Title {get; set;} //other data to bind by window
    //...
}

然后,在初始化窗口时,DataContext必须将其设置为 ViewModel 实例,例如:

MainWindow.DataContext = new WindowViewModel();

在 XAML 中,您将再次使用绑定:

<Grid>
    <ContentPresenter Content="{Binding SomeVm}" />   
</Grid>

我还建议将您的隐式DataTemplates放在generic.xaml字典中,而不是放在窗口中。这样您就可以在整个应用程序中重复使用这些模板。

此外,使用实现通用事件处理的 ViewModelBase 类要好得多,这样您就不需要重新实现INotifyPropertyChanged. 还要尽量避免属性更改通知中的“魔术字符串”。最好使用基于 lambda 的方法或新的Caller Info Attributes。我知道您的示例代码可能已简化,但我正在对其进行评论。

于 2013-09-07T19:13:26.993 回答