我可以实现自己的视图解析服务并让RequestNavigate使用它吗?

时间:2011-09-17 07:15:37

标签: wpf mvvm prism prism-4

我是Prism的新手,我目前正在使用Prism作为概念验证项目重写我们现有的一个应用程序。

应用程序使用MVVM和ViewModel第一种方法:我们的ViewModel由容器解析,IViewResolver服务确定应该连接到哪个视图(使用其他名称约定)。

此时代码(向选项卡控件添加视图)看起来像这样:

var vm = (get ViewModel from somewhere)
IRegion reg = _regionManager.Regions["MainRegion"];
var vw = _viewResolver.FromViewModel(vm); // Spins up a view and sets its DataContext
reg.Add(vw);
reg.Activate(vw);

这一切都很好,但我真的很想使用Prism导航框架为我做所有这些事情,这样我就可以做到这样的事情:

_regionManager.RequestNavigate(
    "MainRegion", 
    new Uri("NameOfMyViewModel", UriKind.Relative)
);

让Prism旋转ViewModel + View,设置DataContext并将视图插入该区域。

通过创建引用ViewModel类型的DataTemplates,我取得了一些成功,例如:

<ResourceDictionary xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:local="clr-namespace:Module01">
<DataTemplate DataType="{x:Type local:TestViewModel}">
<local:TestView />
</DataTemplate>
</ResourceDictionary>

...并且在模块初始化时让模块将相关的资源字典添加到应用程序资源中,但这似乎有些垃圾。

有没有办法从Prism有效地接管视图创建,以便在调用RequestNavigate时我可以查看提供的Uri并根据它调整视图/视图模型?有一个RegionManager.RegisterViewWithRegion的重载,需要一个允许你自己提供视图的委托,我想我就是这样的。

我想我可能需要提供自己的IRegionBehaviorFactory,但我不确定所涉及的是什么(或者即使我走在正确的道路上!)。

任何帮助表示赞赏!

- 注意:最初发布在the prism codeplex site

2 个答案:

答案 0 :(得分:8)

当然可以这样做。我发现Prism v4真的是可扩展的,只要你知道插在哪里就可以了。

在这种情况下,您需要自己的IRegionNavigationContentLoader自定义实现。

以下是如何在引导程序中进行设置(示例来自我自己的一个项目的UnityBootstrapper的子类):

protected override void ConfigureContainer()
{
    // IMPORTANT: Due to the inner workings of UnityBootstrapper, accessing
    // ServiceLocator.Current here will throw an exception!
    // If you want access to IServiceLocator, resolve it from the container directly.
    base.ConfigureContainer();

    // Set up our own content loader, passing it a reference to the service locator
    // (it will need this to resolve ViewModels from the container automatically)
    this.Container.RegisterInstance<IRegionNavigationContentLoader>(
       new ViewModelContentLoader(this.Container.Resolve<IServiceLocator>()));
}

ViewModelContentLoader本身派生自RegionNavigationContentLoader以重用代码,并且看起来像这样:

public class ViewModelContentLoader : RegionNavigationContentLoader
{
    private readonly IServiceLocator serviceLocator;

    public ViewModelContentLoader(IServiceLocator serviceLocator)
        : base(serviceLocator)
    {
        this.serviceLocator = serviceLocator;
    }

    // THIS IS CALLED WHEN A NEW VIEW NEEDS TO BE CREATED
    // TO SATISFY A NAVIGATION REQUEST
    protected override object CreateNewRegionItem(string candidateTargetContract)
    {
        // candidateTargetContract is e.g. "NameOfMyViewModel"

        // Just a suggestion, plug in your own resolution code as you see fit
        var viewModelType = this.GetTypeFromName(candidateTargetContract);
        var viewModel = this.serviceLocator.GetInstance(viewModelType);

        // get ref to viewResolver somehow -- perhaps from the container?
        var view = _viewResolver.FromViewModel(vm);

        return view;
    }

    // THIS IS CALLED TO DETERMINE IF THERE IS ANY EXISTING VIEW
    // THAT CAN SATISFY A NAVIGATION REQUEST
    protected override IEnumerable<object> 
    GetCandidatesFromRegion(IRegion region, string candidateNavigationContract)
    {
        if (region == null) {
            throw new ArgumentNullException("region");
        }

        // Just a suggestion, plug in your own resolution code as you see fit
        var viewModelType = this.GetTypeFromName(candidateNavigationContract);

        return region.Views.Where(v =>
            ViewHasDataContract((FrameworkElement)v, viewModelType) ||
            string.Equals(v.GetType().Name, candidateNavigationContract, StringComparison.Ordinal) ||
            string.Equals(v.GetType().FullName, candidateNavigationContract, StringComparison.Ordinal));
    }

    // USED IN MY IMPLEMENTATION OF GetCandidatesFromRegion
    private static bool 
    ViewHasDataContract(FrameworkElement view, Type viewModelType)
    {
        var dataContextType = view.DataContext.GetType();

        return viewModelType.IsInterface
           ? dataContextType.Implements(viewModelType)
           : dataContextType == viewModelType 
                   || dataContextType.GetAncestors().Any(t => t == viewModelType);
    }

    // USED TO MAP STRINGS OF VIEWMODEL TYPE NAMES TO ACTUAL TYPES
    private Type GetTypeFromName(string typeName)
    {
        // here you need to map the string type to a Type object, e.g.
        // "NameOfMyViewModel" => typeof(NameOfMyViewModel)

        return typeof(NameOfMyViewModel); // hardcoded for simplicity
    }
}

答案 1 :(得分:2)

要停止对“ViewModel第一种方法”的一些疑惑: 您使用更多“控制器方法”,但没有“ViewModel第一种方法”。 “ViewModel第一种方法”是,当你在ViewModel中注入View时,你通过第三方组件(控制器)将ViewModel和View连接起来,顺便提一下(我不想说) “最好”,但是)最松散耦合的方法。

但要回答你的问题: 一种可能的解决方案是为Prism RegionManager编写一个扩展,它完全按照您的描述进行:

    public static class RegionManagerExtensions
    {            
        public static void AddToRegion<TViewModel>(
               this IRegionManager regionManager, string region)
        {
            var viewModel = ServiceLocator.Current.GetInstance<TViewModel>();
            FrameworkElement view;

            // Get View depending on your conventions

            if (view == null) throw new NullReferenceException("View not found.");

            view.DataContext = viewModel;
            regionManager.AddToRegion(region, view);
            regionManager.Regions[region].Activate(view);

        }
    }

然后你可以像这样调用这个方法:

regionManager.AddToRegion<IMyViewModel>("MyRegion");