之前MVC5和之前的版本中,我們要想對View文件的路徑進行控制的話,則必須要對IViewEngine
接口的FindPartialView
或FindView
方法進行重寫,所有的視圖引擎都繼承於該IViewEngine
接口,比如默認的RazorViewEngine
。但新版本MVC6中,對視圖文件的路徑方式卻不太一樣了,目前有兩種方式,一種是通過RazorViewEngine
,另外一種是通過新特性IViewLocationExpander
接口。
通過RazorViewEngine來控制View路徑
在新版的RazorViewEngine
中,該類提供了兩個虛屬性(AreaViewLocationFormats
和ViewLocationFormats
),可以用於重寫控制,而不必再對FindPartialView
或FindView
方法進行重寫,示例如下:
public class ThemeViewEngine : RazorViewEngine { public ThemeViewEngine(IRazorPageFactory pageFactory, IRazorViewFactory viewFactory, IViewLocationExpanderProvider viewLocationExpanderProvider, IViewLocationCache viewLocationCache) : base(pageFactory, viewFactory, viewLocationExpanderProvider, viewLocationCache) { } public override IEnumerable<string> AreaViewLocationFormats { get { var value = new Random().Next(0, 1); var theme = value == 0 ? "Theme1" : "Theme2"; // 可通過其它條件,設置皮膚的種類 return base.AreaViewLocationFormats.Select(f => f.Replace("/Views/", "/Views/" + theme + "/")); } } public override IEnumerable<string> ViewLocationFormats { get { var value = new Random().Next(0, 1); var theme = value == 0 ? "Theme1" : "Theme2"; // 可通過其它條件,設置皮膚的種類 return base.ViewLocationFormats.Select(f => f.Replace("/Views/", "/Views/" + theme + "/")); } } }
然後,通過修改MVcOptions的實例屬性ViewEngines即可完成對視圖引擎的替換,代碼如下:
services.AddMvc().Configure<MvcOptions>(options => { options.ViewEngines.Clear(); options.ViewEngines.Add(typeof(ThemeViewEngine)); });
這樣,系統在查找視圖文件的時候,就會按照新注冊的ThemeViewEngine
的邏輯來執行。
通過IViewLocationExpander來控制View路徑
在MVC6中,微軟還提供了另外一種新的方式來控制View文件的路徑,那就是IViewLocationExpander
接口,通過實現該接口即可實現自定義邏輯,並且也可以使用相關的上下文對象。示例如下:
public class ThemeViewLocationExpander : IViewLocationExpander { public void PopulateValues(ViewLocationExpanderContext context) { var value = new Random().Next(0, 1); var theme = value == 0 ? "Theme1" : "Theme2"; context.Values["theme"] = theme; } public virtual IEnumerable<string> ExpandViewLocations(ViewLocationExpanderContext context, IEnumerable<string> viewLocations) { return viewLocations.Select(f => f.Replace("/Views/", "/Views/" + context.Values["theme"] + "/")); } }
在上述自定義的IViewLocationExpander
中,實現了2個方法分別是PopulateValues
和ExpandViewLocations
,PopulateValues
方法可以讓我們想ViewLocationExpanderContext
上下文中添加響應的鍵值對以便後續使用,通過,我們可以利用通過該上下文對象,來查找ActionContext
和HttpContext
對象,以便利用這些對象做響應的判斷操作;而ExpandViewLocations
方法,只會在沒有View緩存或在View緩存裡找不到對應key的View文件時才會調用該方法,在該方法內,我們可以動態返回視圖的位置。
最後,我們在Startup.cs
裡通過修改RazorViewEngineOptions
實例對象的ViewLocationExpanders
屬性,來實現注冊目的,代碼如下:
services.Configure<RazorViewEngineOptions>(options => { options.ViewLocationExpanders.Add(typeof(ThemViewLocationExpander)); });