概要
Question:
为什么在使用ViewModel时没有显示自定义验证错误消息。
Answer:
自定义验证应该应用于ViewModel而不是Class。有关示例代码,请参阅@ JaySilk84的答案结尾。
MVC3,项目使用
我在我的项目中验证了View中的数据注释和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>
ViewModel
public class AgencyVM
{
public Agency Agency { get; set; }
public Person Person { get; set; }
}
当在视图中显示代理商时,显示验证错误(DOB低于18)。显示ViewModel时,不显示错误。自定义验证总是捕获错误,导致ModelState.IsValid失败并重新显示视图。任何人都可以复制这个吗?关于为什么以及如何修复的任何想法?
更新3:
作为一项临时工作,我通过向ValidationResult添加一个参数,将Validation更改为字段级别1(而不是模型级别1):
if (DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18.", new [] { "DOB" }); }
现在的问题是错误消息显示在字段的旁边而不是表单的顶部(由于用户将返回到没有可见错误的表单,因此表示手风琴视图不好信息)。为了解决这个次要问题,我将此代码添加到Controller 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的回复)?
答案 0 :(得分:3)
现在的问题是您的模型是嵌套的,错误消息被放置在Agency
下的ModelState中,没有.DOB,因为您没有在ValidationResult
中指定它。 ValidationMessageFor()
帮助器正在查找名为 Agency.DOB 的密钥(请参阅下面ValidationMessageFor()帮助程序中的相关代码):
string fullHtmlFieldName = htmlHelper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(expression);
FormContext clientValidation = htmlHelper.ViewContext.GetFormContextForClientValidation();
if (!htmlHelper.ViewData.ModelState.ContainsKey(fullHtmlFieldName) && clientValidation == null)
return (MvcHtmlString) null;
GetFullHtmlFieldName()
正在返回 Agency.DOB ,而非代理商
我认为如果你将DOB添加到ValidationResult
它将起作用:
public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{
if (DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18.", new List<string>() { "DOB" }); }
}
ValidationResult
的第二个参数将告诉它在ModelState中使用哪个键(默认情况下它将附加父对象Agency
),因此ModelState将有一个名为 Agency的键.DOB 这是你ValidationMessageFor()
正在寻找的东西。
修改强>
<击>
如果您不想进行字段级验证,则不需要Html.ValidationMessageFor()
。你只需要ValidationSummary()
。
击>
该视图将AgencyVM
视为模型。如果您希望它正确验证,那么将验证置于AgencyVM
级别并让它验证子对象。或者,您可以对子对象进行验证,但父对象(AgencyVM)必须将其聚合到视图中。您可以做的另一件事就是保持原样并将ValidationSummary(true)
更改为ValidationSummary(false)
。这会将ModelState中的所有内容打印到摘要中。我认为从Agency
删除验证并将其置于AgencyVM
可能是最好的方法:
public class AgencyVM : IValidatableObject
{
public Agency Agency { get; set; }
public Person Person { get; set; }
public IEnumerable<ValidationResult> Validate(ValidationContext validationContext)
{
if (Agency.DOB > DateTime.Now.AddYears(-18)) { yield return new ValidationResult("Must be over 18."); }
if (string.IsNullOrEmpty(Agency.Name)) { yield return new ValidationResult("Need a name"); }
}
}