2009-11-19 30 views
0

从ASP.NET MVC 1.0切换到ASP.NET MVC 2 Beta后,我遇到了一些不同的行为。我检查了突破性的变化,但不清楚问题出在哪里。ASP.NET MVC 2 Beta - 默认模型绑定器

该问题与默认模型联编程序和实现IDataErrorInfo的模型有关。

的性质(IDataErrorInfo.Item):

public string this[string columnName] 

不再被调用为每个属性。我错过了什么?

+0

如果是我的重复:http://stackoverflow.com/questions/1760039/is-idataerrorinfo-ignored-during-model-validation-in-mvc-2 – LukLed 2009-11-19 08:35:27

+0

似乎没有答案。验证工作,但行为不同。在我的情况下,简单的类型,如System.Nullable 。 – Rudy 2009-11-19 18:22:26

回答

1

经过一些进一步的调试工作,我相信我明白为什么在我的特殊情况下IDataErrorInfo.Item没有被调用。下面的代码是在ASP.NET MVC 2测试版用于验证IDataErrorInfo属性:

internal sealed class DataErrorInfoPropertyModelValidator : ModelValidator 
{ 
    public DataErrorInfoPropertyModelValidator(ModelMetadata metadata, ControllerContext controllerContext) 
     : base(metadata, controllerContext) 
    { 
    } 

    public override IEnumerable<ModelValidationResult> Validate(object container) 
    { 
     if (Metadata.Model != null) 
     { 
      var castContainer = container as IDataErrorInfo; 
      if (castContainer != null) 
      { 
       string errorMessage = castContainer[Metadata.PropertyName]; 
       if (!String.IsNullOrEmpty(errorMessage)) 
       { 
        return new[] { new ModelValidationResult { Message = errorMessage } }; 
       } 
      } 
     } 
     return Enumerable.Empty<ModelValidationResult>(); 
    } 
} 

我的模型包含一个属性,是System.Nullable<int>,当模型绑定从HTML结合后的空字符串,Metadata.Model等于null,因此验证不会运行。

这与ASP.NET MVC 1.0有着根本的区别,在这种情况下,此方案将验证器一直激活到调用IDataErrorInfo.Item

我只是没有按照预期的方式使用某些东西?

+0

感谢您报告此问题。我和其他一些MVC开发人员进行了交流,我们怀疑在预览版2和Beta版之间的某个时间点进行了无效检查,但之后变得没有必要。我提出了一个错误来取消支票。 – Levi 2009-11-20 19:17:53

+0

我欣赏你看着这个! – Rudy 2009-11-20 19:31:36

1

DefaultModelBinder在MVC 1.0:

protected virtual void OnPropertyValidated(ControllerContext controllerContext, ModelBindingContext bindingContext, PropertyDescriptor propertyDescriptor, object value) 
{ 
    IDataErrorInfo model = bindingContext.Model as IDataErrorInfo; 
    if (model != null) 
    { 
     string str = model[propertyDescriptor.Name]; 
     if (!string.IsNullOrEmpty(str)) 
     { 
      string key = CreateSubPropertyName(bindingContext.ModelName, propertyDescriptor.Name); 
      bindingContext.ModelState.AddModelError(key, str); 
     } 
    } 
} 

DefaultModelBinder在MVC 2.0测试版:

protected virtual void OnPropertyValidated(ControllerContext controllerContext, ModelBindingContext bindingContext, PropertyDescriptor propertyDescriptor, object value) 
{ 
    ModelMetadata metadata = bindingContext.PropertyMetadata[propertyDescriptor.Name]; 
    metadata.Model = value; 
    string prefix = CreateSubPropertyName(bindingContext.ModelName, metadata.PropertyName); 
    foreach (ModelValidator validator in metadata.GetValidators(controllerContext)) 
    { 
     foreach (ModelValidationResult result in validator.Validate(bindingContext.Model)) 
     { 
      bindingContext.ModelState.AddModelError(CreateSubPropertyName(prefix, result.MemberName), result.Message); 
     } 
    } 
    if ((bindingContext.ModelState.IsValidField(prefix) && (value == null)) && !TypeHelpers.TypeAllowsNullValue(propertyDescriptor.PropertyType)) 
    { 
     bindingContext.ModelState.AddModelError(prefix, GetValueRequiredResource(controllerContext)); 
    } 
} 

它不使用IDataErrorInfo的这个[字符串COLUMNNAME]属性...好像一个漏洞,因为DefaultModelBinder仍然使用Error属性。至少是不一致的。

编辑

我用反射镜,发现DataErrorInfoPropertyModelValidator似乎并没有被使用,所以我创建了自己的类:

public class DataErrorInfoModelPropertyValidatorProvider : ModelValidatorProvider 
{ 
    // Methods 
    public override IEnumerable<ModelValidator> GetValidators(ModelMetadata metadata, ControllerContext context) 
    { 
     if (metadata == null) 
     { 
      throw new ArgumentNullException("metadata"); 
     } 
     if (context == null) 
     { 
      throw new ArgumentNullException("context"); 
     } 

     var validators = new List<ModelValidator>(); 
     validators.Add(new DataErrorInfoPropertyModelValidator(metadata, context)); 
     return validators; 
    } 

    internal sealed class DataErrorInfoPropertyModelValidator : ModelValidator 
    { 
     // Methods 
     public DataErrorInfoPropertyModelValidator(ModelMetadata metadata, ControllerContext controllerContext) 
      : base(metadata, controllerContext) 
     { 
     } 

     public override IEnumerable<ModelValidationResult> Validate(object container) 
     { 
      if (container != null) 
      { 
       IDataErrorInfo info = container as IDataErrorInfo; 
       if (info != null) 
       { 
        string str = info[Metadata.PropertyName]; 
        if (!string.IsNullOrEmpty(str)) 
        { 
         ModelValidationResult[] resultArray = new ModelValidationResult[1]; 
         ModelValidationResult result = new ModelValidationResult(); 
         result.Message = str; 
         resultArray[0] = result; 
         return resultArray; 
        } 
       } 
      } 
      return Enumerable.Empty<ModelValidationResult>(); 
     } 
    } 
} 

然后我用:

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

它工作:)这只是临时解决方案。将要在最后的MVC 2

编辑

我也改变if (base.Metadata.Model != null)到如果(容器!= NULL)在DataErrorInfoPropertyModelValidatorValidate()方法进行修正。

+0

对metaData.GetValidators的调用应产生一个DataErrorInfoPropertyModelValidator。调用其Validate应导致直接调用IDataErrorInfo接口。它似乎没有做到这一点。 – Rudy 2009-11-19 21:04:14