棱镜/WPF 中的新外壳/窗口显示错误的视图



受到这个答案的启发,我创建了一个通用的Shell(Prism,WPF(,如下所示:

<Window x:Class="VRMDataLogger.UI.Shell" xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" xmlns:prism="http://prismlibrary.com/" Title="My App" Height="450" Width="800">
    <Grid>
        <ContentControl prism:RegionManager.RegionManager="{Binding RegionManager}" prism:RegionManager.RegionName="MainShellRegion" />
    </Grid>
</Window>
public partial class Shell : Window
{
    public Shell(IRegionManager regionManager, Type contentType)
    {
        RegionManager = regionManager;
        InitializeComponent();
        RegionManager.RegisterViewWithRegion("MainShellRegion", contentType);
    }
    public IRegionManager RegionManager { get; }
}

初始 shell 在 App.CreateShell() 中创建:

protected override Window CreateShell()
{
    return new Shell(Container.Resolve<IRegionManager>(), typeof(StartScreen));
}

这工作正常,正确的视图显示在初始外壳中。

然后,我尝试从StartScreenViewModel创建第二个 Shell,它显示了不同的视图:

var shell = new Shell(RegionManager.CreateRegionManager(), typeof(MainScreen));
shell.Show();

这将打开一个新窗口,但它显示的视图与第一个窗口(StartScreen(相同,而不是MainScreen。我在这里做错了什么?

所以我也在考虑摆脱该地区的东西。但我认为这样做我会失去的是将依赖注入到我的视图模型中的良好功能

一点也不。事实上,您可以更好地控制视图模型的创建方式。

当视图优先时,您的视图模型通常由ViewModelLocator在导航时创建,然后传递NavigationParameters(如果有(。

如果手动创建视图模型并通过数据模板绑定视图,则可以完全控制视图模型的创建。最简单的选择是为视图模型注入工厂(如 Func<MainScreenViewModel> (,然后从容器中获得完整的依赖项注入。

internal class StartScreenViewModel
{
    public StartScreenViewModel( Func<MainScreenViewModel> mainScreenViewModelFactory )
    {
        GoToMainScreenCommand = new DelegateCommand( () => new Shell( mainScreenViewModelFactory() ).Show() );
    }
    public DelegateCommand GoToMainScreenCommand { get; }
}

当然,如果需要,您可以使用更复杂的手工工厂(请参阅此答案(。

尝试为每个外壳使用唯一的区域名称:

public partial class Shell : Window
{
    public Shell(IRegionManager regionManager, Type contentType)
    {
        RegionManager = regionManager;
        InitializeComponent();
        MainRegionName = Guid.NewGuid().ToString();
        RegionManager.RegisterViewWithRegion(MainRegionName, contentType);
    }
    public string MainRegionName { get; }
    public IRegionManager RegionManager { get; }
}

XAML:

<ContentControl prism:RegionManager.RegionManager="{Binding RegionManager}"
                prism:RegionManager.RegionName="{Binding MainRegionName, RelativeSource={RelativeSource AncestorType=Window}}" />

最新更新