UrlHelper.Action включает нежелательные дополнительные параметры
У меня есть метод в контроллере ApplicationsController
, в котором мне нужно получить базовый URL-адрес для метода действия:
public ActionResult MyAction(string id)
{
var url = Url.Action("MyAction", "Applications");
...
}
Проблема заключается в том, что это включает string id
из текущих данных маршрута, когда мне нужен URL без (URL-адрес используется для извлечения контента из CMS при поиске по URL-адресу).
Я пробовал передавать null
и new { }
в качестве параметра routeValues
безрезультатно.
Соответствующий маршрут выглядит следующим образом (прежде всего другие маршруты):
routes.MapLowercaseRoute(
name: "Applications",
url: "applications/{action}/{id}",
defaults: new { controller = "Applications",
action = "Index", id = UrlParameter.Optional });
Я видел пару других вопросов, которые касаются этого, но ни у кого из них, похоже, нет жизнеспособного решения. В настоящее время я прибегаю к жесткому кодированию пути в контроллере; однако, я хотел бы иметь возможность абстрагировать это в фильтр действий, поэтому мне нужно иметь возможность генерировать URL-адрес.
Существует ли чистый/обычный способ предотвратить это поведение?
Ответы
Ответ 1
Закончилось обходить это с помощью другого подхода. Единственный способ, которым я мог придумать, чтобы предотвратить добавление произвольно названных значений маршрута в сгенерированный URL, заключался в том, чтобы временно удалить их из RouteData
при вызове Url.Action
. Я написал несколько методов расширения, чтобы облегчить это:
public static string NonContextualAction(this UrlHelper helper, string action)
{
return helper.NonContextualAction(action,
helper.RequestContext.RouteData.Values["controller"].ToString());
}
public static string NonContextualAction(this UrlHelper helper, string action,
string controller)
{
var routeValues = helper.RequestContext.RouteData.Values;
var routeValueKeys = routeValues.Keys.Where(o => o != "controller"
&& o != "action").ToList();
// Temporarily remove routevalues
var oldRouteValues = new Dictionary<string, object>();
foreach (var key in routeValueKeys)
{
oldRouteValues[key] = routeValues[key];
routeValues.Remove(key);
}
// Generate URL
string url = helper.Action(routeValues["Action"].ToString(),
routeValues["Controller"].ToString());
// Reinsert routevalues
foreach (var kvp in oldRouteValues)
{
routeValues.Add(kvp.Key, kvp.Value);
}
return url;
}
Это позволяет мне сделать это в фильтре действий, где я не обязательно буду знать, что такое имена параметров для действия (и поэтому не может просто передать анонимный объект, как в других ответах).
Все еще очень интересно узнать, есть ли у кого-то более элегантное решение.
Ответ 2
Хорошо, я вижу проблему. Это что-то называемое "Повторное использование переменной сегмента". При создании маршрутов исходящих URL-адресов и поиске значений для каждой из переменных сегмента в шаблоне URL-маршрутов система маршрутизации будет искать значения из текущего запроса. Это поведение, которое смущает многих программистов и может привести к длительной сессии отладки. Система маршрутизации стремится выполнить сопоставление с маршрутом в той мере, в какой она будет повторно использовать значения переменных сегмента из входящего URL-адреса. Поэтому я думаю, что вам нужно переопределить значение, как предложил Жюльен:
var url = Url.Action("MyAction", "Applications", new { id = "" })
Ответ 3
Используйте значение null или empty для id
, чтобы предотвратить использование Url.Action
текущего:
var url = Url.Action("MyAction", "Applications", new { id = "" })