ASP.NET MVC настраиваемая маршрутизация для поиска - PullRequest
31 голосов
/ 19 декабря 2008

Вот мой сценарий. Для примера скажем, что мне нужно вернуть список автомобилей на основе критериев поиска. Я хотел бы иметь один вид для отображения результатов, так как вывод будет одинаковым, но мне нужно несколько способов добраться туда. Например, у меня может быть форма с текстовым полем для поиска по годам. У меня может быть другая отдельная страница, которая содержит гиперссылку на все красные автомобили Toyota. Как мне справиться с этими несколькими сценариями в одном представлении и контроллере. Моя дилемма в том, что поиск может содержать несколько вариантов: год, марку, модель и т. Д., Но я не знаю, где их разместить.

Каков наилучший подход для этого? Должен ли я определить параметры в маршрутизации или использовать строки запроса и т. Д.?

Ответы [ 3 ]

78 голосов
/ 19 декабря 2008

Вариант 1

Конечно, вы всегда можете выбрать способ / car / search /? Vendor = Toyota & color = Red & model = Corola, и я думаю, что это будет хорошо для вас.

routes.MapRoute(
    "CarSearch",
    "car/search",
    new { controller = "car", action = "search" }
);

В этом случае вы можете получить параметры из Request.Params в действии.

Вариант 2

Или вы можете определить параметры в таблице маршрутизации, но AFAIK потребуется создать набор правил для всех возможных комбинаций, потому что порядок параметров имеет значение, например:

        routes.MapRoute(
            "CarSearch1",
            "car/search/vendor/{vendor}/color/{color}/model/{model}",
            new {controller = "car", action = "search"}
        );

        routes.MapRoute(
            "CarSearch2",
            "car/search/color/{color}/vendor/{vendor}/model/{model}",
            new {controller = "car", action = "search"}
        );

        routes.MapRoute(
            "CarSearch3",
            "car/search/model/{model}/color/{color}/vendor/{vendor}",
            new {controller = "car", action = "search"}
        );

... и так далее. Это правда, если вы используете стандартный MvcRouteHandler.

Но это был легкий путь:)

Вариант 3

Сложный, но, на мой взгляд, самый элегантный способ - создать собственную реализацию IRouteHandler - она ​​даст вам гораздо большую гибкость в порядке следования параметров. Но опять же, это трудный путь, не соглашайтесь, если у вас есть простое приложение. Так, например, как это сделать (очень простой пример):

Добавить новый маршрут в список маршрутов:

routes.Add
    (
        new Route
            (
                "car/search/{*data}",
                new RouteValueDictionary(new {controller = "car", action = "search", data = ""}),
                new MyRouteHandler()
            )
    );

Добавьте классы, которые будут настраивать стандартную цепочку обработки запросов:

class MyRouteHandler : IRouteHandler
{
    public IHttpHandler GetHttpHandler(RequestContext requestContext)
    {
        return new MyHttpHandler(requestContext);
    }
}

class MyHttpHandler : MvcHandler
{
    public MyHttpHandler(RequestContext requestContext) : base(requestContext)
    {
    }

    protected override void ProcessRequest(HttpContextBase httpContext)
    {
        IController controller = new CarController();
        (controller as Controller).ActionInvoker = new MyActionInvoker();
        controller.Execute(RequestContext);
    }
}

class MyActionInvoker : ControllerActionInvoker
{
    protected override ActionResult InvokeActionMethod(MethodInfo methodInfo, IDictionary<string, object> parameters)
    {
        // if form of model/{model}/color/{color}/vendor/{vendor}
        var data = ControllerContext.RouteData.GetRequiredString("data");
        var tokens = data.Split('/');

        var searchParams = new Dictionary<string, string>();
        for (var i = 0; i < tokens.Length; i++)
        {
            searchParams.Add(tokens[i], tokens[++i]);
        }

        parameters["searchParams"] = searchParams;

        return base.InvokeActionMethod(methodInfo, parameters);
    }
}

В контроллере:

public ActionResult Search(IDictionary<string, string> searchParams)
{
    ViewData.Add
        (
            // output 'model = Corola, color = red, vendor = Toyota'
            "SearchParams",
            string.Join(", ", searchParams.Select(pair => pair.Key + " = " + pair.Value).ToArray())
        );
    return View();
}

И будет работать с любым порядком параметров поиска:

/car/search/vendor/Toyota/color/red/model/Corola
/car/search/color/red/model/Corola/vendor/Toyota
/car/search/model/Corola/color/red/vendor/Toyota

Но также не забудьте создать логику генерации ссылок, потому что Html.ActionLink и Html.RenderLink не будут давать вам URL в красивой форме / car / search / model / Corola / color / red / vendor / Toyota, поэтому нужно будет сделать собственный генератор ссылок.

Итак, если вам нужна действительно гибкая маршрутизация - вам лучше пойти по этому сложному пути:)

0 голосов
/ 19 декабря 2008

Что-то в этом роде должно делать то, что вы ищете. Обратите внимание, что есть два разных метода действия, но оба они возвращают вызов DisplayResults () - так что они в конечном итоге используют одно и то же представление с разными ViewData.

public class SearchController : Controller {

    public ActionResult ByColor(Color[] colors) {
         List<Car> results = carRepository.FindByColor(colors);
         return(DisplayResults(result));
    }

    public ActionResult ByMake(string make) {
         List<Car> results = carRepository.FindByMake(make);
         return(DisplayResults(results));
    }

    private ActionResult DisplayResults(IList<Car> results) {

        // Here we explicitly return the view /Views/Search/Results.aspx
        // by specifying the view name in the call to View();
        return(View("Results", results));
    }
}
0 голосов
/ 19 декабря 2008

Каждый метод (действие) на контроллере принимает разные параметры, но создает одну и ту же коллекцию результатов поиска. Тогда каждый будет

return View("SearchResult", searchResultCollection);

Все они используют одно и то же представление, SearchResult.aspx.

...