これが私のシナリオです。たとえば、検索条件に基づいて車のリストを返す必要があるとします。出力は同じになるので、結果を表示するために単一のビューが必要ですが、そこに到達するにはいくつかの方法が必要です。たとえば、年ごとに検索するテキストボックス付きのフォームがあるとします。すべての赤いトヨタ車のハイパーリンクを含む別のページがあるかもしれません。同じビューとコントローラーでこれらの複数のシナリオを処理するにはどうすればよいですか。私のジレンマは、検索にいくつかのオプション(年、メーカー、モデルなど)が含まれる可能性があることですが、それらをどこに配置するかわかりません。
これに対する最善のアプローチは何ですか?ルーティングでパラメータを定義する必要がありますか、それともクエリ文字列などを使用する必要がありますか?
オプション1
もちろん、/ car/search /?vendor = Toyota&color = Red&model = Corolaの方法はいつでも選択できます。それはあなたにとって良いことだと思います。
routes.MapRoute(
"CarSearch",
"car/search",
new { controller = "car", action = "search" }
);
この場合、実行中のRequest.Paramsからパラメーターを取得できます。
オプション2
または、ルーティングテーブルでパラメータを定義することもできますが、パラメータの順序が重要であるため、考えられるすべての組み合わせに対して一連のルールを作成する必要があります。たとえば、次のようになります。
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を使用する場合はこれが当てはまります。
しかし、それは簡単な方法でした:)
オプション
難しいですが、最もエレガントな方法は、独自の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は/ car/search/model/Corola/color/red/vendor/Toyotaのきれいな形式のURLを提供しないため、リンク生成ロジックを作成することも忘れないでください。カスタムリンクジェネレータを作成します。
したがって、本当に柔軟なルーティングが必要な場合は、この難しい方法を使用することをお勧めします:)
これらの線に沿った何かがあなたが求めていることをするはずです。 2つの異なるアクションメソッドがあることに注意してください。ただし、どちらも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));
}
}
コントローラーの各メソッド(アクション)は異なるパラメーターを取りますが、検索結果の同じコレクションを作成します。次に、それぞれが
return View("SearchResult", searchResultCollection);
それらはすべて同じビュー、SearchResult.aspxを使用します。