У меня есть два метода действий, которые противоречат друг другу. В принципе, я хочу иметь возможность перейти к одному и тому же виду, используя два разных маршрута, либо по идентификатору элемента, либо по названию объекта и его родителям (элементы могут иметь одинаковое имя для разных родителей). Для фильтрации списка можно использовать термин поиска.ASP.NET MVC неоднозначные методы действия
Например ...
Items/{action}/ParentName/ItemName
Items/{action}/1234-4321-1234-4321
Вот мои методы действий (существуют также Remove
методы действия) ...
// Method #1
public ActionResult Assign(string parentName, string itemName) {
// Logic to retrieve item's ID here...
string itemId = ...;
return RedirectToAction("Assign", "Items", new { itemId });
}
// Method #2
public ActionResult Assign(string itemId, string searchTerm, int? page) { ... }
А вот маршруты ...
routes.MapRoute("AssignRemove",
"Items/{action}/{itemId}",
new { controller = "Items" }
);
routes.MapRoute("AssignRemovePretty",
"Items/{action}/{parentName}/{itemName}",
new { controller = "Items" }
);
Я понимаю, почему возникает ошибка, так как параметр page
может быть нулевым, но я не могу найти оптимальный способ его решения. С чего начать? Я думал о расширении подписки Method #1
, чтобы включить параметры поиска и переместить логику в Method #2
на частный метод, который они оба вызывали, но я не верю, что это действительно устранит двусмысленность.
Любая помощь была бы принята с благодарностью.
Фактическое решение (основано на ответ Леви)
Я добавил следующий класс ...
public class RequireRouteValuesAttribute : ActionMethodSelectorAttribute {
public RequireRouteValuesAttribute(string[] valueNames) {
ValueNames = valueNames;
}
public override bool IsValidForRequest(ControllerContext controllerContext, MethodInfo methodInfo) {
bool contains = false;
foreach (var value in ValueNames) {
contains = controllerContext.RequestContext.RouteData.Values.ContainsKey(value);
if (!contains) break;
}
return contains;
}
public string[] ValueNames { get; private set; }
}
А потом украсили методы действий ...
[RequireRouteValues(new[] { "parentName", "itemName" })]
public ActionResult Assign(string parentName, string itemName) { ... }
[RequireRouteValues(new[] { "itemId" })]
public ActionResult Assign(string itemId) { ... }
Спасибо за публикацию фактической реализации. Это помогает людям с подобными проблемами. Как и сегодня. :-P –
Удивительно! Предложение незначительного изменения: (imo действительно полезно) 1) params string [] valueNames, чтобы сделать объявление атрибута более кратким и (предпочтение) 2) заменить тело метода IsValidForRequest на 'return ValueNames.All (v => controllerContext.RequestContext.RouteData. Values.ContainsKey (v)); ' –
Привет, Джон, я думаю, что я ничего не беру, потому что где параметры запроса в RouteData? – fravelgue