<string, Drink>
のディクショナリがある場合、アイテムがdictionary.Values
プロパティを使用するように、.Name
をWPFリストボックスにどのようにバインドしますか?
struct Drink
{
public string Name { get; private set; }
public int Popularity { get; private set; }
public Drink ( string name, int popularity )
: this ( )
{
this.Name = name;
this.Popularity = popularity;
}
}
項目コントロールでItemsSource
を設定すると、ソースオブジェクトの列挙子へのバインディングが作成されます。 Dictionary<T1, T2>
の列挙子のタイプはIEnumerable<KeyValuePair<T1, T2>>
です。したがって、アイテムテンプレートでは、Key
プロパティとValue
プロパティにバインドし、パス構文を使用してキーと値の特定のプロパティを取得できます。
ここに例があります。まず、ディクショナリを作成してデータを設定し、それをリソースディクショナリに追加するコード(ディクショナリをデータバインディングに公開する方法はたくさんありますが、これは簡単です)。
namespace WpfApplication17
{
/// <summary>
/// Interaction logic for Window1.xaml
/// </summary>
public partial class Window1 : Window
{
public Window1()
{
Dictionary<string, Drink> d = new Dictionary<string, Drink>();
d.Add("A", new Drink("Nehi", 0));
d.Add("B", new Drink("Moxie", 1));
d.Add("C", new Drink("Vernor's", 2));
d.Add("D", new Drink("Canfield's", 3));
Resources["Drinks"] = d;
InitializeComponent();
}
public class Drink
{
public Drink(string name, int popularity)
{
Name = name;
Popularity = popularity;
}
public string Name { get; set; }
public int Popularity { get; set; }
}
}
}
これで、ListBox
にデータを入力するためのXAML(ListView
の方が簡単です。これは、テンプレートをこれほど複雑にして、見栄えをよくするために定義する必要がないためです)。
<Window x:Class="WpfApplication17.Window1"
xmlns="http://schemas.Microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.Microsoft.com/winfx/2006/xaml"
Title="Window1"
Height="300"
Width="300">
<Grid Margin="10">
<ListBox ItemsSource="{DynamicResource Drinks}" Grid.IsSharedSizeScope="True">
<ListBox.ItemTemplate>
<DataTemplate>
<Grid>
<Grid.ColumnDefinitions>
<ColumnDefinition SharedSizeGroup="Key" />
<ColumnDefinition SharedSizeGroup="Name" />
<ColumnDefinition SharedSizeGroup="Popularity" />
</Grid.ColumnDefinitions>
<TextBlock Margin="2" Text="{Binding Key}" Grid.Column="0"/>
<TextBlock Margin="2" Text="{Binding Value.Name}" Grid.Column="1"/>
<TextBlock Margin="2" Text="{Binding Value.Popularity}" Grid.Column="2"/>
</Grid>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
</Grid>
</Window>
ListView
のXAMLははるかに単純で、起動時に表示がより適切になります。
<ListView ItemsSource="{DynamicResource Drinks}">
<ListView.View>
<GridView>
<GridViewColumn Header="Key"
DisplayMemberBinding="{Binding Key}" />
<GridViewColumn Header="Name"
DisplayMemberBinding="{Binding Value.Name}" />
<GridViewColumn Header="Popularity"
DisplayMemberBinding="{Binding Value.Popularity}" />
</GridView>
</ListView.View>
</ListView>
フォローアップの質問に答えるには:
Adam NathanのWindows Presentation Foundation Unleashedをお勧めします。パネルを使用したレイアウトの章では、グリッドがどのように機能するかを詳細に説明しています。 Grid
は、多くの点でかなり直感に反しています。多くのアイテムを含む単一のGrid
を作成したいと考えていますが、Grid
の行と列の数は動的ではありません。したがって、代わりに、アイテムごとにGrid
を作成し、共有サイズ機能を使用して、各Grid
の列が同じサイズであることを確認します。 ListView
には独自の癖がありますが、一般的な「グリッドに複数のアイテムを表示する」ユースケースの方がはるかに簡単です。
DynamicResource
は、StaticResource
とよく似たマークアップ拡張機能です。違いは、XAMLパーサーが解析時にStaticResource
を解決するときに、参照されるリソースがリソースディクショナリにない場合、例外をスローすることです。 DynamicResource
は、後で項目が辞書に追加された場合に参照を解決します。これには多少のパフォーマンスコストがかかりますが、ほとんどの場合無視できます。 XAMLはStaticResource
で解析されるため、InitializeComponent
を使用すると、私が投稿したコードは機能します。しかし、それを覚える必要がないので、コードでリソースディクショナリに追加するものにバインドする場合は、デフォルトでDynamicResource
を使用します。 XAMLが解析される前または後に作成されます。
ツールボックスについては、VS2010で使い始めるかもしれませんが、2008年に使用できないほどバグが多いことがわかりました。とにかくあまり役に立ちません。レイアウト作業のほとんどすべてをエディターで行い、一部はKaxamlで行います。 2008年のビジュアルエディターは、実際にWPFの学習を行ったと思います難しいそれは私とXAMLの間に抽象化レイヤー(それ自体が私とWPFオブジェクトモデルの間の抽象化レイヤー)を課したためです。そして、それはあまり良い抽象化レイヤーではありません。何を非表示にし、何を表示するかを決定するための設計上の決定は、私には正しいものではないようです。また、地獄のようにバギーです。
ListBox.DisplayMemberPath = "Name";
リストボックスにDataContextの 'Name'プロパティを使用するように指示します