ایجاد یک HtmlHelper سفارشی با پشتیبانی از UnobtrusiveValidationAttributes
پنج شنبه 28 آبان 1394 8:04 AM
همانطور که میدانید، در MVC برای اعتبارسنجی دادهها در سمت کلاینت از کتابخانهی jquery استفاده میشود. مایکروسافت از طریق jquery.validate.unobtrusive و گسترش کتابخانهی jquery.validate توانسته منطق خود را برای اعتبارسنجی دادهها در سمت کلاینت پیاده سازی کند.
برای این منظور MVC به کنترلهایی که باید اعتبارسنجی شوند، خصوصیاتی را از طریق Data Attribute اضافه میکند. برای مثال اگر در مدل خود فیلد ایمیل را به شکل زیر امضاء کرده باشید:
1
2
3
4
5
|
[Display(Name = "رایانامه" )] [Required(AllowEmptyStrings = false , ErrorMessage = "رایانامه خود را وارد کنید." )] [RegularExpression( "\\w+([-+.']\\w+)*@\\w+([-.]\\w+)*\\.\\w+([-.]\\w+)*" , ErrorMessage = "نشانی رایانامه پذیرفتنی نمیباشد." )] [ExistField(Action = "EmailExist" , Namespace = "Parsnet.Controllers" , Controller = "Account" , ErrorMessage = "این رایانامه پیشتر به کار گرفته شده است." )] public string Email { get ; set ; } |
و در View مورد نظر از Htmlhlper مربوطه به شکل زیر استفاده کرده باشید:
1
|
@Html.TextBoxFor(m => m.Email, new { @ class = "form-control en" , placeholder = @Html.DisplayNameFor(m => m.Email) }) |
در نهایت، Html خروجی در سمت کلاینت به شکل زیر خواهد بود:
1
|
< input data-val = "true" data-val-existfiledvalidator = "این رایانامه پیشتر به کار گرفته شده است." data-val-existfiledvalidator-url = "/account/emailexist" data-val-regex = "نشانی رایانامه پذیرفتنی نمیباشد." data-val-regex-pattern = "\w+([-+.']\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*" data-val-required = "رایانامه خود را وارد کنید." id = "Email" name = "Email" placeholder = "رایانامه" value = "" type = "text" > |
و در اینجا کتابخانهی اعتبارسنجی MVC با استفاده از همین خصوصیات *-data، اطلاعات مورد نیاز را جهت نمایش، اعتبارسنجی، تنظیم و بکارگیری، مورد استفاده قرار میدهد.
در یکی از پروژههایی که در حال کار کردن بر روی آن هستم لازم شد تا این اطلاعات اعتبارسنجی به یک تگ span اعمال شوند. سناریوی مورد نظر به این صورت است که در بخش پروفایل کاربر، کاربر میتواند اطلاعات خود را بصورت inline ویرایش کنید. برای اینکار از کتابخانه X-editable استفاده کردم که از این لینک قابل دریافت است.
ابتدا اطلاعات موردنیاز در یک تگ span نمایش داده میشوند و در ادامه کاربر پس از کلیک بر روی آیکن ویرایش، امکان تغییر آن فیلد را دارد. برای اعتبارسنجی دادهها لازم بود تا تمامی اطلاعات مورد نیاز اعتبارسنجی در سمت کلاینت را به شکلی در اختیار داشته باشم و به ذهنم رسید تا با ایجاد یک Helper سفارشی، خصوصیات موردنظر را به تگ span اعمال کنم و سپس در سمت کلاینت از آن استفاده کنم. در واقع با اینکار با استفاده از همان کلاس مدل و این Helper سفارشی، از وارد کردن دستی دادهها و خصوصیات اجتناب کنم. (تصور کنید چیزی حدود 30 فیلد که هرکدام حداقل 4 خصوصیت دارند)
با نگاهی به سورس MVC دیدم پیاده سازی این قابلیت چندان سخت نیست و به راحتی با ایجاد یک Helper سفارشی، منطق خود را پیاده سازی و اعتبارسنجی در سمت کلاینت را به راحتی اعمال کردم.
برای ایجاد این Helper سفارشی ابتدا یک کلاس استاتیک ایجاد کنید و با استفاده از extension Methodها یک helper جدید را ایجاد کنید:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
|
namespace Parsnet { public static MvcHtmlString SpanFor<TModel, TProperty>( this HtmlHelper<TModel> htmlHelper, Expression<Func<TModel, TProperty>> expression, object htmlAttributes) { var sb = new StringBuilder(); var span = new TagBuilder( "span" ); var metadata = ModelMetadata.FromLambdaExpression<TModel, TProperty>(expression, htmlHelper.ViewData); var name = ExpressionHelper.GetExpressionText(expression); var fullName = htmlHelper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(name); var value = "" ; if (metadata.Model != null && metadata.Model.GetType() == typeof (List<IdentityProvider.IdentityRole>)) { var modelList = (List<IdentityProvider.IdentityRole>)metadata.Model; value = String.Join( "، " , modelList.Select(r => r.Name)); } else { value = htmlHelper.FormatValue(metadata.Model, null ); } span.MergeAttributes< string , object >(((IDictionary< string , object >)HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes))); var fieldName = fullName.Split( '.' )[1]; span.MergeAttribute( "data-name" , fieldName, true ); span.MergeAttributes< string , object >(htmlHelper.GetUnobtrusiveValidationAttributes(name, metadata)); sb.Append(span.ToString(TagRenderMode.StartTag)); sb.Append(value); sb.Append(span.ToString(TagRenderMode.EndTag)); return new MvcHtmlString(sb.ToString()); } } } |
ما در این helper سفارشی از عبارتهای لامبدا استفاده میکنیم و با استفاده از این عبارات، فیلد مورد نظر مدل خود را به helper معرفی میکنیم. آرگومان htmlAttributes در متد helper نیز برای دریافت خصوصیات اضافی helper است؛ خصوصیاتی مانند class، id, style و غیره.
با استفاده از کلاس TagBuilder تگ مورد نظر خود را ایجاد میکنیم. در اینجا من تگ span را ایجاد کردهام که شما میتوانید هر تگ دلخواه دیگری را نیز ایجاد کنید. اولین مرحله، استخراج اطلاعات موردنیاز از metadata مدل است که در خط زیر با پردازش عبارت لامبدا اینکار صورت میگیرد:
1
|
var metadata = ModelMetadata.FromLambdaExpression<TModel, TProperty>(expression, htmlHelper.ViewData); |
سپس نام فیلد مورد نظر را از مدل استخراج میکنیم:
1
2
|
var name = ExpressionHelper.GetExpressionText(expression); var fullName = htmlHelper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(name); |
کدهای فوق نام فیلد جاری (در اینجا Email) را از MetaData برای ما استخراج میکند. متغیر value برای نگهداری مقدار این فیلد از مدل است. مرحله بعد استخراج مقدار فیلد و انتساب آن به متغیر value است.
در سناریوی من کاربر میتواند زمینهی فعالیت خود را انتخاب کند که به صورت IdentityRole پیاده سازی شده است. من در اینجا چک میکنیم که اگر نوع دادهای این فیلد List<IdentityProvider.IdentityRole> بود زمینه فعالیت کاربر را از طریق "،" از هم جدا کرده و به صورت یک رشته تبدیل میکنم. در غیر اینصورت همان مقدار عادی فیلد را بکار میگیرم.
1
2
3
4
5
6
7
8
9
|
if (metadata.Model != null && metadata.Model.GetType() == typeof (List<IdentityProvider.IdentityRole>)) { var modelList = (List<IdentityProvider.IdentityRole>)metadata.Model; value = String.Join( "، " , modelList.Select(r => r.Name)); } else { value = htmlHelper.FormatValue(metadata.Model, null ); } |
سپس خصوصیات سفارشی خود را که بصورت attributeهای HTML هستند، در خط زیر به تگ سفارشی اعمال میشوند:
1
|
span.MergeAttributes< string , object >(((IDictionary< string , object >)HtmlHelper.AnonymousObjectToHtmlAttributes(htmlAttributes))); |
مهمترین مرحله که در واقع هدف اصلی من بود، استخراج خصوصیتهای *-data برای اعتبارسجی است که در خط زیر اینکار صورت گرفته است:
1
|
span.MergeAttributes< string , object >(htmlHelper.GetUnobtrusiveValidationAttributes(name, metadata)); |
نحوهی استفاده از این helper سفارشی هم خیلی ساده است:
1
|
@Html.SpanFor(m => m.Profile.Email, new { @ class = "editor" , data_type = "text" }) |
و در نهایت HTML خروجی به شکل زیر است:
1
|
<span class = "editor" data-name= "Email" data-type= "text" data-val= "true" data-val-existfiledvalidator= "این رایانامه پیشتر به کار گرفته شده است." data-val-existfiledvalidator-url= "/account/emailexist" data-val-regex= "نشانی رایانامه پذیرفتنی نمیباشد." data-val-regex-pattern= "\w+([-+.']\w+)*@\w+([-.]\w+)*\.\w+([-.]\w+)*" data-val-required= "رایانامه خود را وارد کنید." >alireza_s_84@yahoo.com</span> |
دیدن شکلهای زیر خالی از لطف نیستند:
و پس از ویرایش:
البته برای درک بهتر این موضوع سعی خواهم کرد تا با یک مثال عملی کامل، نحوهی پیاده سازی را در همینجا قرار دهم.