У меня есть два метода действий, которые противоречат друг другу. По сути, я хочу иметь возможность получить одно и то же представление с использованием двух разных маршрутов, либо по идентификатору элемента, либо по имени элемента и его родителя (элементы могут иметь одно и то же имя у разных родителей). Поисковый термин может использоваться для фильтрации списка.
Например...
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) { ... }
return ValueNames.All(v => controllerContext.RequestContext.RouteData.Values.ContainsKey(v));
contains = ...
contains = controllerContext.RequestContext.RouteData.Values.ContainsKey(value) || controllerContext.RequestContext.HttpContext.Request.Params.AllKeys.Contains(value);
ActionResult DoSomething(Person p)
где Person
имеет различные простые свойства, такие как Name
, и запросы к нему делаются с именами свойств напрямую (например, /dosomething/?name=joe+someone&other=properties
).
controllerContext.HttpContext.Request[value] != null
вместо controllerContext.RequestContext.RouteData.Values.ContainsKey(value)
; но хорошая работа тем не менее.