在Controller ASP.NET MVC 4中调用RenderAction用于仪表板(多)视图

时间:2013-09-03 09:28:30

标签: c# asp.net-mvc asp.net-mvc-4

我有一个模板HTML字符串,它在各个未知点都有{{Url}}个占位符,与我的应用程序中的某些控制器/操作有关。我需要做的是在视图中渲染最终的html之前将html渲染到这些占位符中。

在视图中,我只需调用Html.RenderAction("Action","Controller")即可返回我需要的字符串。 但是,我需要在控制器代码中调用此方法,例如(这是简化的):

在“仪表板”控制器中:

var templateHtml = GetTemplateHtml();

//The following line doesn't compile
var html = Html.RenderAction("Index","PowerAnalysisDashpart");

ViewBag.Html = templateHtml.Replace("{{PowerAnalysisDashpart}}",html)

然后在“仪表板视图”中:

<div id="content">
    @Html.Raw(ViewBag.Html)
</div>

如何调用“RenderAction”来获取控制器中的渲染HTML字符串?

修改 似乎对我想要实现的目标感到困惑。基本上,我们需要管理员能够创建一个HTML模板,它将有效地为我们的应用程序的各种不同页面提供框架。 iframe可以使用,但我们更喜欢在服务器上构建所有HTML。

4 个答案:

答案 0 :(得分:8)

我认为你想要HtmlHelper.Action而不是RenderActionRenderAction将内容写入响应流。 Action将其作为字符串返回。

http://msdn.microsoft.com/en-us/library/ee721266(v=vs.108).aspx

动作在ChildActionExtensions类中定义,该类位于System.Web.Mvc.Html命名空间中,因此您的控制器代码需要使用该命名空间。

Action将Html作为字符串返回,而RenderAction返回void,因为RenderAction将其Html输出直接写入与父文档内联的当前响应流。这就是它从视图而不是控制器工作的原因。

要在MVC 4中创建HtmlHelper实例,您可以使用:

HtmlHelper helper = new HtmlHelper(new ViewContext(ControllerContext, new WebFormView(ControllerContext, "Index"), new ViewDataDictionary(), new TempDataDictionary(), new StringWriter()), new ViewPage());

正如其他评论员在相关的SO问题上所说的那样,这并不是真正的意图(通过hacky方式来实现HtmlHelper显而易见)。我不确定它比你的替代解决方案好多了。如果可以的话,最好重构成更多的MVC,尽管有时我觉得这不是一种选择。

答案 1 :(得分:2)

所以,找到了一个解决方案,但它非常hacky,只需必须更清洁!

我有一个共享的部分视图“_view”,其中只包含此内容:

@{ Html.RenderAction("Index",(string)ViewBag.Controller); }

我使用了此博客中的代码:http://approache.com/blog/render-any-aspnet-mvc-actionresult-to/为ActionResult类编写扩展方法。

然后在我的控制器中我这样做:

var templateHtml = GetTemplateHtml();
ViewBag.Controller = "PowerAnalysis";
var html = View("_view").Capture(ControllerContext);
ViewBag.Html = htmlTemplate.Replace("{{PowerAnalysis}}", html);

//Repeat for other pages

return View();

在我从博客中使用的扩展方法的代码下面:

public class ResponseCapture : IDisposable {
        private readonly HttpResponseBase response;
        private readonly TextWriter originalWriter;
        private StringWriter localWriter;
        public ResponseCapture(HttpResponseBase response) {
            this.response = response;
            originalWriter = response.Output;
            localWriter = new StringWriter();
            response.Output = localWriter;
        }
        public override string ToString() {
            localWriter.Flush();
            return localWriter.ToString();
        }
        public void Dispose() {
            if (localWriter != null) {
                localWriter.Dispose();
                localWriter = null;
                response.Output = originalWriter;
            }
        }
    }


    public static class ActionResultExtensions {
        public static string Capture(this ActionResult result, ControllerContext controllerContext) {
            using (var it = new ResponseCapture(controllerContext.RequestContext.HttpContext.Response)) {
                result.ExecuteResult(controllerContext);
                return it.ToString();
            }
        }
    }

答案 2 :(得分:-2)

我认为您需要部分视图,您可以在主视图中呈现部分视图

 @Html.Partial("myPartailView")

详细检查链接

http://www.codeproject.com/Tips/617361/Partial-View-in-ASP-NET-MVC-4

答案 3 :(得分:-2)

要将视图渲染为字符串,您可以按照以下方法操作:

var html = RenderRazorViewToString("viewName",model); //pass model if you are binding model in view


// It will return your view as string
[NonAction]
public string RenderRazorViewToString(string viewName, object model)
{
    ViewData.Model = model;
    using (var sw = new StringWriter())
    {
        var viewResult = ViewEngines.Engines.FindPartialView(ControllerContext, viewName);
        var viewContext = new ViewContext(ControllerContext, viewResult.View, ViewData, TempData, sw);
        viewResult.View.Render(viewContext, sw);
        viewResult.ViewEngine.ReleaseView(ControllerContext, viewResult.View);
        return sw.GetStringBuilder().ToString();
    }
}