如何通过HierarchicalDataTemplate设置DataContext?

时间:2015-04-15 14:47:57

标签: wpf treeview hierarchicaldatatemplate

我目前对此感到生气:如何通过Hierarchical树传播DataContext? (鉴于MyItem的子项已知且已修复,目标是设置框架的源URI和Da​​taContext) (另外还会在以后添加其他类别和子项目)


首先,这是一个特定的项目定义:

namespace WpfApplication1
{
    using System.Collections.Generic;

    public class MyItem
    {
        public string Name { get; set; }

        public List<MyItem> GetItems()
        {
            return new List<MyItem>() { 
                new MyItem() { Name = "Item A" }, 
                new MyItem() { Name = "Item B" } 
            };
        }
    }
}

其次,特定DataTemplate选择器的定义:

namespace WpfApplication1
{
    using System;
    using System.Collections.Generic;
    using System.Linq;
    using System.Windows.Controls;
    using System.Windows;

    public class MyTemplateSelector : DataTemplateSelector
    {
        public static List<DataTemplate> DataTemplates = new List<DataTemplate>();

        public override System.Windows.DataTemplate SelectTemplate(object item, System.Windows.DependencyObject container)
        {
            var datatemplate = DataTemplates.FirstOrDefault(dt => dt.DataType as Type == item.GetType());

            return datatemplate ?? base.SelectTemplate(item, container);
        }


    }
}

第三,这是MainWindow

<Window x:Class="WpfApplication1.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:my="clr-namespace:WpfApplication1"
        Title="MainWindow" Height="250" Width="800" >

    <Window.Resources>

        <ObjectDataProvider x:Key="ItemsProvider" ObjectType="{x:Type my:MyItem}" MethodName="GetItems" />

        <my:MyTemplateSelector x:Key="MySelector"  />

        <CompositeCollection x:Key="fixedsubitems">
            <HeaderedItemsControl Header="category #i" >
                <HeaderedItemsControl Header="Products j" />
                <HeaderedItemsControl Header="Products k" />
            </HeaderedItemsControl>
        </CompositeCollection>

        <HierarchicalDataTemplate x:Key="myItemTemplate" DataType="{x:Type my:MyItem}" ItemsSource="{Binding Source={StaticResource fixedsubitems}}" ItemTemplateSelector="{StaticResource MySelector}">
            <!--<HierarchicalDataTemplate.ItemContainerStyle>
                <Style TargetType="TreeViewItem" >
                    <Setter Property="DataContext" Value="{Binding RelativeSource={RelativeSource Self}}" ></Setter>
                </Style>
            </HierarchicalDataTemplate.ItemContainerStyle>-->
            <TextBlock Text="{Binding Path=Name}"/>
        </HierarchicalDataTemplate>

        <HierarchicalDataTemplate x:Key="HeaderedItemsHierarchicalTemplate" DataType="{x:Type HeaderedItemsControl}" ItemsSource="{Binding Path=Items}" ItemTemplateSelector="{StaticResource MySelector}">
            <StackPanel Orientation="Horizontal" ContextMenu="{Binding ContextMenu}" >
                <TextBlock Text="{Binding Header}" Margin="3" />
            </StackPanel>
        </HierarchicalDataTemplate>

        <CompositeCollection x:Key="sections" >
            <HeaderedItemsControl Header="Section #1" >
                <HeaderedItemsControl Header="Entry 1.1" >
                </HeaderedItemsControl>
                <HeaderedItemsControl Header="My items" ItemsSource="{Binding Source={StaticResource ItemsProvider}}" >
                </HeaderedItemsControl>
            </HeaderedItemsControl>
            <HeaderedItemsControl Header="Section #2" >
                <HeaderedItemsControl Header="Entry 2.1" >
                </HeaderedItemsControl>
                <HeaderedItemsControl Header="Entry 2.2" >
                </HeaderedItemsControl>
            </HeaderedItemsControl>
        </CompositeCollection>

    </Window.Resources>

    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="1*" />
            <ColumnDefinition Width="3*" />
        </Grid.ColumnDefinitions>

        <ScrollViewer VerticalScrollBarVisibility="Auto" DataContext="{Binding Source={StaticResource sections}}">
            <ListBox Name="list" ItemsSource="{Binding}" HorizontalContentAlignment="Stretch" SelectedItem="{Binding Path=/}">
                <ListBox.ItemTemplate>
                    <DataTemplate >
                        <StackPanel >
                            <Border >
                                <TextBlock Text="{Binding Header}" TextAlignment="Center" VerticalAlignment="Center" Padding="5" />
                            </Border>
                            <Border >
                                <TreeView Name="tv" ItemsSource="{Binding Items}" ItemTemplate="{StaticResource HeaderedItemsHierarchicalTemplate}" >
                                    <TreeView.Resources>
                                        <Style TargetType="TreeViewItem" >
                                            <Setter Property="IsExpanded" Value="True" />
                                        </Style>
                                    </TreeView.Resources>
                                </TreeView>
                            </Border>
                        </StackPanel>
                    </DataTemplate>
                </ListBox.ItemTemplate>
            </ListBox>
        </ScrollViewer>

        <Frame Name="frame" Source="" Grid.Column="1" />

    </Grid>
</Window>

最后,MainWindow代码背后:

using System.Collections;
using System.Diagnostics;
using System.Linq;
using System.Windows;

namespace WpfApplication1
{
    /// <summary>
    /// Interaction logic for MainWindow.xaml
    /// </summary>
    public partial class MainWindow : Window
    {
        public MainWindow()
        {
            InitializeComponent();

            foreach (var res in this.Resources.Cast<DictionaryEntry>())
            {
                if (res.Value is DataTemplate)
                {
                    MyTemplateSelector.DataTemplates.Add(res.Value as DataTemplate);
                }
            }
        }

    }
}

这4个文件显示该窗口

theses 4 files display that window

如何在点击类别或产品等子项目时获取MyItem上下文?

我尝试了几种使用ItemContainerStyle的方法,在DataTemplate资源中定义的Style内部的Setter ......永远不会成功!

1 个答案:

答案 0 :(得分:0)

首先,我使用使用交互式触发器管理的命令更改TreeView代码隐藏交互:

<Window.CommandBindings>
    <CommandBinding Command="Open" Executed="CommandBinding_Executed" />
</Window.CommandBindings>
../..
<TreeView Name="tv" ItemsSource="{Binding Items}" ItemTemplate="{StaticResource HeaderedItemsHierarchicalTemplate}" >
    <i:Interaction.Triggers>
        <i:EventTrigger EventName="SelectedItemChanged">
            <i:InvokeCommandAction Command="Open" CommandParameter="{Binding ElementName=tv, Path=SelectedItem}"/>
        </i:EventTrigger>
    </i:Interaction.Triggers>
    <TreeView.Resources>
        <Style TargetType="TreeViewItem" >
            <Setter Property="IsExpanded" Value="True" />
        </Style>
    </TreeView.Resources>
</TreeView>

使用此相关代码

    private void CommandBinding_Executed(object sender, ExecutedRoutedEventArgs e)
    {
        var tvi = e.OriginalSource as TreeViewItem;
        if (tvi != null && tvi.Tag != null)
        {
            frame.DataContext = tvi.Tag;
            frame.Source = new Uri("Page1.xaml", UriKind.Relative);
        }
    }

2:我将固定的子项分成几个级别:

<CompositeCollection x:Key="fixedsubitems">
    <HeaderedItemsControl Header="category #i" >
        <HeaderedContentControl Header="Products j" />
        <HeaderedContentControl Header="Products k" />
    </HeaderedItemsControl>
</CompositeCollection>

1级的HeaderedItemsControl, 2级的HeaderedContentControl

3:我通过HeaderedItemsControl HierarchicalDataTemplate的ItemContainerStyle设置了我的源代码:

    <HierarchicalDataTemplate.ItemContainerStyle>
        <Style >
            <Setter Property="Control.Tag" Value="{Binding RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type TreeViewItem}, AncestorLevel=2}, Path=DataContext}" ></Setter>
        </Style>
    </HierarchicalDataTemplate.ItemContainerStyle>

4:我复制HeaderedItemsControl HierarchicalDataTemplate以生成HeaderedContentControl HierarchicalDataTemplate并在ItemContainerStyle中设置AncestorLevel = 3

5:我处理框架的LoadCompleted事件

    <Frame Name="frame" Grid.Column="1" LoadCompleted="frame_LoadCompleted" />

使用此相关代码隐藏

    private void frame_LoadCompleted(object sender, System.Windows.Navigation.NavigationEventArgs e)
    {
        var content = frame.Content as FrameworkElement;
        if (content == null)
            return;
        content.DataContext = frame.DataContext;
    }

这是我的Page1:

<Page x:Class="WpfApplication1.Page1"
      xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
      xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
      xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" 
      xmlns:my="clr-namespace:WpfApplication1" 
      xmlns:d="http://schemas.microsoft.com/expression/blend/2008" 
      mc:Ignorable="d" 
      d:DesignHeight="300" d:DesignWidth="300"
      d:DataContext="{d:DesignInstance Type=my:MyItem}"
    Title="Page1" >

    <Grid>
        <TextBlock Text="{Binding Path=Name}" FontSize="16" Background="WhiteSmoke" HorizontalAlignment="Center" VerticalAlignment="Center" />
    </Grid>
</Page>

现在工作! enter image description here