Действия контроллера ASP.NET MVC с пользовательским преобразованием параметров?

Я хочу настроить маршрут ASP.NET MVC, который выглядит следующим образом:

routes.MapRoute(
  "Default", // Route name
  "{controller}/{action}/{idl}", // URL with parameters
  new { controller = "Home", action = "Index", idl = UrlParameter.Optional } // Parameter defaults
);

Это маршруты запросов, которые выглядят так...

Example/GetItems/1,2,3

... для моего действия контроллера:

public class ExampleController : Controller
{
    public ActionResult GetItems(List<int> id_list)
    {
        return View();
    }
}

Вопрос в том, что я настроил для преобразования idl Параметр URL из string в List<int> и вызвать соответствующее действие контроллера?

Я видел похожий вопрос здесь, который использовал OnActionExecuting для предварительной обработки строки, но не изменил тип. Я не думаю, что это будет работать для меня здесь, потому что, когда я переопределить OnActionExecuting в моем контроллере и осмотреть ActionExecutingContext параметр, я вижу, что ActionParameters В словаре уже есть idl ключ с нулевым значением - предположительно, попытка приведения из строки в List<int>... это та часть маршрутизации, которую я хочу контролировать.

Это возможно?

2 ответа

Решение

Хорошей версией является реализация собственной модели Binder. Вы можете найти образец здесь

Я пытаюсь дать вам представление:

public class MyListBinder : IModelBinder
{   
     public object BindModel(ControllerContext controllerContext, ModelBindingContext bindingContext)
     {   
        string integers = controllerContext.RouteData.Values["idl"] as string;
        string [] stringArray = integers.Split(',');
        var list = new List<int>();
        foreach (string s in stringArray)
        {
           list.Add(int.Parse(s));
        }
        return list;  
     }  
}


public ActionResult GetItems([ModelBinder(typeof(MyListBinder))]List<int> id_list) 
{ 
    return View(); 
} 

Как говорит Слфан, связка пользовательских моделей - это путь. Вот еще один подход из моего блога, который является общим и поддерживает несколько типов данных. Он также элегантно использует стандартную реализацию привязки модели:

public class CommaSeparatedValuesModelBinder : DefaultModelBinder
{
    private static readonly MethodInfo ToArrayMethod = typeof(Enumerable).GetMethod("ToArray");

    protected override object GetPropertyValue(ControllerContext controllerContext, ModelBindingContext bindingContext, System.ComponentModel.PropertyDescriptor propertyDescriptor, IModelBinder propertyBinder)
    {
        if (propertyDescriptor.PropertyType.GetInterface(typeof(IEnumerable).Name) != null)
        {
            var actualValue = bindingContext.ValueProvider.GetValue(propertyDescriptor.Name);

            if (actualValue != null && !String.IsNullOrWhiteSpace(actualValue.AttemptedValue) && actualValue.AttemptedValue.Contains(","))
            {
                var valueType = propertyDescriptor.PropertyType.GetElementType() ?? propertyDescriptor.PropertyType.GetGenericArguments().FirstOrDefault();

                if (valueType != null && valueType.GetInterface(typeof(IConvertible).Name) != null)
                {
                    var list = (IList)Activator.CreateInstance(typeof(List<>).MakeGenericType(valueType));

                    foreach (var splitValue in actualValue.AttemptedValue.Split(new[] { ',' }))
                    {
                        list.Add(Convert.ChangeType(splitValue, valueType));
                    }

                    if (propertyDescriptor.PropertyType.IsArray)
                    {
                        return ToArrayMethod.MakeGenericMethod(valueType).Invoke(this, new[] { list });
                    }
                    else
                    {
                        return list;
                    }
                }
            }
        }

        return base.GetPropertyValue(controllerContext, bindingContext, propertyDescriptor, propertyBinder);
    }
}
Другие вопросы по тегам