Html Helper下拉列表在提交/在数据库中将值切换到顶部选项

时间:2016-01-06 16:01:29

标签: asp.net-mvc forms enums html-helper viewbag

我正在填写表单,但是当从下拉列表中选择一个选项并单击“提交”时,无论我选择哪个选项,它都会解析最常见的选项。显示的值永远不会改变,因此您将其保留为默认选项'请选择...'然后点击提交,这样就可以选择...'但数据库中的条目始终是显示在下拉列表顶部的条目。

以下是模型:

public enum Medium
{
    [Description("Teleconference & Report")]
    Teleconference_Report,
    [Description("Email & Telephone")]
    Email_Telephone
}

[Required]
[Display(Name = "Medium")]
public Medium Medium { get; set; }

以下是表格中的字段:

<div class="form-group">
    @Html.LabelFor(model => model.Medium, htmlAttributes: new { @class = "control-label col-md-2" })
    <div class="col-md-5">
        @Html.DropDownList("MediumID", null, "Please select...", htmlAttributes: new { @class = "form-control" })
        @Html.ValidationMessageFor(model => model.Medium, "", new { @class = "text-danger" })
    </div>
</div>

&#34; MediumID&#34;使用viewbag填充DropDownList,该视图包设置为以下任何返回值:

// Puts all of the mediums of communication into a user friendly dropdownlist.
public List<SelectListItem> GetMediumList()
{
    List<SelectListItem> mediumList = new List<SelectListItem>();

    foreach (Medium state in EnumToList<Medium>())
    {
        mediumList.Add(new SelectListItem
        {
            Text = GetEnumDescription(state),
            Value = state.ToString(),
        });
    }

    return mediumList;
}

下面显示另一个名为&#39;频率&#39;的枚举的表单部分,但这些不会更改为用户友好的字符串(并且工作正常)。

<div class="form-group">
    @Html.LabelFor(model => model.Frequency, htmlAttributes: new { @class = "control-label col-md-2" })
    <div class="col-md-5">
        @Html.EnumDropDownListFor(model => model.Frequency, "Please select...", htmlAttributes: new { @class = "form-control" })
        @Html.ValidationMessageFor(model => model.Frequency, "", new { @class = "text-danger" })
    </div>
</div>

下面显示了将枚举转换为用户友好字符串的两种方法:

// Returns a 'user friendly', readable version of the enum.
public static string GetEnumDescription(Enum value)
{
    FieldInfo fi = value.GetType().GetField(value.ToString());

    DescriptionAttribute[] attributes =
        (DescriptionAttribute[])fi.GetCustomAttributes(typeof(DescriptionAttribute), false);

    if (attributes != null && attributes.Length > 0)
        return attributes[0].Description;
    else
        return value.ToString();
}

// Puts all of the same enums into a list.
public static IEnumerable<T> EnumToList<T>()
{
    Type enumType = typeof(T);

    // Can't use generic type constraints on value types,
    // so have to do check like this.
    if (enumType.BaseType != typeof(Enum))
        throw new ArgumentException("T must be of type System.Enum");

    Array enumValArray = Enum.GetValues(enumType);
    List<T> enumValList = new List<T>(enumValArray.Length);

    foreach (int val in enumValArray)
    {
        enumValList.Add((T)Enum.Parse(enumType, val.ToString()));
    }

    return enumValList;
}

最后,这里是方法签名,其中字段是绑定/绑定的:

[HttpPost]
[ValidateAntiForgeryToken]
public ActionResult Create([Bind(Include = "Point,ApplicationID,MediumID,Frequency,StartDate,EndDate")] TouchPoint touchPoint)

在此方法中,使用以下内容将下拉列表传递给视图:

ViewBag.MediumID = GetMediumList();

非常感谢任何帮助。

1 个答案:

答案 0 :(得分:1)

您的模型有一个名为Medium的属性,但您的视图不会绑定到该属性。您生成的<select>的名称MediumID在您的模型中不存在,因此提交时Medium的默认值为Teleconference_Report(第一个枚举值)

将视图更改为

 @Html.DropDownListFor(m => m.Medium, (IEnumerable<SelectListItem>)ViewBag.MediumID, "Please select...", new { @class = "form-control" })

虽然我建议将ViewBag属性名称更改为MediumList,以使其更清楚地显示其集合。更好的是,使用具有属性public IEnumerable<SelectListItem> MediumList { get; set; }的视图模型,以便视图可以为@Html.DropDownListFor(m => m.Medium, Model.MediumList, .... )

您还需要更改[Bind]属性以包含"Medium"(并删除"MediumID"),但使用视图模型意味着不需要[Bind]属性。

附注:您不需要[Required]属性,除非您想使用ErrorMessage = "..."属性添加特定的错误消息(默认情况下始终需要enum,除非您创建财产nullable)。