我有一个与Html.BeginForm()
非常相似的块的自定义实现。实施基本如下:
public class MyBlock : IDisposable {
private readonly HtmlHelper _html;
public MyBlock(HtmlHelper hml) {
this._html.ViewContext.Writer.WriteLine("BEGIN");
}
public void Dispose() {
this._html.ViewContext.Writer.WriteLine("END");
}
}
然后在我看来,我可以这样做:
@using (new MyBlock(Html)) {
@: some content
}
获得:
BEGIN
some content
END
一切正常。但是,当我在“razor片段”中使用我的块时,我遇到麻烦,e。 G。将一些剃刀内容传递给以Func<object, HelperResult>
为参数的函数。例如,我有另一个HtmlHelper函数定义如下:
public static IHtmlString Content(this HtmlHelper @this, Func<object, HelperResult> razor) {
return razor(null);
}
@* use in a view as: *@
@{
var razorContent = Html.Content(@<div>Some razor content</div>);
}
@razorContent
但是,当我执行以下操作时,内部内容呈现时没有外部内容:
@{
var content =Html.Content(
@<text>
@using (new MyBlock(Html)) {
@: some content 2
}
<text>
);
}
@content
我认为问题是“Html”仍然引用外部上下文的HtmlHelper,因此BEGIN和END被发送给不同的作者而不是“某些内容2”,但是,我不确定这是案件。
有谁知道(1)出了什么问题,(2)我该如何解决?
答案 0 :(得分:2)
您的问题的部分解决方案是调用HelperResult的WriteTo方法。您可以将Content方法更改为以下内容:
public static void Content(this HtmlHelper @this, Func<object, HelperResult> razor)
{
razor(null).WriteTo(@this.ViewContext.Writer);
}
然后以这种方式使用它:
@{ Html.Content(
@<text>
@using (new MyBlock(Html)) {
@: some content 2
}
</text>
);
}
修改
如果您希望将值作为IHtmlString或任何其他字符串返回,则可以执行此操作:
public static IHtmlString Content(this HtmlHelper @this, Func<object, HelperResult> razor)
{
using (MemoryStream ms = new MemoryStream())
using (TextWriter tw = new StreamWriter(ms))
{
Delegate @delegate = (Delegate)razor;
WebViewPage target = (WebViewPage)@delegate.Target;
TextWriter tmp = target.Html.ViewContext.Writer;
try
{
target.Html.ViewContext.Writer = tw;
razor(null).WriteTo(tw);
tw.Flush();
ms.Seek(0, SeekOrigin.Begin);
TextReader tr = new StreamReader(ms);
return MvcHtmlString.Create(tr.ReadToEnd());
}
finally
{
target.Html.ViewContext.Writer = tmp;
}
}
}