Como redireciono para a ação anterior no ASP.NET MVC?


122

Vamos supor que eu tenho algumas páginas

  • some.web/articles/details/5
  • some.web/users/info/bob
  • some.web/foo/bar/7

que pode chamar um controlador utilitário comum como

locale/change/es ou authorization/login

Como faço para obter esses métodos ( change, login) para redirecionar para as ações anteriores ( details, info, bar), enquanto passando os parâmetros anteriores a eles ( 5, bob, 7)?

Em resumo: como redireciono para a página que acabei de visitar após executar uma ação em outro controlador?

Respostas:


156

experimentar:

public ActionResult MyNextAction()
{
    return Redirect(Request.UrlReferrer.ToString());
}

Como alternativa, tocando no que Darin disse, tente o seguinte:

public ActionResult MyFirstAction()
{
    return RedirectToAction("MyNextAction",
        new { r = Request.Url.ToString() });
}

então:

public ActionResult MyNextAction()
{
    return Redirect(Request.QueryString["r"]);
}

Apenas uma sugestão: você pode usar "Redirecionar" explicitamente, é mais difícil testar o seu controlador de unidade. É melhor usar um "RedirectToAction".
Syd

Eu recomendo usar Request.Url.AbsolutePath.ToString()como o AccountController's LogOnmétodo contém verificações para início da URL com '/', etc.
fulvio

1
@gotnull Request.Url.AbsolutePath será redirecionado para a mesma ação. Qual não é a saída desejada. Temos que voltar para a segunda última ação. Para isso, poderíamos escrever: return Redirect (ControllerContext.HttpContext.Request.UrlReferrer.ToString ());
Rahatur #

22
@nathanridley: Isso não funciona para solicitações POST. Digamos que o usuário esteja ativo GET Indexentão GET Edit. O URL de referência é então, Indexmas quando o usuário faz POST Edito referenciador agora é Editda solicitação GET anterior. Como posso garantir que POST Editsabe o URL ao qual o usuário se referiu GET Edit?
one.beat.consumer

UrlReferrer é NULL quando eu estava em alguma página e quero ver o que sei, obtive um erro digitando o URL na barra de endereços. Por que quando eu insiro um URL na barra de endereços, ele não pode determinar o UrlReferrer?
Qmaster

46

Se você deseja redirecionar a partir de um botão na tela, você pode usar:

@Html.ActionLink("Back to previous page", null, null, null, new { href = Request.UrlReferrer})

28

Se você não está preocupado com o teste de unidade, basta escrever:

return Redirect(ControllerContext.HttpContext.Request.UrlReferrer.ToString());

9

Uma sugestão de como fazer isso, de modo que:

  1. o URL de retorno sobrevive à solicitação POST de um formulário (e quaisquer validações com falha)
  2. o URL de retorno é determinado a partir do URL de referência inicial
  3. sem usar TempData [] ou outro estado do servidor
  4. lida com a navegação direta para a ação (fornecendo um redirecionamento padrão)

.

public ActionResult Create(string returnUrl)
{
    // If no return url supplied, use referrer url.
    // Protect against endless loop by checking for empty referrer.
    if (String.IsNullOrEmpty(returnUrl)
        && Request.UrlReferrer != null
        && Request.UrlReferrer.ToString().Length > 0)
    {
        return RedirectToAction("Create",
            new { returnUrl = Request.UrlReferrer.ToString() });
    }

    // Do stuff...
    MyEntity entity = GetNewEntity();

    return View(entity);
}

[AcceptVerbs(HttpVerbs.Post)]
public ActionResult Create(MyEntity entity, string returnUrl)
{
    try
    {
        // TODO: add create logic here

        // If redirect supplied, then do it, otherwise use a default
        if (!String.IsNullOrEmpty(returnUrl))
            return Redirect(returnUrl);
        else
            return RedirectToAction("Index");
    }
    catch
    {
        return View();  // Reshow this view, with errors
    }
}

Você pode usar o redirecionamento dentro da visualização da seguinte maneira:

<% if (!String.IsNullOrEmpty(Request.QueryString["returnUrl"])) %>
<% { %>
    <a href="<%= Request.QueryString["returnUrl"] %>">Return</a>
<% } %>

9

No Mvc usando html simples na View View com java script onclick

<input type="button" value="GO BACK" class="btn btn-primary" 
onclick="location.href='@Request.UrlReferrer'" />

Isso funciona muito bem. a esperança ajuda alguém.

@JuanPieterse já respondeu usando, por @Html.ActionLinkisso, se possível, alguém pode comentar ou responder usando@Url.Action


6

Passe um parâmetro returnUrl (url codificado) para as ações de alteração e login e redirecione dentro para esse returnUrl especificado. Sua ação de login pode ser algo como isto:

public ActionResult Login(string returnUrl) 
{
    // Do something...
    return Redirect(returnUrl);
}

5

Estou usando o .Net Core 2 MVC, e este funcionou para mim, no uso do controlador HttpContext.Request.Headers["Referer"];


1

Você pode retornar à página anterior usando a ViewBag.ReturnUrlpropriedade


1

Para construir dinamicamente o returnUrl em qualquer View, tente o seguinte:

@{
    var formCollection =
        new FormCollection
            {
                new FormCollection(Request.Form),
                new FormCollection(Request.QueryString)
            };

    var parameters = new RouteValueDictionary();

    formCollection.AllKeys
        .Select(k => new KeyValuePair<string, string>(k, formCollection[k])).ToList()
        .ForEach(p => parameters.Add(p.Key, p.Value));
}

<!-- Option #1 -->
@Html.ActionLink("Option #1", "Action", "Controller", parameters, null)

<!-- Option #2 -->
<a href="/Controller/Action/@object.ID?returnUrl=@Url.Action(ViewContext.RouteData.Values["action"].ToString(), ViewContext.RouteData.Values["controller"].ToString(), parameters)">Option #2</a>

<!-- Option #3 -->
<a href="@Url.Action("Action", "Controller", new { object.ID, returnUrl = Url.Action(ViewContext.RouteData.Values["action"].ToString(), ViewContext.RouteData.Values["controller"].ToString(), parameters) }, null)">Option #3</a>

Isso também funciona em páginas de layout, vistas parciais e auxiliares de HTML

Relacionado: URL de retorno dinâmico do MVC3 (o mesmo, mas de qualquer controlador / ação)


0

Para o ASP.NET Core, você pode usar o atributo asp-route- *:

<form asp-action="Login" asp-route-previous="@Model.ReturnUrl">

Outro exemplo de detalhes: imagine que você tem um controlador de veículo com ações

Índice

Detalhes

Editar

e você pode editar qualquer veículo a partir do Índice ou a partir de Detalhes, portanto, se você clicou em editar no índice, deve retornar ao índice após a edição e se clicar em editar nos detalhes, deve retornar aos detalhes após a edição.

//In your viewmodel add the ReturnUrl Property
public class VehicleViewModel
{
     ..............
     ..............
     public string ReturnUrl {get;set;}
}



Details.cshtml
<a asp-action="Edit" asp-route-previous="Details" asp-route-id="@Model.CarId">Edit</a>

Index.cshtml
<a asp-action="Edit" asp-route-previous="Index" asp-route-id="@item.CarId">Edit</a>

Edit.cshtml
<form asp-action="Edit" asp-route-previous="@Model.ReturnUrl" class="form-horizontal">
        <div class="box-footer">
            <a asp-action="@Model.ReturnUrl" class="btn btn-default">Back to List</a>
            <button type="submit" value="Save" class="btn btn-warning pull-right">Save</button>
        </div>
    </form>

No seu controlador:

// GET: Vehicle/Edit/5
    public ActionResult Edit(int id,string previous)
    {
            var model = this.UnitOfWork.CarsRepository.GetAllByCarId(id).FirstOrDefault();
            var viewModel = this.Mapper.Map<VehicleViewModel>(model);//if you using automapper
    //or by this code if you are not use automapper
    var viewModel = new VehicleViewModel();

    if (!string.IsNullOrWhiteSpace(previous)
                viewModel.ReturnUrl = previous;
            else
                viewModel.ReturnUrl = "Index";
            return View(viewModel);
        }



[HttpPost]
    public IActionResult Edit(VehicleViewModel model, string previous)
    {
            if (!string.IsNullOrWhiteSpace(previous))
                model.ReturnUrl = previous;
            else
                model.ReturnUrl = "Index";
            ............. 
            .............
            return RedirectToAction(model.ReturnUrl);
    }
Ao utilizar nosso site, você reconhece que leu e compreendeu nossa Política de Cookies e nossa Política de Privacidade.
Licensed under cc by-sa 3.0 with attribution required.