概括
Question:
为什么使用 ViewModel 时不显示自定义验证错误消息。
Answer:
自定义验证应应用于 ViewModel 而不是 Class。有关示例代码,请参阅@JaySilk84 答案的结尾。
MVC3,项目使用
- jquery-1.7.2.min.js
- 现代主义-2.5.3.js
- jquery-ui-1.8.22.custom.min.js(由 jQuery.com 为 Accordion 插件生成)
- jquery.validate.min.js 和
- jquery.validate.unobtrusive.min.js
我在我的项目中对 View 中的 dataannotations 和 Controller 中的 ModelState.AddModelError 进行了验证,所以我知道我已经正确配置了所有验证代码。
但是使用自定义验证会在代码中生成错误,但不会显示错误消息。
public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{ if (DOB > DateTime.Now.AddYears(-18))
{ yield return new ValidationResult("Must be 18 or over."); } }
在 POST 操作中深入调试,自定义验证会导致模型状态失败,并且错误消息被放置在正确的值字段中,但是当模型被发送回视图时,错误消息不会显示。在控制器中,我也有 ModelState.AddModelError 代码,并且它的消息确实显示。对于一个工作而不是另一个工作,如何以不同的方式处理?如果不是这样,还有什么会阻止错误消息显示?
更新 1:
我正在使用 ViewModel 在视图中创建模型。我删除了 ViewModel 并开始显示错误消息,只要我将 ViewModel 添加回消息中,就会再次停止显示。有没有人成功使用 ViewModel 的自定义验证?有什么你需要做的额外工作才能让它工作吗?
更新 2:
我用这两个简单的类(Agency 和 Person)创建了一个新的 MVC3 项目。
public class Agency : IValidatableObject
{
public int Id { get; set; }
public string Name { get; set; }
public DateTime DOB { get; set; }
public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{
if (DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18."); }
}
}
public class Person
{
public int Id { get; set; }
public string Name { get; set; }
}
这是控制器代码
public ActionResult Create()
{
return View();
}
//
// POST: /Agency/Create
[HttpPost]
public ActionResult Create(Agency agency)
{
if (ModelState.IsValid)
{
db.Agencies.Add(agency);
db.SaveChanges();
return RedirectToAction("Index");
}
return View(agency);
}
//[HttpPost]
//public ActionResult Create(AgencyVM agencyVM)
//{
// if (ModelState.IsValid)
// {
// var agency = agencyVM.Agency;
// db.Agencies.Add(agency);
// db.SaveChanges();
// return RedirectToAction("Index");
// }
// return View(agencyVM);
//}
风景
@model CustValTest.Models.Agency
@*@model CustValTest.Models.AgencyVM*@
@* When using VM (model => model.Name) becomes (model => model.Agency.Name) etc. *@
@{
ViewBag.Title = "Create";
}
<h2>Create</h2>
<script src="@Url.Content("~/Scripts/jquery.validate.min.js")" type="text/javascript"></script>
<script src="@Url.Content("~/Scripts/jquery.validate.unobtrusive.min.js")" type="text/javascript"></script>
@using (Html.BeginForm()) {
@Html.ValidationSummary(true)
<fieldset>
<legend>Agency</legend>
<div class="editor-label">
@Html.LabelFor(model => model.Name)
</div>
<div class="editor-field">
@Html.EditorFor(model => model.Name)
@Html.ValidationMessageFor(model => model.Name)
</div>
<div class="editor-label">
@Html.LabelFor(model => model.DOB)
</div>
<div class="editor-field">
@Html.EditorFor(model => model.DOB)
@Html.ValidationMessageFor(model => model.DOB)
</div>
<p>
<input type="submit" value="Create" />
</p>
</fieldset>
}
<div>
@Html.ActionLink("Back to List", "Index")
</div>
视图模型
public class AgencyVM
{
public Agency Agency { get; set; }
public Person Person { get; set; }
}
当视图中仅显示代理时,将显示验证错误(18 岁以下的 DOB)。当 ViewModel 出现时,错误不会显示。自定义验证始终会捕获错误并导致 ModelState.IsValid 失败并重新呈现视图。任何人都可以复制这个吗?关于为什么以及如何解决的任何想法?
更新 3:
作为一种临时解决方法,我通过向 ValidationResult 添加一个参数,将 Validation 更改为一级字段(相对于模型一级):
if (DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18.", new [] { "DOB" }); }
现在的问题是错误消息显示在字段旁边而不是表单顶部(这在手风琴视图中不好,因为用户将返回到表单而没有可见的错误消息)。为了解决这个次要问题,我将此代码添加到控制器 POST 操作中。
ModelState.AddModelError(string.Empty, errMsgInvld);
return View(agencyVM);
}
string errMsgInvld = "There was an entry error, please review the entire form. Invalid entries will be noted in red.";
问题仍未得到解答,为什么 ViewModel 不显示模型级错误消息(有关此内容的更多信息,请参阅我对 JaySilk84 的回复)?