WPF связывает ComboBox с перечислением (с изюминкой) - PullRequest
14 голосов
/ 27 мая 2009

Ну, проблема в том, что у меня есть это перечисление, НО я не хочу, чтобы в выпадающем списке отображались значения перечисления. Это перечисление:

public enum Mode
    {
        [Description("Display active only")]
        Active,
        [Description("Display selected only")]
        Selected,
        [Description("Display active and selected")]
        ActiveAndSelected
    }

Таким образом, в ComboBox вместо отображения Active, Selected или ActiveAndSelected я хочу отобразить свойство DescriptionProperty для каждого значения перечисления. У меня есть метод расширения, называемый GetDescription () для перечисления:

public static string GetDescription(this Enum enumObj)
        {
            FieldInfo fieldInfo =
                enumObj.GetType().GetField(enumObj.ToString());

            object[] attribArray = fieldInfo.GetCustomAttributes(false);

            if (attribArray.Length == 0)
            {
                return enumObj.ToString();
            }
            else
            {
                DescriptionAttribute attrib =
                    attribArray[0] as DescriptionAttribute;
                return attrib.Description;
            }
        }

Так есть ли способ связать перечисление с ComboBox и показать его содержимое с помощью метода расширения GetDescription?

Спасибо!

Ответы [ 6 ]

19 голосов
/ 27 мая 2009

Я бы предложил DataTemplate и ValueConverter. Это позволит вам настроить способ его отображения, но вы все равно сможете прочитать свойство SelectedItem комбобокса и получить фактическое значение перечисления.

ValueConverters требует много стандартного кода, но здесь нет ничего слишком сложного. Сначала вы создаете класс ValueConverter:

public class ModeConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter,
        CultureInfo culture)
    {
        return ((Mode) value).GetDescription();
    }
    public object ConvertBack(object value, Type targetType, object parameter,
        CultureInfo culture)
    {
        throw new NotSupportedException();
    }
}

Поскольку вы преобразуете только перечисляемые значения в строки (для отображения), вам не нужен ConvertBack - это только для сценариев двустороннего связывания.

Затем вы помещаете экземпляр ValueConverter в ваши ресурсы, что-то вроде этого:

<Window ... xmlns:WpfApplication1="clr-namespace:WpfApplication1">
    <Window.Resources>
        <WpfApplication1:ModeConverter x:Key="modeConverter"/>
    </Window.Resources>
    ....
</Window>

Тогда вы готовы предоставить ComboBox DisplayTemplate, который форматирует свои элементы с помощью ModeConverter:

<ComboBox Name="comboBox" ...>
    <ComboBox.ItemTemplate>
        <DataTemplate>
            <TextBlock Text="{Binding Converter={StaticResource modeConverter}}"/>
        </DataTemplate>
    </ComboBox.ItemTemplate>
</ComboBox>

Чтобы проверить это, я также добавил Label, который показал бы мне фактическое значение SelectedItem, и он действительно показал, что SelectedItem - это перечисление вместо отображаемого текста, что я и хотел бы:

<Label Content="{Binding ElementName=comboBox, Path=SelectedItem}"/>
7 голосов
/ 10 апреля 2010

Вот как я это делаю с MVVM. На моей модели я бы определил мое перечисление:

    public enum VelocityUnitOfMeasure
    {
        [Description("Miles per Hour")]
        MilesPerHour,
        [Description("Kilometers per Hour")]
        KilometersPerHour
    }

В моей ViewModel я предоставляю свойство, которое предоставляет возможные варианты выбора в виде строки, а также свойство для получения / установки значения модели. Это полезно, если мы не хотим использовать каждое значение перечисления в типе:

    //UI Helper
    public IEnumerable<string> VelocityUnitOfMeasureSelections
    {
        get
        {
            var units = new []
                            {
                               VelocityUnitOfMeasure.MilesPerHour.Description(),
                               VelocityUnitOfMeasure.KilometersPerHour.Description()
                            };
            return units;
        }
    }

    //VM property
    public VelocityUnitOfMeasure UnitOfMeasure
    {
        get { return model.UnitOfMeasure; }
        set { model.UnitOfMeasure = value; }
    }

Кроме того, я использую универсальный EnumDescriptionCoverter:

public class EnumDescriptionConverter : IValueConverter
{
    //From Binding Source
    public object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    {
        if (!(value is Enum)) throw new ArgumentException("Value is not an Enum");
        return (value as Enum).Description();
    }

    //From Binding Target
    public object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
    {
        if (!(value is string)) throw new ArgumentException("Value is not a string");
        foreach(var item in Enum.GetValues(targetType))
        {
            var asString = (item as Enum).Description();
            if (asString == (string) value)
            {
                return item;
            }
        }
        throw new ArgumentException("Unable to match string to Enum description");
    }
}

И, наконец, с видом я могу сделать следующее:

<Window.Resources>
    <ValueConverters:EnumDescriptionConverter x:Key="enumDescriptionConverter" />
</Window.Resources>
...
<ComboBox SelectedItem="{Binding UnitOfMeasure, Converter={StaticResource enumDescriptionConverter}}"
          ItemsSource="{Binding VelocityUnitOfMeasureSelections, Mode=OneWay}" />
6 голосов
/ 27 мая 2009

Мне нравится, как ты думаешь. Но GetCustomAttributes использует отражение. Как это повлияет на ваше выступление?

Проверьте это сообщение: WPF - отображение перечислений в элементе управления ComboBox http://www.infosysblogs.com/microsoft/2008/09/wpf_displaying_enums_in_combob.html

3 голосов
/ 27 мая 2009

Я предлагаю вам использовать расширение разметки, которое я уже разместил здесь , с небольшой модификацией:

[MarkupExtensionReturnType(typeof(IEnumerable))]
public class EnumValuesExtension : MarkupExtension
{
    public EnumValuesExtension()
    {
    }

    public EnumValuesExtension(Type enumType)
    {
        this.EnumType = enumType;
    }

    [ConstructorArgument("enumType")]
    public Type EnumType { get; set; }

    public override object ProvideValue(IServiceProvider serviceProvider)
    {
        if (this.EnumType == null)
            throw new ArgumentException("The enum type is not set");
        return Enum.GetValues(this.EnumType).Select(o => GetDescription(o));
    }
}

Затем вы можете использовать его так:

<ComboBox ItemsSource="{local:EnumValues local:Mode}"/>

РЕДАКТИРОВАТЬ: предложенный мною метод будет привязывать к списку строк, что нежелательно, поскольку мы хотим, чтобы SelectedItem имел тип Mode Было бы лучше удалить часть .Select (...) и использовать привязку с пользовательским преобразователем в ItemTemplate.

3 голосов
/ 27 мая 2009

Вопросы использования отражения и атрибутов в стороне, есть несколько способов, которыми вы могли бы сделать это, но я думаю, что лучший способ - это просто создать небольшой класс модели представления, который переносит значение перечисления:

public class ModeViewModel : ViewModel
{
    private readonly Mode _mode;

    public ModeViewModel(Mode mode)
    {
        ...
    }

    public Mode Mode
    {
        get { ... }
    }

    public string Description
    {
        get { return _mode.GetDescription(); }
    }
}

Кроме того, вы можете использовать ObjectDataProvider.

0 голосов
/ 17 января 2013

Я сделал это так:

<ComboBox x:Name="CurrencyCodeComboBox" Grid.Column="4" DisplayMemberPath="." HorizontalAlignment="Left" Height="22"   Margin="11,6.2,0,10.2" VerticalAlignment="Center" Width="81" Grid.Row="1" SelectedValue="{Binding currencyCode}" >
            <ComboBox.ItemsPanel>
                <ItemsPanelTemplate>
                    <VirtualizingStackPanel/>
                </ItemsPanelTemplate>
            </ComboBox.ItemsPanel>
        </ComboBox>

в коде я установил itemSource:

CurrencyCodeComboBox.ItemsSource = [Enum].GetValues(GetType(currencyCode))
...