2012-09-12 111 views
2

我在将数据绑定到集合项目集合时遇到问题(我也无法正确说明我的问题)。让我们通过使用psudo模型的示例让每个人都更容易。MVC添加和删除子集合中的项目

可以说我有下面的例子型号:

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; } 
} 

...和实例视图模型:

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] 
      */ 
     } 
} 

我怎么能保证,当我提交表单选定日子将绑定到周?

+0

你有什么问题/错误体验?我们可以看到你的控制器处理这个表单的行为吗? –

+1

我也有兴趣看到填充你的ViewModel的控制器动作。有趣的是,你的Month类已经有了Weeks的引用,并且间接引用了Days(到Weeks),但你已经将这些额外的字段添加到了你的ViewModel中。为什么? –

+0

我添加了一些psudo操作方法。我是MVC的新手,我遵循一种理念,即视图不应该运行查询(在视图模型中发送视图所需的所有内容)。如果你不认为这是必要的,那么我当然可以改变事物。我的主要问题(我有困难阐述)是,我不知道复选框的“名称”属性需要为模型联编程序分配一天。我会再增加一小段来更好地解释这种情况。 – jbaum012

回答

0

看起来最容易做的事情是让一个目标表单来填充型IEnumerable<DayModel>,其中DayModel被定义为一个数据结构:

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

你可以让你的剃刀代码因为是大部分,但随后当谈到渲染复选框,你可以做这样的事情:

@{ 
    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动作试图绑定到视图模型,这将大大简化你的生活。