asp.net将具有视图模型的表单发布到控制器操作方法

时间:2014-02-25 12:01:38

标签: asp.net asp.net-mvc forms post actionmethod

我有一个相当复杂的表格,我需要发布到我的MVC控制器。

这是我最初传递给创建视图的View模型:

public class EditViewModel
{
    public Service service { get; set; }
    public bool sms { get; set; }
    public bool email { get; set; }
    public string userId { get; set; }
}

这是我的观点(简化):

@model IList<Service_Monitor_Web_Interface.Models.ViewModels.EditViewModel>
@{
  ViewBag.Title = "Configure User Notifications";
  Layout = "~/Views/Shared/_Layout.cshtml";
}

<h2>@ViewBag.Title</h2>

@using (Html.BeginForm("Edit", "Users", FormMethod.Post, new { @class = "stdform stdform2", role = "form" }))
{
@Html.AntiForgeryToken()
<hr />

<p>
    <label><u> Service:</u> </label>
    <span class="field">
        <u>Notification Methods:</u>
    </span>
</p>


for (int i = 0; i < Model.Count; i++)
{
    <p>
        <label>@Model[i].service.Name</label>
        <span class="field">
            @Html.CheckBoxFor(model => model[i].sms)
            SMS &nbsp;&nbsp;
            @Html.CheckBoxFor(model => model[i].email)
            Email &nbsp;&nbsp;
        </span>
    </p>
}
<br clear="all" /><br />

<p class="stdformbutton">
    <button class="submit radius2">Save</button>
    <input type="reset" class="reset radius2" value="Reset Form" />
</p>
}

这是我控制器中的Action方法:

    //
    // POST: /Users/Edit
    [HttpPost]
    public ActionResult Edit(IList<EditViewModel> viewModel)
    {
        return View(viewModel);
    }

如何在控制器上接收视图模型时绑定它? 目前,当我调试action方法时,会收到一个看起来像这样的ViewModel:

enter image description here

如何获取服务,userId不能为空?

1 个答案:

答案 0 :(得分:2)

请注意,在帮助者的lambdas中,例如在model => service.sms右侧部分(service.sms)中,正式不是从左侧部分(model)派生的。这会导致结果输入的所有name属性相同,并为您提供您不期望的请求参数。

标准做法是在循环案例中使用for foreach的{​​{1}}。这样就可以正确生成生成的html的名称属性:

for(int i=0; i<Model.Count; i++)
{
    <p>
        <label>@Model[i].service.Name</label>
        <span class="field">                
            @Html.CheckBoxFor(model => model[i].sms)
            SMS &nbsp;&nbsp;               
            @Html.CheckBoxFor(model => model[i].email)
            Email &nbsp;&nbsp;
        </span>
    </p>
}

请注意,这需要Model来实现IList而不是IEnumerable

更新。对于没有任何用户界面的其他值,您可以使用隐藏字段,以便用户看不到它们,然后将其发布到服务器:

<label>@Model[i].service.Name</label>
<span class="field">                
    @Html.CheckBoxFor(model => model[i].sms)
    SMS &nbsp;&nbsp;               
    @Html.CheckBoxFor(model => model[i].email)
    Email &nbsp;&nbsp;
    @Html.HiddenFor(mode => model[i].userId)
    @Html.HiddenFor(mode => model[i].service.Name)
    ...other field of service you want to be posted...
</span>