MVVM Light 一个窗口承载两个视图

 

原文地址:http://www.codeproject.com/Articles/323187/MVVMLight-Using-Two-Views

本文并不是对原文逐句翻译,为了尽量通俗易懂,本人对原文有所增删。

如果你对MVVM还不太熟悉可以先看看本人之前的关于MVVM的博客:

http://www.cnblogs.com/zhangzhi19861216/archive/2013/03/19/WPF-MVM.html

如果你对MVVMLight不太了解,在阅读本文前,最好先阅读本人之前的一片译文:

http://www.cnblogs.com/zhangzhi19861216/archive/2013/03/20/2971400.html

在之前的文章中,我快速展示了如何用MVVMLight创建一个单View、单Window的WPF应用程序。但是WPF应用程序的趋势是一个窗口承载多个视图,

以便减少弹出的对话框或子窗口。本文展示了用MVVM模式构建一个简单的一个窗口承两个视图的应用程序。

在开始之前:

1、 需要VS2010

2、 确保你安装了NUget

3、 用Manage Nuget Package 引用并且添加 MVVM Light。

4、 这篇文章的源码在 github上。

注意,为了简洁起见,博文中的很多代码,特别是Xaml省略了,所以读者最好去github下载源码进行阅读。

承载多个视图

这个应用程序的结构类似之前的文章,我们有个MainWindow,ViewModelLocator和MainViewModel。

一张图片胜过千言万语,所以事不宜迟,这里是VS2010中的项目结构视图:

该项目为典型的MVVM风格:3个文件夹分别为Model,ViewModel和View。在当前项目,我们没有任何的Model,因此它可以忽略不计。

我们从View开始:在Views文件夹下我们创建两个UserControl——FirstView.xaml和SecondView.xaml。

FirstView.xaml和我们上篇文章 一致,而SecondView.xaml仅包含一个Lable元素。

所有涉及两个视图切换的操作都在MainViewModel.cs, MainWindow.xaml, and App.xaml当中.

首先看看MainWindow XAML,如下所示:

  1. <Window x:Class="TwoViews.MainWindow"
  2. DataContext="{Binding Main,
  3. Source={StaticResource Locator}}">
  4. <Grid>
  5. <Grid.RowDefinitions>
  6. <RowDefinition Height="Auto" />
  7. <RowDefinition Height="Auto" />
  8. </Grid.RowDefinitions>
  9.  
  10. <ContentControl Content="{Binding CurrentViewModel}" />
  11.  
  12. <DockPanel Grid.Row="1" >
  13. <Button Command="{Binding SecondViewCommand}"
  14. Content="Second View"
  15. DockPanel.Dock="Right" />
  16. <Button Command="{Binding FirstViewCommand}"
  17. Content="First View"
  18. DockPanel.Dock="Left" />
  19. </DockPanel>
  20. </Grid>
  21. </Window>

看上面的代码,像之前一样,我们使用ViewModelLocator绑定我们的Main  视图模型到MainWindow。然而,这一次,我们有一个ContentControl绑定到一个称为CurrentViewModel(定义在MainViewModel当中)的新属性,并且有两个按钮分别绑定一个命令。

获取源码(译者所建项目):

files.cnblogs.com/zhangzhi19861216/ MVVMLigntStudy.rar

我们打开项目运行一下:

(First View)

(Second View)

我们看到,当分别点击按钮,有两个View在切换(实际上是我们的ViewModel在更新)。

让我们来看看MainViewModel.cs的代码:

  1. public class MainViewModel : ViewModelBase
  2. {
  3.  
  4. private ViewModelBase _currentViewModel;
  5.  
  6. readonly static SecondViewModel _secondViewModel = new SecondViewModel();
  7.  
  8. readonly static FirstViewModel _firstViewModel = new FirstViewModel();
  9.  
  10. public ViewModelBase CurrentViewModel
  11. {
  12. get
  13. {
  14. return _currentViewModel;
  15. }
  16. set
  17. {
  18. if (_currentViewModel == value)
  19. return;
  20. _currentViewModel = value;
  21. RaisePropertyChanged("CurrentViewModel");
  22. }
  23. }
  24.  
  25. public ICommand FirstViewCommand { get; private set; }
  26.  
  27. public ICommand SecondViewCommand { get; private set; }
  28.  
  29. public MainViewModel()
  30. {
  31. CurrentViewModel = MainViewModel._firstViewModel;
  32. FirstViewCommand = new RelayCommand(() => ExecuteFirstViewCommand());
  33. SecondViewCommand = new RelayCommand(() => ExecuteSecondViewCommand());
  34. }
  35.  
  36. private void ExecuteFirstViewCommand()
  37. {
  38. CurrentViewModel = MainViewModel._firstViewModel;
  39. }
  40.  
  41. private void ExecuteSecondViewCommand()
  42. {
  43. CurrentViewModel = MainViewModel._secondViewModel; ;
  44. }
  45. }

MainViewModel定义了三个ViewModel(_currentViewModel、_secondViewModel 、_firstViewModel),和两命令(FirstViewCommand,SecondViewCommand),两个命令分别调用如下两个方法:

  1. private void ExecuteFirstViewCommand()
  2. {
  3. CurrentViewModel = MainViewModel._firstViewModel;
  4. }
  5.  
  6. private void ExecuteSecondViewCommand()
  7. {
  8. CurrentViewModel = MainViewModel._secondViewModel; ;
  9. }

每次触发命令(点击FirstView或者SecondView按钮),其实都是在_firstViewModel和_secondViewModel之间切换——把它们赋给CurrentViewModel,而MainWindow.xaml中的ContentControl元素绑定的就是CurrentViewModel,因此跟着更新。

疑惑:ViewModel之间的切换,如何引起View的不同显示呢?

接下来看看App.xaml:

  1. <Application x:Class="TwoViews.App"
  2. xmlns:views="clr-namespace:TwoViews.Views"
  3. xmlns:vm="clr-namespace:TwoViews.ViewModels"
  4. StartupUri="MainWindow.xaml"
  5. >
  6. <Application.Resources>
  7. <vm:ViewModelLocator x:Key="Locator" />
  8. <DataTemplate DataType="{x:Type vm:SecondViewModel}">
  9. <views:SecondView />
  10. </DataTemplate>
  11. <DataTemplate DataType="{x:Type vm:FirstViewModel}">
  12. <views:FirstView />
  13. </DataTemplate>
  14. </Application.Resources>
  15. </Application>

在这里,我们使用DataTemplate 将ViewModel通过View表现出来。

例如,毫不夸张地说:“如果我的数据类型应该是FirstViewModel,那么WPF框架应该呈现的是FirstView用户控件。

译者注:在WPF中我们可以使用DataTemplate为自己的数据(视图模型中的数据)定制显示方式,也就是说虽然某数据数据是一定的,

但我们可以做到(通过View)让它的表现方式多种多样。

本文就翻译到这里,翻译的章节次序和原文不太相符。因为我觉得这样更容易理解。

如果文中出现错误或者歧义,请指正。

译文源码:files.cnblogs.com/zhangzhi19861216/ MVVMLigntStudy.rar

原文源码:https://github.com/barrylapthorn/MvvmLightExamples

译者注:

看到原文下面留言:Nothing to download? 估计是进入原文源码下载页面,读者找不到源码下载地址。

提示:进入原文下载页面后,点击Zip按钮即可,这主要是很多读者对github不太熟悉。

如果你对MVVMLight感兴趣,而你的英文实在不怎么好,推荐你阅读如下国内作者的博客:

http://www.cnblogs.com/phoenixtrees/archive/2011/05/01/2033847.html

http://blog.csdn.net/duanzilin/article/details/6387639

接下来我还会翻译更多关于WPF的文章,希望大家多多支持,多交流。

MVVM Light 一个窗口承载两个视图的更多相关文章

  1. 一个ActionResult中定位到两个视图—<团委项目>

         在使用MVC做项目的时候一般的情况就是一个ActionResult一个视图,这样对应的Return View();就可以找到下面对应的视图,这是根据一个原则,"约定大于配置&quo ...

  2. 一个ActionResult中定位到两个视图—<团委项目>

    在使用MVC做项目的时候一般的情况就是一个ActionResult一个视图,这样对应的Return View();就可以找到下面对应的视图,这是根据一个原则,“约定大于配置”,但是我们有的时候需要在一 ...

  3. MVVM Light 新手入门(3) :ViewModel / Model 中定义“事件” ,并在View中调用 (无参数调用)

    今天学习MVVM架构中“事件”的添加并调用,特记录如下,学习资料均来自于网络,特别感谢翁智华 的 利刃 MVVMLight 6:命令基础 在MVVM Light框架中,事件是WPF应用程序中UI与后台 ...

  4. Mvvm Light Toolkit 入门

    原文:Mvvm Light Toolkit 入门 前言 之前学习UWP的时候就一直看到有关MVVM的资料但是一直没有系统的去学,最近正好有时间,特地来攻破这个点,顺便学习一下VS与GitHub的链接和 ...

  5. 【MVVM Light】Messager的使用

    一.前言       在MVVM编程的模式中,有时候我们会遇到一个很尴尬的情况: 若干个xaml.cs都复用一个ViewModel,当ViewModel想传递一个特定的消息给某一个xaml.cs的时候 ...

  6. Mvvm Light Toolkit for WPF/Silverlight系列之搭建mvvmlight开发框架

    Mvvm Light Toolkit for WPF/Silverlight系列之搭建mvvmlight开发框架   本章节,我将通过示例介绍如何搭建mvvmlight开发环境.示例中的我会针对wpf ...

  7. MVVM Light须要注意的10个问题

    MVVM Light须要注意的10个问题 从使用XAML技术基础開始(实际上并非非常久曾经).我便关注MVVM(Model – View – ViewModel)模式.偶然接触到MVVM Light不 ...

  8. WPF: WpfWindowToolkit 一个窗口操作库的介绍

    在 XAML 应用的开发过程中,使用MVVM 框架能够极大地提高软件的可测试性.可维护性.MVVM的核心思想是关注点分离,使得业务逻辑从 View 中分离出来到 ViewModel 以及 Model ...

  9. MVVM Light Toolkit使用指南

    原文:MVVM Light Toolkit使用指南 原文地址:  https://blog.csdn.net/ldld1717/article/details/77040077 概述 MVVM Lig ...

随机推荐

  1. Linux下Join命令

    Linux下Join命令 最近新上线算法,打算分析起点书籍点击率的波动,原来已经有流程每天每本书籍的点击率数据(文件).之前这种情况都是写代码对不同天的进行合并,后来发现linux下直接就有join命 ...

  2. centos 5.4 安装nodejs + npm(转)

    而在安装nodejs的时候,需要用到,所以需要手动安装bz2库. sudo yum install -y bzip2* cd Python-/Modules/zlib ./configure make ...

  3. struts2 struts.xml配置文件详解

    <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN&quo ...

  4. stack例子

    栈使用在括号匹配中的例子 程序如下: #include<stack> #include<iostream> using namespace std; int main() { ...

  5. BeagleBone Black Linux驱动程序开发入门(1): LED驱动程序

    这篇文章展示如何在BBB平台上编写LED驱动程序,本文的程序是根据国嵌S3C2440的LED驱动的例子并结合内核中OMAP系列的gpio操作来改的.本文中的程序包括驱动程序模块和用户空间程序.废话不多 ...

  6. 字符的截取方法使用的是Substring 和三目运算符

    substring(0,m.title.length>11?11:m.title.length)

  7. c语言,strcpy

    #include <stdio.h> #include <string.h> int main() {  char string[10];  char *str="a ...

  8. xml_03

    1.xml 2.对于XML文档的约束   |-DTD      <!DOCTYPE 根元素 [       <!ELEMENT 元素名 (xx)>       <!ATTLIS ...

  9. Apache Avro 与 Thrift 比较

    http://www.tbdata.org/archives/1307 Avro和Thrift都是跨语言,基于二进制的高性能的通讯中间件. 它们都提供了数据序列化的功能和RPC服务. 总体功能上类似, ...

  10. 深入理解Java的接口和抽象类 _摘抄

    http://www.cnblogs.com/dolphin0520/p/3811437.html 原文 深入理解Java的接口和抽象类 对于面向对象编程来说,抽象是它的一大特征之一.在Java中,可 ...