将带有标志的单个枚举绑定到 WPF 中的多个复选框?

Binding A Single enum with Flags to Multiple Checkboxes in WPF?

我有以下枚举:

[Flags]
public enum SubscriberLevel {
    BRONZE = 1,
    SILVER1 = 2,
    SILVER2 = 4,
    SILVER = SubscriberLevel.SILVER1 | SubscriberLevel.SILVER2,
    GOLD1 = 8,
    GOLD2 = 16,
    GOLD3 = 32,
    GOLD = SubscriberLevel.GOLD1 | SubscriberLevel.GOLD2 | SubscriberLevel.GOLD3,
    DIAMOND1 = 64,
    DIAMOND2 = 128,
    DIAMOND3 = 256,
    DIAMOND = SubscriberLevel.DIAMOND1 | SubscriberLevel.DIAMOND2 | SubscriberLevel.DIAMOND3,
    ALL = SubscriberLevel.BRONZE | SubscriberLevel.SILVER | SubscriberLevel.GOLD | SubscriberLevel.DIAMOND
}

我正在开发一个控件,该控件应该允许用户通过一系列复选框指定此枚举中的一个、多个或所有标志,列表中的每个枚举一个。

我在控件中有一个 Enum 成员,还有一个 属性 来公开 Enum。 控件本身继承了 INotify属性Changed 方法,属性 的 Setter 在设置值时正确调用 On属性Changed 事件处理程序。

我什至在控件中看到了一些类似的功能,但我没有看到它是否正常运行:

说用户某某取消勾选ALL选项;我希望每个复选框都通过清除来反映这一点,但我没有看到这种情况发生。

截至目前,我认为我的问题是我将这些复选框的 Checked 状态绑定到 Enum 本身,而我需要做的是将 Enum 绑定到这些选中的框,并让每个检查框表示枚举集中的特定标志。

我在调试控件时注意到,当我取消选中该复选框时,该值未被设置。这是为什么?

我知道我需要使用转换器来完成此操作,但就目前而言;我如何在 XAML 方面完成这项工作;将控件的 Datacontext 定义为控件本身,然后将枚举标志绑定到它们各自的复选框?

这是复选框之一的示例:

<CheckBox
    x:Name="chkAll" Grid.Row="1" VerticalContentAlignment="Center" VerticalAlignment="Center"
    IsChecked="{Binding Path=SubLevel, Converter={StaticResource ETBC},
    ConverterParameter={x:Static Enums:SubscriberLevel.ALL}, Mode=TwoWay}"/>

这是转换器:

public class EnumToBoolConverter : IValueConverter {
    public object Convert( object value, Type targetType, object parameter, System.Globalization.CultureInfo culture ) {
        return ( ( Enum )value ).HasFlag( ( Enum )parameter );
    }

    public object ConvertBack( object value, Type targetType, object parameter, System.Globalization.CultureInfo culture ) {
        return value.Equals( true ) ? parameter : Binding.DoNothing;
    }
}

这是控件的代码(就目前而言):

public partial class UCSubscriptionLevels : UserControl, INotifyPropertyChanged {
    private SubscriberLevel _SubLevel = SubscriberLevel.ALL;
    public event PropertyChangedEventHandler PropertyChanged;
    public UCSubscriptionLevels( ) { InitializeComponent( ); }

    /// <summary>
    /// Get or Set Subscriber Levels to Build.
    /// </summary>
    public SubscriberLevel SubLevel {
        get { return this._SubLevel; }
        set {
            this._SubLevel = value;
            if ( this.PropertyChanged != null )
                this.PropertyChanged( this, new PropertyChangedEventArgs( "SubLevel" ) );
        }
    }       
}

编辑

好久不见;我忘记了依赖属性,所以我选择在控件上实现依赖 属性 而不是 属性 成员。行为仍然很粗略。这是控件中 属性 的新代码:

public static readonly DependencyProperty SubscriberLevelProperty =
            DependencyProperty.Register( "SubLevel", typeof( SubscriberLevel ),
            typeof( UCSubscriptionLevels ), new PropertyMetadata( SubscriberLevel.ALL ) );
.
.
.
/// <summary>
/// Get or Set Subscriber Levels to Build.
/// </summary>
public SubscriberLevel SubLevel {
    get { return ( SubscriberLevel )this.GetValue( UCSubscriptionLevels.SubscriberLevelProperty ); }
    set { this.SetValue( UCSubscriptionLevels.SubscriberLevelProperty, value ); }
}

但是,它仍然无法正常运行...

还有;我不得不更改枚举(Bronze = 0 到 1,等等)。

好的;这是一种非常烦人的解决方法,并且可能与您可以获得的正确解决方案相去甚远,但我能够想到一种方法来处理我正在尝试做的事情。这一切都在后面的代码中完成; XAML.

中根本没有绑定发生

首先是依赖关系属性:

public static readonly DependencyProperty SubscriberLevelProperty =
            DependencyProperty.Register( "SubLevel", typeof( SubscriberLevel ),
            typeof( UCSubscriptionLevels ), new FrameworkPropertyMetadata( SubscriberLevel.ALL, FrameworkPropertyMetadataOptions.None, ( S, E ) => {
                SubscriberLevel NV = ( SubscriberLevel )E.NewValue;
                UCSubscriptionLevels sender = S as UCSubscriptionLevels;
                sender.GetAllChildren<CheckBox>( ).ToList( ).ForEach( chk => {
                    chk.Checked -= sender.CheckedChanged;
                    chk.Unchecked -= sender.CheckedChanged;
                    chk.IsChecked = NV.HasFlag( sender.dctChkSL[chk] );
                    chk.Checked += new RoutedEventHandler( sender.CheckedChanged );
                    chk.Unchecked += new RoutedEventHandler( sender.CheckedChanged );
                } );
            } ) );

此处的主要变化是 属性 自行处理传入的数据更改。我以前用其他东西做过这个并且取得了一定程度的成功,所以我在这里试了一下。

在 Loaded 事件中,控件为每个 Checkbox Checked 和 Unchecked 事件添加一个事件处理程序。此回调删除该事件处理程序(以避免无限循环),调整已检查状态,如果它在控件字典中找到,则将其标记为已检查;否则,取消选中它,然后再次将事件处理程序添加到事件中。

事件处理程序所做的就是,嗯

private void CheckedChanged( object sender, RoutedEventArgs e ) { this.SubLevel ^= this.dctChkSL[sender as CheckBox]; }.

这是字典:

private Dictionary<CheckBox, SubscriberLevel> dctChkSL;

这是字典和复选框 Checked/Unchecked 事件处理程序的初始化程序。

this.dctChkSL = new Dictionary<CheckBox, SubscriberLevel>( ) {
    {this.chkAll, SubscriberLevel.ALL}, {this.chkBronze, SubscriberLevel.BRONZE},
    {this.chkSilver, SubscriberLevel.SILVER}, {this.chkSilver1, SubscriberLevel.SILVER1},
    {this.chkSilver2, SubscriberLevel.SILVER2},
    {this.chkGold, SubscriberLevel.GOLD}, {this.chkGold1, SubscriberLevel.GOLD1},
    {this.chkGold2, SubscriberLevel.GOLD2}, {this.chkGold3, SubscriberLevel.GOLD3},
    {this.chkDiamond, SubscriberLevel.DIAMOND}, {this.chkDiamond1, SubscriberLevel.DIAMOND1},
    {this.chkDiamond2, SubscriberLevel.DIAMOND2}, {this.chkDiamond3, SubscriberLevel.DIAMOND3}
};
this.Loaded += ( S, E ) =>
    this.GetAllChildren<CheckBox>( ).ToList( ).ForEach( chk => {
        chk.Checked += new RoutedEventHandler( this.CheckedChanged );
        chk.Unchecked += new RoutedEventHandler( this.CheckedChanged );
    } );

尽管我一直希望有一个更 eloquent 的解决方案,但由于我真的找不到,所以必须这样做,直到有人可以提供更好的答案。