1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

[WinUI3] 3ステートのCheckBox に Binding できない件へ対応した話

Last updated at Posted at 2025-01-18

IsChecked のバインドが途切れてしまう

WinUI3 と MVVM で開発しているアプリで3ステートのチェックボックスを使うことになりました。
チェックボックスは IsThreeState プロパティを True にすることで 選択/非選択/不確定 の3つの状態を表すことができます。
それぞれの状態では IsCheckedプロパティが True/False/null に対応します。

image.png

ですが IsCheckedプロパティに bool? の値をバインドしたところ、
バインド直後は動作するのですが、UI上でチェックボックスをクリックするとバインドが外れてしまい、以降はバインドが連動しなくなってしまいました。

view.xaml
<!-- IsCheckedのバインドが途中で外れてしまう -->
<CheckBox IsChecked="{Binding SelectedState.Value, Mode=OneWay}"
          IsThreeState="True" />
viewmodel.cs
public ReadOnlyReactiveProperty<bool?> SelectedState { get; }

この問題に対して解決できましたので記事にします。

原因

Binding の仕様でした。
Binding は null を伝えることができず、そのために発生した不具合でした。

調べたところ WPF の CheckBox でも同じ原因で同じ不具合が発生するようです。

もし x:Bind が使える場所なら

もし、あなたのアプリの CheckBox コントロールが x:Bind を使える場所にあるなら x:Bind を使いましょう。
それだけで解決です。

x:Bind は nullable のバインドに対応していますので、バインドが外れることなく動作します。

view.xaml
<!-- x:Bind だとUIをクリックしても外れない -->
<CheckBox IsChecked="{x:Bind SelectedState.Value, Mode=OneWay}"
          IsThreeState="True" />

3ステートのプロパティと バインディングコンバータでバインドしてみる

しかし、開発中のアプリでは ListView の DataTemplate の中に CheckBox があり、DataTemplate のなかでは x:Bind が使えませんので別の方法が必要でした。

最初に試したのは バインドするプロパティは enum にして、バインディングコンバータで true/false/null に変換してみる方法です。

結論としては、この方法は効果がありませんでした。

view.xaml
<!-- enum をバインディングコンバータで null に変換。効果はなかった。 -->
<CheckBox IsChecked="{Binding SelectedThreeState.Value, Mode=OneWay, Converter={StaticResource CheckBoxThreeStateConverter}}"
          IsThreeState="True" />
viewmodel.cs
public ReadOnlyReactiveProperty<CheckBoxThreeStates> SelectedThreeState { get; }
    // 3ステートを表すenum
    public enum CheckBoxThreeStates
    {
        Unselected,
        Selected,
        Indeterminate
    }
    // バインディングコンバータ
    internal class CheckBoxThreeStateConverter : IValueConverter
    {
        public object Convert(object value, Type targetType, object parameter, string language)
        {
            if (value == null) return false;
            return (CheckBoxThreeStates)value switch
            {
                CheckBoxThreeStates.Selected => true,
                CheckBoxThreeStates.Unselected => false,
                CheckBoxThreeStates.Indeterminate => null,
                _ => false
            };
        }

        public object ConvertBack(object value, Type targetType, object parameter, string language)
        {
            throw new NotImplementedException();
        }
    }

カスタムコントロールにする

効果があったのはこの対策でした。

State という依存プロパティを持たせて、これに bool? の値をバインドする。
State の変化時に内部で IsChecked に反映させる。
という間接的なバインドをしたところバインドは外れることなく期待通りに動作しました。

以下のブログを参考にしました。

view.xaml
<!-- 期待通りに動いた。Stateに bool? をバインドして使う。 -->
<custom:BindableCheckBox
        State="{Binding SelectedState.Value, Mode=OneWay}"
        IsThreeState="True" />
viewmodel.cs
public ReadOnlyReactiveProperty<bool?> SelectedState { get; }
custom control
    public partial class BindableCheckBox : CheckBox
    {
        public BindableCheckBox()
        {
            this.DefaultStyleKey = typeof(BindableCheckBox);
        }

        public static readonly DependencyProperty StateProperty =
            DependencyProperty.Register(
               nameof(State),
               typeof(object),
               typeof(BindableCheckBox),
               new PropertyMetadata(null, (s, e) =>
               {
                   if (s is BindableCheckBox cb)
                   {
                       try
                       {
                           var newState = (bool?)e.NewValue;
                           if (cb.IsChecked != newState)
                               cb.IsChecked = newState;
                       }
                       finally
                       {
                       }
                   }
               }));

        public bool? State
        {
            get { return (bool?)GetValue(StateProperty); }
            set { SetValue(StateProperty, value); }
        }
    }

カスタムコントロールなので Generic.xaml に標準スタイルを作成してあげる必要があります。
この辺りはカスタムコントロールの一般的な話なので割愛します。

Generic.xaml
    <!-- こんな感じでデフォルトスタイルを作成 -->
    <Style TargetType="local:BindableCheckBox" BasedOn="{StaticResource DefaultBindableCheckBoxStyle}" />

    <!-- 標準CheckBoxのスタイルを真似ただけ -->
    <Style x:Key="DefaultBindableCheckBoxStyle" TargetType="local:BindableCheckBox">
        <Setter Property="Background" Value="{ThemeResource CheckBoxBackgroundUnchecked}" />
        <Setter Property="Foreground" Value="{ThemeResource CheckBoxForegroundUnchecked}" />
        <Setter Property="BorderBrush" Value="{ThemeResource CheckBoxBorderBrushUnchecked}" />
        (・・・以下略・・・)

おわりに

私はこの障害(仕様)への対策に1日費やしてしまいました。
この記事が誰かの助けにならんことを。

1
1
1

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?