の辞書がある場合、項目が プロパティを使用するように、 を WPF ListBox に<string, Drink>
バインドするにはどうすればよいでしょうか。dictionary.Values
.Name
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;
}
}
ベストアンサー1
アイテム コントロールに を設定すると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; }
}
}
}
次に、 を設定するための XAML を示しますListBox
(ただし、 の方が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>
の XAML はListView
はるかにシンプルで、表示もよりきれいです。
<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>
追加の質問にお答えします:
アダム・ネイサンのWindows Presentation Foundation の解放パネルを使用したレイアウトの章では、グリッドの仕組みについてかなり詳しく説明しています。 はGrid
多くの点でかなり直感に反しています。Grid
多数のアイテムを含む を 1 つ作成したいと考えますが、 の行と列の数はGrid
動的ではありません。 そのため、代わりに各アイテムに対して を作成し、共有サイズ機能を使用して、各 の列が同じサイズになるGrid
ようにします。 には独自の癖がありますが、一般的な「グリッドに複数のアイテムを表示する」という使用例では、はるかに簡単です。Grid
ListView
DynamicResource
は、 とよく似た動作をするマークアップ拡張機能ですStaticResource
。違いは、XAML パーサーがStaticResource
を解析しながら解決するときに、参照されているリソースがリソース ディクショナリにない場合は例外がスローされることです。 は、DynamicResource
項目が後でディクショナリに追加された場合に参照を解決します。これには多少のパフォーマンス コストがかかりますが、ほとんどの場合は無視できます。私が投稿したコードはStaticResource
、 を使用する場合に機能します。XAML は で解析されるためですInitializeComponent
。しかし、私はそれを覚えておくのは好きではないので、DynamicResource
コードでリソース ディクショナリに追加するものにバインドする場合は、デフォルトで を使用し、XAML が解析される前か後に作成されるかを気にしないようにしています。
ツールボックスについては、VS2010で使い始めるかもしれませんが、2008のツールボックスはバグだらけで使い物になりません。それにあまり役に立ちません。レイアウト作業はほとんどエディタで行い、一部はKaxamlで行っています。2008のビジュアルエディタはWPFの学習を実際に楽にしてくれたと思います。もっと強くなぜなら、XAML と WPF オブジェクト モデルの間に抽象化レイヤーが課せられたからです (XAML 自体も WPF オブジェクト モデルと XAML の間に抽象化レイヤーです)。そして、これはあまり良い抽象化レイヤーではありません。何を非表示にし、何を表示するかを決定する設計上の決定は、私には正しいものではないように思えます。また、バグも非常に多いです。