gpt4 book ai didi

c# - 将带有标志的单个枚举绑定(bind)到 WPF 中的多个复选框?

转载 作者:太空狗 更新时间:2023-10-29 23:50:33 28 4
gpt4 key购买 nike

我有以下枚举:

[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 的属性。控件本身继承了 INotifyPropertyChanged 方法,属性的 Setter 在设置值时正确调用 OnPropertyChanged 事件处理程序。

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

enter image description here

假设用户某某取消选中 ALL 选项;我希望每个复选框也通过清除来反射(reflect)这一点,但我没有看到这种情况发生。

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

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

我知道我需要使用一个转换器来完成这个,但是现在;我如何着手在 XAML 端完成此操作;将控件的 Datacontext 定义为控件本身,然后将 Enum 标志绑定(bind)到它们各自的复选框?

这是其中一个复选框的示例:

<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 中根本没有发生任何绑定(bind)。

首先是依赖属性:

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;

这是字典和 Checkboxes 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 );
} );

尽管我一直希望有一个更有说服力的解决方案,但我并没有真正找到一个解决方案,所以必须这样做,直到有人可以提供更好的答案为止。

关于c# - 将带有标志的单个枚举绑定(bind)到 WPF 中的多个复选框?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/31011611/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com