Why?
I've created this post because I really love WPF and MVVM, but I hate ViewModelsLocator or however it is called. Generally I'm talking about this totally ugly static class that can be called God-Class, where we have all view model instances resolved. It is probably the most lazy solution and the most often used. Of course I'm not talking about an enterprise solutions like Prism, where everything is done and you don't need to create an ugly locator because you have ViewModelLocationProvider where you can define that you want to use DI (e.g. Unity). I'm rather targeting simple frameworks helping working with MVVM, like MVVM Light, Simple MVVM...and more.I decided to make a different solution, clean enough to forget about adding view model locator getters every time I'm adding new screen or new ViewModel. What is also important I didn't want to add too much boilerplate code.
First of all, why we need something like a ViewModelLocator? When we use DI with IoC we will face a problem with injecting some services to ViewModels. And here problems come, because how to tell WPF that we want to inject services to constructor before instantiating it and putting as a DataContext for our View?
The Code
I just show you the code and then try to explain it a little bit. Look below.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
using Microsoft.Practices.Unity; | |
using System; | |
using System.Collections.Generic; | |
using System.Reflection; | |
using System.Windows; | |
using System.Windows.Markup; | |
namespace PP.Blog.WpfViewModelsResolver | |
{ | |
class ViewModelsResolver | |
{ | |
public void ConfigureViewModels(IUnityContainer container) | |
{ | |
Assembly currentAssembly = Assembly.GetCallingAssembly(); | |
IEnumerable<Type> allAssemblyTypes = currentAssembly.GetTypes(); | |
foreach (var type in allAssemblyTypes) | |
{ | |
if (this.ViewModelNamesOnly(type) != null) | |
{ | |
Type viewType = this.GetMatchingViewType(type); | |
if (viewType != null) | |
{ | |
this.AddDataTemplate(viewType, type); | |
} | |
Type interfaceType = this.GetMatchingInterfaceType(type); | |
if (interfaceType != null) | |
{ | |
container.RegisterType(interfaceType, type); | |
} | |
} | |
} | |
} | |
private Type GetMatchingInterfaceType(Type viewModelType) | |
{ | |
var typeName = viewModelType.FullName; | |
var namespacePath = typeName.Substring(0, typeName.Length - viewModelType.Name.Length); // Get namespace path | |
var interfaceTypeName = namespacePath + "I" + viewModelType.Name; | |
return viewModelType.Assembly.GetType(interfaceTypeName, false); | |
} | |
private Type GetMatchingViewType(Type viewModelType) | |
{ | |
var typeName = viewModelType.FullName; | |
var viewTypeName = typeName.Substring(0, typeName.Length - "Model".Length); // Get matching | |
return viewModelType.Assembly.GetType(viewTypeName, false); | |
} | |
private string ViewModelNamesOnly(Type type) | |
{ | |
string typeName = type.FullName; | |
if (typeName.EndsWith("ViewModel") && !type.IsInterface) | |
{ | |
return type.Name; | |
} | |
return null; | |
} | |
private void AddDataTemplate(Type viewType, Type viewModelType) | |
{ | |
DataTemplate dataTemplate = this.CreateTemplate(viewType, viewModelType); | |
if (!Application.Current.Resources.Contains(dataTemplate.DataTemplateKey)) | |
{ | |
Application.Current.Resources.Add(dataTemplate.DataTemplateKey, dataTemplate); | |
} | |
else | |
{ | |
Console.WriteLine("Duplicated resource key: " + dataTemplate.DataTemplateKey.ToString()); | |
} | |
} | |
private DataTemplate CreateTemplate(Type viewType, Type viewModelType) | |
{ | |
const string xamlTemplate = | |
"<DataTemplate DataType=\"{{x:Type xViewModels:{0}}}\"><xViews:{1}></xViews:{1}></DataTemplate>"; | |
var xaml = String.Format(xamlTemplate, viewModelType.Name, viewType.Name); | |
var context = new ParserContext(); | |
context.XamlTypeMapper = new XamlTypeMapper(new string[0]); | |
context.XamlTypeMapper.AddMappingProcessingInstruction("xViewModels", viewModelType.Namespace, viewModelType.Assembly.FullName); | |
context.XamlTypeMapper.AddMappingProcessingInstruction("xViews", viewType.Namespace, viewType.Assembly.FullName); | |
context.XmlnsDictionary.Add("", "http://schemas.microsoft.com/winfx/2006/xaml/presentation"); | |
context.XmlnsDictionary.Add("x", "http://schemas.microsoft.com/winfx/2006/xaml"); | |
context.XmlnsDictionary.Add("xViewModels", "xViewModels"); | |
context.XmlnsDictionary.Add("xViews", "xViews"); | |
return (DataTemplate)XamlReader.Parse(xaml, context); | |
} | |
} | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
public void ConfigureViewModels(IUnityContainer container) | |
{ | |
Assembly currentAssembly = Assembly.GetCallingAssembly(); | |
IEnumerable<Type> allAssemblyTypes = currentAssembly.GetTypes(); | |
foreach (var type in allAssemblyTypes) | |
{ | |
if (this.ViewModelNamesOnly(type) != null) | |
{ | |
Type viewType = this.GetMatchingViewType(type); | |
if (viewType != null) | |
{ | |
this.AddDataTemplate(viewType, type); | |
} | |
Type interfaceType = this.GetMatchingInterfaceType(type); | |
if (interfaceType != null) | |
{ | |
container.RegisterType(interfaceType, type); | |
} | |
} | |
} | |
} |
How it works
As you can see, I'm using Unity DI to resolve ViewModel instance: public void ConfigureViewModels(IUnityContainer container) {...}, but it can be any injector of course, you will just need to slightly change this method to register ViewModels regarding container convention that you are using.Let me just describe it shortly, to clarify what we've got here.
Runtime template maching
The most important method here is not the first one, but the one that is dynamically creating DataTemplate for ViewModel and registers it in our resources. I called it CreateTemplate, yep brilliant, I know :-) Lets look closer on it.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
private DataTemplate CreateTemplate(Type viewType, Type viewModelType) | |
{ | |
const string xamlTemplate = | |
"<DataTemplate DataType=\"{{x:Type xViewModels:{0}}}\"><xViews:{1}></xViews:{1}></DataTemplate>"; | |
var xaml = String.Format(xamlTemplate, viewModelType.Name, viewType.Name); | |
var context = new ParserContext(); | |
context.XamlTypeMapper = new XamlTypeMapper(new string[0]); | |
context.XamlTypeMapper.AddMappingProcessingInstruction("xViewModels", viewModelType.Namespace, viewModelType.Assembly.FullName); | |
context.XamlTypeMapper.AddMappingProcessingInstruction("xViews", viewType.Namespace, viewType.Assembly.FullName); | |
context.XmlnsDictionary.Add("", "http://schemas.microsoft.com/winfx/2006/xaml/presentation"); | |
context.XmlnsDictionary.Add("x", "http://schemas.microsoft.com/winfx/2006/xaml"); | |
context.XmlnsDictionary.Add("xViewModels", "xViewModels"); | |
context.XmlnsDictionary.Add("xViews", "xViews"); | |
return (DataTemplate)XamlReader.Parse(xaml, context); | |
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<ResourceDictionary | |
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" | |
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml" | |
xmlns:v="clr-namespace:PP.Blog.Views" | |
xmlns:vm="clr-namespace:PP.Blog.ViewModels"> | |
<DataTemplate DataType="{x:Type vm:HomeViewModel}"> | |
<v:HomeView /> | |
</DataTemplate> | |
.... | |
</ResourceDictionary> |
The second most important part is adding created DataTemplate instance to resources, this is much simpler, no parsers are required :-)
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
private void AddDataTemplate(Type viewType, Type viewModelType) | |
{ | |
DataTemplate dataTemplate = this.CreateTemplate(viewType, viewModelType); | |
if (!Application.Current.Resources.Contains(dataTemplate.DataTemplateKey)) | |
{ | |
Application.Current.Resources.Add(dataTemplate.DataTemplateKey, dataTemplate); | |
} | |
else | |
{ | |
Console.WriteLine("Duplicated resource key: " + dataTemplate.DataTemplateKey.ToString()); | |
} | |
} |
There is one very important assumption in my code, you will need to follow MVVM naming convention, so xxxViewModel.cs, xxxView.xaml, xxxView.xaml.cs and each ViewModel must have a corresponding interface as a marker for DI / IoC container, IxxxViewModel.cs.
Hope the post helps! Feel free to comment. Thanks.
Comments
Post a Comment