これは非常に簡単だと思います。モデル内のリスト内のアイテムの表示名を表示する正しい方法を見つけることができないようです。
私の簡易モデル:
public class PersonViewModel
{
public long ID { get; set; }
private List<PersonNameViewModel> names = new List<PersonNameViewModel>();
[Display(Name = "Names")]
public List<PersonNameViewModel> Names { get { return names; } set { names = value; } }
}
および名前:
public class PersonNameViewModel
{
public long ID { get; set; }
[Display(Name = "Set Primary")]
public bool IsPrimary { get; set; }
[Display(Name = "Full Name")]
public string FullName { get; set; }
}
次に、人の名前をすべて表示するテーブルを作成し、DisplayNameFor FullNameを取得します。明らかに、
@Html.DisplayNameFor(model => model.Names.FullName);
動作しない、そして
@Html.DisplayNameFor(model => model.Names[0].FullName);
名前がなければ中断します。ここで表示名を取得する「最良の方法」はありますか?
リストにアイテムがなくても、これは実際に機能します。
@Html.DisplayNameFor(model => model.Names[0].FullName)
MVCは式を実際に実行する代わりに解析するため、機能します。これにより、リストに要素がなくても正しいプロパティと属性を見つけることができます。
パラメータ(上記のmodel
)を使用する必要さえないことに注意してください。これも機能します:
@Html.DisplayNameFor(dummy => Model.Names[0].FullName)
これと同様に:
@{ Namespace.Of.PersonNameViewModel dummyModel = null; }
@Html.DisplayNameFor(dummyParam => dummyModel.FullName)
それを行う別の方法があり、それはより明確だと思います:
public class Model
{
[Display(Name = "Some Name for A")]
public int PropA { get; set; }
[Display(Name = "Some Name for B")]
public string PropB { get; set; }
}
public class ModelCollection
{
public List<Model> Models { get; set; }
public Model Default
{
get { return new Model(); }
}
}
そして、ビューで:
@model ModelCollection
<div class="list">
@foreach (var m in Model.Models)
{
<div class="item">
@Html.DisplayNameFor(model => model.Default.PropA): @m.PropA
<br />
@Html.DisplayNameFor(model => model.Default.PropB): @m.PropB
</div>
}
</div>
T-motyのソリューションが好きです。私はジェネリックを使用したソリューションが必要だったので、私のソリューションは本質的にこれです:
public class CustomList<T> : List<T> where T : new()
{
public static async Task<CustomList<T>> CreateAsync(IQueryable<T> source)
{
return new CustomList<T>(List<T> list); //do whatever you need in the contructor, constructor omitted
}
private T defaultInstance = new T();
public T Default
{
get { return defaultInstance; }
}
}
ビューでこれを使用することは、彼の例と同じです。空のオブジェクトの単一のインスタンスを作成するため、Defaultを参照するたびに新しいインスタンスを作成することはありません。
new T()を呼び出すには、new()制約が必要です。モデルクラスにデフォルトのコンストラクタがない場合、またはコンストラクタに引数を追加する必要がある場合は、これを使用できます。
private T defaultInstance = (T)Activator.CreateInstance(typeof(T), new object[] { "args" });
ビューには@ modelのような行があります:
@model CustomList<Namespace.Of.PersonNameViewModel.Model>