Tuesday, 15 November 2022

Using HostBuilder, ServiceProvider and Dependency Injection with Windows Forms on .NET Core 3

 First of all, after creating a .NET Core Windows Forms application, we need to add the NuGet package Microsoft.Extensions.Hosting to the project. It will allow us to use HostBuilder and, moreover, it automatically imports a bunch of other required packages. Now let’s open the Program.cs file and add the following code:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
private static void Main()
{
    // ...
    var host = Host.CreateDefaultBuilder()
             .ConfigureAppConfiguration((context, builder) =>
             {
                 // Add other configuration files...
                 builder.AddJsonFile("appsettings.local.json", optional: true);
             })
             .ConfigureServices((context, services) =>
             {
                 ConfigureServices(context.Configuration, services);
             })
             .ConfigureLogging(logging =>
             {
                 // Add other loggers...
             })
             .Build();
 
    var services = host.Services;
    var mainForm = services.GetRequiredService<MainForm>();
    Application.Run(mainForm);
}
 
private static void ConfigureServices(IConfiguration configuration,
    IServiceCollection services)
{
    // ...
    services.AddSingleton<MainForm>();
}

HostBuilder configuration at lines 4-18 follows the same structure we have already presented in the previous article, so refer to it for more information. After that, at lines 21 we retrive the MainForm we added in the Service Collection (line 29) and finally we start the application using the Application.Run method (line 22).

We can now run the application: everything will work as expected. And now we can leverage all the features that .NET Core 3 provides. Let’s add also a file named appsettings.json to the root folder of the project. Set its Build Action property to Content and Copy to Output Directory to Copy if newer:

1
2
3
4
5
6
7
{
  "AppSettings": {
    "StringSetting": "Value",
    "IntegerSetting": 42,
    "BooleanSetting": true
  }
}

This file is automatically loaded and made available to the application by the CreateDefaultBuilder method we saw before. Then, we create an AppSettings.cs file to hold configuration settings. This file will map the settings that we write in appsettings.json:

1
2
3
4
5
6
7
8
public class AppSettings
{
    public string StringSetting { get; set; }
  
    public int IntegerSetting { get; set; }
  
    public bool BooleanSetting { get; set; }
}

Moreover, create also a sample service with its interface:

1
2
3
4
5
6
7
8
9
public interface ISampleService
{
    string GetCurrentDate();
}
  
public class SampleService : ISampleService
{
    public string GetCurrentDate() => DateTime.Now.ToLongDateString();
}

Now we must register these services in the IoC Container, as usual:

1
2
3
4
5
6
7
8
9
10
private void ConfigureServices(IConfiguration configuration,
    IServiceCollection services)
{
    services.Configure<AppSettings>
            (configuration.GetSection(nameof(AppSettings)));
 
    services.AddScoped<ISampleService, SampleService>();
 
    //...
}

Remember that the MainForm itself is in the IoC Container. So, when we get it from the Service Provider, it will automatically be injected with all the required services. We just need to modify its constructor:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
private readonly IServiceProvider serviceProvider;
private readonly ISampleService sampleService;
private readonly AppSettings settings;
 
public MainForm(IServiceProvider serviceProvider,
                ISampleService sampleService,
                IOptions<AppSettings> settings)
{
    InitializeComponent();
 
    this.serviceProvider = serviceProvider;
    this.sampleService = sampleService;
    this.settings = settings.Value;
}

Running this code, we’ll obtain a result like the following:

The .NET Core 3.0 Windows Forms application with dependecies injected

The .NET Core 3.0 Windows Forms application with dependecies injected

Finally, let’s try to add a second Form to the application and open it using a button from the Main Form. But, before doing that, remember that, at the time of writing, the Windows Forms Designer for .NET Core is available only in Visual Studio 16.5 Preview. If you haven’t it yet, you can refer to the complete sample you’ll find at the end of the article.

So, register the new form in the Service Collection (inside Program.cs file):

1
2
3
4
5
6
private static void ConfigureServices(IConfiguration configuration,
    IServiceCollection services)
{
    // ...
    services.AddTransient<SecondForm>();
}

At line 5 we register the Form as transient dependency, meaning that, everytime we try to get a reference to it, we’ll get a new instance. Of course, we can use AddScoped or AddSingleton as well. Keep in mind that, in the context of a desktop application, Scoped and Singleton get always the same instance, because in this case we have a scope as long as our application runs.

Then, in MainForm.cs add the code to open the new Form when clicking the button:

1
2
3
4
5
private void OpenSecondFormButton_Click(object sender, EventArgs e)
{
    var form = serviceProvider.GetRequiredService<SecondForm>();
    form.ShowDialog(this);
}

We use the ServiceProvider we passed to the MainForm constructor to get a reference to SecondForm and we call its ShowDialog method (lines 3-4). But SecondForm is itself in the IoC Container, so in turn it can receive the dependencies it needs in the constructor:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public partial class SecondForm : Form
{
    private readonly ISampleService sampleService;
    private readonly AppSettings settings;
 
    public SecondForm(ISampleService sampleService,
                      IOptions<AppSettings> settings)
    {
        InitializeComponent();
 
        this.sampleService = sampleService;
        this.settings = settings.Value;
    }
}
Opening a second Form with the Service Provider

Opening a second Form with the Service Provider


No comments:

Post a Comment