嵌入在另一个标签助手代码中的标签助手不呈现

Posted

技术标签:

【中文标题】嵌入在另一个标签助手代码中的标签助手不呈现【英文标题】:Tag Helper Embedded in Another Tag Helper's Code Doesn't Render 【发布时间】:2018-05-12 20:53:32 【问题描述】:

我正在尝试使用 .net Core 2.0 中的以下自定义标签助手来简化长表单的创建:

@model ObApp.Web.Models.ViewComponents.FormFieldViewModel

<span>Debug - Paramater value: @Model.FieldFor</span>
<div class="form-group">
    <label asp-for="@Model.FieldFor"></label>
    <input asp-for="@Model.FieldFor" class="form-control" />
</div>

这看起来很简单,但是当我使用时,我得到了一个意想不到的(对我来说)结果:

<vc:form-field field-for="PersonEmail"></vc:form-field>

预期结果

<span>Debug - Paramater value: PersonEmail</span>
<div class="form-group">
    <label for="PersonEmail">Email</label>
    <input name="PersonEmail" class="form-control" id="PersonEmail" 
        type="text" value="PersonEmail">
</div>

实际结果

<span>Debug - Paramater value: PersonEmail</span>
<div class="form-group">
    <label for="FieldFor">FieldFor</label>
    <input name="FieldFor" class="form-control" id="FieldFor" 
        type="text" value="PersonEmail">
</div>

我尝试从 @Model.FieldFor 周围删除引号,以及其他一些语法更改。

有什么建议吗?

谢谢!

【问题讨论】:

这可能是一个操作顺序。换句话说,asp-for 处理发生在 Razor 实际取消引用模型属性值之前。如果是这种情况,我不确定您实际上可以做些什么。但是,如果这就是您的视图组件的全部内容,那么它应该是一个视图组件。改为创建自定义 taghelper。 两个优点。谢谢你。我尝试将它作为标签助手并遇到嵌套助手未处理的相同问题。我正在研究如何正确嵌套标签助手,或者,如果这不起作用,将使用我自己的代码构建整个助手,而不是嵌套 asp 标签助手。如果没有人超过我,我会在我让它工作时在这里发布结果。 【参考方案1】:

正如其他人向我指出的那样,可能无法按照我发布此问题时最初想要的方式直接嵌入标签助手。结果,我将代码重构为以编程方式“新建”所需的标签助手。

我的最终解决方案比我预期的要多得多,但从长远来看,它将节省大量时间来开发我计划的表单密集型应用程序。

目标

我的目标是通过使用这个自定义标签助手来加速表单的创建,例如:

<formfield asp-for="OrganizationName"></formfield>

要生成这些内置的 Razor 标签助手:

<div class="form-group">
    <div class="row">
        <label class="col-md-3 col-form-label" for="OrganizationName">Company Name</label>
        <div class="col-md-9">
            <input name="OrganizationName" class="form-control" id="OrganizationName" type="text" value="" data-val-required="The Company Name field is required." data-val="true" data-val-maxlength-max="50" data-val-maxlength="Maximum company name length is 50 characters.">
            <span class="field-validation-valid" data-valmsg-replace="true" data-valmsg-for="OrganizationName"></span>
        </div>
    </div>
</div>

我最初的工作解决方案

这是针对简单案例的首过测试解决方案。 IE。用于默认的硬编码类和文本框输入类型。

using Microsoft.AspNetCore.html;
using Microsoft.AspNetCore.Mvc.Rendering;
using Microsoft.AspNetCore.Mvc.TagHelpers;
using Microsoft.AspNetCore.Mvc.ViewFeatures;
using Microsoft.AspNetCore.Razor.TagHelpers;
using System.Collections.Generic;
using System.Threading.Tasks;

namespace ObApp.Web.TagHelpers

    // Builds form elements to generate the following (for example):
    // <div class="form-group">
    //     <div class="row">
    //         <input ... >Email</input>
    //         <div>
    //             <input type="text" ... />
    //             <span class="field-validation-valid ... ></span>
    //         </div>
    //     </div>
    // </div>

    public class FormfieldTagHelper : TagHelper
    
        private const string _forAttributeName = "asp-for";
        private const string _defaultWraperDivClass = "form-group";
        private const string _defaultRowDivClass = "row";
        private const string _defaultLabelClass = "col-md-3 col-form-label";
        private const string _defaultInputClass = "form-control";
        private const string _defaultInnerDivClass = "col-md-9";
        private const string _defaultValidationMessageClass = "";

        public FormfieldTagHelper(IHtmlGenerator generator)
        
            Generator = generator;
        

        [HtmlAttributeName(_forAttributeName)]
        public ModelExpression For  get; set; 

        public IHtmlGenerator Generator  get; 

        [ViewContext]
        [HtmlAttributeNotBound]
        public ViewContext ViewContext  get; set; 

        public override async Task ProcessAsync(TagHelperContext context, TagHelperOutput output)
        
            // Replace this parent tag helper with div tags wrapping the entire form block
            output.TagName = "div";
            output.Attributes.SetAttribute("class", _defaultWraperDivClass);

            // Manually new-up each child asp form tag helper element
            TagHelperOutput labelElement = await CreateLabelElement(context);
            TagHelperOutput inputElement = await CreateInputElement(context);
            TagHelperOutput validationMessageElement = await CreateValidationMessageElement(context);

            // Wrap input and validation with column div
            IHtmlContent innerDiv = WrapElementsWithDiv(
                    new List<IHtmlContent>()
                    
                        inputElement,
                        validationMessageElement
                    ,
                    _defaultInnerDivClass
                );

            // Wrap all elements with a row div
            IHtmlContent rowDiv = WrapElementsWithDiv(
                    new List<IHtmlContent>()
                    
                        labelElement,
                        innerDiv
                    ,
                    _defaultRowDivClass
                );

            // Put everything into the innerHtml of this tag helper
            output.Content.SetHtmlContent(rowDiv);
        

        private async Task<TagHelperOutput> CreateLabelElement(TagHelperContext context)
        
            LabelTagHelper labelTagHelper = 
                new LabelTagHelper(Generator)
                
                    For = this.For,
                    ViewContext = this.ViewContext
                ;

            TagHelperOutput labelOutput = CreateTagHelperOutput("label");

            await labelTagHelper.ProcessAsync(context, labelOutput);

            labelOutput.Attributes.Add(
                new TagHelperAttribute("class", _defaultLabelClass));

            return labelOutput;
        

        private async Task<TagHelperOutput> CreateInputElement(TagHelperContext context)
        
            InputTagHelper inputTagHelper = 
                new InputTagHelper(Generator)
                
                    For = this.For,
                    ViewContext = this.ViewContext
                ;

            TagHelperOutput inputOutput = CreateTagHelperOutput("input");

            await inputTagHelper.ProcessAsync(context, inputOutput);

            inputOutput.Attributes.Add(
                new TagHelperAttribute("class", _defaultInputClass));

            return inputOutput;
        

        private async Task<TagHelperOutput> CreateValidationMessageElement(TagHelperContext context)
        
            ValidationMessageTagHelper validationMessageTagHelper = 
                new ValidationMessageTagHelper(Generator)
                
                    For = this.For,
                    ViewContext = this.ViewContext
                ;

            TagHelperOutput validationMessageOutput = CreateTagHelperOutput("span");

            await validationMessageTagHelper.ProcessAsync(context, validationMessageOutput);

            return validationMessageOutput;
        

        private IHtmlContent WrapElementsWithDiv(List<IHtmlContent> elements, string classValue)
        
            TagBuilder div = new TagBuilder("div");
            div.AddCssClass(classValue);
            foreach(IHtmlContent element in elements)
            
                div.InnerHtml.AppendHtml(element);
            

            return div;
        

        private TagHelperOutput CreateTagHelperOutput(string tagName)
        
            return new TagHelperOutput(
                tagName: tagName,
                attributes: new TagHelperAttributeList(),
                getChildContentAsync: (s, t) =>
                
                    return Task.Factory.StartNew<TagHelperContent>(
                            () => new DefaultTagHelperContent());
                
            );
        
    

后续步骤/建议改进

这适用于没有验证错误的文本框。在调整默认 CSS 之后,我计划采取的下一步是:

    在出现验证错误时显示正确的 CSS 类属性以进行格式化。此时标签助手将为我“工作”。 将硬编码的 CSS 类移到站点配置文件中。 绑定到视图中的 HTML 属性以允许传入非默认类。另一种方法是通过 ViewModel 传入非默认表单类。 检测非文本框输入类型并相应地设置格式。

感谢@Chris Pratt 让我在这方面朝着正确的方向开始。

【讨论】:

【参考方案2】:

在 .net core 中,asp-for="@Model.FieldFor" 中不需要@,如下例所示

@model ObApp.Web.Models.ViewComponents.FormFieldViewModel

<span>Debug - Paramater value: @Model.FieldFor</span>
<div class="form-group">
    <label asp-for="model.FieldFor"></label>
    <input asp-for="model.FieldFor" class="form-control" />
</div>

【讨论】:

asp-for="model.FieldFor" 生成错误,但 asp-for="FieldFor" 在语法上是正确的。但是我仍然有同样的原始问题。我开始认为这是在取消引用属性之前剃刀处理的问题。我确实喜欢上面@Chris 的想法,它应该是一个标签助手而不是一个视图组件,所以我接下来会尝试。谢谢 - 尽管它没有解决问题,但您的评论教会了我一些关于在 Razor 中引用模型属性的有用信息!

以上是关于嵌入在另一个标签助手代码中的标签助手不呈现的主要内容,如果未能解决你的问题,请参考以下文章

有没有办法获取选项标签值并将其传递给车把中的助手?

ASP.NET Core MVC 标签助手参数中的字符串文字

如何使用标签助手或 html 助手在选择选项中显示 Font Awesome 图标? [复制]

ASP.NET CORE MVC 选择标签助手 - 如何在不使用 ModelView 的情况下设置选定值

使用 concat 助手渲染组件

自定义jsp标签