View-models make use of Services to execute the operations requested by the user, such as create, update or retrieve a list of customers or products. View-models also make use of Services to log the user activity, show dialogs or display a text in the status-bar by sending a message to the shell view.
Services contains the core functionality of the application. We distinguish two kinds of services:
- Application Services – implement core functionality needed by the infrastructure of the application. This functionality in independent of the business of the application and can be reused in other solutions. Examples of application services are Navigation Service or Message Service that can be reused for any other application.
- Domain Services (or Business Services) – implements the functionality specific for the business of the application. Examples of domain services are Customer Services or Product Services that are specific for a product management application.
The following diagram shows the two group of services used in this application:
Here is a brief description of the Application Services used in this application:
Service | Description |
---|---|
Navigation Service |
Expose the functionality to navigate back and forward to a different view. It also offers the possibility to open a view in a new window. |
Message Service |
Enables communication between different components of the application without having to know anything about each other. The communication between components are based on a publishers-subscribers pattern. |
Log Service |
Offers the methods to write logs to a local repository to keep track of the user activity for debugging or auditing purposes. |
Login Service |
Implements the authentication and authorization mechanism to access the application. |
Dialog Service |
Offers methods to display a dialog message to show information or ask for confirmation. |
IFilePickerService |
Allows the application access to the file system with a dialog to select a file |
ISettingsService |
Stores and provides configuration and setting values needed by the application |
Context Service |
Exposes properties and methods related to the current execution context. This service is used internally to manage the execution in a multi-window environment where each window is executed in a different thread. |
The domain services in this application offer CRUD operations (Create, Read, Update, Delete) over the business entities. We have a specific service for Customers, Orders, OrderItems and Products.
To see the common methods used in these services, let’s examine the Customer service:
GetCustomer(id)
– get a single customer by its id.GetCustomers(request)
– get a collection of customers using the request parameter.GetCustomers(skip, take, request)
– same as GetCustomers(request) but returns only ‘take’ number of items starting from the ‘skip’ parameter.GetCustomersCount(request)
– return the number of Customers using the request parameter.UpdateCustomer(customer)
– update or create a new Customer with the values contained in the customer parameter.DeleteCustomer(customer)
– delete the Customer specified by the customer parameter.
There is also a LookupTables Service used to retrieve information for common Tables such as Categories or CountryCodes. This service is used, for example, to get the name of a country by its code, or the tax rate for a specific tax type.
Since we are using MVVM, and the ViewModel-First approach, we will use a Navigation Service abstraction to facilitate the ViewModel-based navigation.
This kind of navigation, oposed to View-based navigation, is the navigation that uses a ViewModel as the subject that determines the navigation. The View isn't specified explicitly. Instead, there is a mechanism to associate each ViewModel with its corresponding View. This is where our Navigation Service comes in. It will perform the navigation itself, but also the will glue the ViewModel with its View.
This contract is in charge of the Navigation of the app. The service is agnostic of the platform that is going to use it and that's why is located in the Inventory.ViewModels project.
Let's take a look to the interface:
public interface INavigationService
{
bool IsMainView { get; }
bool CanGoBack { get; }
void Initialize(object frame);
bool Navigate<TViewModel>(object parameter = null);
bool Navigate(Type viewModelType, object parameter = null);
Task<int> CreateNewViewAsync<TViewModel>(object parameter = null);
Task<int> CreateNewViewAsync(Type viewModelType, object parameter = null);
void GoBack();
Task CloseViewAsync();
}
Let's review each one of these methods:
Method | Usage |
---|---|
Initialize | This method is needed in order to initialize the Navigation Service |
Navigate | There are 2 overloads of this Method. Both the Type of the destination ViewModel is needed plus some arguments |
CreateNewViewAsync | The inventory app example, allow us to create a View in a new Window. There are also 2 overloads of this method, and both need the target ViewModel to load |
GoBack | Navigate back in the stack |
CloseViewAsync | Close the actual Window |
The Navigation Service sits between the View and the ViewModel. As the Navigate method takes the type of the ViewModel, our implementation will have to find the View that is associated with it. Let's review now how this interface is implemented.
The interface is implemented in the Inventory.App project by the NavigationService
class.
The Navigation Service will need a mechanism to associate Views to ViewModels. In our implementation, this is done using an internal dictionary.
Whenever the Navigate
method is called, this dictionary will be queried for the Type of the View that corresponds to the ViewModel.
Let's take a look to the implementation of the Navigate
method:
public bool Navigate(Type viewModelType, object parameter = null)
{
if (Frame == null)
{
throw new InvalidOperationException("Navigation frame not initialized.");
}
return Frame.Navigate(GetView(viewModelType), parameter);
}
In the first place, we're checking whether the Frame
is null. The Frame property is the object that will perform the navigation at the UI side. It's usually set at the very beginning of the execution. In our application, inside the ShellView.InitializeNavigation()
method, where the service is resolved.
In the second place, where are telling the Frame
to navigate to the View associated to viewModelType
. The lookup is done in GetView
:
static public Type GetView(Type viewModel)
{
if (_viewModelMap.TryGetValue(viewModel, out Type view))
{
return view;
}
throw new InvalidOperationException($"View not registered for ViewModel '{viewModel.FullName}'");
}
In order to know the association between a ViewModel and its View, some kind of registration is needed. For this effect, we will expose a method in our implementation:
static public void Register<TViewModel, TView>() where TView : Page
{
if (!_viewModelMap.TryAdd(typeof(TViewModel), typeof(TView)))
{
throw new InvalidOperationException($"ViewModel already registered '{typeof(TViewModel).FullName}'");
}
}
It just adds the type of the ViewModel and the type of the View in a common dictionary.
The registration is usually done at the beginning of the execution, so all entries are available as soon as possible. In our application, you can locate the registration in the Startup
class, inside the ConfigureNavigation
method.
private static void ConfigureNavigation()
{
NavigationService.Register<ShellViewModel, ShellView>();
NavigationService.Register<MainShellViewModel, MainShellView>();
NavigationService.Register<DashboardViewModel, DashboardView>();
...
}
In our implementation, there are a few additional properties and methods to enable more advanced navigation scenarios, like to pop a new Window or close it programmatically.
In our application, there are scenarios where we will feature multitasking by opening new Windows. It's provided by the CreateNewViewAsync
methods.
Also, we made the functionality symmetrical by exposing the CloseViewAsync
method, that is very easy to implement thanks to the ApplicationViewSwitcher
static class.
public async Task CloseViewAsync()
{
int currentId = ApplicationView.GetForCurrentView().Id;
await ApplicationViewSwitcher.SwitchAsync(MainViewId, currentId, ApplicationViewSwitchingOptions.ConsolidateViews);
}
Another service that it's important to know is the one represented by the interface IMessageService
. The mission of this service is to send messages between decoupled components (Views, ViewModels, etc), and once the subscriber receive the message, react to it executing some action. This pattern is known as the Event Aggregator Pattern:
The IMessageService
exposes the following methods:
public interface IMessageService
{
void Subscribe<TSender>(object target, Action<TSender, string, object> action) where TSender : class;
void Subscribe<TSender, TArgs>(object target, Action<TSender, string, TArgs> action) where TSender : class;
void Unsubscribe(object target);
void Unsubscribe<TSender>(object target) where TSender : class;
void Unsubscribe<TSender, TArgs>(object target) where TSender : class;
void Send<TSender, TArgs>(TSender sender, string message, TArgs args) where TSender : class;
}
Methods | Description |
---|---|
Subscribe | This is used to subscribe your class to a specific event. When we subscribe to an event, we need to indicate the sender (TSender ), the target object, and the action to execute when the message is received |
Unsubscribe | Unsubscribe an event already registered. It's important to unsubscribe events in order to avoid memory leaks |
Send | Communicate to subscriber that a specific event has occurred. When we send a message we are passing the sender (TSender ), the message that identifies the event occurred (message ) and additional arguments when necessary |
The best thing to understand how it works is to check how it's being used in the Inventory App. For example, lets see how the CustomersViewModel
loads the detail of a Customer, when it is selected from the List of Customers. In this process the following actors are involved:
CustomerListViewModel
: When an element of the list is selected, this ViewModel will use theIMessageService
to send to the Customer selected to the possible subscribers.
private TModel _selectedItem = default(TModel);
public TModel SelectedItem
{
get => _selectedItem;
set
{
if (Set(ref _selectedItem, value))
{
if (!IsMultipleSelection)
{
MessageService.Send(this, "ItemSelected", _selectedItem);
}
}
}
}
CustomersViewModel
: This ViewModel will Subcribe to the Customers list ItemSelected event. It will also unsubscribe from this event when necessary.
public void Subscribe()
{
MessageService.Subscribe<CustomerListViewModel>(this, OnMessage);
CustomerList.Subscribe();
CustomerDetails.Subscribe();
CustomerOrders.Subscribe();
}
private async void OnMessage(CustomerListViewModel viewModel, string message, object args)
{
if (viewModel == CustomerList && message == "ItemSelected")
{
await ContextService.RunAsync(() =>
{
OnItemSelected();
});
}
}
private async void OnItemSelected()
{
if (CustomerDetails.IsEditMode)
{
StatusReady();
CustomerDetails.CancelEdit();
}
CustomerOrders.IsMultipleSelection = false;
var selected = CustomerList.SelectedItem;
if (!CustomerList.IsMultipleSelection)
{
if (selected != null && !selected.IsEmpty)
{
await PopulateDetails(selected);
await PopulateOrders(selected);
}
}
CustomerDetails.Item = selected;
}
We are executing the OnItemSelected
method when we received from the CustomerListViewModel
a message ItemSelected
.
Finally, as we previously mentioned, it's important to control the subscriptions and unsubcriptions of the events to prevent memory leaks. We will do this overriding the events OnNavigatedTo
and OnNavigatingFrom
methods of our Views:
protected override async void OnNavigatedTo(NavigationEventArgs e)
{
ViewModel.Subscribe();
await ViewModel.LoadAsync(e.Parameter as CustomerListArgs);
}
protected override void OnNavigatingFrom(NavigatingCancelEventArgs e)
{
ViewModel.Unload();
ViewModel.Unsubscribe();
}
The purpouse of this section is to show how an Event Aggregator Pattern can be implemented. The key to unsderstand the whole process are the Subscriber
and Subscriptions
internal classes.
This class is just a Diccionary of actions associated with a Type
. We will use it to store the action to execute when a Subscriber receives a message.
class Subscriptions
{
private Dictionary<Type, Delegate> _subscriptions = null;
public Subscriptions()
{
_subscriptions = new Dictionary<Type, Delegate>();
}
public bool IsEmpty => _subscriptions.Count == 0;
public void AddSubscription<TSender, TArgs>(Action<TSender, string, TArgs> action)
{
_subscriptions.Add(typeof(TArgs), action);
}
public void RemoveSubscription<TArgs>()
{
_subscriptions.Remove(typeof(TArgs));
}
public void TryInvoke<TArgs>(object sender, string message, TArgs args)
{
var argsType = typeof(TArgs);
foreach (var keyValue in _subscriptions.Where(r => r.Key.IsAssignableFrom(argsType)))
{
var action = keyValue.Value;
action?.DynamicInvoke(sender, message, args);
}
}
}
The Subscriber class is the responsable of:
- Container of the subscriptions defined for a specific
Type
. - Create the
WeakReference
between the Publisher and the Subscriber. - Unregister subscriptions.
- Invoke the action to execute when the subscriber receives a message.
class Subscriber
{
private WeakReference _reference = null;
private Dictionary<Type, Subscriptions> _subscriptions;
public Subscriber(object target)
{
_reference = new WeakReference(target);
_subscriptions = new Dictionary<Type, Subscriptions>();
}
public object Target => _reference.Target;
public bool IsEmpty => _subscriptions.Count == 0;
public void AddSubscription<TSender, TArgs>(Action<TSender, string, TArgs> action)
{
if (!_subscriptions.TryGetValue(typeof(TSender), out Subscriptions subscriptions))
{
subscriptions = new Subscriptions();
_subscriptions.Add(typeof(TSender), subscriptions);
}
subscriptions.AddSubscription(action);
}
public void RemoveSubscription<TSender>()
{
_subscriptions.Remove(typeof(TSender));
}
public void RemoveSubscription<TSender, TArgs>()
{
if (_subscriptions.TryGetValue(typeof(TSender), out Subscriptions subscriptions))
{
subscriptions.RemoveSubscription<TArgs>();
if (subscriptions.IsEmpty)
{
_subscriptions.Remove(typeof(TSender));
}
}
}
public void TryInvoke<TArgs>(object sender, string message, TArgs args)
{
var target = _reference.Target;
if (_reference.IsAlive)
{
var senderType = sender.GetType();
foreach (var keyValue in _subscriptions.Where(r => r.Key.IsAssignableFrom(senderType)))
{
var subscriptions = keyValue.Value;
subscriptions.TryInvoke(sender, message, args);
}
}
}
}
Once we understand how the internal classes Subscriptions
and Subscriber
work, the implementation of the IMessageService
becomes much simplier. These are the implementation of main three methods of IMessageService
:
- Subscribe: We are just simply creating a new
Subscriber
in case it wasn't already defined, and associate anAction
to be executed.
public void Subscribe<TSender, TArgs>(object target, Action<TSender, string, TArgs> action) where TSender : class
{
if (target == null)
throw new ArgumentNullException(nameof(target));
if (action == null)
throw new ArgumentNullException(nameof(action));
lock (_sync)
{
var subscriber = _subscribers.Where(r => r.Target == target).FirstOrDefault();
if (subscriber == null)
{
subscriber = new Subscriber(target);
_subscribers.Add(subscriber);
}
subscriber.AddSubscription<TSender, TArgs>(action);
}
}
- Unsubscribe: Just remove a subscription for the collection.
public void Unsubscribe(object target)
{
if (target == null)
throw new ArgumentNullException(nameof(target));
lock (_sync)
{
var subscriber = _subscribers.Where(r => r.Target == target).FirstOrDefault();
if (subscriber != null)
{
_subscribers.Remove(subscriber);
}
}
}
- Send: Execute the actions associated to Subscribers when a specific event occurs.
public void Send<TSender, TArgs>(TSender sender, string message, TArgs args) where TSender : class
{
if (sender == null)
throw new ArgumentNullException(nameof(sender));
foreach (var subscriber in GetSubscribersSnapshot())
{
// Avoid sending message to self
if (subscriber.Target != sender)
{
subscriber.TryInvoke(sender, message, args);
}
}
}
Allows for top exception and optionally all the InnerExceptions Logging
- WriteAsync: Logs the top exception when MustExploreDeepExceptions property in the class VirtualCollection is false (by default) or the top Exception and all InnerExceptions when true. MustExploreDeepExceptions is optionally set in the constructor:
VirtualCollection(ILogService logService, int rangeSize = 16, bool mustExploreDeepExceptions=false)
// log top Exception only
WriteAsync(LogType type, string source, string action, string message, string description)
// log top Exception and all Inner ones
WriteAsync(LogType type, string source, string action, Exception ex)