web-dev-qa-db-ja.com

別のフォルダーから部分的にレンダリング(共有されない)

別のフォルダーからパーシャル(ユーザーコントロール)をビューにレンダリングさせるにはどうすればよいですか?プレビュー3では、完全なパスでRenderUserControlを呼び出していましたが、プレビュー5にアップグレードすると、これはもう不可能になります。代わりにRenderPartialメソッドを取得しましたが、探している機能が提供されていません。

235
Boris Callens

ビューへのパスとファイル拡張子を含めるだけです。

かみそり:

@Html.Partial("~/Views/AnotherFolder/Messages.cshtml", ViewData.Model.Successes)

ASP.NETエンジン:

<% Html.RenderPartial("~/Views/AnotherFolder/Messages.ascx", ViewData.Model.Successes); %>

それがあなたの問題でない場合、RenderUserControlで動作するために使用したコードを含めてください。

419
Elijah Manor

私の場合、私はMvcMailer(https://github.com/smsohan/MvcMailer)を使用しており、「共有」ではない別のフォルダーから部分ビューにアクセスしたかったのです。上記のソリューションは機能しませんでしたが、相対パスを使用すると機能しました。

@Html.Partial("../MyViewFolder/Partials/_PartialView", Model.MyObject)
25
Aaron Sherman

この他のパスを頻繁に使用している場合は、常にパスを指定することなく、これを永続的に修正できます。デフォルトでは、ViewフォルダーとSharedフォルダーの部分ビューを確認しています。ただし、追加したいとします。

Modelsフォルダーにクラスを追加します。

public class NewViewEngine : RazorViewEngine {

   private static readonly string[] NEW_PARTIAL_VIEW_FORMATS = new[] {
      "~/Views/Foo/{0}.cshtml",
      "~/Views/Shared/Bar/{0}.cshtml"
   };

   public NewViewEngine() {
      // Keep existing locations in sync
      base.PartialViewLocationFormats = base.PartialViewLocationFormats.Union(NEW_PARTIAL_VIEW_FORMATS).ToArray();
   }
}

次に、Global.asax.csファイルに次の行を追加します。

ViewEngines.Engines.Add(new NewViewEngine());
25
Paul

Views/AccountフォルダーにあるmyPartial.ascxというユーザーコントロールの場合、次のように記述します。

<%Html.RenderPartial("~/Views/Account/myPartial.ascx");%>
7
Rahatur

うまく機能しているように見える回避策を作成しました。アクション名の検索、ビューの検索などのために別のコントローラーのコンテキストに切り替える必要があることがわかりました。これを実装するために、HtmlHelperの新しい拡張メソッドを作成しました。

public static IDisposable ControllerContextRegion(
    this HtmlHelper html, 
    string controllerName)
{
    return new ControllerContextRegion(html.ViewContext.RouteData, controllerName);
}

ControllerContextRegionは次のように定義されます:

internal class ControllerContextRegion : IDisposable
{
    private readonly RouteData routeData;
    private readonly string previousControllerName;

    public ControllerContextRegion(RouteData routeData, string controllerName)
    {
        this.routeData = routeData;
        this.previousControllerName = routeData.GetRequiredString("controller");
        this.SetControllerName(controllerName);
    }

    public void Dispose()
    {
        this.SetControllerName(this.previousControllerName);
    }

    private void SetControllerName(string controllerName)
    {
        this.routeData.Values["controller"] = controllerName;
    }
}

ビュー内でこれを使用する方法は次のとおりです。

@using (Html.ControllerContextRegion("Foo")) {
    // Html.Action, Html.Partial, etc. now looks things up as though
    // FooController was our controller.
}

コードでcontrollerルートコンポーネントを変更しないようにする必要がある場合、これに望ましくない副作用が生じる可能性がありますが、これまでのコードでは、このアプローチにマイナス面はないようです。

5
Jacob

WebFormsViewEngineのベースであるVirtualPathProviderViewEngineは、パスの前にある「〜」および「/」文字をサポートするため、上記の例が機能するはずです。

サンプルではパス「〜/ Account/myPartial.ascx」を使用していることに気付きましたが、ユーザーコントロールはViews/Accountフォルダーにあることを述べました。やってみました

<%Html.RenderPartial("~/Views/Account/myPartial.ascx");%>

それとも質問のタイプミスですか?

カスタムビューエンジンを作成し、ViewEngineResultを返すメソッドを用意します。この例では、_options.ViewLocationFormatsを上書きしてフォルダーディレクトリを追加します。

public ViewEngineResult FindView(ActionContext context, string viewName, bool isMainPage)
        {
            var controllerName = context.GetNormalizedRouteValue(CONTROLLER_KEY);
            var areaName = context.GetNormalizedRouteValue(AREA_KEY);

            var checkedLocations = new List<string>();
            foreach (var location in _options.ViewLocationFormats)
            {
                var view = string.Format(location, viewName, controllerName);
                if (File.Exists(view))
                {
                    return ViewEngineResult.Found("Default", new View(view, _ViewRendering));
                }
                checkedLocations.Add(view);
            }

            return ViewEngineResult.NotFound(viewName, checkedLocations);
        }

例: https://github.com/AspNetMonsters/pugzor

これを試してみてください

~/Views/Shared/parts/UMFview.ascx

コードの前に~/Views/を配置します

0
mounir