ItemsControl with multiple DataTemplates for a viewmodel

前端 未结 4 1495
陌清茗
陌清茗 2020-11-29 04:59

is it possible to bind an itemscontrol with canvas as template to multiple DataTemplates?

I have 2 collections and depending on the type I would like to display a di

相关标签:
4条回答
  • 2020-11-29 05:11

    Have a look at the Data template selector: here or here.

    0 讨论(0)
  • 2020-11-29 05:14

    You can create multiple ObservableCollections and then bind your ItemsSource to a CompositeCollection which joins those collections.

    Then in your XAML you can create different DataTemplates for the respective types using the DataType property which like styles gets automatically applied if it is placed in the resources. (You can also create the composite in XAML which is shown on MSDN, if the CollectionContainers should be bound that is a bit more difficult though)

    Example code:

    ObservableCollection<Employee> data1 = new ObservableCollection<Employee>(new Employee[]
    {
        new Employee("Hans", "Programmer"),
        new Employee("Elister", "Programmer"),
        new Employee("Steve", "GUI Designer"),
        new Employee("Stefan", "GUI Designer"),
        new Employee("Joe", "Coffee Getter"),
        new Employee("Julien", "Programmer"),
    });
    ObservableCollection<Machine> data2 = new ObservableCollection<Machine>(new Machine[]
    {
        new Machine("E12", "GreedCorp"),
        new Machine("E11", "GreedCorp"),
        new Machine("F1-MII", "CommerceComp"),
        new Machine("F2-E5", "CommerceComp")
    });
    CompositeCollection coll = new CompositeCollection();
    coll.Add(new CollectionContainer() { Collection = data1 });
    coll.Add(new CollectionContainer() { Collection = data2 });
    Data = coll;
    
    <ItemsControl ItemsSource="{Binding Data}">
        <ItemsControl.ItemsPanel>
            <ItemsPanelTemplate>
                <VirtualizingStackPanel/>
            </ItemsPanelTemplate>
        </ItemsControl.ItemsPanel>
        <ItemsControl.Resources>
            <DataTemplate DataType="{x:Type local:Employee}">
                <StackPanel Orientation="Horizontal">
                    <TextBlock Text="{Binding Name}"/>
                    <TextBlock Text=" ("/>
                    <TextBlock Text="{Binding Occupation}"/>
                    <TextBlock Text=")"/>
                </StackPanel>
            </DataTemplate>
            <DataTemplate DataType="{x:Type local:Machine}">
                <StackPanel Orientation="Horizontal">
                    <TextBlock Text="{Binding Model}"/>
                    <TextBlock Text=" - "/>
                    <TextBlock Text="{Binding Manufacturer}"/>
                </StackPanel>
            </DataTemplate>
        </ItemsControl.Resources>
    </ItemsControl>
    

    Here i use a different panel but it should be the same for a canvas.

    0 讨论(0)
  • 2020-11-29 05:20

    You could have ObservableCollection<object> in your ViewModel and bind the ItemsControl's Source to this collection.

    Then, to get a different look for different types of data, you could two DataTemplates without x:Key, but with properly set DataType in your Resources. The ItemsControl will then automatically select the appropriate DataTemplate for your item.

    0 讨论(0)
  • 2020-11-29 05:27

    Another option with less code behind would be to define two ListBoxes, each with their own templates and bound to their own collections. Define them each in the same physical space and just control which one is visible based on your state. You could even do this with the Visual State Manager and custom states.

    0 讨论(0)
提交回复
热议问题