質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

ただいまの
回答率

90.51%

  • C#

    9048questions

    C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

  • WPF

    826questions

    Windows Presentation Foundation (WPF) は、魅力的な外観のユーザー エクスペリエンスを持つ Windows クライアント アプリケーションを作成するための次世代プレゼンテーション システムです

DataContextで複数のクラスを使用するには

解決済

回答 1

投稿

  • 評価
  • クリップ 3
  • VIEW 7,128

cube

score 22

前提・実現したいこと

クラスと関連付けられたComboBoxを複数配置したいと考え、以下のようなコードを書きました。
このコードでは省略してますが、コンボボックスで選択した項目に設定されている変数をBindingで表示や変更させることなどもできています。
(Personを選んだら年齢を表示。Bookを選んだら著作者表示など)

しかし、以下のコードではPersonAddのボタンの処理と、BookAddのボタンの処理を共存させられません。

該当のソースコード

using System.Windows;

namespace WpfApplication2
{
    /// <summary>
    /// MainWindow.xaml の相互作用ロジック
    /// </summary>

    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();
        }

        private void PersonAddButton_Click(object sender, RoutedEventArgs e)
        {
            var List = this.DataContext as PersonList;
            List.PersonList_Items.Add(new Person { Name = "新しいPerson" });
        }

        private void BookAddButton_Click(object sender, RoutedEventArgs e)
        {
            var List = this.DataContext as BookList;
            List.BookList_Items.Add(new Book { Name = "新しいBook" });
        }
    }
}
using System.Collections.ObjectModel;

namespace WpfApplication2
{
    //Personのクラス
    public class Person
    {
        public string Name { get; set; }
        public int Age { get; set; }
    }

    //Bookのクラス
    public class Book
    {
        public string Name { get; set; }
        public string Author { get; set; }
    }

    //--------------------------------------------

    public class PersonList
    {
        public PersonList()
        {
            PersonList_Items = new ObservableCollection<Person> { };
        }
        public ObservableCollection<Person> PersonList_Items { get; set; }
        public Person PersonList_SelectedItem { get; set; }
    }

    public class BookList
    {
        public BookList()
        {
            BookList_Items = new ObservableCollection<Book> { };
        }
        public ObservableCollection<Book> BookList_Items { get; set; }
        public Book BookList_SelectedItem { get; set; }
    }
}
<Window x:Class="WpfApplication2.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        xmlns:local="clr-namespace:WpfApplication2"
        mc:Ignorable="d"
        Title="MainWindow" Height="100" Width="270">

    <Window.DataContext>
        <local:PersonList/>
    </Window.DataContext>

    <Grid>
        <ComboBox x:Name="comboBox" Margin="10,10,132,0" VerticalAlignment="Top"
                   ItemsSource="{Binding Path=PersonList_Items}" DisplayMemberPath="Name" />
        <Button x:Name="PersonAddButton" Content="PersonAdd" Margin="10,37,177,0" VerticalAlignment="Top"
                 Click="PersonAddButton_Click"/>

        <ComboBox x:Name="comboBox1" Margin="135,10,7,0" VerticalAlignment="Top"
                   ItemsSource="{Binding Path=BooktList_Items}" DisplayMemberPath="Name" />
        <Button x:Name="BookAddButton" Content="BookAdd" Margin="135,37,52,0" VerticalAlignment="Top"
                 Click="BookAddButton_Click"/>
    </Grid>
</Window>

問題点

<Window.DataContext>
        <local:PersonList/>
</Window.DataContext>

xamlのこの部分が、"PersonList"か"BookList"どちらかしか設定できません。
"PersonList"にするとBookAddを押したときに。"BookList"にするとPersonAddを押したときにエラーになります。

PersonListとBookListを同一のクラスの入れ子クラスにしたりもしましたが、上手くいきませんでした。

質問

どのようにすれば2つの処理を共存させられますでしょうか?
ご教授のほどよろしくお願いいたします。

補足

C#どころかプログラムも初心者なので、盛大な勘違いをしているかもしれませんがご容赦ください。

  • 気になる質問をクリップする

    クリップした質問は、後からいつでもマイページで確認できます。

    またクリップした質問に回答があった際、通知やメールを受け取ることができます。

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+2

Windowと1対1で対応するViewModelクラスを作成してDataContextに設定するのが良いかと思います。
例えば以下のようなクラスです。

//新規に追加するクラス
    class MainWindowViewModel
    {
        public PersonList PersonList { get; private set; }
        public BookList BookList { get; private set; }

        public MainWindowViewModel()
        {
            PersonList = new PersonList();
            BookList = new BookList();
        }
    }

XAML側は以下のようになります。

//一部省略
    <Window.DataContext>
        <local:MainWindowViewModel/>
    </Window.DataContext>

    <Grid>
        <ComboBox x:Name="comboBox" Margin="10,10,132,0" VerticalAlignment="Top"
                   ItemsSource="{Binding Path=PersonList.PersonList_Items}" DisplayMemberPath="Name" />
        <Button x:Name="PersonAddButton" Content="PersonAdd" Margin="10,37,177,0" VerticalAlignment="Top"
                 Click="PersonAddButton_Click"/>

        <ComboBox x:Name="comboBox1" Margin="135,10,7,0" VerticalAlignment="Top"
                   ItemsSource="{Binding Path=BookList.BookList_Items}" DisplayMemberPath="Name" />
        <Button x:Name="BookAddButton" Content="BookAdd" Margin="135,37,52,0" VerticalAlignment="Top"
                 Click="BookAddButton_Click"/>
    </Grid>

クリックイベント処理は以下のようになります。
※ただ通常WPFではViewであるWindowのコードビハインドに余計なロジックが含まれるのは好ましくなく、
可能であればこのような処理はViewModel側で実装してXAMLでBindすることが多いです。

//一部省略
        private void PersonAddButton_Click(object sender, RoutedEventArgs e)
        {
            var vm = this.DataContext as MainWindowViewModel;
            vm.PersonList.PersonList_Items.Add(new Person { Name = "新しいPerson" });
        }

        private void BookAddButton_Click(object sender, RoutedEventArgs e)
        {
            var vm = this.DataContext as MainWindowViewModel;
            vm.BookList.BookList_Items.Add(new Book { Name = "新しいBook" });
        }

投稿

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2016/09/19 09:57

    ありがとうございます。
    望んだとおりにできました。

    キャンセル

同じタグがついた質問を見る

  • C#

    9048questions

    C#はマルチパラダイムプログラミング言語の1つで、命令形・宣言型・関数型・ジェネリック型・コンポーネント指向・オブジェクティブ指向のプログラミング開発すべてに対応しています。

  • WPF

    826questions

    Windows Presentation Foundation (WPF) は、魅力的な外観のユーザー エクスペリエンスを持つ Windows クライアント アプリケーションを作成するための次世代プレゼンテーション システムです