У мене є два способи дії, які суперечать один одному. В основному, я хочу мати змогу дістатися до одного і того ж виду, використовуючи два різні маршрути, або за ідентифікатором елемента, або за іменем елемента, і його батьком (елементи можуть мати однакове ім’я для різних батьків). Пошуковий термін може використовуватися для фільтрації списку.
Наприклад...
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)
; але приємна робота.