Бритва: если модель - это список <>, то @Html.LabelFor создает пустое поле "для"

Если модель представляет собой список объектов, то @Html.LabelFor model => model[i].member) создает пустой атрибут for. DisplayName работает правильно, привязка к модели также отлично работает. Единственное, что не работает, это атрибут for.

Ниже приведены две версии кода MVC. Первые 3 фрагмента кода предназначены для комбинации модели, контроллера и представления, которая не работает (модель представляет собой список). Второй набор из 3 фрагментов кода работает. На этот раз список завернут в объект.

Что мне не хватает?

Модель:

public class ViewModel {
    public bool ClickMe { get; set; }
}

Контроллер:

public ActionResult LabelForViewModel() {
    List<ViewModel> model = new List<ViewModel>() {
        new ViewModel() { ClickMe = true}
    };
    return View(model);
}

Вид:

@model List<ModelBinding.Models.ViewModel>
// form code removed for brevity
@for (var i = 0; i < Model.Count; i++) {
    <div class="form-group">
        @Html.LabelFor(model => model[i].ClickMe)
        <div class="col-md-10">
            @Html.EditorFor(model => model[i].ClickMe, new { htmlAttributes = new { @class = "form-control" } })
            @Html.ValidationMessageFor(model => model[i].ClickMe, "", new { @class = "text-danger" })
        </div>
    </div>
}

Разметка:

<div class="form-group">
    <label for="">ClickMe</label>
    <div class="col-md-10">
        <input name="[0].ClickMe" class="form-control check-box" type="checkbox" checked="checked" value="true" data-val-required="The ClickMe field is required." data-val="true"><input name="[0].ClickMe" type="hidden" value="false">
        <span class="field-validation-valid text-danger" data-valmsg-replace="true" data-valmsg-for="[0].ClickMe"></span>
    </div>
</div>

Обратите внимание, что поле for пусто и вход не имеет поля id (это поле используется в поле "для" ).

Я создал модель обертки, которая содержит список объектов, модифицировала контроллер и представление, а затем она работает нормально. В этом случае поле "for" заполняется правильно, и все работает так, как ожидалось.

public class ListWrapper {
    public List<ViewModel> ViewModels { get; set; }
}

Вид:

@for (var i = 0; i < Model.ViewModels.Count; i++) {
    <div class="form-group">
        @Html.LabelFor(model => model.ViewModels[i].ClickMe)
        <div class="col-md-10">
            @Html.EditorFor(model => model.ViewModels[i].ClickMe, new { htmlAttributes = new { @class = "form-control" } })
            @Html.ValidationMessageFor(model => model.ViewModels[i].ClickMe, "", new { @class = "text-danger" })
        </div>
    </div>
}

Разметка:

<div class="form-group">
    <label for="ViewModels_0__ClickMe">ClickMe</label>
    <div class="col-md-10">
        <input name="ViewModels[0].ClickMe" class="form-control check-box" id="ViewModels_0__ClickMe" type="checkbox" checked="checked" value="true" data-val-required="The ClickMe field is required." data-val="true"><input name="ViewModels[0].ClickMe" type="hidden" value="false">
        <span class="field-validation-valid text-danger" data-valmsg-replace="true" data-valmsg-for="ViewModels[0].ClickMe"></span>
    </div>
</div>

Итак, кажется, что атрибут for пуст, только если модель представляет собой список объектов.

Это известная проблема? Или, может быть, я что-то упустил?

Ответ 1

Это поведение является результатом структуры, соответствующей спецификациям HTML-4. В HTML-4

Идентификаторы ID и NAME должны начинаться с буквы ([A-Za-z]), за которой может следовать любое количество букв, цифр ([0-9]), дефис ( "-" ), подчеркивание ( "_" ), двоеточие ( ":" ) и периоды ( "." ).

Когда ваша модель представляет собой коллекцию, и вы используете @Html.EditorFor(m => m[i].SomeProperty) для создания элементов управления формой, метод генерирует атрибуты name и id, основанные на имени свойства. В этом случае атрибут name для первого элемента в коллекции будет

name="[0].SomeProperty"

но чтобы избежать столкновений с селекторами jQuery, метод заменяет символы ., [ и ] символом подчеркивания _ при создании атрибута id, который приведет к

id="_0__SomeProperty"

но поскольку это недопустимо в соответствии со спецификацией HTML-4, хелпер не выводит его в html.

То же самое происходит при использовании @Html.LabelFor() (так как связанный элемент управления формы не имеет атрибута id, значение атрибута for не выводится в html.

Обратите внимание, что в HTML-5 атрибут id будет действителен, поэтому, надеюсь, это поведение будет удалено в будущем.

Второй пример работает, потому что id="ViewModels_0__ClickMe "начинается с буквы и поэтому является допустимым.

Боковое примечание. Вы можете решить первую проблему, создав собственные атрибуты id, например

@Html.LabelFor(model => model[i].ClickMe, new { for = string.Format("item{0}", i) })
@Html.EditorFor(model => model.ViewModels[i].ClickMe, new { htmlAttributes = new { @class = "form-control", id = string.Format("item{0}", i) } })

Ответ 2

Объявление @model присваивает объект модели переменной Model (обратите внимание на "M" ). Но тогда вы читаете от Model, который не был назначен.

Попробуйте следующее:

  @model List<ModelBinding.Models.ViewModel>
  // form code removed for brevity
        @for (var i = 0; i < Model.Count; i++) {
            <div class="form-group">
                @Html.LabelFor(model => Model[i].ClickMe)
                <div class="col-md-10">
                    @Html.EditorFor(model => Model[i].ClickMe, new { htmlAttributes = new { @class = "form-control" } })
                    @Html.ValidationMessageFor(model => Model[i].ClickMe, "", new { @class = "text-danger" })
                </div>
            </div>
        }

Это не объясняет, почему ваш второй пример работает вообще, но эй, попробуйте.