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 = "" })