Forcing EditorFor to prefix input items on view with Class Name?

前端 未结 2 1142
心在旅途
心在旅途 2021-01-12 17:58

I have an EditorFor:

<%: Html.EditorFor(model => model.Client, \"ClientTemplate\", new { editing = false })%>

This will bind comin

相关标签:
2条回答
  • 2021-01-12 18:14

    Try something like:

    <% Html.BeginHtmlFieldPrefixScope("Client") {
      Html.EditorFor(model => model.Client, "ClientTemplate", new { editing = false });
    <% } %>
    

    Every field you make with EditorFor, LabelFor and the likes will be prefixed.

    EDIT: Here's the extension method I'm using, sorry!

    public static IDisposable BeginHtmlFieldPrefixScope(this HtmlHelper html, string htmlFieldPrefix)
    {
      return new HtmlFieldPrefixScope(html.ViewData.TemplateInfo, htmlFieldPrefix);
    }
    

    ...and the class...

    private class HtmlFieldPrefixScope : IDisposable
    {
        private readonly TemplateInfo templateInfo;
        private readonly string previousHtmlFieldPrefix;
    
        public HtmlFieldPrefixScope(TemplateInfo templateInfo, string htmlFieldPrefix)
        {
            this.templateInfo = templateInfo;
    
            previousHtmlFieldPrefix = templateInfo.HtmlFieldPrefix;
            templateInfo.HtmlFieldPrefix = htmlFieldPrefix;
        }
    
        public void Dispose()
        {
            templateInfo.HtmlFieldPrefix = previousHtmlFieldPrefix;
        }
    }
    

    See the link mentioned by Kohan in the comments below.

    0 讨论(0)
  • 2021-01-12 18:14

    MVC3 HTML Name Clashing

    Cutting and pasting that does not work in MVC3. To get the extension to work, I had to create a class file:

    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Web;
    using System.Web.Mvc;
    
    namespace incMvcSite.Classes {
        public static class HtmlPrefixScopeExtensions {
            public static IDisposable BeginHtmlFieldPrefixScope(this HtmlHelper html, string htmlFieldPrefix) {
                return new HtmlFieldPrefixScope(html.ViewData.TemplateInfo, htmlFieldPrefix);
            }
    
            private class HtmlFieldPrefixScope : IDisposable {
                private readonly TemplateInfo templateInfo;
                private readonly string previousHtmlFieldPrefix;
    
                public HtmlFieldPrefixScope(TemplateInfo templateInfo, string htmlFieldPrefix) {
                    this.templateInfo = templateInfo;
    
                    previousHtmlFieldPrefix = templateInfo.HtmlFieldPrefix;
                    templateInfo.HtmlFieldPrefix = htmlFieldPrefix;
                }
    
                public void Dispose() {
                    templateInfo.HtmlFieldPrefix = previousHtmlFieldPrefix;
                }
            }
        }
    }
    

    In the Razor (.cshtml) file, I added the following:

    @using incMvcSite.Classes
    @using(Html.BeginHtmlFieldPrefixScope("Permission")) {
        <fieldset>
            <legend>Permission</legend>
    
            // The Html.EditorFor's would go here...
        </fieldset>
    }
    

    Notice the using to bring me extension class into scope. That allows the second using line to work.

    Now the problem is that when posting back, the object is not updated. In my controller, I used a second parameter to specify my prefix:

    TryUpdateModel(modelUser.Permission, "Permission");
    

    This added the prefix to all field in the HTML, and the TryUpdateModel loaded the object with prefixed control names. Now you can properly namespace your controls for embedded edit lists, and for partial views of models with the same property names.

    0 讨论(0)
提交回复
热议问题