Skip to content

Latest commit

 

History

History
333 lines (245 loc) · 20.5 KB

TechnicalDeepDive.md

File metadata and controls

333 lines (245 loc) · 20.5 KB

Technical Deep Dive

Contents:

Design Goals

PhotoSharingApp has an architecture that reflects best practices for larger app projects. The following are high-level architectural goals:

  • Flexibility: Easy-to-maintain code, allowing components to be easily updated as new capabilities are developed and integrated
  • Extensibility: Allowing parts of the app to be easily added or replaced with alternative implementations
  • Quality: Allowing services and components to be tested

In order to achieve the above goals, these are the more detailed key design principles:

  • The Open/Closed Principle (OCP) - The OCP states that "software entities (classes, modules, functions, and so on) should be open for extension, but closed for modification" (Meyer, Bertrand (1988))
  • Separation of concerns (SoC) - The code is divided into distinct features with as little overlap in functionality as possible
  • Single Responsibility Principle (SRP) - Each code component should be responsible only for a specific feature or functionality.
  • Dependency Inversion Principle (DIP) - This principle states that software modules should not depend on concretions, but on abstractions. In this sample code, interfaces (such as ICameraEngine, IPhotoService) are being used to hide a specific implementation in order to reduce coupling.
  • Don't Repeat Yourself (DRY) - Specific functionality should be implemented in only one place, not duplicated across multiple components. A good example in PhotoSharingApp would be the process of making sure that the user is signed in for actions such as uploading photos or posting comments. Doing this check and offering a way to sign in should not be duplicated across the app although it is being used in different places. Therefore, the class PhotoSharingApp.Universal.Services.AuthEnforcementHandler which implements IAuthEnforcementHandler.cs takes care of that by offering the method Task CheckUserAuthentication();.

More details about the above design principles can be found here.

MVVM

PhotoSharingApp is based on a clear separation between Views, ViewModels and Models. A separation between those aspects has a couple of advantages:

  • Supports design goal of separation of concerns
  • Increases application testability
  • Natural XAML pattern

For more details on the MVVM pattern and an introduction into Views, ViewModels and Models can be found on MSDN.

You will find all Views, ViewModels and Models located in their respective namespaces within PhotoSharingApp.Universal.

In PhotoSharingApp, binding is initialized in the code-behind of the View (such as CategoriesPage.xaml.cs) as follows:

_viewModel = ServiceLocator.Current.GetInstance<CategoriesViewModel>();
DataContext = _viewModel;

Throughout the project, ViewModels inherit from ViewModelBase.cs to get INotifyPropertyChanged support (which is implemented in ObservableObjectBase.cs) and make use of other shared functionality:

Architecture

Models support the INotifyPropertyChanged interface as well by directly inheriting from ObservableObjectBase.cs.

Design ViewModels

In the PhotoSharingApp.Universal.ViewModels.Design namespace you will find ViewModel classes that are designed to show data at design-time in the Visual Studio Designer. Design-time data helps with the design process of certain pages and does not require running the app in order to verify design changes.

An example of design-time data is in the LeaderboardDesignViewModel.cs, which provides test data from the dummy service in the Visual Studio Designer:

Architecture

The XAML page uses DesignInstance to specify the design-time data context as follow:

d:DataContext="{d:DesignInstance design:LeaderboardDesignViewModel, IsDesignTimeCreatable=True}"

Design-time ViewModels in PhotoSharingApp are usually minimalistic to mimic basic UI data for the Visual Studio designer. The following implementation of LeaderboardDesignViewModel uses the dummy service to display example images:

/// <summary>
/// The design-time ViewModel for Leaderboards view.
/// </summary>
class LeaderboardDesignViewModel
{
    public LeaderboardDesignViewModel()
    {
        var service = new PhotoDummyService();
        Leaderboard = service.LeaderboardData;
    }

    public Leaderboard Leaderboard { get; set; }
}

Design-time data is helpful when creating the UI and its controls while previewing real data to get a feeling of how the UI looks like at runtime.

Dependency Injection & Registries

To minimize dependencies between different parts of the application, PhotoSharingApp has been designed to make use of interfaces in order to hide the actual implementation. With dependency injection, classes do not directly instantiate other objects that are needed, instead they offer constructors accepting interfaces that allow passing in any valid implementations. In PhotoSharingApp, the process of creating those instances is delegated to Unity.

Dependencies are registered and configured in UnityBootstrapper.cs, which requires registries to implement IRegistry. With Unity as the dependency container, all dependencies are being resolved automatically.

As part of the app initialization you will find the following in App.xaml.cs:

UnityBootstrapper.Init();
UnityBootstrapper.ConfigureRegistries();

By calling Init(), the UnityBootstrapper class adds available registries to an internal list:

private static void AddRegistries()
{
   Registries.Add(new NavigationBarRegistry(Container));
   Registries.Add(new ServicesRegistry(Container));
   Registries.Add(new ViewModelRegistry(Container));
   Registries.Add(new ViewRegistry());
}

And configures each of these in a second step:

public static void ConfigureRegistries()
{
    AddRegistries();
    Registries.ForEach(r => r.Configure());
}

A registry is a concept of grouping dependency registrations; for example, ServicesRegistry.cs adds implementations of a number of services to the Unity container:

public void Configure()
{
    ...
    Container.RegisterType<INavigationFacade, NavigationFacade>();

    Container.RegisterType<IAuthenticationHandler, AuthenticationHandler>();
    Container.RegisterType<IPhotoService, IntegrationServiceClient>();

    Container.RegisterType<ICameraEngine, CameraEngine>();
    Container.RegisterType<ILicensingFacade, LicensingFacade>();
    ...
}

UI pages resolve their ViewModels using the ServiceLocator as follows:

viewModel = ServiceLocator.Current.GetInstance<CategoriesViewModel>(loadData);
DataContext = viewModel;

Constructor injection is automatically performed with the dependencies registered in Unity:

    public CategoriesViewModel(INavigationFacade navigationFacade, IPhotoService photoService,
        TelemetryClient telemetryClient, IAuthEnforcementHandler authEnforcementHandler)
    {
        _navigationFacade = navigationFacade;
        _photoService = photoService;
        _telemetryClient = telemetryClient;
        _authEnforcementHandler = authEnforcementHandler;
        ...

Navigation

In PhotoSharingApp we follow the facade pattern for the page navigation and in-app purchases.

The Universal Windows Platform allows page navigation without restricting the actual type of navigation parameters. In a more complex app you might have many-to-one relationships in terms of navigation: For example, you can navigate to the photo details page by clicking on a photo in the photo collection or by clicking on a photo in the featured photos area. In both cases, you would call the Frame.Navigate(System.Type sourcePageType, object parameter) method from different places in the code. The fact that the navigation parameter is of type object can make it difficult to keep those navigation requests in sync with the target page. Moreover, in the case that a page's navigation parameters need to be adjusted, it can be difficult to track these changes to ensure type safety.

In PhotoSharingApp, you will find the INavigationFacade which covers navigation to all available pages and accepts only parameters that those specific pages will actually understand.

/// <summary>
/// Navigates to the photo stream view.
/// </summary>
/// <param name="category">The category.</param>
void NavigateToPhotoStream(Category category);

Internally, the NavigationFacade class wraps the given parameter and calls a private Navigate method which does serialization if needed, performs the actual navigation and passes the data in form of a (serialized) StreamViewModelArgs instance to StreamPage.

Navigate(typeof(StreamViewModel), new StreamViewModelArgs(category));

The navigation parameter can then be received as follows in StreamPage.xaml.cs:

var args = SerializationHelper.Deserialize<StreamViewModelArgs>(e.Parameter as string);

By abstracting navigation and routing every navigation request through INavigationFacade you can make sure that you pass the right parameter to the target page. Implementing the interface, INavigationFacade, will also help once input parameters need to be changed for a page as now each usage of a navigation method in INavigationFacade can be traced back to the caller at compile-time.

Navigation Bar

All items in the navigation bar implement the INavigationBarMenuItem interface and are registered in NavigationBarRegistry.cs.

If we look at the example in NavigationBarRegistry, this registry is adding implementations of INavigationBarMenuItem to the Unity container:

public void Configure()
{
    // Top items
    Container.RegisterTypeWithName<INavigationBarMenuItem, CategoriesNavigationBarMenuItem>();
    Container.RegisterTypeWithName<INavigationBarMenuItem, CameraNavigationBarMenuItem>();
    Container.RegisterTypeWithName<INavigationBarMenuItem, WelcomeNavigationBarMenuItem>();
    Container.RegisterTypeWithName<INavigationBarMenuItem, LeaderboardNavigationBarMenuItem>();
    Container.RegisterTypeWithName<INavigationBarMenuItem, ProfileNavigationBarMenuItem>();

    // Bottom items
    Container.RegisterTypeWithName<INavigationBarMenuItem, SettingsNavigationBarMenuItem>();
}

At this point valid implementations of INavigationBarMenuItem have been registered, so that those can be resolved in AppShellViewModel.cs using the ServiceLocator as follows:

NavigationBarMenuItems = ServiceLocator.Current
   .GetAllInstances<INavigationBarMenuItem>()
   .Where( i => i.Position == NavigationBarItemPosition.Top )
   .ToList();

BottomNavigationBarMenuItems = ServiceLocator.Current
   .GetAllInstances<INavigationBarMenuItem>()
   .Where( i => i.Position == NavigationBarItemPosition.Bottom )
   .ToList();

For example, you want to add a new feedback menu item that allows users to navigate to a feedback page, adding this item into the navigation bar can be done with the following steps:

  1. Add a new class (such as "FeedbackNavigationBarMenuItem") in the PhotoSharingApp.Universal.NavigationBar namespace and implement INavigationBarMenuItem.cs.

  2. Register your new class in NavigationBarRegistry.cs:

    Container.RegisterTypeWithName<INavigationBarMenuItem, FeedbackNavigationBarMenuItem>();

Service communication

Communication between the app and service is handled in the Services namespace. IPhotoService is the central part which defines all available service calls. By abstracting service calls behind the IPhotoService interface, you can easily switch between a dummy service which serves static data for exploring purposes and a real hosted service implementation, such as an Azure App Service.

IPhotoService provides all methods that are needed with respect to retrieving and uploading data to the service.

interface IPhotoService
{
    /// <summary>
    /// Uploads the photo.
    /// </summary>
    /// <param name="stream">The memory stream.</param>
    /// <param name="localPath">The local path.</param>
    /// <param name="caption">The caption.</param>
    /// <param name="categoryId">The id of the assocaited category.</param>
    /// <returns>The uploaded photo.</returns>
    Task<Photo> UploadPhoto(Stream stream, string localPath, string caption, string categoryId);

    /// <summary>
    /// Gets the photo details for the given photo id.
    /// </summary>
    /// <param name="photoId">The photo identifier.</param>
    /// <returns>The photo.</returns>
    Task<Photo> GetPhotoDetails(string photoId);
    ...

In PhotoSharingApp, IPhotoService is mainly used by ViewModels to interact with the service. The abstraction behind an interface allows easily replacing this service with an alternative implementation or switching between different implementations at runtime from the DebugPage.xaml. ServiceClient.cs is the implementation that is used to communicate with an Azure App Service, while PhotoDummyService is the implementation that shows static test data:

Debug Page

Contract Models Conversions

In PhotoSharingApp.Portable you will find data contracts that are used by the app to communicate to the service. These classes have the single purpose, to make sure the app and service agree on the same contracts, therefore they are very lightweight.

Besides storing data, data models in an app can offer helper methods, additional properties which do formatting or implement INotifyPropertyChanged to support data binding. Therefore, in this example, the app's data model is different from the data that is transferred between service and app.

To give an example, the CategoryContract.cs class simply contains two properties for the Id and Name:

/// <summary>
/// The category data contract.
/// </summary>
public class CategoryContract
{
    /// <summary>
    /// The category Id.
    /// </summary>
    public string Id { get; set; }

    /// <summary>
    /// The category name.
    /// </summary>
    public string Name { get; set; }
}

The Category.cs class provides data binding functionality as well as an additional HasErrors property for input validation:

/// <summary>
/// Represents a category.
/// </summary>
public class Category : ObservableObjectBase
{
    private string _id;
    private string _name;

    /// <summary>
    /// Gets the data model validation result.
    /// Returns true, if it contains one or more errors.
    /// Otherwise, false.
    /// </summary>
    public bool HasErrors
    {
        get
        {
            return string.IsNullOrWhiteSpace(_name)
                   || _name.Length < 2
                   || _name.StartsWith("my", StringComparison.CurrentCultureIgnoreCase);
        }
    }

    /// <summary>
    /// Gets or sets the Id.
    /// </summary>
    public string Id
    {
        get { return _id; }
        set
        {
            if (value != _id)
            {
                _id = value;
                NotifyPropertyChanged();
            }
        }
    }

    /// <summary>
    /// Gets or sets the name.
    /// </summary>
    public string Name
    {
        get { return _name; }
        set
        {
            if (value != _name)
            {
                _name = value;
                NotifyPropertyChanged();

                // The name has changed, so we need to update the
                // object's validation status.
                NotifyPropertyChanged(() => HasErrors);
            }
        }
    }
}

Conversion methods can be found in the PhotoSharingApp.Universal.ContractModelConverterExtensions namespace. A conversion from data model to data contract is usually performed when data is sent to the service, a conversion from data contract to data model when data is received from the service.

In-app purchase

In-app purchases in PhotoSharingApp have two abstraction layers:

  • CurrentAppProxy.cs acts as a proxy to allow switching between Windows.ApplicationModel.Store.CurrentApp and Windows.ApplicationModel.Store.CurrentAppSimulator.
  • ILicensingFacade.cs acts as a facade on top of CurrentAppProxy. The motivation for this design is that purchasing gold includes multiple steps: Check if user is signed in, perform purchase using Windows.ApplicationModel.Store.CurrentApp and notify PhotoSharingApp service to increase the user's gold count. Additionally, you may want to track any purchase requests by telemetry.

As you can see there are multiple processes involved to perform a gold purchase which may be triggered by different parts in the app. With that in mind in-app purchases are being routed through ILicensingFacade similar to INavigationFacade.