LabelFor和TextBoxFor不生成相同的id

Gle*_*rde 5 asp.net-mvc html-helper asp.net-mvc-2

使用以下代码时,字段的id和标签的for属性中的id不相同.

<%: Html.LabelFor(x => x.Localizations["en"]) %>   => Localizations[en]
<%: Html.TextBoxFor(x=> x.Localizations["en"]) %>  => Localizations_en_

<%: Html.LabelFor(x => x.Localizations["en"].Property) %>
       => Localizations[en]_Property
<%: Html.TextBoxFor(x=> x.Localizations["en"].Property) %>
       => Localizations_en__Property
Run Code Online (Sandbox Code Playgroud)

我在反射器中跟踪代码,发现生成值的方式不同.不使用相同的帮助方法.

LabelFor用途HtmlHelper.GenerateIdFromName和TextBoxFor用途TagBuilder#GenerateId.

有没有人知道这个的原因,或解决方法(除了编写你自己的整套输入/ textarea /选择助手)?或者这是一个错误?

更新:

因为我总是使用标签文本的第二个参数为标签使用html帮助器,我修改它以使用与表单字段助手相同的id生成代码.

public static MvcHtmlString LabelFor<TModel, TValue>(this HtmlHelper<TModel> helper, Expression<Func<TModel, TValue>> expression, string labelText)
{
    // The main part of this code is taken from the internal code for Html.LabelFor<TModel, TValue>(...).
    var metaData = ModelMetadata.FromLambdaExpression(expression, helper.ViewData);
    var fieldName = ExpressionHelper.GetExpressionText(expression);

    TagBuilder builder = new TagBuilder("label");
    // Generate the id as for the form fields (adds it to self).
    builder.GenerateId(fieldName);
    // Use the generated id for the 'for' attribute.
    builder.Attributes.Add("for", builder.Attributes["id"]);
    // Remove the id again.
    builder.Attributes.Remove("id");
    builder.SetInnerText(labelText);
    return MvcHtmlString.Create(builder.ToString());
}
Run Code Online (Sandbox Code Playgroud)

这解决了我的直接问题,但它没有回答为什么实现看起来像在MVC2中的问题.如果有原因的话.

顺便说一下:没有必要在HTML5中实际修改id/for属性,因为^~[]如果你愿意的话,拥有一个id是完全合法的.所有主流浏览器都支持它.Mathias Bynens很好地解释了这一点.

更新2:

这实际上并不能解决问题,因为DefaultModelBinder无论如何都无法绑定它.MVC 2中的字段名称生成器似乎不支持在字典中使用嵌套对象,因为它生成:

<input type="text" name="Dict[en]" value="(VALUE)">
Run Code Online (Sandbox Code Playgroud)

而不是模型绑定器想要的东西:

<input type="hidden" name="Dict[0].Key" value="en">
<input type="text" name="Dict[0].Value" value="(VALUE)">
Run Code Online (Sandbox Code Playgroud)

奇怪的是它以这种方式开箱即用.

我已经尝试为它创建一个自定义模型绑定器,但无论我尝试使用它,我都无法使用MVC2:

ModelBinders.Binders.Add(typeof(IDictionary<string,object>), new DictionaryModelBinder());
ModelBinders.Binders.Add(typeof(IDictionary<string,string>), new DictionaryModelBinder());
ModelBinders.Binders.Add(typeof(IDictionary), new DictionaryModelBinder());
ModelBinders.Binders.Add(typeof(Dictionary), new DictionaryModelBinder());
Run Code Online (Sandbox Code Playgroud)

所以现在它似乎又回到了手动创建隐藏.Key字段的名称属性值.

Cra*_*ntz 1

MVC 故意使用id特殊字符,这些字符在 jQuery/CSS3 选择器中很重要。这是因为当 ID 中存在“保留”(由 jQuery/CSS3)字符时,选择器语法会变得复杂。

它不会这样做name因为那里没有必要并且会妨碍绑定。

如果LabelFor实际上没有指向相应的TextBoxFor. 但我认为错误是在 中LabelFor,而不是在 中TextBoxFor