web-dev-qa-db-ja.com

Dictionary <T>をWPFリストボックスにバインド

<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;
    }
}
21
Joan Venge

項目コントロールで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オブジェクトモデルの間の抽象化レイヤー)を課したためです。そして、それはあまり良い抽象化レイヤーではありません。何を非表示にし、何を表示するかを決定するための設計上の決定は、私には正しいものではないようです。また、地獄のようにバギーです。

48
Robert Rossney

ListBox.DisplayMemberPath = "Name";

リストボックスにDataContextの 'Name'プロパティを使用するように指示します

1
Stephan