リストスキャフォールドビューの出力にレンダリングされた[Display(Name="Some Title")]
DataAnnotations "Some Title"を取得するにはどうすればよいですか?
このクラスの厳密に型指定されたリストスキャフォールドビューを作成します。
public class CompanyHoliday
{
[Key]
public int Id { get; set; }
[Required]
[Display(Name = "Datum")]
[DataType(System.ComponentModel.DataAnnotations.DataType.Date)]
public DateTime Date { get; set; }
[Required]
[StringLength(50)]
[Display(Name = "Feiertag Name")]
public string Name { get; set; }
}
ビューは次のようになります。
@model IEnumerable<Zeiterfassung.Domain.CompanyHoliday>
@{
ViewBag.Title = "Year";
}
<h2>Year</h2>
<p>
@Html.ActionLink("Create New", "Create")
</p>
<table>
<tr>
<th></th>
<th>
Date
</th>
<th>
Name
</th>
</tr>
@foreach (var item in Model) {
<tr>
<td>
@Html.ActionLink("Edit", "Edit", new { id=item.Id }) |
@Html.ActionLink("Details", "Details", new { id=item.Id }) |
@Html.ActionLink("Delete", "Delete", new { id=item.Id })
</td>
<td>
@String.Format("{0:g}", item.Date)
</td>
<td>
@item.Name
</td>
</tr>
}
</table>
ただし、テーブルヘッダーに「Date」と「Name」は必要ありません。 「データム」と「Feiertag名」を動的にレンダリングしたい。
実際の列ヘッダーのタイトルは、DisplayDataAnnotationから取得する必要があります。
どうすればよいですか?
少し古いトピックですが、これを処理するための拡張メソッドを思いつきました。
私のモデルは次のとおりです。
public class MyModel
{
[Display(Name = "Some Property")]
public string SomeProp { get; set; }
}
このメソッドを静的クラスにドロップします。
namespace Web.Extensions
{
public static class HtmlHelperExtensions
{
public static MvcHtmlString DisplayNameFor<TModel, TProperty>(this HtmlHelper<IEnumerable<TModel>> helper, Expression<Func<TModel, TProperty>> expression)
{
var name = ExpressionHelper.GetExpressionText(expression);
name = helper.ViewContext.ViewData.TemplateInfo.GetFullHtmlFieldName(name);
var metadata = ModelMetadataProviders.Current.GetMetadataForProperty(() => Activator.CreateInstance<TModel>(), typeof(TModel), name);
return new MvcHtmlString(metadata.DisplayName);
}
}
}
次に、IEnumerable<MyModel>
を取得しているビューで、次のように使用できます。
@using Web.Extensions
@model IEnumerable<MyModel>
<table>
<tr>
<th>
@Html.DisplayNameFor(m => m.AvgElecCost)
</th>
生成されるHTMLは次のようになります。
<th>
Some Property
</th>
それが役に立てば幸い!
これが私が従ったパターンです。これは、リストがnullになることはないことを前提としていますが、空にすることはできますが、nullリストの場合は条件を簡単に短絡できます。私はブライアンの拡張メソッドが好きです。
@if (Model.BurgersList.Count > 0)
{
var meta = Model.BurgersList.First();
<table>
<tr>
<th>
@Html.DisplayNameFor(m => meta.Title)
</th>
<th>
@Html.DisplayNameFor(m => meta.HasMustard)
</th>
//etc....
@foreach (var item in Model.AssignmentDefinitions)
{
<tr>
<td>
@Html.DisplayFor(m => item.Title)
</td>
<td>
@Html.DisplayFor(m => item.HasMustard)
</td>
//etc...
}
</table>
}
else
{
@:No burgers available. Create(usually make this an action link) a new one.
}
//MVC4 has the DisplayNameFor Extensions
public static class HtmlHelperExtensions
{
public static MvcHtmlString DisplayNameFor<TModel, TProperty>(this HtmlHelper<IEnumerable<TModel>> helper, Expression<Func<TModel, TProperty>> expression)
{
return DisplayNameFor(expression);
}
public static MvcHtmlString DisplayNameFor<TModel, TProperty>(this HtmlHelper<TModel> helper, Expression<Func<TModel, TProperty>> expression)
{
return DisplayNameFor(expression);
}
private static MvcHtmlString DisplayNameFor<TModel, TProperty>(Expression<Func<TModel, TProperty>> expression)
{
var metadata = ModelMetadata.FromLambdaExpression<TModel, TProperty>(expression, new ViewDataDictionary<TModel>());
var htmlFieldName = ExpressionHelper.GetExpressionText(expression);
string s = metadata.DisplayName ?? (metadata.PropertyName ?? htmlFieldName.Split(new char[] { '.' }).Last<string>());
return new MvcHtmlString(HttpUtility.HtmlEncode(s));
}
}