MVC エリアごとに 1 つのプロジェクトに分割された大規模な MVC 4 サイトがあります。RazorGenerator を使用して、展開用にすべてのビューをプロジェクト アセンブリにプリコンパイルしPrecompiledMvcEngine
、ViewEngine
.
Shared
別のアセンブリからのビューを共有したい新しいエリアを作成しましたがInvalidOperationException
、部分ビューを見つけようとすると、DisplayTemplates または EditorTemplates のいずれも見つからないようです。
この質問で説明されている問題に似ていると思います。
RazorGenerator App_Start の私のコードは次のようなものです:
var assemblies = new List<Tuple<string, Assembly>>()
{
Tuple.Create("Areas/Area1", typeof(ABC.Project1.AreaRegistration).Assembly),
Tuple.Create("Areas/Area2", typeof(ABC.Project2.AreaRegistration).Assembly),
};
// Get rid of the default view engine
ViewEngines.Engines.Clear();
foreach ( var assembly in assemblies )
{
var engine = new PrecompiledMvcEngine(assembly.Item2, assembly.Item1) {
UsePhysicalViewsIfNewer = HttpContext.Current.Request.IsLocal
};
// Allow sharing of Area1 Shares views with Area2
if (assembly.Item1 == "Areas/Area2")
{
var sharedPaths = new[] { "~/Areas/Area1/Views/Shared/{0}.cshtml" };
engine.ViewLocationFormats = engine.ViewLocationFormats.Concat(sharedPaths).ToArray();
engine.MasterLocationFormats = engine.MasterLocationFormats.Concat(sharedPaths).ToArray();
engine.PartialViewLocationFormats = engine.PartialViewLocationFormats.Concat(sharedPaths).ToArray();
}
ViewEngines.Engines.Insert(0, engine);
VirtualPathFactoryManager.RegisterVirtualPathFactory(engine);
}
のような Area2 内のビューで部分参照に遭遇すると@Html.Partial("Partials/Footer")
、例外が発生します。Razor が正しいパスを探しているようです
System.InvalidOperationException: The partial view 'Partials/Footer' was not found or no view engine supports the searched locations. The following locations were searched:
~/Areas/Area2/Views/Home/Partials/Footer.cshtml
~/Areas/Area2/Views/Shared/Partials/Footer.cshtml
~/Views/Home/Partials/Footer.cshtml
~/Views/Shared/Partials/Footer.cshtml
~/Areas/Area1/Views/Shared/Partials/Footer.cshtml
at System.Web.Mvc.HtmlHelper.FindPartialView(ViewContext viewContext, String partialViewName, ViewEngineCollection viewEngineCollection)
のソース コードを見るとPrecompiledMvcEngine
、アセンブリ内のビューのみを検索しているように見えます。
私は当初、ビュー システムがパスを解決しようとするときに、登録されているすべての ViewEngine を調べると考えていましたが、それは間違った仮定のようです (そうしない理由もわかります)。
複数のアセンブリ間でビューを共有する方法はありますか?
アップデート
PrecompiledMvcEngine
コンストラクターでアセンブリのリストを受け取るカスタム バージョンを作成することで、この問題を回避しました。コアの変更は次のとおりです。
_mappings = new Dictionary<string, Type>(StringComparer.OrdinalIgnoreCase);
foreach (var kvp in assemblies)
{
var baseVirtualPath = NormalizeBaseVirtualPath(kvp.Key);
var assembly = kvp.Value;
var mapping = from type in assembly.GetTypes()
where typeof(WebPageRenderingBase).IsAssignableFrom(type)
let pageVirtualPath = type.GetCustomAttributes(inherit: false).OfType<PageVirtualPathAttribute>().FirstOrDefault()
where pageVirtualPath != null
select new KeyValuePair<string, Type>(CombineVirtualPaths(baseVirtualPath, pageVirtualPath.VirtualPath), type);
foreach (var map in mapping)
{
_mappings.Add(map);
}
}
このアプローチのより良い代替案や落とし穴はありますか?