Razor不会在HiddenFor中呈现隐藏的精确PK

Nea*_*alR 5 asp.net-mvc entity-framework razor asp.net-mvc-4

在我的ASP MVC 4站点遇到一个奇怪的问题.如果用户打开create表单并尝试将代理添加到我们的数据库,我们自然会首先在字段上运行验证.如果出现错误,我们会将代理保存为未完成状态,并将用户重定向回创建页面.

当用户尝试重新保存代理时出现错误.在第一个回发后,代理将保存到数据库中并生成PK.然而,在第二个回发后,PK被发送到服务器,其值0不是刚刚自动生成的值.

HiddenForCreate视图上添加了一个,但是0每次都会显示一个值.

我还逐步完成了代码,以确保在.Save调用之后生成PK,在调用时仍然存在,return View并且还确保Model.ID在呈现视图时属性包含相同的值.

无论如何,如果我右键单击页面并查看源,隐藏字段将呈现如下:

<input data-val="false" data-val-number="The field ID must be a number." data-val-required="The ID field is required." id="ID" name="ID" type="hidden" value="0" />
Run Code Online (Sandbox Code Playgroud)

模型

public partial class AgentTransmission
{
    public int ID { get; set; }
    .
    .
    .
}
Run Code Online (Sandbox Code Playgroud)

视图

@model MonetModelFromDb.Models.AgentTransmission

@{
    ViewBag.Title = "Create new Agent";
}

@Html.HiddenFor(model => model.CreatedDate, new { data_val = "false" })
@Html.HiddenFor(model => model.CreatedOperator, new { data_val = "false" })
@Html.HiddenFor(model => model.ReferenceNumber, new { data_val = "false" })
@Html.HiddenFor(model => model.Region, new { data_val = "false" })
@Html.HiddenFor(model => model.INDDist, new { data_val = "false" })
@Html.HiddenFor(model => model.LastChangeDate, new { data_val = "false" })
@Html.HiddenFor(model => model.LastChangeOperator, new { data_val = "false" })
@Html.HiddenFor(model => model.EditTaxId, new { data_val = "false" })
@Html.HiddenFor(model => model.ParentId, new { data_val = "false" })
@Html.HiddenFor(model => model.IsSubstat, new { data_val = "false" })
@Html.HiddenFor(model => model.ID, new { data_val = "false" })
Run Code Online (Sandbox Code Playgroud)

渲染HiddenFor部分

<input data-val="false" data-val-date="The field CreatedDate must be a date." id="CreatedDate" name="CreatedDate" type="hidden" value="" />
<input data-val="false" id="CreatedOperator" name="CreatedOperator" type="hidden" value="" />
<input data-val="false" id="Region" name="Region" type="hidden" value="NM-834" />
<input data-val="false" id="INDDist" name="INDDist" type="hidden" value="834" />
<input data-val="false" data-val-date="The field LastChangeDate must be a date." data-val-required="The LastChangeDate field is required." id="LastChangeDate" name="LastChangeDate" type="hidden" value="4/8/2015 10:43:30 AM" />
<input data-val="false" id="LastChangeOperator" name="LastChangeOperator" type="hidden" value="TYPCLS" />
<input data-val="false" data-val-required="The EditTaxId field is required." id="EditTaxId" name="EditTaxId" type="hidden" value="False" />
<input data-val="false" data-val-number="The field ParentId must be a number." id="ParentId" name="ParentId" type="hidden" value="" />
<input data-val="false" data-val-required="The IsSubstat field is required." id="IsSubstat" name="IsSubstat" type="hidden" value="False" />
<input data-val="false" data-val-number="The field ID must be a number." data-val-required="The ID field is required." id="ID" name="ID" type="hidden" value="0" />
Run Code Online (Sandbox Code Playgroud)

调节器

    [HttpPost]
    [MonetAuthorize]
    public ActionResult Create(AgentTransmission agenttransmission, bool andAddAgent = false)
    {
    .
    .
    .
        //Determine if this is first POST or not
        if (agenttransmission.ID > 0)
        {
            db.Entry(agenttransmission).State = EntityState.Modified;
        }
        else
        {
            db.AgentTransmission.Add(agenttransmission);
        }
        db.SaveChanges();

        //Send back to view if errors pressent
        if (!String.IsNullOrWhiteSpace(errorMsg))
        {
            return View(agenttransmission);
        }           
    }
Run Code Online (Sandbox Code Playgroud)

编辑

如果我删除HiddenFor帮助器并简单地剪切/粘贴渲染的input标签,我就能捕获相关的PK值.但是,这有点hacky所以我希望找到一个更优雅的解决方案(如果可能的话)

<input data-val="false" id="ID" name="ID" type="hidden" value="@Model.ID" />
Run Code Online (Sandbox Code Playgroud)

Chr*_*att 6

这里经常提到这个问题的变化.基本上它归结为ModelState对象以及它的值覆盖视图的实际模型上的值的事实.发布表单时,0将在ModelState您的ID属性的对象中设置.在post操作中,您保存实体,这会导致其ID属性更新,但0仍在ModelState.当你返回视图时,再次0设置为值ID,因为,再次,这是什么ModelState.

它以这种方式工作的原因最好用一个例子来解释.假设您有一个表单,您正在编辑现有实体,该实体的Name属性设置为"Foo".用户在表单中将其更改为"Bar",然后发布表单.但是,他们忽略了填写必填字段,因此错误会阻止保存更新.此时会发生什么?如果我们使用模型值,该Name字段将重置为"Foo".但是,如果ModelState使用,则该Name字段保留用户对"Bar"的修改.在后一种情况下,他们只是修复错误并再次发布.在前者中,他们必须重新制作他们之前对表单所做的所有更改,这显然是一种非常糟糕的用户体验.

现在,至于如何解决这个问题.最好的方法是遵循PRG模式(Post-Redirect-Get).如果提交良好,并且您成功保存了更改,则不要返回视图,即使您希望用户能够立即进行其他更改.如果您需要,只需重定向回相同的操作,但重定向过程足以清除,ModelState以便用户现在与从数据库中提取的更新模型进行交互.

如果这不可行,那么你可以简单地清除ModelState.我建议不要完全清除它,因为你可能会引起一些非常真实的用户沮丧,如上例所示.如果你真的不能做重定向,那么试着只清除ModelState你真正需要的值.

ModelState.Remove("ID");
Run Code Online (Sandbox Code Playgroud)