1

如果这是 MVC3 的问题,会有关于此的帖子,但我找不到。我一定做错了什么。我有一个简单的视图(Index.cshtml),它使用 for 循环遍历列表。在每次迭代中,我输出两个文本输入,其中包含来自列表项之一的值。

@{Html.BeginForm();}
@Html.Encode("\n")
@for (int i = 0; i < Model.SortOptions.Count; i++ )
{
    @Html.TextBoxFor(m => m.SortOptions[i].ColumnName);
    @Html.Encode("\n");
    @Html.TextBoxFor(m => m.SortOptions[i].Direction);
    @Html.Encode("\n");
}
<input type="submit" value="Submit" />
@{Html.EndForm();}

我有两个视图控制器,一个用于 GET 请求,一个用于 POST。POST 版本向列表中添加的项目与 GET 版本不同。这就是问题所在。重新加载页面后,文本框的值与在 GET 上加载页面时的值相同。

起初我认为这一定是缓存问题,但如果我修改代码(如下所示),手动添加文本输入并将值注入 html,新值将发送到浏览器。

@{Html.BeginForm();}
@Html.Encode("\n")
@for (int i = 0; i < Model.SortOptions.Count; i++ )
{
    var columnNameName = string.Format("SortOptions[{0}].ColumnName", i);
    var columnNameID = string.Format("SortOptions_{0}__ColumnName", i);
    var directionName = string.Format("SortOptions[{0}].Direction", i);
    var directionID = string.Format("SortOptions_{0}__Direction", i);

<input type="hidden" name="@columnNameName" id="@columnNameID" value="@Model.SortOptions[i].ColumnName" />
<input type="hidden" name="@directionName" id="@directionID" value="@Model.SortOptions[i].Direction" />

}
<input type="submit" value="Submit" />
@{Html.EndForm();}

我已经逐步完成了代码,以确保模型在将它们发送到视图时包含预期值。我什至通过单步执行视图中的代码来检查列表的值。它似乎具有正确的值,但是当我在浏览器中查看它时,它的值应与页面响应 GET 请求时对应的值相对应。这是编辑器模板的问题吗?我刚开始使用 mvc3 和 razor 引擎,所以有很多我不知道的。任何帮助,将不胜感激。

----- 更新:添加了控制器代码----

    [HttpGet]
    public ActionResult Index()
    {
        var inv = new InventoryEntities();
        var model = new IndexModel(inv);
        model.SortOptions = new List<SortOption>();
        model.SortOptions.Add(new SortOption { ColumnName = "Model", Direction = SortDirection.Ascending });
        model.SortOptions.Add(new SortOption { ColumnName = "Make", Direction = SortDirection.Ascending });
        //Load data
        model.LoadEquipmentList();

        return View(model);
    }


    [HttpPost]
    [OutputCache(Duration = 1)]
    public ActionResult Index(List<SortOption> sortOptions, SortOption sort)
    {
        var inv = new InventoryEntities();
        var model = new IndexModel(inv);
        ModelState.Remove("SortOptions");
        model.SortOptions = new List<SortOption>();
        model.SortOptions.Add(new SortOption { ColumnName = "Type", Direction = SortDirection.Descending });
        model.SortOptions.Add(new SortOption { ColumnName = "SubType", Direction = SortDirection.Descending });
        model.EquipmentList = new List<EquipmentListItem>();
        model.EquipmentList.Add(new EquipmentListItem { ID = 3, AssignedTo = "Mike", Location = "Home", Make = "Ford", Model = "Pinto", Selected = false, SubType = "Car", Type = "Vehicle" });

        return View(model);
    }
4

2 回答 2

0

我突然想到了几件事——没有看到更多的东西很难说,但是……这两者都可以这样重写。额外的@符号不是必需的。

@using(Html.BeginForm()) {
    Html.Encode("\n")
    for (int i = 0; i < Model.SortOptions.Count; i++ ) {
        Html.TextBoxFor(m => m.SortOptions[i].ColumnName);
        Html.Encode("\n");
        Html.TextBoxFor(m => m.SortOptions[i].Direction);
        Html.Encode("\n");
    }
    <input type="submit" value="Submit" />
}


@using (Html.BeginForm()) { 
    Html.Encode("\n");
    for (int i = 0; i < Model.SortOptions.Count; i++ ) {
        var columnNameName = string.Format("SortOptions[{0}].ColumnName", i);
        var columnNameID = string.Format("SortOptions_{0}__ColumnName", i);
        var directionName = string.Format("SortOptions[{0}].Direction", i);
        var directionID = string.Format("SortOptions_{0}__Direction", i);

        <input type="hidden" name="@columnNameName" id="@columnNameID" value="@Model.SortOptions[i].ColumnName" />
        <input type="hidden" name="@directionName" id="@directionID" value="@Model.SortOptions[i].Direction" />

    }
    <input type="submit" value="Submit" />
}

否则你的东西看起来是对的,我看不出有什么问题。

于 2011-04-09T01:37:49.070 回答
0

请记住,诸如 Html 助手等TextBoxFor在绑定其值时首先使用模型状态,然后再使用模型。让我们考虑一个非常简单的例子来说明这意味着什么:

public class HomeController : Controller
{
    public ActionResult Index()
    {
        return View(new MyViewModel { Name = "foo" });
    }

    [HttpPost]
    public ActionResult Index(MyViewModel model)
    {
        model.Name = "bar";
        return View(model);
    }
}

和观点:

@model MyViewModel
@using (Html.BeginForm())
{
    @Html.TextBoxFor(x => x.Name)
    <input type="submit" value="OK" />
}

现在,当您提交表单时,您会期望文本框中的值会更改"bar"为您在 POST 操作中输入的值,但值不会更改。这是因为模型状态中的键值已经存在,Name其中包含用户输入的内容。因此,如果您希望它起作用,您需要从模型状态中删除原始值:

[HttpPost]
public ActionResult Index(MyViewModel model)
{
    // remove the original value if you intend to modify it here
    ModelState.Remove("Name");
    model.Name = "bar";
    return View(model);
}

同样的事情也发生在你的场景中。因此,您可能需要在 POST 操作中从模型状态中删除正在修改的值。

于 2011-04-09T07:04:46.567 回答