zoukankan      html  css  js  c++  java
  • PRISM 4

    I am using Prism 4 with MEF Extensions and the MVVM pattern. During initialization in a module I call RegisterViewWithRegion(RegionNames.MyRegion, typeof(MyView)) which works perfectly when the view is constructed like this:

    [Export]
    [PartCreationPolicy(CreationPolicy.NonShared)]
    public partial class MyView : UserControl
    {
        public MyView()
        {
         ....
    

    The view gets registered and everything is fine. As soon as I change the Export to a Custom Export Attribute the view can't be found anymore, although it is still in the container. This Custom Export Attribute is taken from the Stock Trader RI:

    [AttributeUsage(AttributeTargets.Class, AllowMultiple = false)]
    [MetadataAttribute]
    public class ViewExportAttribute : ExportAttribute, IViewRegionRegistration
    {
        public ViewExportAttribute()
            : base(typeof(object))
        { }
    
        public ViewExportAttribute(string viewName)
            : base(viewName, typeof(object))
        {
            ViewName = viewName;
        }
    
        public string RegionName { get; set; }
        public string ViewName { get; set; }
    
    }
    

    and the interface is

    public interface IViewRegionRegistration
    {
        string RegionName { get; }
        string ViewName { get; }
    }
    

    By changing the Export Attribute to

    [ViewExport(ViewName = "MyView", RegionName = RegionNames.MyRegion)]
    [PartCreationPolicy(CreationPolicy.NonShared)]
    public partial class MyView : UserControl
    {
        public MyView()
        {
        ....
    

    when calling RegisterViewWithRegion it throws an error: Activation error occured while trying to get instance of type MyView, key ""

    Any advice? I was looking at this part of code the whole day without finding a solution.

    shareimprove this question
     
        
    Later that night... I finally found out that is has something to do with this part in the Custom Export Attribute:base(typeof(object)) - but still no knowledge of how to solve the RegisterViewWithRegion problem... – okieh Jun 7 '11 at 21:14

    4 Answers

    Another day, another way... I will try to answer my question even though I have only limited knowledge about PRISM. In other words: I'm still learning.

    The Custom Export Attribute taken from the Stock Trade RI is used by the AutoPopulateExportedViewsBehavior. This behavior adds a view to its region automatically by checking the Export Attribute for the region name then adds the view to the corresponding region. But all views with this Custom Attribute now have a contract name of "object" which makes it impossible for the ServiveLocator to find them. This Custom Attribute is for a scenario with fixed region/view links. A solution when working with a Custom Export Attribute is to get all exports of type "object" and the appropriate metadata:

    MyView view;
    var myList = container.GetExports<object, IViewRegionRegistration>();
    foreach (Lazy<object, IViewRegionRegistration> lazy in myList)
    {
        if (lazy.Metadata.ViewName == "MyView")
        {
            view = lazy.Value as MyView;
            region.Add(view);
            break;
        }
    }
    

    But I think when using ViewInjection and Prism Navigation it is better to just use the default [Export] attribute, then everything works smoothly.

    shareimprove this answer
     
     

    Are you configuring the aggregate catalog in your MEF bootstrapper? If so, are you adding the assembly that contains your ViewExportAttribute and AutoPopulateExportedViewsBehavior classes? I believe this happens in the StockTraderRI's bootstrapper with this line:

    this.AggregateCatalog.Catalogs.Add(new AssemblyCatalog(typeof(StockTraderRICommands).Assembly));
    

    The StockTraderRICommands class is in the same assembly as the ViewExportAttribute and AutoPopulateExportedViewsBehavior classes.

    shareimprove this answer
     
        
    I had the same issue as the original asker and this was the solution. – Dylan Nov 30 '11 at 16:02

    The custom export attribute passes typeof(object) to the base constructor, which changes the contract exported so that it no longer matches the import. Change it so it calls the parameterless constructor.

    As far as the activation error you'll need to look at the exception in more detail. The root cause is probably there somewhere, perhaps buried under an InnerException.

    shareimprove this answer
     

    I encountered exactly the same problem and it was a hard one for a MEF/PRISM beginner. okieh describes the problem very well, I just want to post an alternative solution, coming from theStocktraderUI sample application:

    The solution works (/seems to work) if you want View discovery without any form of config file, etc. where you have to register your views.

    1. Modify the ViewExport custom event

    [Export]
    [AttributeUsage(AttributeTargets.Class, AllowMultiple = false)]
    [MetadataAttribute]
    public sealed class ViewExportAttribute : ExportAttribute, IViewRegionRegistration
    {
        public ViewExportAttribute()
            : base(typeof(UserControl))
        { }
    
        public string ViewName { get { return base.ContractName; } }
    
        public string RegionName { get; set; }
    }
    

    The [Export] attribute is added and the base constructor is now called with UserControl instead of object. That way it can be discovered by MEF.

    2. Modify AutoPopulateExportedViewsBehavior

    [ImportMany(typeof(UserControl))]
    public Lazy<UserControl, IViewRegionRegistration>[] RegisteredViews { get; set; }
    

    The [ImportMany] attribute is added and the type of the Lazy initializiation is changed to UserControl. Now, all UserControls with IViewRegionRegistration-implementing MetaData-type are imported.

    That's basically it. You can use the [ViewExport] as before. Note, that Views are limited to (sub)types of UserControl. I suppose this can be modified if you want to. And make sure your aggregate catalog imports the ViewExportAttribute and the AutoPopulateExportedViewsBehavior, as Nicolaus said...

    This way, you don't need additional interfaces for your views and can still discover everything without hardcoded registration.

    I hope it helps and let me know, if I missed any drawbacks of my solution.

    http://stackoverflow.com/questions/6271167/prism-4-registerviewwithregion-custom-export-attributes

  • 相关阅读:
    消息队列中间件的技术选型分析
    数据库和缓存一致性的问题
    《RocketMQ 安装和使用》
    RocketMQ原理讲解系列文章
    阿里巴巴开源项目
    RocketMQ与Kafka对比(18项差异)
    对象初始化
    pytest_05_fixture之conftest.py
    pytest_04_测试用例setup和teardown
    Python与MogoDB交互
  • 原文地址:https://www.cnblogs.com/tianciliangen/p/4983074.html
Copyright © 2011-2022 走看看