DropDownListFor not respecting Selected property of SelectList

前端 未结 2 1291
北海茫月
北海茫月 2020-12-17 19:37

I have the following line of code:

@Html.DropDownListFor(x => x.TimeOption, new SelectList(Model.TimeOptions, \"Value\", \"Name\", (int)Model.TimeOption))         


        
相关标签:
2条回答
  • 2020-12-17 20:24

    This tends to stump people a lot - apparently the roadmap for MVC 5 includes more intuitive DropDownList helpers.

    When you do:

    @Html.DropDownListFor(x => x.TimeOption ...)
    

    ... the HTML helper will do everything in its power not to use the value you set to Selected = true in your SelectList. It will:

    • Try to get it from ModelState - which it won't get on the GET request
    • Try to get it from ViewData - which it will get, because you have a TimeOption property on your model.

    Only if both of those attempts fail, will it succumb, and use what you asked it to use.

    So, in your case, it will use the current value of TimeOption on your model as the selected value. Which should be fine, because that's what you asked for in your SelectList anyway ((int)Model.TimeOption), right?

    Wrong. Because it will not cast your property (enum?) to int like you do. It will convert it to a string. For an enum that yields the enumeration's name - e.g. PastMonth. If TimeOption is a custom class of yours, it will be the result of ToString().

    Then, when it doesn't find that string as one of your <select> values (because they're all integers), it decides not to make anything selected.

    In short: SelectListItem.Selected has no effect on the helper. If your helper call specifies a name (either literally or through a lambda expression), the helper will use anything in ModelState or ViewData with that name, and convert it to a string.

    In most cases this happens with enumerations. The easiest way to get around it is to use the enumeration's name, rather than its int representation, as the value for the SelectListItem.

    0 讨论(0)
  • 2020-12-17 20:27

    You've already got a good explanation above as to why its not working, but just to add my 2p

    Generally I prefer using an enumerable of SelectListItems as the source of dropdownitems, although either should work. It's just preferable doing it this way as there's no need to specified which item is selected - the MVC helper will infer it from the one who's Value matches the TimeOption.ToString() method.

    ie

    @Html.DropDownListFor(x => x.TimeOption, ((IEnumerable<TimeOption>)Model.TimeOptions).ToList().ConvertAll(y => new SelectListItem(){Text = y.Name, Value = y.Value.ToString()})
    

    NB - I'm not sure what type Model.TimeOptions is here - if its not IEnumerable< TimeOption> then replace it. (if its already a list then .ToList() is not necessary)

    0 讨论(0)
提交回复
热议问题