我有两种互相矛盾的行动方式。基本上,我希望能够使用两种不同的路径来获得相同的视图,通过项目的 ID 或者项目的名称和它的父视图(项目可以在不同的父视图中使用相同的名称)。可以使用搜索词来筛选列表。
比如说..。
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
参数可以是 null,但是我不知道解决这个问题的最佳方法。我的设计一开始就很糟糕吗?我已经考虑过扩展 Method #1
的签名以包含搜索参数,并将 Method #2
中的逻辑移动到一个它们都会调用的私有方法,但我不认为这会实际解决模棱两可的问题。
如果你能帮忙,我将不胜感激。
实际解 (基于 Levi 的答案)
我添加了以下类..。
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) { ... }