別のフォルダーからパーシャル(ユーザーコントロール)をビューにレンダリングさせるにはどうすればよいですか?プレビュー3では、完全なパスでRenderUserControlを呼び出していましたが、プレビュー5にアップグレードすると、これはもう不可能になります。代わりにRenderPartialメソッドを取得しましたが、探している機能が提供されていません。
ビューへのパスとファイル拡張子を含めるだけです。
かみそり:
@Html.Partial("~/Views/AnotherFolder/Messages.cshtml", ViewData.Model.Successes)
ASP.NETエンジン:
<% Html.RenderPartial("~/Views/AnotherFolder/Messages.ascx", ViewData.Model.Successes); %>
それがあなたの問題でない場合、RenderUserControlで動作するために使用したコードを含めてください。
私の場合、私はMvcMailer(https://github.com/smsohan/MvcMailer)を使用しており、「共有」ではない別のフォルダーから部分ビューにアクセスしたかったのです。上記のソリューションは機能しませんでしたが、相対パスを使用すると機能しました。
@Html.Partial("../MyViewFolder/Partials/_PartialView", Model.MyObject)
この他のパスを頻繁に使用している場合は、常にパスを指定することなく、これを永続的に修正できます。デフォルトでは、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());
Views/AccountフォルダーにあるmyPartial.ascxというユーザーコントロールの場合、次のように記述します。
<%Html.RenderPartial("~/Views/Account/myPartial.ascx");%>
うまく機能しているように見える回避策を作成しました。アクション名の検索、ビューの検索などのために別のコントローラーのコンテキストに切り替える必要があることがわかりました。これを実装するために、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
ルートコンポーネントを変更しないようにする必要がある場合、これに望ましくない副作用が生じる可能性がありますが、これまでのコードでは、このアプローチにマイナス面はないようです。
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);
}
これを試してみてください
~/Views/Shared/parts/UMFview.ascx
コードの前に~/Views/
を配置します