HappyCoding.Hosting 1.1.0

dotnet add package HappyCoding.Hosting --version 1.1.0
NuGet\Install-Package HappyCoding.Hosting -Version 1.1.0
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="HappyCoding.Hosting" Version="1.1.0" />
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add HappyCoding.Hosting --version 1.1.0
#r "nuget: HappyCoding.Hosting, 1.1.0"
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install HappyCoding.Hosting as a Cake Addin
#addin nuget:?package=HappyCoding.Hosting&version=1.1.0

// Install HappyCoding.Hosting as a Cake Tool
#tool nuget:?package=HappyCoding.Hosting&version=1.1.0

Generic Host for WinUI applications

The .Net Hosting extensions look like they are only for ASP .Net but they are not. The documentation now introduces them as part of the .Net fundamentals and they can very well be used in any scenario including console and desktop applications.

A host is an object that encapsulates an app's resources and lifetime functionality, such as:

  • Dependency injection (DI)
  • Configuration (in-memory, json, secrets, environment variables, command line)
  • Logging
  • Diagnostics
  • Different types of File Providers
  • App startup/shutdown and IHostedService implementations

➡ What if we could leverage all of that in a WinUI application?

You really don't need a dozen of Nugets from many maintained or no longer maintained sources. All of that can be done with just the .Net framework. Let's see how.

Before we start

The starting point for the application will be the basic WinUI application with a custom Main entry point. Not that it is absolutely required, but it will give us full control of the setup of the Generic Host environment and will make the WinUI subsystem just another service for which we manage the lifecycle.

See https://github.com/abdes/winui-override-main for the basics of how to override the default entry point.

A hosted service for WinUI

A typical WinUI application runs in a UI thread, assumed to be the main thread, and for which the dispatched is initialized in the Main entry point.

What we want is to have that thread be a background service, hosted by the generic host, along with the other application services. This will give us the advantage of managing the UI just as another service, and of course getting all the goodies of the Generic Host listed above.

The implementation, extensively documented in the source code, adds a UserInterfaceHostedService which runs the 'UI Thread' as a background service. We can decide, based on the options in the HostingContext if we want the lifetime of that 'UI Thread' to be linked to the application lifetime or not. In other words, we can decide of termination of the 'UI Thread' results in termination of the application or not, and vice versa.

An extension method for the builder

To simplify the setup of the User Interface hosted service, an extension method to the host builder is provided.

This greatly simplifies the host building as it is now just a matter of calling that extension method.

    private static void Main(string[] args)
    {
        // Use a default application host builder, which comes with logging,
        // configuration providers for environment variables, command line,
        // appsettings.json and secrets.
        var builder = Host.CreateApplicationBuilder(args);

        // You can further customize and enhance the builder with additional
        // configuration sources, logging providers, etc.

        // Setup and provision the hosting context for the User Interface
        // service.
        ((IHostApplicationBuilder)builder).Properties.Add(
            key: HostingExtensions.HostingContextKey,
            value: new HostingContext() { IsLifetimeLinked = true });

        // Add the WinUI User Interface hosted service as early as possible to
        // allow the UI to start showing up while you continue setting up other
        // services not required for the UI.
        var host = builder.ConfigureWinUI<App, MainWindow>().Build();

        // Finally start the host. This will block until the application
        // lifetime is terminated through CTRL+C, closing the UI windows or
        // programmatically.
        host.Run();
    }

Why the UI hosted service does not create the MainWindow?

Simply because not all applications are single window applications and it is not obvious in a multi-window application which window will terminate the UI when closed. Therefore, it is much more preferable to continue having the Application responsible for managing its Windows.

Can this be adapted to WPF, etc. ?

Yes. Simply implement the classes inside WinUI namespace for any other UI framework.

Product Compatible and additional computed target framework versions.
.NET net8.0-windows10.0.22621 is compatible. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

NuGet packages

This package is not used by any NuGet packages.

GitHub repositories

This package is not used by any popular GitHub repositories.

Version Downloads Last updated
1.1.0 219 12/3/2023
1.0.1 100 11/27/2023
1.0.0 71 11/27/2023

More unit testing and some refactoring to further improve code quality.