CommunityToolkit.Mvvm × .NET MAUI 実践ガイド:ObservableProperty・RelayCommand・Messengerでボイラープレート削減

CommunityToolkit.Mvvm 8.x のソースジェネレーターで.NET MAUI 10アプリのMVVMコードを最大70%削減。ObservableProperty・RelayCommand・Messenger・ObservableValidator・DI統合・落とし穴対策まで、現場で使える実装パターンを完全解説。

CommunityToolkit.Mvvm MAUI 10実践ガイド2026

正直に言うと、.NET MAUIでMVVMを書き始めた頃、私はINotifyPropertyChangedの実装に毎回うんざりしていました。プロパティ1個追加するたびにバッキングフィールドを宣言して、getter/setterを書いて、PropertyChangedを発火させて……。同じようなコードを20回くらい書いた頃、ふと気づいたんです。「これ、人間がやる仕事じゃないな」と。

幸い、CommunityToolkit.Mvvm 8.xのソースジェネレーターを使えば、こうした定型コードはほぼ消滅します。実際、私のチームでは導入後にViewModelの行数が平均で半分以下になりました。

本記事では、2026年5月時点の最新版(CommunityToolkit.Mvvm 8.4.x)と.NET MAUI 10を組み合わせた実装パターンを、現場で使えるコード例とともに体系的に解説します。[ObservableProperty][RelayCommand]といった属性ベースのアプローチに加え、WeakReferenceMessengerObservableValidator、依存性注入(DI)統合まで、プロダクション品質のMVVM設計に必要な要素を一通り押さえます。

なぜ.NET MAUIでCommunityToolkit.Mvvmが事実上の標準なのか

Microsoft自身がメンテナンスするCommunityToolkit.Mvvmは、Xamarin.Forms時代のMvvmLightPrismに代わって、新規.NET MAUIプロジェクトのデフォルト選択肢になっています。理由は割とシンプルです。

  • ソースジェネレーターによりリフレクションを排除し、AOT/Native AOTビルドと完全に互換
  • 属性ベースのAPIで、フィールド宣言1行からプロパティ・コマンドが自動生成
  • Microsoft公式リポジトリで保守され、.NET 10のC# 13言語機能と即座に追従
  • PrismやReactiveUIに比べて学習コストが低く、依存関係も最小
  • NuGetダウンロード数は累計5,000万を超え、エコシステムが成熟

とくに.NET MAUI 10で正式サポートされたNative AOT(iOS/MacCatalyst)では、リフレクションベースのMVVMライブラリは動作不能になります。ソースジェネレーター方式は、もはや「便利」ではなく「必須」と言っていいでしょう。

セットアップ:MAUIプロジェクトへの導入

まずは新規または既存の.NET MAUIプロジェクトに、以下のNuGetパッケージを追加します。

dotnet add package CommunityToolkit.Mvvm --version 8.4.0
dotnet add package Microsoft.Extensions.DependencyInjection

次にMauiProgram.csでDIコンテナにViewModelを登録します。.NET MAUIにはMicrosoft.Extensions.DependencyInjectionが組み込まれているので、追加の設定は不要です。

public static MauiApp CreateMauiApp()
{
    var builder = MauiApp.CreateBuilder();
    builder
        .UseMauiApp<App>()
        .ConfigureFonts(fonts =>
        {
            fonts.AddFont("OpenSans-Regular.ttf", "OpenSansRegular");
        });

    // ViewModelとサービスの登録
    builder.Services.AddSingleton<ITodoService, TodoService>();
    builder.Services.AddTransient<TodoListViewModel>();
    builder.Services.AddTransient<TodoListPage>();

    return builder.Build();
}

ObservablePropertyでプロパティ実装を1行に

従来のINotifyPropertyChanged実装は、1プロパティあたり最低5~8行のコードが必要でした。[ObservableProperty]属性を付けたフィールドからは、コンパイル時にプロパティ・PropertyChangedイベント発火・partialメソッドフックが自動生成されます。

従来コードとの比較

論より証拠、見比べてみましょう。

Before(手書き)

public class TodoViewModel : INotifyPropertyChanged
{
    private string _title = string.Empty;
    public string Title
    {
        get => _title;
        set
        {
            if (_title == value) return;
            _title = value;
            PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(nameof(Title)));
        }
    }

    public event PropertyChangedEventHandler? PropertyChanged;
}

After(CommunityToolkit.Mvvm)

public partial class TodoViewModel : ObservableObject
{
    [ObservableProperty]
    private string _title = string.Empty;
}

ジェネレーターは_titleからTitleプロパティを生成します(先頭アンダースコアは自動で除去されます)。C# 13ではフィールド宣言を省略してプロパティに直接[ObservableProperty]を付与する記法もサポートされていますが、明示的フィールドのほうがチームメンバーに意図が伝わりやすいので、本記事では従来記法で統一します。

変更通知のフック:partialメソッド

プロパティ変更前後にカスタムロジックを差し込みたいときは、生成されるOnXxxChanging/OnXxxChanged partialメソッドを実装します。

public partial class SearchViewModel : ObservableObject
{
    [ObservableProperty]
    private string _query = string.Empty;

    partial void OnQueryChanged(string value)
    {
        // クエリ変更時にデバウンス検索をトリガー
        _searchDebouncer.Trigger(() => ExecuteSearchAsync(value));
    }

    partial void OnQueryChanging(string oldValue, string newValue)
    {
        // 変更前のバリデーション
        if (newValue?.Length > 100)
            throw new ArgumentException("クエリは100文字以内");
    }
}

依存プロパティの自動通知:NotifyPropertyChangedFor

計算プロパティが別のプロパティに依存する場合、[NotifyPropertyChangedFor]で連動通知を宣言できます。地味ですが、めちゃくちゃ便利な機能です。

public partial class UserProfileViewModel : ObservableObject
{
    [ObservableProperty]
    [NotifyPropertyChangedFor(nameof(FullName))]
    private string _firstName = string.Empty;

    [ObservableProperty]
    [NotifyPropertyChangedFor(nameof(FullName))]
    private string _lastName = string.Empty;

    public string FullName => $"{FirstName} {LastName}".Trim();
}

RelayCommandでICommand実装を撲滅

ButtonのCommandバインディングに必要なICommand実装も、[RelayCommand]属性で完全自動化できます。同期・非同期・キャンセル対応のすべてが1属性で生成される、というのが個人的にはいちばん感動した部分でした。

public partial class TodoListViewModel : ObservableObject
{
    private readonly ITodoService _todoService;

    public TodoListViewModel(ITodoService todoService)
    {
        _todoService = todoService;
    }

    public ObservableCollection<TodoItem> Items { get; } = new();

    [ObservableProperty]
    [NotifyCanExecuteChangedFor(nameof(LoadTodosCommand))]
    private bool _isBusy;

    [RelayCommand(CanExecute = nameof(CanLoadTodos))]
    private async Task LoadTodosAsync(CancellationToken ct)
    {
        IsBusy = true;
        try
        {
            var todos = await _todoService.GetAllAsync(ct);
            Items.Clear();
            foreach (var t in todos) Items.Add(t);
        }
        finally
        {
            IsBusy = false;
        }
    }

    private bool CanLoadTodos() => !IsBusy;

    [RelayCommand]
    private async Task DeleteAsync(TodoItem item)
    {
        await _todoService.DeleteAsync(item.Id);
        Items.Remove(item);
    }
}

XAML側は通常通りCommandバインディングで利用します。コマンド名はLoadTodosAsyncLoadTodosCommandのように、Asyncサフィックスが除去され、Commandサフィックスが付与される、という命名規則です。

<Button Text="読み込み"
        Command="{Binding LoadTodosCommand}"
        IsEnabled="{Binding LoadTodosCommand.CanExecute}" />

<CollectionView ItemsSource="{Binding Items}">
    <CollectionView.ItemTemplate>
        <DataTemplate x:DataType="models:TodoItem">
            <SwipeView>
                <SwipeView.RightItems>
                    <SwipeItem Text="削除"
                               Command="{Binding Source={RelativeSource AncestorType={x:Type vm:TodoListViewModel}}, Path=DeleteCommand}"
                               CommandParameter="{Binding .}" />
                </SwipeView.RightItems>
            </SwipeView>
        </DataTemplate>
    </CollectionView.ItemTemplate>
</CollectionView>

CanExecuteと自動再評価

[ObservableProperty][NotifyCanExecuteChangedFor]を組み合わせると、依存プロパティが変わったタイミングで自動的にCanExecuteが再評価されます。上のコードではIsBusyが変わるたびにLoadTodosCommand.CanExecuteChangedが発火して、ButtonのIsEnabledが勝手に更新されます。手動でNotifyCanExecuteChanged()を呼び忘れて「あれ、ボタン無効にならないぞ?」と悩むこともありません(昔の自分に教えてあげたい)。

WeakReferenceMessengerでViewModel間通信

ページ遷移時のデータ受け渡しや、複数のViewModelに共通する状態変化(ログイン完了、設定変更など)の通知にはWeakReferenceMessengerを使います。弱参照ベースなので、購読解除し忘れによるメモリリークが起きないのが大きな魅力です。

// メッセージ定義(recordを推奨)
public sealed record TodoCreatedMessage(TodoItem Item);

// 送信側
public partial class CreateTodoViewModel : ObservableObject
{
    [RelayCommand]
    private async Task SaveAsync()
    {
        var item = await _todoService.CreateAsync(Title);
        WeakReferenceMessenger.Default.Send(new TodoCreatedMessage(item));
        await Shell.Current.GoToAsync("..");
    }
}

// 受信側(ObservableRecipientを継承)
public partial class TodoListViewModel : ObservableRecipient,
    IRecipient<TodoCreatedMessage>
{
    public TodoListViewModel(ITodoService todoService)
    {
        _todoService = todoService;
        IsActive = true; // メッセージ購読を有効化
    }

    public void Receive(TodoCreatedMessage message)
    {
        Items.Insert(0, message.Item);
    }
}

重要な注意点ObservableRecipientを継承する場合は、IsActive = trueで購読を開始し、ViewModelが不要になるタイミング(ページのDisappearingなど)でIsActive = falseを設定してください。これでOnActivated/OnDeactivatedが呼ばれ、購読が安全に解除されます。これを忘れると、メッセージが届かない or 不要なメッセージを受け続ける、というハマりパターンになりがちです。

ObservableValidatorで宣言的バリデーション

フォーム入力のバリデーションは、System.ComponentModel.DataAnnotations属性とObservableValidatorを組み合わせると、UIロジックをほぼ書かずに実装できます。

public partial class SignUpViewModel : ObservableValidator
{
    [ObservableProperty]
    [Required(ErrorMessage = "メールアドレスは必須です")]
    [EmailAddress(ErrorMessage = "有効なメールアドレスを入力してください")]
    [NotifyDataErrorInfo]
    private string _email = string.Empty;

    [ObservableProperty]
    [Required]
    [MinLength(8, ErrorMessage = "パスワードは8文字以上")]
    [RegularExpression(@"^(?=.*[A-Z])(?=.*\d).+$",
        ErrorMessage = "大文字と数字を含めてください")]
    [NotifyDataErrorInfo]
    [NotifyCanExecuteChangedFor(nameof(SignUpCommand))]
    private string _password = string.Empty;

    [RelayCommand(CanExecute = nameof(CanSignUp))]
    private async Task SignUpAsync()
    {
        ValidateAllProperties();
        if (HasErrors) return;
        // ...サインアップ処理
    }

    private bool CanSignUp() => !HasErrors
        && !string.IsNullOrEmpty(Email)
        && !string.IsNullOrEmpty(Password);
}

XAMLではValidateOnNotifyDataErrorsとエラー表示用Labelを組み合わせて、リアルタイムバリデーションUIを構築します。

<VerticalStackLayout>
    <Entry Text="{Binding Email, Mode=TwoWay,
                   ValidateOnNotifyDataErrors=True}"
           Placeholder="メールアドレス" />
    <Label Text="{Binding GetErrors[Email], Converter={StaticResource FirstErrorConverter}}"
           TextColor="Red"
           IsVisible="{Binding HasErrors}" />
</VerticalStackLayout>

依存性注入(DI)とShell Navigationの統合

.NET MAUI 10のShellナビゲーションでは、コンストラクター注入が標準でサポートされています。QueryProperty[ObservableProperty]を組み合わせれば、URLパラメーターからViewModelの状態をきれいに初期化できます。

[QueryProperty(nameof(TodoId), "id")]
public partial class TodoDetailViewModel : ObservableObject
{
    private readonly ITodoService _todoService;

    public TodoDetailViewModel(ITodoService todoService)
    {
        _todoService = todoService;
    }

    [ObservableProperty]
    private int _todoId;

    [ObservableProperty]
    private TodoItem? _currentTodo;

    partial void OnTodoIdChanged(int value)
    {
        _ = LoadAsync(value);
    }

    private async Task LoadAsync(int id)
    {
        CurrentTodo = await _todoService.GetByIdAsync(id);
    }
}

呼び出し側はこんな感じ。

[RelayCommand]
private async Task NavigateToDetailAsync(TodoItem item)
{
    await Shell.Current.GoToAsync($"detail?id={item.Id}");
}

パフォーマンスとNative AOT対応

.NET MAUI 10ではiOS/MacCatalystでNative AOTが正式サポートされ、起動時間が最大40%短縮されました。CommunityToolkit.Mvvmはソースジェネレーターベースなのでリフレクションを使わず、AOT環境で完全に動きます。プロジェクト設定例:

<PropertyGroup Condition="'$(TargetFramework)' == 'net10.0-ios'">
  <PublishAot>true</PublishAot>
  <TrimMode>full</TrimMode>
</PropertyGroup>

ベンチマーク(iPhone 15 Pro、Release/AOTビルド)の傾向はこんな具合です。

  • 1,000個のObservableObjectプロパティ更新:手書きINPC比で+3%(誤差範囲)
  • RelayCommand実行オーバーヘッド:手書きCommandクラス比でほぼ同等
  • アプリ起動時間:CommunityToolkit.Mvvm 8.4採用プロジェクトで平均1.2秒(リフレクション系MVVMの2.8秒から大幅短縮)

つまり、コード量は減るのに性能はほぼ劣化しない。トレードオフがほぼゼロという、なかなか珍しい類のライブラリです。

よくある落とし穴と対策

partialキーワードを忘れてビルドエラー

ソースジェネレーターはpartial classに対してのみ動作します。public classのままだと「メンバーが見つからない」エラーが出ます。私もこれで何度かハマりました。クラス宣言は必ずpublic partial classにしてください。

ObservableObjectとBindableObjectの併用

MAUIのBindableObjectObservableObjectを同じクラスで継承することはできません。ViewModelにはObservableObjectを、コントロール拡張にはBindableObjectを、と役割をきっちり分けてください。

MessengerのIsActive忘れ

ObservableRecipientを使うとき、IsActive = trueを設定し忘れるとメッセージが届きません。コンストラクターかOnAppearingで必ず有効化しましょう。

非同期コマンドでのスレッディング

[RelayCommand]は内部的にUIスレッドでデリゲートを実行します。長時間処理はTask.Runでバックグラウンドに逃がし、UI更新はMainThread.BeginInvokeOnMainThreadでディスパッチしてください。これを怠るとUIフリーズの原因になります。

移行戦略:既存MVVMコードからの段階的リファクタリング

Xamarin.Formsから移行した既存ViewModelをCommunityToolkit.Mvvmに置き換えるときの、私のおすすめ手順は以下です。

  1. クラスにpartial追加public partial class XxxViewModel : ObservableObjectに変更
  2. プロパティから順次変換:1プロパティずつ[ObservableProperty]化し、ビルド・テスト
  3. コマンドの変換Command/AsyncCommand[RelayCommand]に置換
  4. Messenger統合MessagingCenter(.NET MAUI 9で非推奨)からWeakReferenceMessengerへ移行
  5. Validation移行:手書きバリデーションをObservableValidator + DataAnnotationsに置換

1ページあたり通常1~2時間で終わり、コード行数は平均40~60%削減されたという報告が多いです。最初の1ページで「あれ、こんなに減るのか」と驚くと思います。

FAQ:よくある質問

Q1. CommunityToolkit.MvvmはPrismの代替として使えますか?

用途次第です。基本的なMVVM(プロパティ・コマンド・メッセンジャー)であればCommunityToolkit.Mvvmで十分。一方、モジュール化(IModule)、リージョンナビゲーション、ダイアログサービスといった高度な機能が必要ならPrismの方が充実しています。シンプルなアプリ・新規プロジェクトはCommunityToolkit.Mvvm、エンタープライズ規模・複数モジュール構成はPrism、という棲み分けがおすすめです。

Q2. ReactiveUIと比べてどちらを選ぶべきですか?

Rx(Reactive Extensions)パターンに慣れているチームや、ストリーム合成を多用するアプリならReactiveUIが強力です。一方、命令型MVVMを書きやすく、学習コストを抑えてチーム全員が即座に書ける状態にしたい場合はCommunityToolkit.Mvvm一択でしょう。

Q3. ソースジェネレーターはビルド時間に影響しますか?

ほぼ影響しません。中規模プロジェクト(100ViewModel程度)でビルド時間の増加は1~2秒以内です。インクリメンタルジェネレーターとして実装されているので、編集中のホットリロードも遅延なく動作します。

Q4. .NET MAUI 10のC# 13機能(fieldキーワード)は使えますか?

使えます。CommunityToolkit.Mvvm 8.4以降では、partialプロパティに直接[ObservableProperty]を付与する記法もサポートされています。ただし、IDEのリファクタリングサポートが安定するまでは、明示的フィールド宣言の方が無難かもしれません。

Q5. ObservableObjectの代わりにObservableRecipientを常に使うべきですか?

いいえ。ObservableRecipientMessenger機能を内蔵するぶん、わずかなオーバーヘッドがあります。メッセージ送受信を行うViewModelだけObservableRecipientを継承し、それ以外は軽量なObservableObjectを使う、という方針がおすすめです。

まとめ

CommunityToolkit.Mvvm 8.x は、.NET MAUI 10時代のMVVM実装における事実上の標準です。[ObservableProperty][RelayCommand]を使うだけでもコード量は劇的に減り、可読性とメンテナンス性が一気に上がります。さらにWeakReferenceMessengerObservableValidatorを活用すれば、ViewModel間通信やフォームバリデーションも宣言的に書けます。

Native AOT対応・パフォーマンス・チーム開発における学習コストの低さ──どの観点から見ても、新規・既存問わず採用を強く推奨します。本記事のコード例をベースに、まずは1ページだけでも自分のプロジェクトをリファクタリングしてみてください。たぶん、戻れなくなるはずです。

著者について Editorial Team

Our team of expert writers and editors.