Loading

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)
        {
            //...
        }
    }

注意

如果在弹窗服务当中设置区域, 在窗口再次打开的同时会提示区域名称已注册的问题, 则你需要在关闭前移除原有的所有区域名即可。

 regionManager.Regions.Remove("ShowRegion");

总结

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

posted @ 2022-01-14 14:16  痕迹g  阅读(3055)  评论(3编辑  收藏  举报