问题
I want to create custom client-side validator, but I want define validation rules via Data Annotations attributes at business logic layer. How can I access model validation attributes in runtime?
I want to write 'generator', which will convert this code:
public class LoginModel
{
[Required]
[MinLength(3)]
public string UserName { get; set; }
[Required]
public string Password { get; set; }
}
into this one:
var loginViewModel= {
UserName: ko.observable().extend({ minLength: 3, required: true }),
Password: ko.observable().extend({ required: true })
};
But not from .cs sources, of course. =)
Maybe reflection?
UPD
I've found this method: MSDN. But can't understand how to use it.
回答1:
This is the universal way how to do this:
private string GenerateValidationModel<T>()
{
var name = typeof(T).Name.Replace("Model", "ViewModel");
name = Char.ToLowerInvariant(name[0]) + name.Substring(1);
var validationModel = "var " + name + " = {\n";
foreach (var prop in typeof(T).GetProperties())
{
object[] attrs = prop.GetCustomAttributes(true);
if (attrs == null || attrs.Length == 0)
continue;
string conds = "";
foreach (Attribute attr in attrs)
{
if (attr is MinLengthAttribute)
{
conds += ", minLength: " + (attr as MinLengthAttribute).Length;
}
else if (attr is RequiredAttribute)
{
conds += ", required: true";
}
// ...
}
if (conds.Length > 0)
validationModel += String.Format("\t{0}: ko.observable().extend({{ {1} }}),\n", prop.Name, conds.Trim(',', ' '));
}
return validationModel + "};";
}
Usage:
string validationModel = GenerateValidationModel<LoginModel>();
Output:
var loginViewModel = {
UserName: ko.observable().extend({ minLength: 3, required: true}),
Password: ko.observable().extend({ required: true}),
};
It's good idea to cache the output
回答2:
As commented above - I believe T4 might be worth a shot here. A huge benefit is that it's not executed at runtime (though it can, if that's your requirement) and you can avoid all possible issues with runtime file generation. Hopefully a sufficient starting point:
<#@ template language="C#" debug="True" hostspecific="true" #>
<#@ output extension="js" #>
<#@ assembly name="System.Core" #>
<#@ assembly name="EnvDTE" #>
<#@ import namespace="System.Collections.Generic" #>
<#@ import namespace="System.Linq" #>
<#@ import namespace="EnvDTE" #>
<#
var serviceProvider = Host as IServiceProvider;
if (serviceProvider == null)
{
throw new InvalidOperationException("Host is not IServiceProvider");
}
var dte = serviceProvider.GetService(typeof(DTE)) as DTE;
if (dte == null)
{
throw new InvalidOperationException("Unable to resolve DTE");
}
var project = dte.Solution.Projects
.OfType<Project>()
.Single(p => p.Name == "ConsoleApplication2");
var model = project.CodeModel
.CodeTypeFromFullName("MyApp.LoginModel")
as CodeClass;
//might want to have a list / find all items matching some rule
#>
var <#= Char.ToLowerInvariant(model.Name[0])
+ model.Name.Remove(0, 1).Replace("Model", "ViewModel") #>= {
<#
foreach (var property in model.Members.OfType<CodeProperty>())
{
var minLength = property.Attributes
.OfType<CodeAttribute>()
.FirstOrDefault(a => a.Name == "MinLength");
var required = property.Attributes
.OfType<CodeAttribute>()
.FirstOrDefault(a => a.Name == "Required");
var koAttributes = new List<String>();
if (minLength != null)
koAttributes.Add("minLength: " + minLength.Value);
if (required != null)
koAttributes.Add("required: true");
#>
<#= property.Name #>: ko.observable().extend({<#=
String.Join(", ", koAttributes) #>}),
<#
}
#>
}
来源:https://stackoverflow.com/questions/19638105/get-data-annotations-attributes-from-model