Коллекции MVC ViewModel являются нулевыми при обратной передаче

У меня есть модель представления, которая содержит ссылку на класс и 2 коллекции IEnumerable.

public class BuildingTypeViewModel
{
    public Static_Item BuildingType { get; set; }
    public IEnumerable<Reading_Type> ReadingTypes { get; set; }
    public IEnumerable<Building_Type_Reading> BuildingReadings { get; set; }
}

Я заполняю эту ViewModel в действии Edit на контроллере

    public ActionResult Edit(int id)
    {            
        Static_Item staticItem = db.Static_Item.Find(id);

        BuildingTypeViewModel model = new BuildingTypeViewModel
        {
            BuildingType = staticItem,
            ReadingTypes=db.Reading_Type.ToList(),
            BuildingReadings = db.Building_Type_Reading.Where(bt => bt.UN_Building_Type == staticItem.UN_Building_Type).ToList()
        };

        return View(model);
    }

Тип здания в модели представления — это класс с идентификатором и описанием, а данные будут примерно такими:

UN_Building_Type=1, Description = "Hospital"

IEnumerable для Reading_Type будет таким:

UN_Reading_Type = 1, Description = "Electric"    
UN_Reading_Type = 2, Description = "Gas"

IEnumerable Building_Type_Readings будет таким:

UN_Building_Type_Readings=1, UN_Building_Type=1, UN_Reading_Type = 1, Typical=300, Good=150
UN_Building_Type_Readings=2, UN_Building_Type=1, UN_Reading_Type = 2, Typical=800, Good=400

Я загружаю эти данные в свое представление:

@model SSE.Enterprise.EE_Web_Portal.Models.BuildingTypeViewModel

@using (Html.BeginForm())
{
    @Html.AntiForgeryToken()

    <div class="form-horizontal">
        <h4>Static_Item</h4>
        <hr />
        @Html.ValidationSummary(true, "", new { @class = "text-danger" })
        @Html.HiddenFor(model => model.BuildingType.UN_Building_Type)

        <div class="form-group">
            @Html.LabelFor(model => model.BuildingType.Description, htmlAttributes: new { @class = "control-label col-md-2" })
            <div class="col-md-10">
                @Html.EditorFor(model => model.BuildingType.Description, new { htmlAttributes = new { @class = "form-control" } })
                @Html.ValidationMessageFor(model => model.BuildingType.Description, "", new { @class = "text-danger" })
            </div>
        </div>

        <div class="form-group">
            <label class="control-label col-md-2">Building Readings</label>
            <div class="col-md-10">
                <table class="table">
                    <tr>
                        <th/>
                        <th>
                            @Html.LabelFor(model=>model.BuildingReadings.FirstOrDefault().Typical)
                        </th>
                        <th>
                            @Html.LabelFor(model => model.BuildingReadings.FirstOrDefault().Good)
                        </th>
                    </tr>

                    @foreach (var item in Model.ReadingTypes)
                    {
                        <tr>
                            <td>
                                @Html.Label(item.Description)
                            </td>
                            <td>
                                @Html.EditorFor(model => model.BuildingReadings.FirstOrDefault(b => b.UN_Reading_Type == item.UN_Reading_Type).Typical, new { htmlAttributes = new { @class = "form-control" } })
                            </td>
                            <td>
                                @Html.EditorFor(model => model.BuildingReadings.FirstOrDefault(b => b.UN_Reading_Type == item.UN_Reading_Type).Good, new { htmlAttributes = new { @class = "form-control" } })
                            </td>
                        </tr>
                    }
                </table>
            </div>
        </div>

        <div class="form-group">
            <div class="col-md-offset-2 col-md-10">
                <input type="submit" value="Save" class="btn btn-default" />
            </div>
        </div>
    </div>
}

<div>
    @Html.ActionLink("Back to List", "Index")
</div>

Вот мой метод постбэка редактирования.

    [HttpPost]
    [ValidateAntiForgeryToken]
    public ActionResult Edit([Bind(Include = "BuildingType")] BuildingTypeViewModel model)
    {   
        if (ModelState.IsValid)
        {
            db.Entry(model).State = EntityState.Modified;
            db.SaveChanges();
            return RedirectToAction("Index");
        }
        return View(model);
    }

Когда страница отправляется обратно, 2 коллекции IEnumerable являются нулевыми, поэтому я не могу сохранить введенные данные.

Есть идеи?

Ta


person Sun    schedule 12.11.2015    source источник
comment
разве в вашей начальной форме не должно быть более подробной информации? например: использование (Html.BeginForm(Action, Controller, FormMethod.Post,...   -  person Mark Homer    schedule 12.11.2015
comment
@MarkHomer: Нет. Все это опционально. Если вы не передаете действие и контроллер, используются текущие действие и контроллер. По умолчанию используется POST, поэтому его также не нужно устанавливать.   -  person Chris Pratt    schedule 12.11.2015
comment
Вы не можете использовать цикл foreach для создания элементов управления формы - проверьте html, который вы генерируете. Атрибуты name не имеют никакого отношения к вашему mdoel. Вам нужно использовать цикл for (и свойства вашей коллекции должны быть IList<T>, а не IEnumerable<T>. Или, в качестве альтернативы, вам нужно использовать пользовательский EditorTemplates для typeof Reading_Type и Building_Type_Reading   -  person    schedule 13.11.2015
comment
@StephenMuecke, глядя на представление, я бы согласился с этим, хотя не смотрел слишком внимательно, все выглядит немного беспорядочно, лол, похоже на кошмар для отладки, я бы пересмотрел дизайн   -  person Mark Homer    schedule 13.11.2015
comment
@MarkHomer, всякий раз, когда вы видите что-то настолько ужасное, как m=> m.BuildingReadings.FirstOrDefault(b => b.UN_Reading_Type == item.UN_Reading_Type).Typical, вы можете быть уверены, что это нужно переделать :)   -  person    schedule 13.11.2015


Ответы (1)


Прежде всего, указав атрибут Bind(Include = "BuildingType"), вы указываете MVC привязывать только это единственное свойство. Удалите атрибут, и MVC попытается связать ваши 2 коллекции IEnumerable.

Затем проверьте свои @Html.EditorFor звонки. Я не уверен, что MVC может понять FirstOrDefault внутри. Старайтесь избегать селекторов LINQ внутри вашего представления.

И после этого, как уже упоминал @will, попробуйте изменить IEnumerable на List.

person holdenmcgrohen    schedule 12.11.2015
comment
Ты прав. Для правильной привязки вам необходимо проиндексировать элементы в коллекции, т. е. @Html.EditorFor(model => model.BuildingReadings[0].Typical) - person Chris Pratt; 12.11.2015
comment
Я удалил атрибут Bind и LINQ, и он остался прежним. - person Sun; 12.11.2015
comment
Попробуйте посмотреть исходный HTML-код страницы, которую вы получаете. Ваши входные данные должны выглядеть так: <input name="BuildingReadings[0].Typical" ... >, <input name="BuildingReadings[0].Good" ... > и т. д. Если ничего не помогает, вы всегда можете воспользоваться любым руководством по привязке к коллекциям в MVC (например, этот), затем постепенно изменяйте его, чтобы он соответствовал вашей модели представления, и смотрите, на каком этапе привязка не удалась. - person holdenmcgrohen; 13.11.2015