По-видимому, можно динамически присоединять атрибуты DataAnnotation к свойствам объекта во время выполнения и как таковые достичь динамической валидации.
Может ли кто-нибудь предоставить образец кода на этом?
По-видимому, можно динамически присоединять атрибуты DataAnnotation к свойствам объекта во время выполнения и как таковые достичь динамической валидации.
Может ли кто-нибудь предоставить образец кода на этом?
MVC имеет крючок для предоставления собственного ModelValidatorProvider. По умолчанию MVC 2 использует подкласс класса ModelValidatorProvider, называемый DataAnnotationsModelValidatorProvider, который может использовать атрибуты System.DataAnnotations.ComponentModel.ValidationAttribute для проверки.
DataAnnotationsModelValidatorProvider использует отражение, чтобы найти все атрибуты ValidationAttributes и просто прокручивает коллекцию для проверки ваших моделей. Все, что вам нужно сделать, это переопределить метод GetValidators и ввести собственные атрибуты из того источника, который вы выберете. Я использую эту методику для проверки условных условий, свойства с атрибутом DataType.Email всегда передаются через регулярное выражение и используют этот метод для извлечения информации из базы данных, чтобы применить более ограничительные проверки для пользователей, не имеющих полномочий.
В следующем примере просто говорится: "Всегда делать какие-либо свойства FirstName":
public class CustomMetadataValidationProvider : DataAnnotationsModelValidatorProvider
{
protected override IEnumerable<ModelValidator> GetValidators(ModelMetadata metadata, ControllerContext context, IEnumerable<Attribute> attributes)
{
//go to db if you want
//var repository = ((MyBaseController) context.Controller).RepositorySomething;
//find user if you need it
var user = context.HttpContext.User;
if (!string.IsNullOrWhiteSpace(metadata.PropertyName) && metadata.PropertyName == "FirstName")
attributes = new List<Attribute>() {new RequiredAttribute()};
return base.GetValidators(metadata, context, attributes);
}
}
Все, что вам нужно сделать, это зарегистрировать поставщика в файле Global.asax.cs:
protected void Application_Start()
{
ModelValidatorProviders.Providers.Add(new CustomMetadataValidationProvider());
AreaRegistration.RegisterAllAreas();
RegisterRoutes(RouteTable.Routes);
}
Конечный результат:
с этой моделью:
public class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
public DateTime Birthday { get; set; }
}
В вашем global.asax вам нужно очистить ModelValidatorProviders перед добавлением нового. В противном случае он добавит каждую аннотацию два раза, что даст вам "имена типа проверки в ненавязчивых правилах проверки клиента должны быть уникальными". - Ошибка.
protected void Application_Start()
{
ModelValidatorProviders.Providers.Clear();
ModelValidatorProviders.Providers.Add(new CustomMetadataValidationProvider());
AreaRegistration.RegisterAllAreas();
RegisterRoutes(RouteTable.Routes);
}
Я не думаю, что вы можете добавлять атрибуты к членам во время выполнения, но вы, вероятно, можете использовать собственный поставщик метаданных для обработки этого для вас.
Вы должны проверить этот пост в блоге.
Подход использования пользовательского MetadataValidationProvider
с переопределенным GetValidators
имеет несколько недостатков:
DisplayAttribute
, не связаны с проверкой, поэтому добавление их на этапе проверки не работает.Если вы хотите, чтобы ваши динамически применяемые аннотации данных работали последовательно, вы можете подклассы DataAnnotationsModelMetadataProvider
и DataAnnotationsModelValidatorProvider
. После этого замените фреймворки через ModelMetadataProviders.Current
и ModelValidatorProviders.Providers
при запуске приложения. (Вы можете сделать это в Application_Start
.)
При подклассе встроенных поставщиков систематический и, надеюсь, будущий способ применения ваших собственных атрибутов - переопределить GetTypeDescriptor
. Я сделал это успешно, но это связано с созданием реализации ICustomTypeDescriptor
и PropertyDescriptor
, для чего потребовалось много кода и времени.