通过Scoped RegionManager的区域将视图添加到ItemsControl的ItemsSource

时间:2017-11-18 19:17:55

标签: c# wpf xaml prism region-management

我试图通过某个地区填充ItemsSource ComboBox的衍生物ItemsControl

查看

通过RegionManager将范围prism:RegionManager.RegionManager="{Binding RegionManager}"(在视图模型中找到)分配给视图。

MainWindow.xaml

<Window x:Class="Applications.Testing.Wpf.RegionCreationTester.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
    xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
    xmlns:local="clr-namespace:Applications.Testing.Wpf.RegionCreationTester"
    xmlns:prism="http://www.codeplex.com/prism"
    mc:Ignorable="d"
    Title="MainWindow" Height="350" Width="525"
    prism:RegionManager.RegionManager="{Binding RegionManager}"
    prism:ViewModelLocator.AutoWireViewModel="True">
    <Grid>
        <ComboBox prism:RegionManager.RegionName="{x:Static local:RegionNames.itemsControlRegion}"/>
    </Grid>
</Window>

MainWindow.xaml.cs

namespace Applications.Testing.Wpf.RegionCreationTester
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window, IModelled<MainWindowViewModel>
    {
        public MainWindowViewModel ViewModel
        {
            get
            {
                return (MainWindowViewModel)DataContext;
            }
            set
            {
                DataContext = value;
            }
        }

        public MainWindow()
        {
            InitializeComponent();

            ViewModel.PopulateItemsControl();
        }
    }
}

视图模型

视图模型通过prism:ViewModelLocator.AutoWireViewModel="True"分配给视图的DataContext。

MainWindowViewModel.cs

namespace Applications.Testing.Wpf.RegionCreationTester
{
    public class MainWindowViewModel
    {
        /// <summary>
        /// Gets the <see cref="RegionManager"/> scoped to this control.
        /// </summary>
        /// <remarks>Exists so that child controls can register regions for their own child controls which are also child controls in this control.</remarks>
        public RegionManager RegionManager { get; } = new RegionManager();

        /// <summary>
        /// Adds some child views to the <see cref="RegionNames.itemsControlRegion"/>.
        /// </summary>
        /// <remarks>Normally these views would be resolved using an IoC container but this have been omitted for brevity.</remarks>
        public void PopulateItemsControl()
        {
            var region = RegionManager.Regions[RegionNames.itemsControlRegion];
            region.Add(new TextBlock { Text = "Item #1" });
            region.Add(new Button { Content = "Item #2" });
        }
    }
}

RegionNames.cs

namespace Applications.Testing.Wpf.RegionCreationTester
{
    public static class RegionNames
    {
        public const string itemsControlRegion = "collectionRegion";
    }
}

引导程序

的App.xaml

<Application x:Class="Applications.Testing.Wpf.RegionCreationTester.App"
         xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"/>

App.xaml.cs

namespace Applications.Testing.Wpf.RegionCreationTester
{
    /// <summary>
    /// Interaction logic for App.xaml
    /// </summary>
    public partial class App : Application
    {
        protected override void OnStartup(StartupEventArgs e)
        {
            base.OnStartup(e);
            new RegionCreationTesterBootstrapper().Run();
        }
    }
}

RegionCreationTesterBootstrapper.cs

namespace Applications.Testing.Wpf.RegionCreationTester
{
    public class RegionCreationTesterBootstrapper : UnityBootstrapper
    {
        protected override DependencyObject CreateShell()
            => new MainWindow();

        protected override void InitializeShell()
        {
            base.InitializeShell();

            (Application.Current.MainWindow = (Window)Shell).Show();
        }
    }
}

一旦发生所有区域填充并且应用程序即将运行,我得到一个Prism.Regions.UpdateRegionsException,其中包含一条InnerException,并在"Region with the given name is already registered: collectionRegion" App的行中显示消息new RegionCreationTesterBootstrapper().Run() }。在调用MainWindow的构造函数退出之后,我能够获得断点的最后一行是new MainWindow()中的CreateShell。当我只想注册一次时,为什么我被告知该地区已经注册?我在MainWindow的构造函数中设置了断点,以确认它只被创建一次,即使它不是,它的作用域RegionManager也应该防止发生此异常。我错过了什么?

更新

我刚刚在PopulateItemsControl中注释掉了代码,发现即使只有一个视图添加到该区域也会引发异常,如果没有向该区域添加视图但是访问该区域,则会发生异常(正如在行中所做的那样:var region = RegionManager.Regions[RegionNames.itemsControlRegion];)。因此,现在的问题是访问用于View Injection的作用域RegionManager上的现有区域,以便为其添加视图;我不确定为什么从RegionManager访问某个区域会改变它的状态,这看起来像是Prism中的一个错误,或者可能与懒惰的枚举有关。

1 个答案:

答案 0 :(得分:2)

嗯,你这样做了两次,你只是不知道。当您在ComboBox上设置RegionName附加属性时,将附加将创建具有给定名称的区域的事件处理程序(这是RegionManager的静态部分)。当您在VM中实例化的RegionManager实例尝试访问区域集合时,索引器首先在RegionManager类上调用 static 方法来引发该事件。获得创建区域任务的全局RegionManager实例(当您使用RegionName附加属性时)尚未完成它的作业 - 当您尝试使用您的实例访问该区域时,尚未加载窗口,处理程序尚未被删除,并再次调用它。如果在窗口加载后调用了PopulateItemsControl方法(例如在MainWindow的Loaded事件处理程序中),则不会得到异常,但是您的代码将无法按预期工作。这是因为RegionManager的实例不是“处理”您的collectionRegion,全局RegionManager是。

注入RegionManager实例

如果您的VM中需要RegionManager实例,请使用构造函数注入。

public class MainWindowViewModel : BindableBase
{
    private IRegionManager rm;

    public MainWindowViewModel(IRegionManager manager)
    {
        this.rm = manager;
    }

    public void PopulateItemsControl()
    {
        var region = rm.Regions[RegionNames.itemsControlRegion];
        region.Add(new TextBlock { Text = "Item #1" });
    }
}

依赖注入容器(Unity或您正在使用的任何东西)将在创建VM时解析IRegionManager实例(无论如何,PRISM正在为您完成这项工作,您自己并未对其进行实例化。)

地区范围

RegionManager保留区域集合,不允许具有相同名称的区域。因此,除非您的窗口中有多个ComboBox都具有名为collectionRegion的区域,否则RegionManager(全局的)就可以了。如果您的collectionRegion将具有相同视图类的实例,这些实例都在其自身内定义另一个区域,那么您需要区域范围 - RegionManager实例,这些实例具有这些视图的自己的范围。在这种情况下,Add方法可以为该视图创建RegionManager的本地实例:

IRegion collectionRegion = this.regionManager.Regions["collectionRegion"];
bool makeRegionManagerScope = true;
IRegionManager localRegionManager =
    collectionRegion.Add(view, null, makeRegionManagerScope);