I've searched everywhere, what I want is basically a RenderViewToString for MVC6 that does NOT use the context of a controller or action.
Since custom TagHelpers go into their own class, these context's won't be available.
Currently I have the following code, I'd very much like to put the html in a Razor View, cause this is unacceptable lol.
public class WindowTagHelper : TagHelper
{
public string Title { get; set; }
public async override Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
{
var preContent = await output.GetChildContentAsync();
var id = context.UniqueId;
output.TagName = "div";
output.Attributes["id"] = id;
output.Attributes["class"] = "window";
output.TagMode = TagMode.StartTagAndEndTag;
var windowControls = new StringBuilder();
windowControls.Append("<div class=\"window-controls\">");
windowControls.Append("<ul class=\"window-icon-list\">");
var minimizeIcon = "<li class=\"control-icon\"><a data-window-id=\"{0}\" class=\"control-icon-minimize\" href=\"#\" data-toggle=\"tooltip\" title=\"Minimize Window\"><div class=\"fa fa-minus\"></div></a></li>".InjectWith(id);
var maximizeIcon = "<li class=\"control-icon\"><a data-window-id=\"{0}\" class=\"control-icon-maximize\" href=\"#\" data-toggle=\"tooltip\" title=\"Maximize Window\"><div class=\"fa fa-square-o\"></div></a></li>".InjectWith(id);
var closeIcon = "<li class=\"control-icon\"><a data-window-id=\"{0}\" class=\"control-icon-close\" href=\"#\" data-toggle=\"tooltip\" title=\"Close Window\"><div class=\"fa fa-close\"></div></a></li>".InjectWith(id);
windowControls.Append(minimizeIcon);
windowControls.Append(maximizeIcon);
windowControls.Append(closeIcon);
windowControls.Append("</ul>");
windowControls.Append("</div>");
var content = new StringBuilder();
content.AppendFormat("<div class=\"window-title\">{0}{1}</div>", windowControls.ToString(), Title);
content.Append(preContent.GetContent());
output.Content.AppendHtml(content.ToString());
}
}
Following method does not work, it throws an exception and injecting ICompositeViewEngine as service requires a controller/action context.
public string RenderPartialViewToString(string viewName, object model)
{
if (string.IsNullOrEmpty(viewName))
viewName = ActionContext.ActionDescriptor.Name;
ViewData.Model = model;
using (StringWriter sw = new StringWriter())
{
var engine = Resolver.GetService(typeof(ICompositeViewEngine)) as ICompositeViewEngine;
ViewEngineResult viewResult = engine.FindPartialView(ActionContext, viewName);
ViewContext viewContext = new ViewContext(ActionContext, viewResult.View, ViewData, TempData, sw,new HtmlHelperOptions());
var t = viewResult.View.RenderAsync(viewContext);
t.Wait();
return sw.GetStringBuilder().ToString();
}
}
We achieved it with a custom taghelper and an extension to ViewContext
[ViewContext]
public ViewContext ViewContext { get; set; }
public async override void Process(TagHelperContext context, TagHelperOutput output)
{
var sw = new StringWriter();
// Create a new viewData (viewbag). This will be used in a new ViewContext to define the model we want
ViewDataDictionary viewData = new ViewDataDictionary(ViewContext.ViewData, For.Model);
// Generate a viewContext with our viewData
var viewContext = new ViewContext(ViewContext, ViewContext.View, viewData, ViewContext.TempData, sw, new HtmlHelperOptions());
// Use the viewContext to run the given ViewName
output.Content.Append(new HtmlString(await viewContext.RenderPartialView(ViewName)));
}
public async static Task<string> RenderPartialView(this ViewContext context, string viewName, ICompositeViewEngine viewEngine = null, ViewEngineResult viewResult = null)
{
viewEngine = viewEngine ?? context.HttpContext.RequestServices.GetRequiredService<ICompositeViewEngine>();
viewResult = viewResult ?? viewEngine.FindPartialView(context, viewName);
await viewResult.View.RenderAsync(context);
return context.Writer.ToString();
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With