Я использую описанный ниже подход, чтобы автоматически копировать аннотации данных из моих сущностей в мою модель представления.Это гарантирует, что такие вещи, как значения StringLength и Required, всегда будут одинаковыми для сущности / модели представления.
Работает с использованием конфигурации Automapper, поэтому работает, если свойства именуются на модели представления по-разному, если AutoMapper настроен правильно.
Вам нужно создать собственный ModelValidatorProvider и пользовательский ModelMetadataProvider, чтобы заставить это работать.Моя память о том, почему это немного туманно, но я считаю, что это так и на стороне сервера, и на стороне клиента, а также любое другое форматирование, которое вы выполняете на основе метаданных (например, звездочка рядом с обязательными полями).
Примечание. Я немного упростил свой код, добавив его ниже, поэтому возможны некоторые небольшие проблемы.
Поставщик метаданных
public class MetadataProvider : DataAnnotationsModelMetadataProvider
{
private IConfigurationProvider _mapper;
public MetadataProvider(IConfigurationProvider mapper)
{
_mapper = mapper;
}
protected override System.Web.Mvc.ModelMetadata CreateMetadata(IEnumerable<Attribute> attributes, Type containerType, Func<object> modelAccessor, Type modelType, string propertyName)
{
//Grab attributes from the entity columns and copy them to the view model
var mappedAttributes = _mapper.GetMappedAttributes(containerType, propertyName, attributes);
return base.CreateMetadata(mappedAttributes, containerType, modelAccessor, modelType, propertyName);
}
}
ValidatorПоставщик
public class ValidatorProvider : DataAnnotationsModelValidatorProvider
{
private IConfigurationProvider _mapper;
public ValidatorProvider(IConfigurationProvider mapper)
{
_mapper = mapper;
}
protected override System.Collections.Generic.IEnumerable<ModelValidator> GetValidators(System.Web.Mvc.ModelMetadata metadata, ControllerContext context, IEnumerable<Attribute> attributes)
{
var mappedAttributes = _mapper.GetMappedAttributes(metadata.ContainerType, metadata.PropertyName, attributes);
return base.GetValidators(metadata, context, mappedAttributes);
}
}
Вспомогательный метод. Указан в 2 классах
public static IEnumerable<Attribute> GetMappedAttributes(this IConfigurationProvider mapper, Type sourceType, string propertyName, IEnumerable<Attribute> existingAttributes)
{
if (sourceType != null)
{
foreach (var typeMap in mapper.GetAllTypeMaps().Where(i => i.SourceType == sourceType))
{
foreach (var propertyMap in typeMap.GetPropertyMaps())
{
if (propertyMap.IsIgnored() || propertyMap.SourceMember == null)
continue;
if (propertyMap.SourceMember.Name == propertyName)
{
foreach (ValidationAttribute attribute in propertyMap.DestinationProperty.GetCustomAttributes(typeof(ValidationAttribute), true))
{
if (!existingAttributes.Any(i => i.GetType() == attribute.GetType()))
yield return attribute;
}
}
}
}
}
if (existingAttributes != null)
{
foreach (var attribute in existingAttributes)
{
yield return attribute;
}
}
}
Другие примечания
- Если вы используете внедрение зависимостей, убедитесь, что ваш контейнер еще не заменяет встроенный поставщик метаданных или поставщик валидатора.В моем случае я использовал пакет Ninject.MVC3, который связал один из них после создания ядра, и мне пришлось потом перепривязывать его, чтобы мой класс фактически использовался.Я получал исключения из-за того, что Required разрешено добавлять только один раз, и большую часть дня занимался его поиском.