unity enum description c# enums generics bindinglist

c# - description - Obtenga el valor Enum<T> Descripción



enum java (3)

Enum no tiene un método de Descripción (). Lo mejor que puedes hacer es que tu enum implemente una interfaz que tenga el método Description (). Si haces eso, entonces puedes tener

public static BindingList<KeyValuePair<T extends _interface_, String>> getBindingList()

y luego dentro de eso puedes referirte a

T foo = ...? foo.Description(...);

Tengo mi clase enumHelper que contiene estos:

public static IList<T> GetValues() { IList<T> list = new List<T>(); foreach (object value in Enum.GetValues(typeof(T))) { list.Add((T)value); } return list; }

y

public static string Description(Enum value) { Attribute DescAttribute = LMIGHelper.GetAttribute(value, typeof(DescriptionAttribute)); if (DescAttribute == null) return value.ToString(); else return ((DescriptionAttribute)DescAttribute).Description; }

mi enum es algo así como:

public enum OutputType { File, [Description("Data Table")] DataTable }

Hasta aquí todo bien. Todo el trabajo previo está bien. Ahora quiero agregar un nuevo ayudante para devolver BindingList>, así puedo vincular cualquier enumeración con cualquier combo usando

BindingList<KeyValuePair<OutputType, string>> list = Enum<OutputType>.GetBindableList(); cbo.datasource=list; cbo.DisplayMember="Value"; cbo.ValueMember="Key";

Para eso agregué:

public static BindingList<KeyValuePair<T, string>> GetBindingList() { BindingList<KeyValuePair<T, string>> list = new BindingList<KeyValuePair<T, string>>(); foreach (T value in Enum<T>.GetValues()) { string Desc = Enum<T>.Description(value); list.Add(new KeyValuePair<T, string>(value, Desc)); } return list; }

Pero "Enum.Description (value)" ni siquiera está compilando: Argumento ''1'': no ​​se puede convertir de ''T'' a ''System.Enum''

¿Cómo puedo hacer eso? ¿Es eso posible?

Gracias.


Debes cambiar:

public static string Description(Enum value) { ... }

a

public static string Description(T value) { ... }

entonces acepta un valor de la enumeración. Ahora aquí es donde se vuelve complicado: tienes un valor, pero los atributos decoran el campo que contiene el valor .

En realidad, necesita reflexionar sobre los campos de la enumeración y verificar el valor de cada uno con respecto al valor que se le ha otorgado (los resultados se deben almacenar en caché para el rendimiento):

foreach(var field in typeof(T).GetFields()) { T fieldValue; try { fieldValue = (T) field.GetRawConstantValue(); } catch(InvalidOperationException) { // For some reason, one of the fields returned is {Int32 value__}, // which throws an InvalidOperationException if you try and retrieve // its constant value. // // I am unsure how to check for this state before // attempting GetRawConstantValue(). continue; } if(fieldValue == value) { var attribute = LMIGHelper.GetAttribute(field, typeof(DescriptionAttribute)) as DescriptionAttribute; return attribute == null ? value.ToString() : attribute.Description; } }

Editar abordando la pregunta de seguimiento

Al método FillComboFromEnum le falta el parámetro de tipo para la enumeración. Prueba esto:

public static void FillComboFromEnum<T>(ComboBox Cbo, BindingList<KeyValuePair<T, string>> List) where T : struct

Observe que restringí el tipo para ser una estructura. No es una restricción de enumeración completa, pero está más cerca que nada.


Eche un vistazo a este artículo . Puede hacerlo utilizando System.ComponentModel.DescriptionAttribute o creando su propio atributo:

/// <summary> /// Provides a description for an enumerated type. /// </summary> [AttributeUsage(AttributeTargets.Enum | AttributeTargets.Field, AllowMultiple = false)] public sealed class EnumDescriptionAttribute : Attribute { private string description; /// <summary> /// Gets the description stored in this attribute. /// </summary> /// <value>The description stored in the attribute.</value> public string Description { get { return this.description; } } /// <summary> /// Initializes a new instance of the /// <see cref="EnumDescriptionAttribute"/> class. /// </summary> /// <param name="description">The description to store in this attribute. /// </param> public EnumDescriptionAttribute(string description) : base() { this.description = description; } }

Luego necesita decorar los valores enum con este nuevo atributo:

public enum SimpleEnum { [EnumDescription("Today")] Today, [EnumDescription("Last 7 days")] Last7, [EnumDescription("Last 14 days")] Last14, [EnumDescription("Last 30 days")] Last30, [EnumDescription("All")] All }

Toda la "magia" tiene lugar en los siguientes métodos de extensión:

/// <summary> /// Provides a static utility object of methods and properties to interact /// with enumerated types. /// </summary> public static class EnumHelper { /// <summary> /// Gets the <see cref="DescriptionAttribute" /> of an <see cref="Enum" /> /// type value. /// </summary> /// <param name="value">The <see cref="Enum" /> type value.</param> /// <returns>A string containing the text of the /// <see cref="DescriptionAttribute"/>.</returns> public static string GetDescription(this Enum value) { if (value == null) { throw new ArgumentNullException("value"); } string description = value.ToString(); FieldInfo fieldInfo = value.GetType().GetField(description); EnumDescriptionAttribute[] attributes = (EnumDescriptionAttribute[]) fieldInfo.GetCustomAttributes(typeof(EnumDescriptionAttribute), false); if (attributes != null && attributes.Length > 0) { description = attributes[0].Description; } return description; } /// <summary> /// Converts the <see cref="Enum" /> type to an <see cref="IList" /> /// compatible object. /// </summary> /// <param name="type">The <see cref="Enum"/> type.</param> /// <returns>An <see cref="IList"/> containing the enumerated /// type value and description.</returns> public static IList ToList(this Type type) { if (type == null) { throw new ArgumentNullException("type"); } ArrayList list = new ArrayList(); Array enumValues = Enum.GetValues(type); foreach (Enum value in enumValues) { list.Add(new KeyValuePair<Enum, string>(value, GetDescription(value))); } return list; } }

Finalmente, puede simplemente vincular el cuadro combinado:

combo.DataSource = typeof(SimpleEnum).ToList();