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

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

ただいまの
回答率

90.62%

  • C#

    6869questions

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

  • WPF

    680questions

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

  • XAML

    246questions

    XAML(Extensible Application Markup Language)はWPF、Silverlight、Windows PhoneそしてWindows Store appsでユーザーインターフェースを定義するために使われるXML言語です。

XAMLのみで動的にエレメントを追加したい

解決済

回答 1

投稿

  • 評価
  • クリップ 2
  • VIEW 6,674
退会済みユーザー

退会済みユーザー

現在コード上で動的にエレメントを追加しているコードがあります。
これをXAMLで記述したい(データバインディングしたい)のですが方法が判りません。
わかる方ご教示ください。調査に必要なキーワードのみでも構いません。

詳細:
・動的にエレメントを追加したい。
・追加するエレメントは型も動的なため、FrameworkElementを指定。

<抜粋>
void AddElements(FrameworkElement[] elements)
{
    foreach(var e in elements)
        grid1.Children.Add(e);
}
[XAML]
<Grid Name="grid1">
</Grid>
上記をコードではなくXAMLのみデータバインディングしたいのです。

イメージとしては下記のようにしたい。
(当然ですが下記例では動きません)

<Grid Name="grid1">
{DataBinding Path="Elements"}
</Grid>

以上、よろしくお願いします。
  • 気になる質問をクリップする

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

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

    クリップを取り消します

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

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

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

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

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

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

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

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

    質問の評価を下げる

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

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

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

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

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

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

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

    詳細な説明はこちら

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

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

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

質問への追記・修正、ベストアンサー選択の依頼

  • Tak1wa

    2015/08/26 16:24

    以下質問です。 ・親エレメントがGridである理由は何でしょうか。 ・DataBindingということはMVVMについては理解されており、ViewModelを実装済みでしょうか。 ・エレメントの追加例をいくつか記載できますでしょうか。(何を以てどのエレメントを追加するのか)

    キャンセル

  • 退会済みユーザー

    退会済みユーザー

    2015/08/26 16:37

    質問ありがとうございます。 ・コードには記載しませんでしたが、件数によって動的にカラムを増やしていきたいからです。 (エレメントを横に並べていきたいのと、その時に微妙にレイアウトも変更したい)  レイアウトの詳細な都合上、StackPanel辺りは不適切でした。 ・ViewModelは実装済みです。現在Window側にあるコードを順次XAML+ViewModelに移動させている最中です。 ・エレメントの例はImage,MediaElementと言った画像が表示されるものです。これらのエレメントは今のところ確定でないため、FrameworkElementとしています。 以上、よろしくお願いします。

    キャンセル

回答 1

checkベストアンサー

+1

こんにちは。
何を以てどのエレメントを扱うかによると思いますが、
汎用的なところではDataTemplateが良いでしょうか。
(もちろんTriggerとかでもなんでもできると思いますが)

ViewModelからバインドされる子要素の型によってエレメントをView側で決定します。

<Window x:Class="WpfApplication1.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="clr-namespace:WpfApplication1"
        Title="MainWindow" Height="350" Width="525">
    <Window.DataContext>
        <local:ViewModel />
    </Window.DataContext>
    <Window.Resources>
        <!--DataTemplateとして定義-->
        <DataTemplate DataType="{x:Type local:ItemModelA}">
            <TextBox Text="TypeA" Background="Yellow" />
        </DataTemplate>
        <DataTemplate DataType="{x:Type local:ItemModelB}">
            <CheckBox Content="TypeB" />
        </DataTemplate>
        <DataTemplate DataType="{x:Type local:ItemModelC}">
            <TextBlock Text="TypeC" Foreground="Red" />
        </DataTemplate>
    </Window.Resources>
    <Grid>
        <ItemsControl ItemsSource="{Binding MyProperty}">
            <!--レイアウト定義はどうやっても良いがItemsPanelでやる例-->
            <ItemsControl.ItemsPanel>
                <ItemsPanelTemplate>
                    <UniformGrid Rows="1" />
                </ItemsPanelTemplate>
            </ItemsControl.ItemsPanel>
        </ItemsControl>
    </Grid>
</Window>

public class ViewModel
{
    public ViewModel()
    {
        MyProperty = new System.Collections.ObjectModel.ObservableCollection<ItemModelBase>();
        MyProperty.Add(new ItemModelA());
        MyProperty.Add(new ItemModelB());
        MyProperty.Add(new ItemModelC());
    }

    public System.Collections.ObjectModel.ObservableCollection<ItemModelBase> MyProperty { get; set; }
}

public class ItemModelBase { }
public class ItemModelA : ItemModelBase { }
public class ItemModelB : ItemModelBase { }
public class ItemModelC : ItemModelBase { }

実現したいことが具体的になると若干変わってくるかとは思います。
少なくともViewModel側でElementの参照を保持するのはMVVMの設計原則に反し、メモリリークのリスクも出てきますので避けたほうが良いでしょう。

投稿

  • 回答の評価を上げる

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

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

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

  • 回答の評価を下げる

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

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

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

  • 2015/08/26 17:03

    回答ありがとうございます。
    この方法ですと予め全てのエレメントが定まっている状態でのみ有効な形ですよね。
    (初心者なので理解が違っていたら申し訳ないです)
    適用するエレメントが定まっていないのは、エレメントを生成する部分をプラグインとして外部実装したいからなのです。
    プラグインの追加のみで作られるエレメントを自由に変更したい、と考えております。
    こういった形だとXAMLで定義せずに素直に現状のコードの方がいいのでしょうか?

    キャンセル

  • 2015/08/26 17:09

    なるほど。そういうことだったのですね。
    サンプルソースでは仰るとおり予めすべてのエレメントが決まっている必要があります。

    外部実装の要件にも依ります。

    ただ外だしにしたいだけであれば、Window.Resource部分を外部リソースで定義すれば良いです。
    プラグイン的な動作であれば動的にエレメント生成を行う必要がありますのでDataTemplateSelectorなどを用いて、その中でプラグイン実装してやるのが良いでしょうか。
    もしくは親エレメント自体をカスタムコントロール化してやるという方法もあります。

    キャンセル

  • 2015/08/27 09:29

    リソース部分を外出しにすることも出来るんですね。素晴らしいです。
    普通にコードで生成させていましたが調べてみます。
    ありがとうございました。

    キャンセル

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

  • ただいまの回答率 90.62%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

関連した質問

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

  • C#

    6869questions

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

  • WPF

    680questions

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

  • XAML

    246questions

    XAML(Extensible Application Markup Language)はWPF、Silverlight、Windows PhoneそしてWindows Store appsでユーザーインターフェースを定義するために使われるXML言語です。