App.Web
和App.Views
是我在一个解决方案中的项目,我将我的观点放在App.Views中并使用RazorGenerator预编译。如果我使用App.Web
之类的话,那就很好用,
中查看的虚拟路径
~/Views/Index.cshtml
是我在App.View
它可以在App.Web中成功呈现此视图
public ActionResult Index() {
return View("~/Views/Index.cshtml");
}
但是当我尝试RenderViewToString
时,它会返回null
。
class FakeController : ControllerBase
{
protected override void ExecuteCore() { }
public static string RenderViewToString(string controllerName, string viewName, object viewData)
{
using (var writer = new StringWriter())
{
var routeData = new RouteData();
routeData.Values.Add("controller", controllerName);
var fakeControllerContext = new ControllerContext(new HttpContextWrapper(new HttpContext(new HttpRequest(null, "http://google.com", null), new HttpResponse(null))), routeData, new FakeController());
var razorViewEngine = new RazorViewEngine();
var razorViewResult = razorViewEngine.FindView(fakeControllerContext, viewName, "", false);
var viewContext = new ViewContext(fakeControllerContext, razorViewResult.View, new ViewDataDictionary(viewData), new TempDataDictionary(), writer);
razorViewResult.View.Render(viewContext, writer);
return writer.ToString();
}
}
}
这就是它的全部,
FakeController.RenderViewToString("FakeName", "~/Views/Index.csthml", MessageModel);
This is discussed and probably solved in asp.net core,但我正在使用asp.net mvc 5。
请你帮我弄清楚,为什么它不起作用?
答案 0 :(得分:5)
您正在尝试使用Razor View Engine获取预编译的视图。这是个错误。 Razor引擎根据cshtml文件生成视图。但是,在预编译视图的情况下,RazorGenerator
已将cshtml文件编译为从System.Web.Mvc.WebViewPage
派生的一组类。这些类重写方法Execute()
(由RazorGenerator
基于输入cshtml自动生成),将html写入输出TextWriter
。
原始视图文件(cshtml)不再需要,因此不会与应用程序一起部署。当您调用Razor尝试定位cshtml并基于它构建视图时,它会返回null
视图。
ASP.NET MVC支持多个视图引擎(实现System.Web.Mvc.IViewEngine
的类)。 RazorViewEngine
就是其中之一。 RazorGenerator.Mvc
NuGet包添加了自己的视图引擎(PrecompiledMvcEngine
),该引擎基于预编译视图工作。已注册的视图引擎存储在ViewEngines.Engines
集合中。当您安装RazorGenerator.Mvc
NuGet包时,它会添加注册RazorGeneratorMvcStart
实例的PrecompiledMvcEngine
类:
[assembly: WebActivatorEx.PostApplicationStartMethod(typeof(App.Views.RazorGeneratorMvcStart), "Start")]
namespace App.Views {
public static class RazorGeneratorMvcStart {
public static void Start() {
var engine = new PrecompiledMvcEngine(typeof(RazorGeneratorMvcStart).Assembly) {
UsePhysicalViewsIfNewer = HttpContext.Current.Request.IsLocal
};
ViewEngines.Engines.Insert(0, engine);
// StartPage lookups are done by WebPages.
VirtualPathFactoryManager.RegisterVirtualPathFactory(engine);
}
}
}
您应该使用PrecompiledMvcEngine
而不是RazorViewEngine
的此实例来访问预编译的视图。以下是RenderViewToString
方法的调整代码:
public static string RenderViewToString(string controllerName, string viewName, object viewData)
{
using (var writer = new StringWriter())
{
var routeData = new RouteData();
routeData.Values.Add("controller", controllerName);
var fakeControllerContext = new ControllerContext(new HttpContextWrapper(new HttpContext(new HttpRequest(null, "http://google.com", null), new HttpResponse(null))), routeData, new FakeController());
var viewEngine = ViewEngines.Engines.OfType<PrecompiledMvcEngine>().FirstOrDefault();
if (viewEngine == null)
{
throw new InvalidOperationException("PrecompiledMvcEngine is not registered");
}
var viewResult = viewEngine.FindView(fakeControllerContext, viewName, "", false);
var viewContext = new ViewContext(fakeControllerContext, viewResult.View, new ViewDataDictionary(viewData), new TempDataDictionary(), writer);
viewResult.View.Render(viewContext, writer);
return writer.ToString();
}
}
一个重要的注意事项:您应该将RazorGenerator.Mvc
NuGet包与您的视图(App.Views
)一起安装到项目中,而不是安装到Web应用程序项目中,因为PrecompiledMvcEngine
将当前程序集作为预编译的视图。确保RazorGeneratorMvcStart
没有添加到App.Web
项目中。当我添加对RazorGenerator.Mvc.dll
程序集的引用时,它发生在我身上。