源码网商城,靠谱的源码在线交易网站 我的订单 购物车 帮助

源码网商城

解读ASP.NET 5 & MVC6系列教程(16):自定义View视图文件查找逻辑

  • 时间:2020-10-26 19:51 编辑: 来源: 阅读:
  • 扫一扫,手机访问
摘要:解读ASP.NET 5 & MVC6系列教程(16):自定义View视图文件查找逻辑
之前MVC5和之前的版本中,我们要想对View文件的路径进行控制的话,则必须要对[code]IViewEngine[/code]接口的[code]FindPartialView[/code]或[code]FindView[/code]方法进行重写,所有的视图引擎都继承于该[code]IViewEngine[/code]接口,比如默认的[code]RazorViewEngine[/code]。但新版本MVC6中,对视图文件的路径方式却不太一样了,目前有两种方式,一种是通过[code]RazorViewEngine[/code],另外一种是通过新特性[code]IViewLocationExpander[/code]接口。 [b]通过RazorViewEngine来控制View路径[/b] 在新版的[code]RazorViewEngine[/code]中,该类提供了两个虚属性([code]AreaViewLocationFormats[/code]和[code]ViewLocationFormats[/code]),可以用于重写控制,而不必再对[code]FindPartialView[/code]或[code]FindView[/code]方法进行重写,示例如下:
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));
});
这样,系统在查找视图文件的时候,就会按照新注册的[code]ThemeViewEngine[/code]的逻辑来执行。 [b]通过IViewLocationExpander来控制View路径[/b] 在MVC6中,微软还提供了另外一种新的方式来控制View文件的路径,那就是[code]IViewLocationExpander[/code]接口,通过实现该接口即可实现自定义逻辑,并且也可以使用相关的上下文对象。示例如下:
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"] + "/"));
  }
}
在上述自定义的[code]IViewLocationExpander[/code]中,实现了2个方法分别是[code]PopulateValues[/code]和[code]ExpandViewLocations[/code],[code]PopulateValues[/code]方法可以让我们想[code]ViewLocationExpanderContext[/code]上下文中添加响应的键值对以便后续使用,通过,我们可以利用通过该上下文对象,来查找[code]ActionContext[/code]和[code]HttpContext[/code]对象,以便利用这些对象做响应的判断操作;而[code]ExpandViewLocations[/code]方法,只会在没有View缓存或在View缓存里找不到对应key的View文件时才会调用该方法,在该方法内,我们可以动态返回视图的位置。 最后,我们在[code]Startup.cs[/code]里通过修改[code]RazorViewEngineOptions[/code]实例对象的[code]ViewLocationExpanders[/code]属性,来实现注册目的,代码如下:
services.Configure<RazorViewEngineOptions>(options =>
{
  options.ViewLocationExpanders.Add(typeof(ThemViewLocationExpander));
});
  • 全部评论(0)
联系客服
客服电话:
400-000-3129
微信版

扫一扫进微信版
返回顶部