从ASP.NET MVC 1.0切换到ASP.NET MVC 2 Beta后,我遇到了一些不同的行为。我检查了一些重大变化,但目前还不清楚问题所在。
问题与默认模型绑定器和实现IDataErrorInfo的模型有关。
属性(IDataErrorInfo.Item):
public string this[string columnName]
不再为每个属性调用。我错过了什么?
答案 0 :(得分:1)
MVC 1.0中的DefaultModelBinder:
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);
}
}
}
MVC 2.0 beta中的DefaultModelBinder:
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这个[string columnName]属性......好像是一个bug,因为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)
的{{1}}方法中将Validate()
更改为if(container!= null)。
答案 1 :(得分:1)
经过一些进一步的调试工作后,我相信我理解为什么在我的特定情况下IDataErrorInfo.Item
没有被调用。 ASP.NET MVC 2 Beta中使用以下代码来验证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
。
我是不是按照预期的方式使用了某种东西?