Ресурс по умолчанию для аннотаций данных в ASP.NET MVC
Есть ли способ установить ресурс по умолчанию для проверки аннотаций данных?
Я не хочу делать что-то вроде этого:
[Required(ErrorMessage="Name required.", ErrorMessageResourceType=typeof(CustomDataAnnotationsResources)]
public string Name { get; set; }
Мне хотелось бы что-то вроде этого:
Global.asax
DataAnnotations.DefaultResources = typeof(CustomDataAnnotationsResources);
затем
[Required]
public string Name { get; set; }
кто-нибудь просветит свет!
заблаговременно
ИЗМЕНИТЬ
Моя реальная проблема была в EF Code First CTP4. CTP5 исправить. Спасибо всем.
Ответы
Ответ 1
Вы можете попробовать:
Добавьте этот класс в свой проект:
public class ExternalResourceDataAnnotationsValidator : DataAnnotationsModelValidator<ValidationAttribute>
{
/// <summary>
/// The type of the resource which holds the error messqages
/// </summary>
public static Type ResourceType { get; set; }
/// <summary>
/// Function to get the ErrorMessageResourceName from the Attribute
/// </summary>
public static Func<ValidationAttribute, string> ResourceNameFunc
{
get { return _resourceNameFunc; }
set { _resourceNameFunc = value; }
}
private static Func<ValidationAttribute, string> _resourceNameFunc = attr => attr.GetType().Name;
public ExternalResourceDataAnnotationsValidator(ModelMetadata metadata, ControllerContext context, ValidationAttribute attribute)
: base(metadata, context, attribute)
{
if (Attribute.ErrorMessageResourceType == null)
{
this.Attribute.ErrorMessageResourceType = ResourceType;
}
if (Attribute.ErrorMessageResourceName == null)
{
this.Attribute.ErrorMessageResourceName = ResourceNameFunc(this.Attribute);
}
}
}
и в вашем global.asax добавьте следующее:
// Add once
ExternalResourceDataAnnotationsValidator.ResourceType = typeof(CustomDataAnnotationsResources);
// Add one line for every attribute you want their ErrorMessageResourceType replaced.
DataAnnotationsModelValidatorProvider.RegisterAdapter(typeof(RangeAttribute), typeof(ExternalResourceDataAnnotationsValidator));
Он будет искать свойство с тем же именем, что и тип проверки для сообщения об ошибке. Вы можете изменить это с помощью свойства ResourceNameFunc.
EDIT: AFAIK это работает с MVC2 и далее, поскольку DataAnnotationsModelValidatorProvider был представлен в MVC2.
Ответ 2
Для этого я создал новый класс, который наследуется от RequiredAttribute, и сообщение об ошибке встроено в этот новый класс:
public class RequiredWithMessageAttribute : RequiredAttribute
{
public RequiredWithMessageAttribute()
{
ErrorMessageResourceType = typeof(ValidationResource);
ErrorMessageResourceName = "RequiredErrorMessage";
}
}
Сообщение об ошибке выводится из файла ValidationResource.resx, где я перечисляю сообщение об ошибке следующим образом:
RequiredErrorMessage → "{0} требуется."
где {0} = отображаемое имя.
Затем я аннотирую свои модели следующим образом, поэтому мне не нужно повторять свои сообщения об ошибках:
[RequiredWithMessage]
public string Name { get; set; }
Как только вы это сделаете, появляется сообщение об ошибке ( "Name is required." ), когда проверка не выполняется.
Это правильно работает с проверкой на стороне сервера ASP.NET MVC и проверкой на стороне клиента.
Ответ 3
Я сделал другой подход. Вам все еще нужно наследовать атрибуты DataAnnotation
, но вы можете получить более гибкое решение для перевода.
Код из мой пост в блоге (прочитайте его подробнее)
Конечный результат
public class User
{
[Required]
[LocalizedDisplayNameAttribute("User_Id")]
public int Id { get; set; }
[Required]
[StringLength(40)]
[LocalizedDisplayNameAttribute("User_FirstName")]
public string FirstName { get; set; }
[Required]
[StringLength(40)]
[LocalizedDisplayNameAttribute("User_LastName")]
public string LastName { get; set; }
}
1 Наследовать все атрибуты аннотации данных, такие как
public class RequiredAttribute : System.ComponentModel.DataAnnotations.RequiredAttribute
{
private string _displayName;
public RequiredAttribute()
{
ErrorMessageResourceName = "Validation_Required";
}
protected override ValidationResult IsValid(object value, ValidationContext validationContext)
{
_displayName = validationContext.DisplayName;
return base.IsValid(value, validationContext);
}
public override string FormatErrorMessage(string name)
{
var msg = LanguageService.Instance.Translate(ErrorMessageResourceName);
return string.Format(msg, _displayName);
}
}
2 Inherit DisplayNameAttribute
public class LocalizedDisplayNameAttribute : DisplayNameAttribute
{
private PropertyInfo _nameProperty;
private Type _resourceType;
public LocalizedDisplayNameAttribute(string className, string propertyName)
: base(className + (propertyName == null ? "_Class" : ("_" + propertyName)))
{
}
public override string DisplayName
{
get
{
return LanguageService.Instance.Translate(base.DisplayName) ?? "**" + base.DisplayName + "**";
}
}
}
3. Создайте языковой сервис (вы можете переключиться на любой источник языка)
public class LanguageService
{
private static LanguageService _instance = new LanguageService();
private List<ResourceManager> _resourceManagers = new List<ResourceManager>();
private LanguageService()
{
}
public static LanguageService Instance { get { return _instance; } }
public void Add(ResourceManager mgr)
{
_resourceManagers.Add(mgr);
}
public string Translate(string key)
{
foreach (var item in _resourceManagers)
{
var value = item.GetString(key);
if (value != null)
return value;
}
return null;
}
}
Наконец, вам нужно зарегистрировать строковые таблицы, которые вы используете для перевода сообщений проверки и моделей
LanguageService.Instance.Add(MyNameSpace.ModelResource.ResourceManager);
LanguageService.Instance.Add(MyNameSpace.ValidationResources.ResourceManager);