为什么MVC4验证js auto会将数字验证添加到datetime?

时间:2015-02-18 14:03:35

标签: javascript asp.net-mvc validation asp.net-mvc-4

我最近升级到MVC 4并使用validate.js和validate.unobtrosive.js进行客户端验证。从MVC 4开始,我注意到html输出已经改变了。

MVC3:

<input class="dutchDate date_datepicker hasDatepicker" id="FirstDate" name="FirstDate" type="text" value="18-02-2015">

MVC4:

<input class="dutchDate date_datepicker hasDatepicker" data-val="true" data-val-number="Waarde moet numeriek zijn" id="FirstDate" name="FirstDate" type="text" value="18-02-2015">

现在,我的自定义验证和数字验证会验证datetime字段。我不明白为什么数字验证被添加到输入标签。有一些经验相同吗?我不希望在日期输入上进行数字验证。

型号:

public class TripFilter : IResultFilter
{
    ...
    [DataType(DataType.Date)]
    [DisplayFormat(DataFormatString = "{0:dd-MM-yyyy}", ApplyFormatInEditMode = true)]
    public DateTime? FirstDate { get; set; }
    [DataType(DataType.Date)]
    [DisplayFormat(DataFormatString = "{0:dd-MM-yyyy}", ApplyFormatInEditMode = true)]
    public DateTime? UntilDate { get; set; }
    ...
}

Renderering:

@Html.TextBox("", dt.ToString("dd-MM-yyyy"), new { @class = ViewData["class"] + " dutchDate date_datepicker", @type = "text" }

修改

在检查全局asax之后,我注意到以下几行:

ModelValidatorProviders.Providers.Add(new ClientNumberValidatorProvider());

以下是:

public class ClientNumberValidatorProvider : ClientDataTypeModelValidatorProvider
{
    public override IEnumerable<ModelValidator> GetValidators(ModelMetadata metadata, ControllerContext context)
    {
        bool isNumericField = base.GetValidators(metadata, context).Any();
        if (isNumericField)
            yield return new ClientSideNumberValidator(metadata, context);
    }
}

public class ClientSideNumberValidator : ModelValidator
{
    public ClientSideNumberValidator(ModelMetadata metadata,
        ControllerContext controllerContext)
        : base(metadata, controllerContext) { }

    public override IEnumerable<ModelValidationResult> Validate(object container)
    {
        yield break; // Do nothing for server-side validation
    }

    public override IEnumerable<ModelClientValidationRule> GetClientValidationRules()
    {
        yield return new ModelClientValidationRule
        {
            ValidationType = "number",
            ErrorMessage = Resources.Global.InvalidField
        };
    }
}

这似乎是添加数字验证的原因。让我感到困惑的是,为什么这只发生在DateTime字段而不是字符串字段?

3 个答案:

答案 0 :(得分:1)

我有类似的问题并通过添加

解决了这个问题
DataAnnotationsModelValidatorProvider.AddImplicitRequiredAttributeForValueTypes = false;

Application_Start

中的Global.asax.cs

这适用于所有值类型 - string不是值类型,DateTime是。您不需要更改ClientNumberValidatorProvider

答案 1 :(得分:0)

MVC 4正在生成这些属性,因为您正在使用web.config中设置的不显眼的验证:

 <appSettings>
    <add key="webpages:Version" value="3.0.0.0" />
    <add key="webpages:Enabled" value="false" />
    <add key="ClientValidationEnabled" value="true" />
    <add key="UnobtrusiveJavaScriptEnabled" value="true" />
  </appSettings>

您可以在web.config中将这些属性设置为false:

<add key="ClientValidationEnabled" value="false" />
<add key="UnobtrusiveJavaScriptEnabled" value="false" />

但我想你真的想在客户端验证你的日期,所以你必须使用@Html.TextBoxFor而不是@Html.TextBox

@Html.TextBoxFor(m=>m.FirstDate)

甚至可以更好地在DateTime.cshtml下创建名为Views/Shared/EditorTemplates的模板,然后使用@Html.EditorFor(m=>m.FirstDate) 模板可以是这样的(使用适当的标记和属性):

@model DateTime?
@{
    var value = "";
    const string format = "{0:yyyy-MM-dd HH:mm}";
    if (Model.HasValue) {
        value = string.Format(format, Model.Value);
    }
}

<label class="form-group">
    <i class="glyphicon glyphicon-calendar"></i>
    @Html.TextBox("", null,
        new
        {
            @class = "form-control date", placeholder = ViewData.ModelMetadata.DisplayName ?? ViewData.ModelMetadata.PropertyName,
            data_format = "dd/mm/yy HH:ii P",
            data_initial_value = value
        })
</label>

EditFor帮助器将自动推断您传递的属性的类型,并使用在这种情况下称为类型的模板DateTime。 请注意,该模板也使用@model DateTime?键入,也允许处理空日期。

答案 2 :(得分:0)

我通过调整ClientNumberValidatorProvider解决了我的问题。它现在看起来像这样:

public class ClientNumberValidatorProvider : ClientDataTypeModelValidatorProvider
{
    private static readonly HashSet<Type> numericTypes = new HashSet<Type>(new Type[]
    {
        typeof(byte), typeof(sbyte),
        typeof(short), typeof(ushort),
        typeof(int), typeof(uint),
        typeof(long), typeof(ulong),
        typeof(float), typeof(double), typeof(decimal)
    });

    public override IEnumerable<ModelValidator> GetValidators(ModelMetadata metadata, ControllerContext context)
    {
        if (numericTypes.Contains(metadata.ModelType))
            yield return new ClientSideNumberValidator(metadata, context);
    }
}

如果模型的属性是HashSet中的类型之一,它将仅返回ClientSideNumberValidator。谢谢彼得史密斯指出我的怀疑方向。 MicroSoft也使用HashSet:ClientDataTypeModelValidatorProvider.cs