3

我创建了一个从 ValidationAttribute 派生的自定义验证器。我的理解是它将为客户端脚本生成足够的元数据以自动验证(使用 jquery.validate)。自定义验证器在服务器端工作正常。但它不会在客户端触发错误消息。(其他默认验证器,如“StringLength”在客户端也可以正常工作。)我们如何纠正它?

public class Person
{
    [Required(ErrorMessage = "First name required")]
    public string FirstName { get; set; }

    [CustomStartLetterMatch("FirstName")]
    [StringLength(5,ErrorMessage = "Must be under 5 characters")]
    public string LastName { get; set; }

    [Range(18,50,ErrorMessage="Must be between 18 and 50")]
    public int Age { get; set; }


}


public sealed class CustomStartLetterMatch : ValidationAttribute
{

    private const string _defaultErrorMessage = " First letter of '{0}' must be same as first letetr of '{1}'";
    private string _basePropertyName;

    public CustomStartLetterMatch(string basePropertyName)
        : base(_defaultErrorMessage)
    {
        _basePropertyName = basePropertyName;
    }


    //Override FormatErrorMessage Method
    public override string FormatErrorMessage(string name)
    {
        return string.Format(_defaultErrorMessage, name, _basePropertyName);
    }


    //Override IsValid
    protected override ValidationResult IsValid(object value, ValidationContext validationContext)
    {
        //Get PropertyInfo Object
        var basePropertyInfo = validationContext.ObjectType.GetProperty(_basePropertyName);
        var baseValue = (string)basePropertyInfo.GetValue(validationContext.ObjectInstance, null);
        var currentValue = (string)value;


        string firstLetterBaseValue = baseValue.Substring(0, 1);
        string firstLetterCurrentValue = currentValue.Substring(0, 1);

        //Comparision
        if (!string.Equals(firstLetterBaseValue, firstLetterCurrentValue))
        {
            var message = FormatErrorMessage(validationContext.DisplayName);
            return new ValidationResult(message);
        }

        //Default return - This means there were no validation error
        return null;
    }

}

看法

@model MyValidationTEST.Person

<script src="@Url.Content("~/Scripts/jquery-1.5.1.min.js")" type="text/javascript">  </script>

@*UnObtrusive*@
<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>Person</legend>

    <div class="editor-label">
        @Html.LabelFor(model => model.FirstName)
    </div>
    <div class="editor-field">
        @Html.EditorFor(model => model.FirstName)
        @Html.ValidationMessageFor(model => model.FirstName)
    </div>

    <div class="editor-label">
        @Html.LabelFor(model => model.LastName)
    </div>
    <div class="editor-field">
        @Html.EditorFor(model => model.LastName)
        @Html.ValidationMessageFor(model => model.LastName)
    </div>

    <div class="editor-label">
        @Html.LabelFor(model => model.Age)
    </div>
    <div class="editor-field">
        @Html.EditorFor(model => model.Age)
        @Html.ValidationMessageFor(model => model.Age)
    </div>



    <p>
        <input type="submit" value="Create" />
    </p>
</fieldset>
}

<div>
@Html.ActionLink("Back to List", "Index")
</div>

阅读:

  1. MVC3 中的 IValidatableObject - 客户端验证

  2. 在 ASP.NET MVC3 中,如何使用非常相似但略有不同的视图模型保持 DRY?

  3. http://odetocode.com/Blogs/scott/archive/2011/02/22/custom-data-annotation-validator-part-ii-client-code.aspx

  4. http://bradwilson.typepad.com/blog/2010/10/mvc3-unobtrusive-validation.html

  5. 带有参数的 ASP.NET MVC 3 客户端验证

  6. 如何在客户端将自定义 ValidationAttribute 呈现为“da​​ta-val-xx”属性?

  7. ASP.NET-MVC3 中“自我验证模型”中的客户端验证

4

1 回答 1

5

我的理解是它将为客户端脚本生成足够的元数据以自动验证(使用 jquery.validate)。

你的理解是错误的。您不可能期望有足够的元数据来生成客户端验证。在这种IsValid方法中,您绝对可以做任何事情。您甚至可以调用非托管 C++ 库来执行验证。您不可能期望 ASP.NET MVC 3 会在客户端上反映这一点。

如果要为此类自定义验证逻辑启用客户端验证,则需要实现IClientValidatable并添加自定义适配器。在这个适配器中,您必须重新实现您在服务器上执行的相同逻辑,但这次使用的是 javascript。

这是一个例子。这是另一个

正如您所看到的,客户端验证适用于一些简单的规则,例如Required 和一些东西,一旦您开始进行一些真正的验证,您将不得不自己实现它。

于 2012-02-19T10:48:13.650 回答