这个问题让我疯了好几个小时了......
在我的域中,我有2个彼此相关的实体Sku
和Item
。每个sku可以有很多项目。
public class Sku
{
private readonly EntitySet<Item> items;
public Sku()
{
items = new EntitySet<Item>(AttachItems, DetachItems);
}
public int SkuId { get; set; }
public string LongDescription { get; set; }
public EntitySet<Item> Items
{
get { return items; }
set{ items.Assign(value);}
}
private void AttachItems(Item entity)
{
entity.Sku = this;
}
private static void DetachItems(Item entity)
{
entity.Sku = null;
}
}
public class Item
{
public Sku Sku { get; set; }
public int ItemId { get; set; }
public string Category { get; set; }
public string Description { get; set; }
}
我正在构建一个页面,允许最终用户同时更新sku上的某些字段和每个项目的某些字段。
<% using (Html.BeginForm("Save", "Merchant", FormMethod.Post,
new { enctype = "multipart/form-data" })) { %>
<fieldset>
<legend>Sku</legend>
<p><label for="SkuId">SkuId:</label>
<%= Html.TextBox("SkuId", Model.SkuId,
new{@readonly="readonly",onfocus="this.blur();"}) %></p>
<p><label for="LongDescription">LongDescription:</label>
<%= Html.TextBox("LongDescription", Model.LongDescription) %></p>
</fieldset>
<% for (int i = 0; i < Model.Items.Count; i++) { %>
<fieldset>
<legend>Item</legend>
<p><label for="ItemId">ItemId:</label>
<%= Html.TextBox(string.Format("items[{0}].{1}", i, "ItemId"),
Model.Items[i].ItemId,
new { @readonly = "readonly", onfocus = "this.blur();" })%></p>
<p><label for="Category">Category:</label>
<%= Html.TextBox(string.Format("items[{0}].{1}", i, "Category"),
Model.Items[i].Category)%></p>
<p><label for="Description">Description:</label>
<%= Html.TextBox(string.Format("items[{0}].{1}", i, "Description"),
Model.Items[i].Description)%></p>
</fieldset>
<%} // for-loop %>
<p><input type="submit" value="Save" /></p>
<%} // form %>
我有一些控制器代码可以接受Sku
和EntitySet
Item
,然后将Items
分配给Sku
。< / p>
[AcceptVerbs(HttpVerbs.Post)]
public ActionResult Save(Sku sku, EntitySet<Item> items)
{
if (sku != null)
{
if (items != null)
{
sku.Items.Assign(items);
}
}
// save Sku to repository ...
// return Details view ...
}
这有效,但我注意到除了DefaultModelBinder
的一次旅行之外,每Item
次Sku
两次。绑定Sku
时,将调用Items
的setter,并且绑定器甚至会使用正确的值传递水合Items
集合。但是,在致电items.Assign
后,Items.Count
为0
。这就是我必须重新分配控制器代码中的项目的原因。我期待通过活页夹将项目转移到Items
集合。这应该消除每个项目的额外行程,因为我的控制器方法上的items
参数可以被删除。为什么这不起作用?
答案 0 :(得分:1)
您可能需要为此创建自定义模型绑定器吗?
答案 1 :(得分:1)
希望我能正确理解你的问题......
您是否尝试使用单个参数定义 Sku ?
,而不是使用2个参数定义保存操作?然后,您需要重新定义项目HTML控件,类似于以下示例...
<%=
Html.TextBox
(
string.Format("sku.Items[{0}].{1}", i, "ItemId"),
Model.Items[i].ItemId,
new { @readonly = "readonly", onfocus = "this.blur();" }
)
%>
这样,您就可以直接在Sku对象中填充项目。
另一个可能的解决方案是在Item类中添加一个额外的字段,例如......
Int32 SkuId { get; set; }
这样,您可以为视图中的每个项目定义一个额外的隐藏字段,该字段将自动绑定到控制器上每个项目的SkuId。
<%=
Html.Hidden
(
string.Format("sku.Items[{0}].{1}", i, "SkuId"),
Model.Items[i].SkuId
)
%>
然后,您可以独立于Sku对象更新项目集合。无论你走哪条路,两个集合都必须明确告诉Linq SQL以无论如何更新sku和项目。
您也可以定义自己的活页夹类,但在这种情况下,这可能比它的价值更多。只需遵循ASP.NET MVC约定,我认为你应该能够找到一些可以工作的东西,而不会觉得它是一个黑客。
答案 2 :(得分:0)
我有一个类似的问题,其中EntitySets未在我的ViewModel中正确绑定。
我所做的是创建另一个名为Mvc [YourEntitySetPropertyName]的属性,作为通用列表,它包含EntitySet的私有字段:
public List<InvoiceLineItem> MvcInvoiceLineItemList
{
get { return _invoiceLineItemList.ToList(); }
set { _invoiceLineItemList.AddRange(value); }
}
然后我在视图标记上使用了它而不是EntitySet属性。
不需要传递控制器方法签名中的项目 - 只需通过Sku:
[AcceptVerbs(HttpVerbs.Post)]
public ActionResult Save(Sku sku)
{
if (sku != null)
{
// save Sku to repository ...
// return Details view ...
}
}