Prism区域异常问题分析(导航失效?)

Posted dotNET跨平台

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Prism区域异常问题分析(导航失效?)相关的知识,希望对你有一定的参考价值。

前文

本篇文章主要讨论在WPF当中使用Prism区域导航的失效的问题, 在其它的博客当中也出现了多次讨论这个问题以及对应的解决方法,
例如重写OnInitialized方法等等。我认为这都不是解决问题的根源, 既然如此, 下面我们将来分析Prism的IRegionManager的具体流程。

Prism初始化过程

首先, 我们分析一下下面的代码, 用于创建应用程序的主页

protected override Window CreateShell()
        
            return Container.Resolve<MainWindow>();
        

那么, 在这个MainWindow当中, 如果我们定于以下一个区域, 则可以使用IRegionManager进行导航操作, 如下所示:

<ContentControl Grid.Row="1" prism:RegionManager.RegionName="ShowRegion" />

现在问题就来了, 一旦我们把这个窗口的实例销毁, 自己重新创建一个MainWindow的时候, 你会发现你无法进行导航了, 这个时候就是大家理解的那个问题, 是Region失效了?
并不是, 为了解决这个问题, 我们需要了解Prism框架本身做了什么动作, 查看源代码之后, 发现以下初始化代码:

var shell = CreateShell();  
            if (shell != null)
            
                MvvmHelpers.AutowireViewModel(shell);
                RegionManager.SetRegionManager(shell, _containerExtension.Resolve<IRegionManager>());
                RegionManager.UpdateRegions();
                InitializeShell(shell);
            

了解到, 从一开始获取到MainWindow之后, 陆续进行了上下文绑定, 设置IRegionManager实例以及更新区域的操作。

在这里, 我们至少了解了几个东西。

  • MainWindow的DataContext初始化的时机

  • MainWindow窗口当中IRegionManager的初始化过程

  • 区域刷新的动作

完成了这些动作之后, 最终ShowDialog展示了首页, 于是,我们可以在这里愉快的使用IRegionManager进行导航操作。

分析结果

如果想要实现在某个窗口当中进行导航, 除了定义区域之外, 你还需要做的就是给窗口设置IRegionManager的实例以及刷新区域, 核心就是这两行代码:

RegionManager.SetRegionManager(shell, _containerExtension.Resolve<IRegionManager>());
 RegionManager.UpdateRegions();

既然我们了解到这个之后, 还有一个问题也顺其自然的解决了, 在Prism当中使用弹窗服务不能导航的问题, 因为在Prism框架提供的IDialogService中并没有实现
设置IRegionManager以及刷新区域, 这就是问题的根源, 所以我们必须手动的去修改实现达到支持导航的功能。

弹窗中实现导航

示例: 以下代码, 展示了如何在弹窗当中设置区域以及刷新区域的问题。

var provider = ContainerLocator.Container.Resolve<IContainerProvider>();
            var regionManager = ContainerLocator.Container.Resolve<IRegionManager>();
            var win = provider.Resolve<object>("ShowWindow");
            if (win is Window view)
            
                RegionManager.SetRegionManager(view, regionManager);
                RegionManager.UpdateRegions();

                view.ShowDialog();
            

当然, 你完全可以自行实现IDialogService接口覆盖Prism提供的内部实现, 以达到弹窗支持导航的行为, 例如:

public interface IMyDialogService : IDialogService
    

    

    public class MyDialogService : DialogService, IMyDialogService
    
        public MyDialogService(IContainerExtension containerExtension)
            : base(containerExtension)
        
        

        public new void ShowDialog(string name, IDialogParameters parameters, Action<IDialogResult> callback)
        
            //...
        
    

总结

对于任何窗口, 我们都可以使用IRegionManager进行导航操作, 在第一次框架初始化的时候, 只不过是Prism内部帮我们处理了区域的设置以及刷新行为。
而如何我们想在其它地方使用区域导航, 则需要手动设置区域以及刷新区域即可。

以上是关于Prism区域异常问题分析(导航失效?)的主要内容,如果未能解决你的问题,请参考以下文章

在 Wpf 应用程序中使用 Prism 进行导航

棱镜:贝壳之间的导航?

如何使用 Prism 自定义适配器“导航”?

prism journal导航按钮的可用性探索记录

.NET Core 3 WPF MVVM框架 Prism系列之导航系统

如何在 Prism 中为我的对话窗口设置区域管理器?