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内部帮我们处理了区域的设置以及刷新行为。
而如何我们想在其它地方使用区域导航, 则需要手动设置区域以及刷新区域即可。