MVC在子集合中添加和删除项目

时间:2012-09-12 19:22:39

标签: c# asp.net-mvc razor collections model-binding

我无法将数据绑定到集合项的集合(我也无法正确写入问题)。让我们通过使用psudo模型的示例让每个人都更容易。

假设我有以下示例模型:

public class Month()
{
    public int ID { get; set; }
    public string Name { get; set; }
    public virtual ICollection<Week> Weeks { get; set; }
}

public class Week()
{
    public int ID { get; set; }
    public int MonthID { get; set; }
    public String Name { get; set; }
    public virtual ICollection<Day> Days { get; set; }
}

public class Day()
{
    public int ID { get; set; }
    public String Name { get; set; }
}

...以及一个示例viewmodel:

public class EditMonthViewModel()
{
    public Month Month { get; set; }
    public List<Week> Weeks { get; set; }
    public List<Day> AllDays { get; set; }
}

“编辑操作/查看”的目的是使用户能够编辑一个月,分配到该月的周数,以及添加和删除某个月的几周内的天数。视图可能会有所帮助。

@model myProject.ViewModels.EditMonthViewModel

//...
@using (Html.BeginForm())
{
    //Edit Month Stuff...

    @for(int i = 0; i < Model.Weeks.Count(); i++)
    {
        <h2>@Model.Weeks[i].Name</h2>
        @Html.EditorFor(model => Model.Weeks[i].Name)

        //loop through all possible days
        //Select only days that are assigned to Week[i] 
        @for(int d = 0; d < Model.AllDays.Count(); d ++)
        {
            //This is the focus of this question.  
            //How do you bind the data here?
            <input type="checkbox"
                   name="I have no idea" 
                   @Html.Raw(Model.Weeks[i].Days.Contains(Model.AllDays[d]) ? "checked" : "") />
        }
    }
}

控制器操作方法

public ActionResult Edit(int id)
{
    var viewModel = new EditMonthViewModel();
    viewModel.Month = db.Months.Find(id);
    viewModel.Weeks = db.Weeks.Where(w => w.MonthID == id).ToList();

    viewModel.AllDays = db.Days.ToList();
}

[HttpPost]
public ActionResult Edit(EditMonthViewModel viewModel)
{
    var monthToUpdate = db.Months.Find(viewModel.Month.ID);
    //...

    if(viewModel.Weeks != null)
    {
        foreach (var week in viewModel.Weeks)
        {
            var weekToUpdate = monthToUpdate.Weeks.Single(w => w.ID == week.ID);
            //...

            /*So, I have a collection of weeks that I can grab, 
              but how do I know what was selected? My viewModel only has a 
              list of AllDays, not the days selected for Week[i]
            */
        }
}

如何确保在提交表单时所选日期将与本周绑定?

1 个答案:

答案 0 :(得分:0)

看起来最简单的事情就是让表单填充IEnumerable<DayModel>类型的数据结构,其中DayModel定义为:

public class DayModel
{
    public int WeekId { get; set; }
    public int DayId { get; set; }
    public bool IsIncluded { get; set; }
}

您可以将Razor代码保留为大部分内容,但是在渲染复选框时,您可以执行以下操作:

@{
    var modelIdx = 0;
}

// ...

<input type="hidden" name="days[@modelIdx].WeekId" value="@Model.Weeks[i].Id" />
<input type="hidden" name="days[@modelIdx].DayId" value="@Model.AllDays[d].Id" />
<input type="checkbox" name="days[@modelIdx].IsIncluded" value="@(Model.Weeks[i].Days.Contains(Model.AllDays[d]) ? "checked" : "")" />
@{ modelIdx++; }

然后,您发布的控制器操作可以具有此签名:

[HttpPost]
public ActionResult Edit(IEnumerable<DayModel> days)
{
    //...
}

有助于我的是永远不要混淆视图模型,视图模型应仅用于视图模型(通常是GET操作)和非视图模型(我们称之为普通模型)。避免让你的POST动作试图绑定到视图模型,这将大大简化你的生活。