给出一个模型:
public class Receipt: BaseEntity
{
[Required(AllowEmptyStrings = false,ErrorMessage = "Please Select A Store")]
public Store Store { get; set; }
public List<Item> Items { get; set; }
public Item NewItem { get; private set; }
[Required(AllowEmptyStrings = false,ErrorMessage = "ReceiptModel.EntryDate is required")]
public DateTime EntryDate { get; set; }
public Guid EntryOwner { get; set; }
public string Name { get; set; }
public double ReceiptTotal()
{
return Items.Sum(item => item.ItemPrice);
}
public Receipt()
{
Items = new List<Item>();
EntryOwner = Guid.Empty;
EntryDate = DateTime.Now;
Store = new Store();
NewItem = new Item();
}
}
和控制器:
public class ReceiptController : Controller
{
//
// GET: /Receipt/
public ActionResult Index()
{
Receipt receipt = new Receipt();
return View(receipt);
}
[HttpPost]
public ActionResult AddItem(Receipt receipt)
{
receipt.Items.Add(receipt.NewItem);
if (ModelState.IsValid)
{
return View("Index", receipt);
}
return View("Index", receipt);
}
一个视图:更新后的视图 - 删除了所有部分页面并合并为一个页面
<form class="form-horizontal" method="POST" action="@Url.Action("AddItem","Receipt")">
<div class="jumbotron">
<div id="main" class="panel panel-primary">
<div class="panel-heading"> <h1 class="panel-title">Receipt Builder</h1></div>
<div class="panel-body">
<fieldset>
@Html.LabelFor(model => model.Name)
@Html.EditorFor(model => model.Name)
@Html.LabelFor(model => model.Store.Name)
@Html.EditorFor(model => model.Store.Name)
@Html.LabelFor(model => model.EntryDate)
@Html.EditorFor(model => model.EntryDate)
</fieldset>
</div>
</div>
</div>
<table class="table table-bordered table-condensed table-hover table-responsive table-striped">
<thead>
<tr>
<td>
<label class="control-label">#</label>
</td>
<td>
<label class="control-label">Item</label>
</td>
<td>
<label class="control-label">Unit</label>
</td>
<td>
<label class="control-label">Quantity</label>
</td>
<td>
<label class="control-label">UnitPrice</label>
</td>
<td>
<label class="control-label">Category</label>
</td>
<td>
<label class="control-label">Total</label>
</td>
</tr>
<tr>
<td>
<button class="btn-sm btn-danger" style="border-radius: 50%; background-color: red;" name="Submit" id="Submit" type="submit">+</button>
</td>
<td>
@Html.EditorFor(model => model.NewItem.Name)
</td>
<td>
@Html.EditorFor(model => model.NewItem.UnitType)
</td>
<td>
@Html.EditorFor(model => model.NewItem.Quantity)
</td>
<td>
@Html.EditorFor(model => model.NewItem.UnitPrice)
</td>
<td>
@Html.EditorFor(model => model.NewItem.Category.Name)<br/>
@Html.EditorFor(model => model.NewItem.Category.IsTaxable)
</td>
<td>
@Html.EditorFor(model => model.NewItem.ItemPrice)
</td>
</tr>
</thead>
<tbody>
@foreach (var item in Model.Items)
{
<tr>
<td>
@Model.Items.ToList().IndexOf(item) + 1
</td>
<td>
@Html.LabelFor(titem => item.Name)
</td>
<td>
@Html.LabelFor(titem => item.UnitType)
</td>
<td>
@Html.LabelFor(titem => item.Quantity)
</td>
<td>
@Html.LabelFor(titem => item.UnitPrice)
</td>
<td>
@Html.LabelFor(titem => item.Category.Name)
</td>
<td>
@Html.LabelFor(titem => item.ItemPrice)
</td>
</tr>
}
</tbody>
<tfoot></tfoot>
</table>
答案 0 :(得分:6)
这确实是一个具有约束力的问题。似乎对于像Lists这样的复杂集合,我们实际上无法像我希望的那样利用IEnumerable的好处。这是由于数据在具有不明确名称的帖子中实际传递的方式。这是最终的代码(只需要更新视图)
@for (int i = 0; i < Model.Items.Count(); i++)
{
<tr>
<td>
@(i+1)
</td>
<td>
@Html.TextBoxFor(m => m.Items[i].Name)
</td>
<td>
@Html.TextBoxFor(m => m.Items[i].UnitType)
</td>
<td>
@Html.TextBoxFor(m => m.Items[i].Quantity)
</td>
<td>
@Html.TextBoxFor(m => m.Items[i].UnitPrice)
</td>
<td>
@Html.TextBoxFor(m => m.Items[i].Category.Name)
</td>
<td>
@Model.Items[i].ItemPrice
</td>
</tr>
}
生成的HTML将每个元素命名为其索引,即Item [N] .Name 现在可以正确绑定帖子数据。
答案 1 :(得分:1)
收据中的项目显示为文本,没有控件。因此,当您将数据发布回服务器时,ASP.NET MVC无法从请求正文和LETS Receipt.Items集合中找到此信息。
您可以借助Html.HiddenFor()方法在部分视图中将输入的收据信息添加为隐藏字段。或者只是将输入的项目保存在DB或其他地方,并在返回AddItem视图之前将它们读入Receipt.Items集合。