How to bind RadioButtons to an enum?

前端 未结 9 2081
日久生厌
日久生厌 2020-11-22 03:21

I\'ve got an enum like this:

public enum MyLovelyEnum
{
    FirstSelection,
    TheOtherSelection,
    YetAnotherOne
};

I got a property in

相关标签:
9条回答
  • 2020-11-22 03:49

    I would use the RadioButtons in a ListBox, and then bind to the SelectedValue.

    This is an older thread about this topic, but the base idea should be the same: http://social.msdn.microsoft.com/Forums/en-US/wpf/thread/323d067a-efef-4c9f-8d99-fecf45522395/

    0 讨论(0)
  • 2020-11-22 03:50

    You could use a more generic converter

    public class EnumBooleanConverter : IValueConverter
    {
      #region IValueConverter Members
      public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
      {
        string parameterString = parameter as string;
        if (parameterString == null)
          return DependencyProperty.UnsetValue;
    
        if (Enum.IsDefined(value.GetType(), value) == false)
          return DependencyProperty.UnsetValue;
    
        object parameterValue = Enum.Parse(value.GetType(), parameterString);
    
        return parameterValue.Equals(value);
      }
    
      public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
      {
        string parameterString = parameter as string;
        if (parameterString == null)
            return DependencyProperty.UnsetValue;
    
        return Enum.Parse(targetType, parameterString);
      }
      #endregion
    }
    

    And in the XAML-Part you use:

    <Grid>
        <Grid.Resources>
          <l:EnumBooleanConverter x:Key="enumBooleanConverter" />
        </Grid.Resources>
        <StackPanel >
          <RadioButton IsChecked="{Binding Path=VeryLovelyEnum, Converter={StaticResource enumBooleanConverter}, ConverterParameter=FirstSelection}">first selection</RadioButton>
          <RadioButton IsChecked="{Binding Path=VeryLovelyEnum, Converter={StaticResource enumBooleanConverter}, ConverterParameter=TheOtherSelection}">the other selection</RadioButton>
          <RadioButton IsChecked="{Binding Path=VeryLovelyEnum, Converter={StaticResource enumBooleanConverter}, ConverterParameter=YetAnotherOne}">yet another one</RadioButton>
        </StackPanel>
    </Grid>
    
    0 讨论(0)
  • 2020-11-22 03:52

    You can further simplify the accepted answer. Instead of typing out the enums as strings in xaml and doing more work in your converter than needed, you can explicitly pass in the enum value instead of a string representation, and as CrimsonX commented, errors get thrown at compile time rather than runtime:

    ConverterParameter={x:Static local:YourEnumType.Enum1}

    <StackPanel>
        <StackPanel.Resources>          
            <local:ComparisonConverter x:Key="ComparisonConverter" />          
        </StackPanel.Resources>
        <RadioButton IsChecked="{Binding Path=YourEnumProperty, Converter={StaticResource ComparisonConverter}, ConverterParameter={x:Static local:YourEnumType.Enum1}}" />
        <RadioButton IsChecked="{Binding Path=YourEnumProperty, Converter={StaticResource ComparisonConverter}, ConverterParameter={x:Static local:YourEnumType.Enum2}}" />
    </StackPanel>
    

    Then simplify the converter:

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

    Edit (Dec 16 '10):

    Thanks to anon for suggesting returning Binding.DoNothing rather than DependencyProperty.UnsetValue.


    Note - Multiple groups of RadioButtons in same container (Feb 17 '11):

    In xaml, if radio buttons share the same parent container, then selecting one will de-select all other's within that container (even if they are bound to a different property). So try to keep your RadioButton's that are bound to a common property grouped together in their own container like a stack panel. In cases where your related RadioButtons cannot share a single parent container, then set the GroupName property of each RadioButton to a common value to logically group them.


    Edit (Apr 5 '11):

    Simplified ConvertBack's if-else to use a Ternary Operator.


    Note - Enum type nested in a class (Apr 28 '11):

    If your enum type is nested in a class (rather than directly in the namespace), you might be able to use the '+' syntax to access the enum in XAML as stated in a (not marked) answer to the question Unable to find enum type for static reference in WPF:

    ConverterParameter={x:Static local:YourClass+YourNestedEnumType.Enum1}

    Due to this Microsoft Connect Issue, however, the designer in VS2010 will no longer load stating "Type 'local:YourClass+YourNestedEnumType' was not found.", but the project does compile and run successfully. Of course, you can avoid this issue if you are able to move your enum type to the namespace directly.


    Edit (Jan 27 '12):

    If using Enum flags, the converter would be as follows:

    public class EnumToBooleanConverter : 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;
        }
    }
    

    Edit (May 7 '15):

    In case of a Nullable Enum (that is not asked in the question, but can be needed in some cases, e.g. ORM returning null from DB or whenever it might make sense that in the program logic the value is not provided), remember to add an initial null check in the Convert Method and return the appropriate bool value, that is typically false (if you don't want any radio button selected), like below:

        public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
        {
            if (value == null) {
                return false; // or return parameter.Equals(YourEnumType.SomeDefaultValue);
            }
            return value.Equals(parameter);
        }
    

    Note - NullReferenceException (Oct 10 '18):

    Updated the example to remove the possibility of throwing a NullReferenceException. IsChecked is a nullable type so returning Nullable<Boolean> seems a reasonable solution.

    0 讨论(0)
  • 2020-11-22 03:54

    For the EnumToBooleanConverter answer: Instead of returning DependencyProperty.UnsetValue consider returning Binding.DoNothing for the case where the radio button IsChecked value becomes false. The former indicates a problem (and might show the user a red rectangle or similar validation indicators) while the latter just indicates that nothing should be done, which is what is wanted in that case.

    http://msdn.microsoft.com/en-us/library/system.windows.data.ivalueconverter.convertback.aspx http://msdn.microsoft.com/en-us/library/system.windows.data.binding.donothing.aspx

    0 讨论(0)
  • 2020-11-22 04:00

    I've created a new class to handle binding RadioButtons and CheckBoxes to enums. It works for flagged enums (with multiple checkbox selections) and non-flagged enums for single-selection checkboxes or radio buttons. It also requires no ValueConverters at all.

    This might look more complicated at first, however, once you copy this class into your project, it's done. It's generic so it can easily be reused for any enum.

    public class EnumSelection<T> : INotifyPropertyChanged where T : struct, IComparable, IFormattable, IConvertible
    {
      private T value; // stored value of the Enum
      private bool isFlagged; // Enum uses flags?
      private bool canDeselect; // Can be deselected? (Radio buttons cannot deselect, checkboxes can)
      private T blankValue; // what is considered the "blank" value if it can be deselected?
    
      public EnumSelection(T value) : this(value, false, default(T)) { }
      public EnumSelection(T value, bool canDeselect) : this(value, canDeselect, default(T)) { }
      public EnumSelection(T value, T blankValue) : this(value, true, blankValue) { }
      public EnumSelection(T value, bool canDeselect, T blankValue)
      {
        if (!typeof(T).IsEnum) throw new ArgumentException($"{nameof(T)} must be an enum type"); // I really wish there was a way to constrain generic types to enums...
        isFlagged = typeof(T).IsDefined(typeof(FlagsAttribute), false);
    
        this.value = value;
        this.canDeselect = canDeselect;
        this.blankValue = blankValue;
      }
    
      public T Value
      {
        get { return value; }
        set 
        {
          if (this.value.Equals(value)) return;
          this.value = value;
          OnPropertyChanged();
          OnPropertyChanged("Item[]"); // Notify that the indexer property has changed
        }
      }
    
      [IndexerName("Item")]
      public bool this[T key]
      {
        get
        {
          int iKey = (int)(object)key;
          return isFlagged ? ((int)(object)value & iKey) == iKey : value.Equals(key);
        }
        set
        {
          if (isFlagged)
          {
            int iValue = (int)(object)this.value;
            int iKey = (int)(object)key;
    
            if (((iValue & iKey) == iKey) == value) return;
    
            if (value)
              Value = (T)(object)(iValue | iKey);
            else
              Value = (T)(object)(iValue & ~iKey);
          }
          else
          {
            if (this.value.Equals(key) == value) return;
            if (!value && !canDeselect) return;
    
            Value = value ? key : blankValue;
          }
        }
      }
    
      public event PropertyChangedEventHandler PropertyChanged;
    
      private void OnPropertyChanged([CallerMemberName] string propertyName = "")
      {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
      }
    }
    

    And for how to use it, let's say you have an enum for running a task manually or automatically, and can be scheduled for any days of the week, and some optional options...

    public enum StartTask
    {
      Manual,
      Automatic
    }
    
    [Flags()]
    public enum DayOfWeek
    {
      Sunday = 1 << 0,
      Monday = 1 << 1,
      Tuesday = 1 << 2,
      Wednesday = 1 << 3,
      Thursday = 1 << 4,
      Friday = 1 << 5,
      Saturday = 1 << 6
    }
    
    public enum AdditionalOptions
    {
      None = 0,
      OptionA,
      OptionB
    }
    

    Now, here's how easy it is to use this class:

    public class MyViewModel : ViewModelBase
    {
      public MyViewModel()
      {
        StartUp = new EnumSelection<StartTask>(StartTask.Manual);
        Days = new EnumSelection<DayOfWeek>(default(DayOfWeek));
        Options = new EnumSelection<AdditionalOptions>(AdditionalOptions.None, true, AdditionalOptions.None);
      }
    
      public EnumSelection<StartTask> StartUp { get; private set; }
      public EnumSelection<DayOfWeek> Days { get; private set; }
      public EnumSelection<AdditionalOptions> Options { get; private set; }
    }
    

    And here's how easy it is to bind checkboxes and radio buttons with this class:

    <StackPanel Orientation="Vertical">
      <StackPanel Orientation="Horizontal">
        <!-- Using RadioButtons for exactly 1 selection behavior -->
        <RadioButton IsChecked="{Binding StartUp[Manual]}">Manual</RadioButton>
        <RadioButton IsChecked="{Binding StartUp[Automatic]}">Automatic</RadioButton>
      </StackPanel>
      <StackPanel Orientation="Horizontal">
        <!-- Using CheckBoxes for 0 or Many selection behavior -->
        <CheckBox IsChecked="{Binding Days[Sunday]}">Sunday</CheckBox>
        <CheckBox IsChecked="{Binding Days[Monday]}">Monday</CheckBox>
        <CheckBox IsChecked="{Binding Days[Tuesday]}">Tuesday</CheckBox>
        <CheckBox IsChecked="{Binding Days[Wednesday]}">Wednesday</CheckBox>
        <CheckBox IsChecked="{Binding Days[Thursday]}">Thursday</CheckBox>
        <CheckBox IsChecked="{Binding Days[Friday]}">Friday</CheckBox>
        <CheckBox IsChecked="{Binding Days[Saturday]}">Saturday</CheckBox>
      </StackPanel>
      <StackPanel Orientation="Horizontal">
        <!-- Using CheckBoxes for 0 or 1 selection behavior -->
        <CheckBox IsChecked="{Binding Options[OptionA]}">Option A</CheckBox>
        <CheckBox IsChecked="{Binding Options[OptionB]}">Option B</CheckBox>
      </StackPanel>
    </StackPanel>
    
    1. When the UI loads, the "Manual" radio button will be selected and you can alter your selection between "Manual" or "Automatic" but either one of them must always be selected.
    2. Every day of the week will be unchecked, but any number of them can be checked or unchecked.
    3. "Option A" and "Option B" will both initially be unchecked. You can check one or the other, checking one will uncheck the other (similar to RadioButtons), but now you can also uncheck both of them (which you cannot do with WPF's RadioButton, which is why CheckBox is being used here)
    0 讨论(0)
  • 2020-11-22 04:05

    For UWP, it is not so simple: You must jump through an extra hoop to pass a field value as a parameter.

    Example 1

    Valid for both WPF and UWP.

    <MyControl>
        <MyControl.MyProperty>
            <Binding Converter="{StaticResource EnumToBooleanConverter}" Path="AnotherProperty">
                <Binding.ConverterParameter>
                    <MyLibrary:MyEnum>Field</MyLibrary:MyEnum>
                </Binding.ConverterParameter>
            </MyControl>
        </MyControl.MyProperty>
    </MyControl>
    

    Example 2

    Valid for both WPF and UWP.

    ...
    <MyLibrary:MyEnum x:Key="MyEnumField">Field</MyLibrary:MyEnum>
    ...
    
    <MyControl MyProperty="{Binding AnotherProperty, Converter={StaticResource EnumToBooleanConverter}, ConverterParameter={StaticResource MyEnumField}}"/>
    

    Example 3

    Valid only for WPF!

    <MyControl MyProperty="{Binding AnotherProperty, Converter={StaticResource EnumToBooleanConverter}, ConverterParameter={x:Static MyLibrary:MyEnum.Field}}"/>
    

    UWP doesn't support x:Static so Example 3 is out of the question; assuming you go with Example 1, the result is more verbose code. Example 2 is slightly better, but still not ideal.

    Solution

    public abstract class EnumToBooleanConverter<TEnum> : IValueConverter
    {
        public object Convert(object value, Type targetType, object parameter, string language)
        {
            var Parameter = parameter as string;
    
            if (Parameter == null)
                return DependencyProperty.UnsetValue;
    
            if (Enum.IsDefined(typeof(TEnum), value) == false)
                return DependencyProperty.UnsetValue;
    
            return Enum.Parse(typeof(TEnum), Parameter).Equals(value);
        }
    
        public object ConvertBack(object value, Type targetType, object parameter, string language)
        {
            var Parameter = parameter as string;
            return Parameter == null ? DependencyProperty.UnsetValue : Enum.Parse(typeof(TEnum), Parameter);
        }
    }
    

    Then, for each type you wish to support, define a converter that boxes the enum type.

    public class MyEnumToBooleanConverter : EnumToBooleanConverter<MyEnum>
    {
        //Nothing to do!
    }
    

    The reason it must be boxed is because there's seemingly no way to reference the type in the ConvertBack method; the boxing takes care of that. If you go with either of the first two examples, you can just reference the parameter type, eliminating the need to inherit from a boxed class; if you wish to do it all in one line and with least verbosity possible, the latter solution is ideal.

    Usage resembles Example 2, but is, in fact, less verbose.

    <MyControl MyProperty="{Binding AnotherProperty, Converter={StaticResource MyEnumToBooleanConverter}, ConverterParameter=Field}"/>
    

    The downside is you must define a converter for each type you wish to support.

    0 讨论(0)
提交回复
热议问题