0

这个问题与“如何在 HierarchicalDataTemplate 的 DataType 属性中引用泛型类型? ”的答案密切相关。

我遵循了该答案的基本思想并创建了这个数据结构:

<!-- for DictItemVM<string, Remote.Address> which is a viewmodel for a KeyValuePair<...> -->
<x:Array Type="{x:Type sys:Type}"
         x:Key="KVParamsStringToRemoteAddress"
         xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
         xmlns:sys="clr-namespace:System;assembly=mscorlib"
         xmlns:remote="clr-namespace:Remote"
         xmlns:mvvm="clr-namespace:MVVM">
    <x:Type TypeName="sys:String" />
    <mvvm:GenericType BaseType="{x:Type TypeName=remote:Address}"/>
</x:Array>

<mvvm:GenericType xmlns:mvvm="clr-namespace:MVVM"
                  BaseType="{x:Type TypeName=mvvm:DictItemVM`2}"
                  InnerTypes="{StaticResource KVParamsStringToRemoteAddress}"
                  x:Key="DictItemVMOfStringToRemoteAddress"/>

DictItemVM<T,U>是 a 的视图模型,KeyValuePair<...>源自 BaseVM。BaseVM 有一个 DataTemplate 视图,但我正在努力为DictItemVM<string, Remote.Address>.
Remote.Address 是一个复杂的值类型(存储路径和访问信息)。Remote.Address 有自己的 DataTemplate 视图。
所以现在我有了 StaticResource“DictItemVMOfStringToRemoteAddress”,我想用它来指定一个 DataTemplate:

<DataTemplate x:Key="TestKey" DataType="{StaticResource DictItemVMOfStringToRemoteAddress}">
    <StackPanel>
        <Label Content="UniqueName" />
        <TextBox Text="{Binding UniqueName}" />
        <Label Content="Key"/>
        <TextBox Text="{Binding Key, Mode=OneWay}" IsEnabled="False" />
        <Label Content="Value"/>
        <ContentControl Content="{Binding Value, Mode=OneWay}" />
    </StackPanel>
</DataTemplate>

现在这个 DataTemplate应该被用作一个视图,而是显示 BaseVM 的视图。
有人给我一个关于这个的提示吗?

[编辑:2010-08-09]
我尝试过的一些事情:

在 x:Array 定义中,我替换
<mvvm:GenericType BaseType="{x:Type TypeName=remote:Address}"/>

<x:Type TypeName="remote:Address"/>
因为它基本上就是这样 - 没有区别。

还尝试在标签之间创建 DataType(而不是链接到 StaticResource),如下所示:

<DataTemplate x:Key="TestKey">
    <DataTemplate.DataType>
        <Binding>
            <Binding.Source>
                <mvvm:GenericType 
                  BaseType="{x:Type TypeName=mvvm:DictItemVM`2}">
                    <mvvm:GenericType.InnerTypes>
                        <x:Type TypeName="sys:String" />
                        <x:Type TypeName="remote:Address"/>
                    </mvvm:GenericType.InnerTypes>
                </mvvm:GenericType>
            </Binding.Source>
        </Binding>
    </DataTemplate.DataType>

在 GenericType.InnerTypes 中使用和不使用 x:Array 进行了尝试,都给了我这个错误。

试图从这样的静态属性传递类型:
DataType="{x:Static mvvm:StaticTypes.DictItemVMOfStringToRemoteAddress}"
和这样:
DataType="{Binding Path={x:Static mvvm:StaticTypes.DictItemVMOfStringToRemoteAddress}}"
没有区别。

奇怪的是,这个特定的 DataTemplate 需要有一些x:Key值,而 xaml 资源文件中的所有其他文件都指向常规类型,例如:DataType="{x:Type mvvm:EffectVM}"。如果我删除 x:Key,我会收到错误。

4

1 回答 1

1

我找到了一个解决方案,尽管该解决方案并不令人满意。

KeyValuePair<T,U>在 XAML 中,为要显示的每种类型创建一个 DataTemplate并为其提供一些唯一的 x:Key:

<DataTemplate x:Key="DictItemOfStringAndAddressVM">
    <!-- ... -->
</DataTemplate>

然后在代码隐藏中,创建一个 DataTemplateSelector 并覆盖 SelectTemplate:

public class GenericDataTemplateSelector : System.Windows.Controls.DataTemplateSelector
{
    public override System.Windows.DataTemplate SelectTemplate(object item, System.Windows.DependencyObject container)
    {
        FrameworkElement element = container as FrameworkElement;

        if ((element != null) && (item != null))
        {
            if (item is DictItemVM<string, Remote.Address>)
            {
                return element.FindResource("DictItemOfStringAndAddressVM") as DataTemplate;
            }
            else if(item is SomeOtherComplexType)
            {
                // ...
            }
            else return base.SelectTemplate(item, container);
        }
        return null;
    }
}

再次在 XAML 中,将此类声明为资源:

<mvvm:GenericDataTemplateSelector x:Key="GenDataTempSelect"/>

最后,(在我的情况下)在 ContentControl 中,添加属性:

ContentTemplateSelector="{StaticResource GenDataTempSelect}"

--

缺点:

  • 创建新的 DataTemplate 时,您必须在两个位置更改代码。
  • 每个 ContentControl、ListView、... 都必须设置其适当的属性。
  • 并没有真正回答如何在 WPF 中引用泛型类型的问题!

优点:

  • 易于添加任何结构或复杂性的新类型(享受 C# 相对于 WPF 的所有好处......)
  • WPF 中没有复杂的嵌套类型描述,因为上述解决方案需要。
于 2010-08-11T08:36:24.660 回答