当前位置 : 主页 > 网络编程 > ASP >

asp.net-mvc – 为Razor调整自定义Html助手(它使用HtmlTextWriter,因此返回void)

来源:互联网 收集:自由互联 发布时间:2021-06-24
问题 我为WebFormViewEngine视图编写了一个非常漂亮的菜单Html帮助器.这个引擎允许你的助手返回void,仍然可以使用: @Html.Theseus 这对我的助手很有用,因为它可以使用HtmlTextWriter渲染菜单,直
问题

我为WebFormViewEngine视图编写了一个非常漂亮的菜单Html帮助器.这个引擎允许你的助手返回void,仍然可以使用:

@Html.Theseus

这对我的助手很有用,因为它可以使用HtmlTextWriter渲染菜单,直接渲染到输出流.

但是,在Razor视图中,Html助手应该返回一个值(通常是MvcHtmlString),这是添加到输出中的值.差异小,后果大.

有一种解决方法,正如GvS(见ASP.NET MVC 2 to MVC 3: Custom Html Helpers in Razor)所指出的那样:

如果帮助程序返回void,请执行以下操作:

@{Html.Theseus;}

(基本上,您只是调用方法,而不是渲染到视图中).

虽然仍然很整洁,但这与@ Html.seus不完全相同.所以…

我的代码很复杂,但效果很好,所以不愿意进行主要编辑,即用另一个编写器替换HtmlTextWriter.一段代码如下:

writer.AddAttribute(HtmlTextWriterAttribute.Href, n.Url);
writer.AddAttribute(HtmlTextWriterAttribute.Title, n.Description);
writer.RenderBeginTag(HtmlTextWriterTag.A);
writer.WriteEncodedText(n.Title);
writer.RenderEndTag();

// Recursion, if any
// Snip off the recursion at this level if specified by depth
// Use a negative value for depth if you want to render the entire sitemap from the starting node

    if ((currentDepth < depth) || (depth < 0))
    {
         if (hasChildNodes)
         {
              // Recursive building starts here

              // Open new ul tag for the child nodes 
              // "<ul class='ChildNodesContainer {0} Level{1}'>"; 
              writer.AddAttribute(HtmlTextWriterAttribute.Class, "Level" + currentDepth.ToString());
              writer.RenderBeginTag(HtmlTextWriterTag.Ul);

              // BuildMenuLevel calls itself here to 
              // recursively traverse the sitemap hierarchy, 
              // building the menu as I go.
              // Note: this is where I increase the currentDepth variable!
               BuildChildMenu(currentDepth + 1, depth, n, writer);

              // Close ul tag for the child nodes
              writer.RenderEndTag();
          }
    }

用TagBuilders重写是不会有趣的.就目前而言,它呈现任何类型的菜单,包括我的4guysfromrolla文章中描述的“增量导航”:
Implementing Incremental Navigation with ASP.NET

选项:

我想我可以返回一个空的MvcHtmlString,但这几乎是一个黑客的定义……

唯一的选择是进入夕阳并使用TagBuilder重写帮助程序来构建每个标记,将其添加到StringBuilder,然后构建下一个标记等,然后使用StringBuilder实例创建MvcHtmlString.严重丑陋,除非我能做点什么……

问题:

有办法:

停止HtmlTextWriter渲染到流,而是像StringBuilder一样使用它,在我用来创建MvcHtmlString(或HtmlString)的过程结束时?

听起来不太可能,即使我写的……

PS:

关于HtmlTextWriter的好处是你可以构建大量的标签,而不是像TagBuilder一样逐个构建它们.

与您收到的 other question Razor的回复相反,您不需要返回HtmlString.您的代码现在的问题是您直接写入响应流. Razor内部执行内容,这意味着您可以搞砸响应顺序(参见 similar question).

所以在你的情况下你可能会这样做(虽然我还没有测试过):

public static void Theseus(this HtmlHelper html)
{
    var writer = new HtmlTextWriter(html.ViewContext.Writer);
    ...
}

编辑(跟进以解决您的意见):

Html Helpers完全能够直接返回HtmlString或void并写入上下文编写器.例如,Html.Partial和Html.RenderPartial在Razor中都能正常工作.我认为你困惑的是调用一个版本而不是另一个版本所需的语法.

例如,考虑一个Aspx视图:

<%: Html.Partial("Name") %>
<% Html.RenderPartial("Name") %>

您以不同方式调用每种方法如果你翻转东西,事情将无法奏效.同样在Razor中:

@Html.Partial("Name")
@{ Html.RenderPartial("Name"); }

现在只是碰巧使用void helper的语法在Razor中比Aspx更冗长.但是,两者都工作得很好.除非你的意思是“问题在于html助手无法返回void”.

顺便说一句,如果你真的想用这个语法调用你的助手:@ Html.Theseus()你可以这样做:

public static IHtmlString Theseus(this HtmlHelper html)
{
    var writer = new HtmlTextWriter(html.ViewContext.Writer);
    ...
    return new HtmlString("");
}

但这有点像黑客.

网友评论