[英]Challenges with selecting values in ListBoxFor
最近在我的第一個ASP.Net MVC2 Web應用程序上工作,當我需要在列表框中選擇多個值時,我遇到了一些問題。 我用一些jQuery解決了這個問題,但是繼續編寫了一些非常簡單的代碼來演示。 我使用EF作為模型,有兩個實體 - Customers和HelpDeskCalls:
控制器:
public ActionResult Edit(int id)
{
Customer currCustomer = ctx.Customers.Include("HelpDeskCalls").Where(c => c.ID == id).FirstOrDefault();
List<HelpDeskCall> currCustCalls = (ctx.HelpDeskCalls.Where(h => h.CustomerID == id)).ToList();
List<SelectListItem> currSelectItems = new List<SelectListItem>();
List<String> selectedValues = new List<string>();
foreach (HelpDeskCall currCall in currCustCalls)
{
bool isSelected = (currCall.ID % 2 == 0) ? true : false;
//Just select the IDs which are even numbers...
currSelectItems.Add(new SelectListItem() { Selected = isSelected, Text = currCall.CallTitle, Value = currCall.ID.ToString() });
//add the selected values into a separate list as well...
if (isSelected)
{
selectedValues.Add(currCall.ID.ToString());
}
}
ViewData["currCalls"] = (IEnumerable<SelectListItem>) currSelectItems;
ViewData["currSelected"] = (IEnumerable<String>) selectedValues;
return View("Edit", currCustomer);
}
視圖:
<div class="editor-field">
<%: Html.ListBoxFor(model => model.HelpDeskCalls, new MultiSelectList(Model.HelpDeskCalls, "ID", "CallTitle", (IEnumerable) ViewData["currSelected"]), new { size = "12" })%>
<%: Html.ListBoxFor(model => model.HelpDeskCalls, ViewData["currCalls"] as IEnumerable<SelectListItem>, new { size = "12"}) %>
<%: Html.ListBox("Model.HelpDeskCalls", new MultiSelectList(Model.HelpDeskCalls, "ID", "CallTitle", (IEnumerable)ViewData["currSelected"]), new { size = "12"}) %>
<%: Html.ValidationMessageFor(model => model.HelpDeskCalls) %>
</div>
對於這個示例,我只是選擇均勻的HelpDeskCall.IDs。 我正在為ListBoxFor嘗試兩種不同的語法:一種使用IEnumerable值進行選擇,一種使用IEnumerable的SelectListItems。 默認情況下,當我運行此代碼時,不會對ListBoxFor進行任何選擇,但非強類型ListBox會正確選擇。
我讀這篇文章的ASP.Net和這個線程的SO,但沒有喜悅。 實際上,如果我將覆蓋ToString()添加到我的HelpDeskCall類(如ASP.net線程中所建議的那樣),則選擇所有值,這也是不對的。
如果有人能夠闡明這應該如何發揮作用(以及我錯過或做錯了什么),那么新手就會非常感激。
這是一個說明強類型版本的示例:
模型:
public class MyViewModel
{
public int[] SelectedItemIds { get; set; }
public MultiSelectList Items { get; set; }
}
控制器:
public class HomeController : Controller
{
public ActionResult Index()
{
// Preselect items with id 1 and 3
var selectedItemIds = new[] { 1, 3 };
var model = new MyViewModel
{
Items = new MultiSelectList(
new[]
{
// TODO: Fetch from your repository
new { Id = 1, Name = "item 1" },
new { Id = 2, Name = "item 2" },
new { Id = 3, Name = "item 3" },
},
"Id",
"Name",
selectedItemIds
)
};
return View(model);
}
}
視圖:
<%: Html.ListBoxFor(x => x.SelectedItemIds, Model.Items) %>
我不知道在我使用的MVC3的RTM中這種行為是否已經改變,但似乎選擇和綁定現在開箱即用。 唯一的問題是模型應該包含帶有ID的屬性,如下所示:
public class MyViewModel {
public int[] ItemIDs { get; set; }
}
然后視圖中的以下內容將正常工作,既可以預先選擇正確的值,也可以在發布期間正確綁定:
@Html.ListBoxFor(model => model.ItemIDs, (IEnumerable<SelectListItem>)(new[] {
new SelectListItem() { Value = "1", Text = "1" },
new SelectListItem() { Value = "2", Text = "2" }
}))
我找到了更好的解決方法。 通常的方式來預選選擇列表:
@Html.ListBoxFor(
model => model.Roles,
new MultiSelectList(db.Roles, "Id", "Name")
)
@Html.ValidationMessageFor(model => model.Roles)
不起作用..,從未選擇任何選項,直到:
public ActionResult Edit(int id)
{
var user = db.Users.Find(id);
// this is workaround for http://aspnet.codeplex.com/workitem/4932?ProjectName=aspnet
ViewData["Roles"] = user.Roles.Select(r => r.Id);
return View(user);
}
選定的角色必須存儲在ViewData中,以解決令人討厭的錯誤。
另一個選擇是利用nameof,你可以做這樣的事情;
Html.ListBox(nameof(MainProjectViewModel.Projects), Model.Projects)
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.