Asp.net MVCでは、URL構造は次のようになります
http://example.com/ {controller}/{action}/{id}
各「コントローラー」、たとえば http://example.com/blog には、BlogControllerがあります。
しかし、URLの{controller}部分は事前に決定されていませんが、実行時に動的に決定されます。同じコントローラーに何かをマップする「動的コントローラー」を作成し、値に基づいて何を決定するのですか。する?
{action}と同じことですが、URLの{action}部分も動的である場合、このシナリオをプログラムする方法はありますか?
絶対に!カスタムコントローラーが存在しない場合は、DefaultControllerFactory
をオーバーライドしてカスタムコントローラーを見つける必要があります。次に、動的アクション名を処理するためにIActionInvoker
を作成する必要があります。
コントローラファクトリは次のようになります。
public class DynamicControllerFactory : DefaultControllerFactory
{
private readonly IServiceLocator _Locator;
public DynamicControllerFactory(IServiceLocator locator)
{
_Locator = locator;
}
protected override Type GetControllerType(string controllerName)
{
var controllerType = base.GetControllerType(controllerName);
// if a controller wasn't found with a matching name, return our dynamic controller
return controllerType ?? typeof (DynamicController);
}
protected override IController GetControllerInstance(Type controllerType)
{
var controller = base.GetControllerInstance(controllerType) as Controller;
var actionInvoker = _Locator.GetInstance<IActionInvoker>();
if (actionInvoker != null)
{
controller.ActionInvoker = actionInvoker;
}
return controller;
}
}
次に、アクションの呼び出し元は次のようになります。
public class DynamicActionInvoker : ControllerActionInvoker
{
private readonly IServiceLocator _Locator;
public DynamicActionInvoker(IServiceLocator locator)
{
_Locator = locator;
}
protected override ActionDescriptor FindAction(ControllerContext controllerContext,
ControllerDescriptor controllerDescriptor, string actionName)
{
// try to match an existing action name first
var action = base.FindAction(controllerContext, controllerDescriptor, actionName);
if (action != null)
{
return action;
}
// @ray247 The remainder of this you'd probably write on your own...
var actionFinders = _Locator.GetAllInstances<IFindAction>();
if (actionFinders == null)
{
return null;
}
return actionFinders
.Select(f => f.FindAction(controllerContext, controllerDescriptor, actionName))
.Where(d => d != null)
.FirstOrDefault();
}
}
このコードをもっとたくさん見ることができます ここ 。これは、完全に動的なMVCパイプラインを作成するための私と同僚による古い最初のドラフトの試みです。あなたはそれを参照として自由に使用し、あなたが望むものをコピーすることができます。
編集
私は、そのコードが何をするかについての背景を含めるべきだと考えました。ドメインモデルを中心にMVCレイヤーを動的に構築しようとしていました。したがって、ドメインにProductクラスが含まれている場合は、products\alls
に移動して、すべての製品のリストを表示できます。商品を追加したい場合は、product\add
に移動します。 product\edit\1
に移動して製品を編集できます。エンティティのプロパティを編集できるようにするなどの方法も試しました。したがって、product\editprice\1?value=42
は製品#1の価格プロパティを42に設定します(私のパスは少しずれている可能性があり、正確な構文はもう思い出せません)。これがお役に立てば幸いです。
もう少し考えてみると、動的アクション名を処理するための他の答えよりも少し簡単な方法があるかもしれません。デフォルトのコントローラーファクトリをオーバーライドする必要があります。次のようにルートを定義できると思います。
routes.MapRoute("Dynamic", "{controller}/{command}/{id}", new { action = "ProcessCommand" });
次に、デフォルト/ダイナミックコントローラーで
public ActionResult ProcessCommand(string command, int id)
{
switch(command)
{
// whatever.
}
}
独自のIControllerFactory
を記述して(またはDefaultControllerFactory
から派生して)、それをControllerBuilder
に登録する必要があります。