从ASP.NET MVC 1.0切换到ASP.NET MVC 2 Beta后,我遇到了一些不同的行为。我检查了突破性的变化,但不清楚问题出在哪里。ASP.NET MVC 2 Beta - 默认模型绑定器
该问题与默认模型联编程序和实现IDataErrorInfo的模型有关。
的性质(IDataErrorInfo.Item):
public string this[string columnName]
不再被调用为每个属性。我错过了什么?
从ASP.NET MVC 1.0切换到ASP.NET MVC 2 Beta后,我遇到了一些不同的行为。我检查了突破性的变化,但不清楚问题出在哪里。ASP.NET MVC 2 Beta - 默认模型绑定器
该问题与默认模型联编程序和实现IDataErrorInfo的模型有关。
的性质(IDataErrorInfo.Item):
public string this[string columnName]
不再被调用为每个属性。我错过了什么?
经过一些进一步的调试工作,我相信我明白为什么在我的特殊情况下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
。
我只是没有按照预期的方式使用某些东西?
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)在DataErrorInfoPropertyModelValidator
Validate()
方法进行修正。
对metaData.GetValidators的调用应产生一个DataErrorInfoPropertyModelValidator。调用其Validate应导致直接调用IDataErrorInfo接口。它似乎没有做到这一点。 – Rudy 2009-11-19 21:04:14
如果是我的重复:http://stackoverflow.com/questions/1760039/is-idataerrorinfo-ignored-during-model-validation-in-mvc-2 – LukLed 2009-11-19 08:35:27
似乎没有答案。验证工作,但行为不同。在我的情况下,简单的类型,如System.Nullable。 –
Rudy
2009-11-19 18:22:26