このドキュメントの内容
.NET の汎用ホストを使って Windows フォームアプリケーションを実行する方法をライブラリ化しました。
汎用ホストとは
ASP.NET やサービスアプリケーションを開発している方にとってはおなじみですが、アプリケーションの実行に必要なリソースを管理してくれる実行基盤のことで、あらかじめコンフィグやログ出力の機能が組み込まれています。MAUI も汎用ホストに対応したことからも、.NET アプリケーションの標準的なインフラストラクチャであると言えます。これまで Windows フォームアプリケーションやコンソールアプリケーションを中心に開発してきた方も汎用ホストに慣れておくメリットは十分にあると思います。
公開先
GitHub と Nuget で公開しています。
実装例
エントリポイントで Application.Run メソッドを呼ぶ代わりに、汎用ホスト上でアプリケーションを実行するように書き換えます。
using Microsoft.Extensions.Hosting;
using mxProject.WindowFormHosting;
static void Main(string[] args)
{
#if NET6_0
ApplicationConfiguration.Initialize();
#else
Application.EnableVisualStyles();
Application.SetCompatibleTextRenderingDefault(false);
#endif
// 通常の起動方法
// Application.Run(new Form1());
// Windwosフォームアプリケーションを実行します
// このメソッドの実装を以降で説明します
RunWindowsFormApp(args);
}
以降の実装例に現れる MyAppContext クラスは mxProject.WindowFormHosting.IWindowsFormAppContext インターフェースを実装したクラスです。固有の型のコンテキストを利用する場合は、このインターフェースを実装するコンテキストクラスを定義してください。
メインフォームの型を指定する
/// <summary>
/// メインフォームの型を指定してWindowsフォームアプリケーションを実行します。
/// </summary>
static void RunWindowsFormApp(string[] args)
{
void OnStart(IWindowsFormAppContext context) { }
void OnExit(IWindowsFormAppContext context) { }
IHostBuilder builder = Host.CreateDefaultBuilder(args)
.AddWindowsFormApp<Form1>(onStart: OnStart, onExit: OnExit)
;
builder.Build().Run();
}
メインフォームとコンテキストの型を指定する
/// <summary>
/// メインフォームとコンテキストの型を指定してWindowsフォームアプリケーションを実行します。
/// </summary>
static void RunWindowsFormApp(string[] args)
{
void OnStart(MyAppContext context) { }
void OnExit(MyAppContext context) { }
IHostBuilder builder = Host.CreateDefaultBuilder(args)
.AddWindowsFormApp<MyAppContext, Form1>(onStart: OnStart, onExit: OnExit);
builder.Build().Run();
}
アプリケーション情報を指定する
/// <summary>
/// アプリケーション情報を指定してWindowsフォームアプリケーションを実行します。
/// </summary>
static void RunWindowsFormApp(string[] args)
{
IHostBuilder builder = Host.CreateDefaultBuilder(args)
.AddWindowsFormApp(new MyWindowsFormAppInfo());
builder.Build().Run();
}
internal class MyWindowsFormAppInfo : IWindowsFormAppInfo<MyAppContext>
{
internal MyWindowsFormAppInfo() {}
// 実行時にメインフォームの型を決定できます
public Type StartupObjectType => typeof(Form1)
public void OnStart(MyAppContext context) {}
public void OnExit(MyAppContext context) {}
}
タスクトレイ常駐アプリケーションとして実行する
/// <summary>
/// 通知アイコンの型を指定してタスクトレイ常駐アプリケーションを実行します。
/// </summary>
static void RunWindowsFormApp(string[] args)
{
void OnStart(IWindowsFormAppContext context) { }
void OnExit(IWindowsFormAppContext context) { }
IHostBuilder builder = Host.CreateDefaultBuilder(args)
.AddTaskTrayApp<MyNotifyIconProvider>(onStart: OnStart, onExit: OnExit);
builder.Build().Run();
}
/// <summary>
/// 通知アイコンを生成します。
/// </summary>
internal class MyNotifyIconProvider : INotifyIconProvider
{
public MyNotifyIconProvider(IWindowsFormAppContext context) {
m_Context = context;
}
private readonly IWindowsFormAppContext m_Context;
/// <summary>
/// 通知アイコンを生成します。
/// </summary>
public NotifyIcon CreateNotifyIcon()
{
return new NotifyIcon
{
Icon = new System.Drawing.Icon("sample.ico"),
ContextMenuStrip = CreateContextMenu(),
Text = "サンプルアプリケーション",
Visible = true
};
}
private ContextMenuStrip CreateContextMenu()
{
var menu = new ContextMenuStrip();
menu.Items.Add("フォーム1を表示", null, (sender, e) =>
{
using var form = m_Context.FormProvider.CreateForm<Form1>();
form.ShowDialog();
});
menu.Items.Add("フォーム2を表示", null, (sender, e) =>
{
using var form = m_Context.FormProvider.CreateForm<Form2>();
form.ShowDialog();
});
menu.Items.Add("終了", null, (sender, e) =>
{
Application.Exit();
});
return menu;
}
}
コンテキストの型を指定してタスクトレイ常駐アプリケーションとして実行する
/// <summary>
/// 通知アイコンとコンテキストの型を指定してタスクトレイ常駐アプリケーションを実行します。
/// </summary>
static void RunWindowsFormApp(string[] args)
{
void OnStart(MyAppContext context) { }
void OnExit(MyAppContext context) { }
IHostBuilder builder = Host.CreateDefaultBuilder(args)
.AddTaskTrayApp<MyAppContext, MyNotifyIconProvider>(onStart: OnStart, onExit: OnExit);
builder.Build().Run();
}
/// <summary>
/// 通知アイコンを生成します。
/// </summary>
internal class MyNotifyIconProvider : INotifyIconProvider
{
public MyNotifyIconProvider(MyAppContext context) {
m_Context = context;
}
private readonly MyAppContext m_Context;
// 以降割愛
}