Есть ли способ, чтобы DefaultModelBinder игнорировал пустые элементы при привязке к списку ‹Enum›

У меня есть сценарий, в котором я хотел бы изменить поведение DefaultModelBinder в том, как он привязывается к списку перечислений.

У меня есть перечисление ...

public enum MyEnum { FirstVal, SecondVal, ThirdVal }

и класс для модели ...

public class MyModel
{
    public List<MyEnum> MyEnums { get; set; }
}

а тело POST ...

MyEnums=&MyEnums=ThirdVal

В настоящее время после привязки модели свойство MyEnums будет содержать ...

[0] = FirstVal
[1] = ThirdVal

Был ли способ указать связывателю модели игнорировать пустое значение в опубликованных данных, чтобы свойство MyEnums могло выглядеть следующим образом?

[0] = ThirdVal

person JeremyWeir    schedule 25.02.2010    source источник


Ответы (1)


Вы можете написать собственный связыватель модели для MyModel:

public class MyModelModelBinder : DefaultModelBinder
{
    protected override void SetProperty(
        ControllerContext controllerContext, 
        ModelBindingContext bindingContext, 
        PropertyDescriptor propertyDescriptor, 
        object value)
    {
        if (value is ICollection<MyEnum>)
        {
            var myEnums = controllerContext.HttpContext.Request[propertyDescriptor.Name];
            if (!string.IsNullOrEmpty(myEnums))
            {
                var tokens = myEnums.Split(new [] { ',' }, StringSplitOptions.RemoveEmptyEntries);
                value = tokens.Select(x => (MyEnum)Enum.Parse(typeof(MyEnum), x)).ToList();
            }
        }
        base.SetProperty(controllerContext, bindingContext, propertyDescriptor, value);
    }
}

который зарегистрирован в Application_Start:

protected void Application_Start()
{
    AreaRegistration.RegisterAllAreas();
    RegisterRoutes(RouteTable.Routes);
    ModelBinders.Binders.Add(typeof(MyModel), new MyModelModelBinder());
}

ОБНОВИТЬ:

В соответствии с просьбой в разделе комментариев, вот как сделать предыдущее связыватель более универсальным:

protected override void SetProperty(
    ControllerContext controllerContext, 
    ModelBindingContext bindingContext, 
    PropertyDescriptor propertyDescriptor, 
    object value)
{
    var collection = value as IList;
    if (collection != null && collection.GetType().IsGenericType)
    {
        var genericArgument = collection
            .GetType()
            .GetGenericArguments()
            .Where(t => t.IsEnum)
            .FirstOrDefault();

        if (genericArgument != null)
        {
            collection.Clear();
            var enumValues = controllerContext.HttpContext
                .Request[propertyDescriptor.Name];
            var tokens = enumValues.Split(
                new[] { ',' }, StringSplitOptions.RemoveEmptyEntries);
            foreach (var token in tokens)
            {
                collection.Add(Enum.Parse(genericArgument, token));
            }
        }
    }
    base.SetProperty(controllerContext, bindingContext, propertyDescriptor, value);
}
person Darin Dimitrov    schedule 26.02.2010
comment
Мне это нравится, спасибо. Вы знаете, как это можно сделать для любого типа Enum? - person JeremyWeir; 26.02.2010