我有一个页面,我将两个视图模型发布到控制器,查询和约会.约会在查询中嵌套.用户可以选择在不创建约会的情况下向我们提交查询.
我在视图模型属性上使用内置的MVC必需属性.
我的问题是,当用户选择在没有预约的情况下创建查询时,如何优雅地忽略嵌套的约会视图模型上的验证器并让ModelState.IsValid返回true?
if(!viewModel.CreateAppointment)
{
//ignore the nested view models validation
}
Run Code Online (Sandbox Code Playgroud)
您可以创建一个自定义的IgnoreModelError属性,如下所示,并在视图中使用2个单独的按钮,仅用于查询,另一个用于约会.
// C#
public class IgnoreModelErrorAttribute : ActionFilterAttribute
{
private string keysString;
public IgnoreModelErrorsAttribute(string keys)
: base()
{
this.keysString = keys;
}
public override void OnActionExecuting(ActionExecutingContext filterContext)
{
ModelStateDictionary modelState = filterContext.Controller.ViewData.ModelState;
string[] keyPatterns = keysString.Split(new char[] { ',' },
StringSplitOptions.RemoveEmptyEntries);
for (int i = 0; i < keyPatterns.Length; i++)
{
string keyPattern = keyPatterns[i]
.Trim()
.Replace(@".", @"\.")
.Replace(@"[", @"\[")
.Replace(@"]", @"\]")
.Replace(@"\[\]", @"\[[0-9]+\]")
.Replace(@"*", @"[A-Za-z0-9]+");
IEnumerable<string> matchingKeys = _
modelState.Keys.Where(x => Regex.IsMatch(x, keyPattern));
foreach (string matchingKey in matchingKeys)
modelState[matchingKey].Errors.Clear();
}
}
}
[HttpPost]
[IgnoreModelErrors("Enquiry.Appointment")]
public ActionResult CreateEnquiryOnly(Enquiry enquiry)
{
// Code for enquiry only.
}
[HttpPost]
public ActionResult CreateAppointment(Enquiry enquiry)
{
// Code for appointment.
}
Run Code Online (Sandbox Code Playgroud)
嗯,在使用标准数据属性时,没有办法“优雅地”忽略错误。
不过,您有多种选择。快速而肮脏(即不优雅)的方法是从控制器中的 ModelState 中清除相关错误。
if (some condition) {
ModelState["controlName"].Errors.Clear();
}
Run Code Online (Sandbox Code Playgroud)
您还可以编写自己的使用条件测试的自定义数据属性。就像这里描述的那样:
http://blogs.msdn.com/b/simonince/archive/2011/02/04/conditional-validation-in-asp-net-mvc-3.aspx
第三种方法是避开属性并使用验证框架,例如FluentValidation
最后一个选择是使用 JavaScript 来确定数据的正确状态,然后修改表单操作 URL 以发布到不同的操作方法。然后,您可以使用 Bind 属性来装饰操作方法参数,以排除您不需要的数据项。但是,我不推荐这个,因为它需要客户端参与服务器端验证过程。
| 归档时间: |
|
| 查看次数: |
5014 次 |
| 最近记录: |