簡體   English   中英

在ListBoxFor中選擇值的挑戰

[英]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.

 
粵ICP備18138465號  © 2020-2024 STACKOOM.COM