views:

1592

answers:

2

I'm trying to figure MEF's Constructor Injection attribute. I have no idea how i tell it to load the parameters to the constructor.

This is property i'm trying to load

    [ImportMany(typeof(BUsers))]
    public IEnumerable<BUsers> LoadBUsers { get; set; }

Here is the code that i use to import the assemblies.

 try
 {
            var catalog = new AggregateCatalog();
            catalog.Catalogs.Add(new AssemblyCatalog(System.Reflection.Assembly.GetExecutingAssembly()));
            catalog.Catalogs.Add(new DirectoryCatalog("DI")); 
            var container = new CompositionContainer(catalog);
            container.ComposeParts(this);
 }

Here is the class i'm trying to load

[Serializable]
[Export(typeof(BUsers))]
public class EditProfile : BUsers
{
    [ImportingConstructor]
    public EditProfile(string Method, string Version)
    {

        Version = "2";
        Action = "Edit";
        TypeName = "EditProfile";
    }
+7  A: 

When you use the ImportingConstructor attribute, the parameters to the constructor become imports. By default, what you are importing (the contract name) is based on the type of the parameter or property that your are importing into. So in this case the contract type for both your imports is string, and there's no real difference between the first and second parameter.

It looks like you are trying to use imports to supply configuration values, which isn't necessarily what it was designed for. To get it to do what you want, you should override the contract name for each of the parameters, like this:

[ImportingConstructor]
public EditProfile([Import("Method") string Method, [Import("Version")] string Version)
{ }

Then you need exports for Method and Version in your container. One way to do this is just to add them directly:

var container = new CompositionContainer(catalog);
container.ComposeExportedValue("Method", "MethodValue");
container.ComposeExportedValue("Version", "2.0");
container.ComposeParts(this);

(Note that ComposeExportedValue is actually an extension method defined on the static AttributedModelServices class.)

If you want to read these values from a configuration file of some sort, you could create your own export provider which reads the configuration and provides the values in it as exports to the container.

An alternative way to handle this would be to just import an interface that provides access to the configuration values by name, and get the values you need from the body of the constructor.

Daniel Plaisted
Many Thanks, I was running out of ideas on this one.
alpha
I just downloaded the new one at CodePlex. The method of ComposeExportedValue() is not in the class of CompositionContainer. Where is it?
David.Chu.ca
I think I found the method. It is in the class of AttributedModelServices, where the method is defined as extended method to class CompositionContainer.
David.Chu.ca
@David.Chu.ca Yes, ComposeExportedValue is an extension method on the AttributedModelServices class.
Daniel Plaisted
+2  A: 

I like Danial's solution; however, only one thing I see is the tight coupling of parameter names between the actor (who creates CompopositionContrainer()) and Export part with [ImportingConstructor] for customized CTOR. For example, "Method" has two be matched in both places. It makes hard to maintain the Export part if the actor and Export part are in difference projects.

If it is possible, I would add the second CTOR to the Export part class. For example:

[Export(typeof(BUsers))] 
public class EditProfile : BUsers
{
[ImportingConstructor]
public EditProfile(EditProfileParameters ctorPars)
: this(ctorPars.Method, ctorPars.Version) {}

public EditProfile(string Method, string Version)
{
    Version = "2";
    Action = "Edit";
    TypeName = "EditProfile";
}

The class of EditProfileParameters should be straightforward: two properties of Method and Version:

[Export]
public class EtitProfileParameters{
   public string Method { get; set; }
   public string Version { get; set; }
}

The key point is to add Export attribute to the class. Then MEF should be able to map this class to the parameter of EditProfile's CTOR.

Here is example to add the Export part to container:

var container = new CompositionContainer(catalog);
var instance1 = new EditProfileParameers();
// set property values from config or other resources
container.ComposeExportedValue(instance1);
container.ComposeParts(this);
David.Chu.ca