Действия контроллера 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);
}
}