1.概要

源码及PPT地址:https://github.com/JusterZhu/wemail

视频地址:https://www.bilibili.com/video/BV1KQ4y1C7tg?sharesource=copyweb

(1)Prism概览

03Prism WPF 入门实战 - Region_控件


  • Application:我们开发应用程序,初始化Bootstrapper。
  • Bootstrapper:是用来初始化应用程序级别的组件和服务,它也被用来配置和初始化module catalog和Shell 的View和View Model。
  • Modules:是能够独立开发、测试、部署的功能单元,Modules可以被设计成实现特定业务逻辑的模块(如Profile Management),也可以被设计成实现通用基础设施或服务的模块。
  • Shell是宿主应用程序(host application),modules将会被load到Shell中。Shell定义了应用程序的整体布局和结构,而不关心寄宿其中的Module,Shell通常实现通用的application service和infrastructure,而应用的逻辑则实现在具体的Module中,同时,Shell也提供了应用程序的顶层窗口。
  • Services:是用来实现非UI相关功能的逻辑,例如logging、exception management、data access。Services可以被定义在应用程序中或者是Module中,Services通常被注册在依赖注入容器中,使得其它的组件可以很容易的定位这个服务。
  • Container:注入服务、其他模块依赖。

03Prism WPF 入门实战 - Region_ide_02

03Prism WPF 入门实战 - Region_ide_03


(2)Region

  • Region是应用程序UI的逻辑区域(具体的表现为容器控件),Views在Region中展现,很多种控件可以被用作Region:ContentControl、ItemsControl、ListBox、TabControl。Views能在Regions编程或者自动呈现,Prism也提供了Region导航的支持。这么设计主要为了解耦让内容显示灵活具有多样性。

03Prism WPF 入门实战 - Region_控件_04


在实战项目当中,需根据业务需求来划分Region。

03Prism WPF 入门实战 - Region_初始化_05

(3)RegionManager

RegionManager主要实现维护区域集合、提供对区域的访问、合成视图、区域导航、定义区域。

区域定义方式有两种:



Xaml实现

<ContentControl x:Name=“ContentCtrl” prism:RegionManager.RegionName="ContentRegion" />


C#实现

RegionManager.SetRegionName(ContentCtrl,”ContentRegion”);

public MainWindowViewModel(IRegionManager regionManager)
{
_regionManager = regionManager;
_regionManager.RegisterViewWithRegion("ContentRegion", typeof(MainWindow));
}


(4)RegionAdapter

RegionAdapter(区域适配)主要作用为特定的控件创建相应的Region,并将控件与Region进行绑定,然后为Region添加一些行为。

因为并不是所有的控件都可以作为Region的,需要为需要定义为Region的控件添加RegionAdapter。一个RegionAdapter需要实现IRegionAdapter接口,如果你需要自定义一个RegionAdapter,可以通过继承RegionAdapterBase类来省去一些工作。

Prism为开发者提供了几个默认RegionAdapter:


  • ContentControlRegionAdapter:创建一个SingleActiveRegion并将其与ContentControl绑定
  • ItemsControlRegionAdapter:创建一个AllActiveRegion并将其与ItemsControl绑定
  • SelectorRegionAdapter:创建一个Region并将其与Selector绑定
  • TabControlRegionAdapter:创建一个Region并将其与TabControl绑定

2.详细内容

Region和RegionManager实战应用。

03Prism WPF 入门实战 - Region_控件_06

(1)定义Region及选择好容器控件

<TabControl prism:RegionManager.RegionName="TabRegion" />

(2)ViewModel注册视图到TabRegion当中 public class MainWindowViewModel : BindableBase { //Region管理对象 private IRegionManager _regionManager; private string _title = "Prism Application";

   public string Title
{
get { return _title; }
set { SetProperty(ref _title, value); }
}

public MainWindowViewModel(IRegionManager regionManager)
{
//Prism框架内依赖注入的RegionManager
_regionManager = regionManager;

//在ContentRegion中注册视图TempView(TabItem1)
_regionManager.RegisterViewWithRegion("TabRegion", typeof(TempView));

//TabItem2
_regionManager.RegisterViewWithRegion("TabRegion", typeof(Temp2View));

//TabItem3
_regionManager.RegisterViewWithRegion("WorkRegion", typeof(Temp3View));

//对视图的访问、操作
//var contentRegion = _regionManager.Regions["ContentRegion"];
//contentRegion.Context
//contentRegion.Remove()
//contentRegion.Activate()
//foreach (var item in contentRegion.ActiveViews)
//{
// contentRegion.Activate(item);
//}
}
}


03Prism WPF 入门实战 - Region_初始化_07

RegionAdapter实战应用。

如果在实际开发工作当中遇到了特殊场景需要而Prism并没有设置对应的RegionAdapter。这时候可以通过继承实现RegionAdapterBase内置对象来扩展一个新的RegionAdapter。

(1)实现一个新的RegionAdapter
/// <summary>
/// custom region adapter.
/// </summary>
public class StackPanelRegionAdapter : RegionAdapterBase<StackPanel>
{
public StackPanelRegionAdapter(IRegionBehaviorFactory regionBehaviorFactory) : base(regionBehaviorFactory)
{
}

protected override void Adapt(IRegion region, StackPanel regionTarget)
{
//该事件监听往StackPanel添加view时的操作
region.Views.CollectionChanged += (sender, e) =>
{
//监听到增加操作时则往StackPanel添加Children,枚举出来的操作在后面一段代码中体现
if (e.Action == System.Collections.Specialized.NotifyCollectionChangedAction.Add)
{
regionTarget.Children.Clear();
foreach (var item in e.NewItems)
{
regionTarget.Children.Add(item as UIElement);
}
}
};
}

protected override IRegion CreateRegion()
{
return new Region();
}
}

// Summary:
// Describes the action that caused a System.Collections.Specialized.INotifyCollectionChanged.CollectionChanged
// event.
public enum NotifyCollectionChangedAction
{
//
// Summary:
// An item was added to the collection.
Add,
//
// Summary:
// An item was removed from the collection.
Remove,
//
// Summary:
// An item was replaced in the collection.
Replace,
//
// Summary:
// An item was moved within the collection.
Move,
//
// Summary:
// The contents of the collection changed dramatically.
Reset
}

(2)在App.cs文件中注册新的RegionAdapter

public partial class App
{
/// <summary>
/// 应用程序启动时创建Shell
/// </summary>
/// <returns></returns>
protected override Window CreateShell()
{
return Container.Resolve<MainWindow>();
}

protected override void RegisterTypes(IContainerRegistry containerRegistry)
{
}

/// <summary>
/// 配置区域适配
/// </summary>
/// <param name="regionAdapterMappings"></param>
protected override void ConfigureRegionAdapterMappings(RegionAdapterMappings regionAdapterMappings)
{
base.ConfigureRegionAdapterMappings(regionAdapterMappings);


//添加自定义区域适配对象,会自动适配标记上prism:RegionManager.RegionName的容器控件为Region
regionAdapterMappings.RegisterMapping(typeof(StackPanel), Container.Resolve<StackPanelRegionAdapter>());
}
}

(3)在xaml中使用

<StackPanel prism:RegionManager.RegionName="StackPanelRegion"></StackPanel>