1

WPF 和 MVVM 相当新,我正在尝试将 ContentTemplate(或 ItemTemplate,两者都没有工作)绑定到 C# WPF 程序中的 DataTemplate 属性。我这样做是因为我有一个配置文件,它为每个“条目”定义了不同的“条目显示类型”,以便不必制作无数的视图/视图模型(现在,只有一个通用条目视图模型可以跟踪标签,数据和显示类型,我更愿意保持这种方式以避免不必要的类结构膨胀)。有什么办法可以使这项工作?

这是我尝试过的其中一件事的示例:

XAML:

<ItemsControl IsTabStop="False" ItemsSource="{Binding Path=FNEntries}"Margin="12,46,12,12">
  <ItemsControl.ItemTemplate>
    <DataTemplate>
      <ContentControl ContentTemplate="{Binding Path=TypeView}" />
    </DataTemplate>
  </ItemsControl.ItemTemplate>
</ItemsControl>

CS(在具有 (DataTemplate)TypeView 和 (string)PropertyName 的入口视图模型类构造函数内):

NodeTypeView = (DataTemplate)Application.Current.FindResource("TypeTest");

资源 XAML:

<DataTemplate x:Key="TypeTest">
<TextBlock Margin="2,6">
  <TextBlock Text="{Binding Path=PropertyName}" />
</TextBlock>

当我用它运行时,什么都没有出现。但是,如果我将资源数据模板的内容直接放在内容控件的位置上,一切都会显示得很好(除了它不是我想要的数据驱动方式)。任何帮助/建议将不胜感激。谢谢!

4

1 回答 1

5

我真的会说你大多做错了=)

将模板存储在 ViewModel 中通常是个坏主意,因为您会将图形对象存储在 VM 中。这应该在视图端完成

如果您想要根据项目类型或其他任何类型的变量 DataTemplate,这里有一些替代的“更清洁”的解决方案:

先决条件:您的所有模板都在某处定义为资源。

假设您有ResourceDictionary以下内容,用于测试目的:

<DataTemplate x:Key="Template1" />
<DataTemplate x:Key="Template2" />
<DataTemplate x:Key="Template3" />

解决方案1:使用ItemTemplateSelector

(最干净的解决方案恕我直言)对于这个问题,我会把你重定向到这个优秀的教程,它教我如何使用它 如果我能理解它,你绝对不能 =D

解决方案 2:在您的Binding

让我们稍微改变你的Binding,通过使用转换器使其绑定到当前对象本身

<DataTemplate>
      <ContentControl ContentTemplate="{Binding Converter={StaticResource MyConverter}}" />
    </DataTemplate>

这是您的转换器的样子(注意:value这里的对象是绑定对象,在您的情况下,您正在使用它的类型,所以这个例子也是关于类型的)

public class MyConverter : IValueConverter
{
    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        if (value.GetType() == typeof(WhateverYouWant))
        {
        return (DataTemplate)Application.Current.FindResource("OneTemplate");
        } 
        else if (value.getType() == typeof(AnotherTypeHere))
        {
        return (DataTemplate)Application.Current.FindResource("AnotherTemplate");
        }
        // other cases here...
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        return value; //We don't care about this!
    }
}

这将为您解决问题

我想这两种解决方案都可以工作,并且更清洁,但请注意,这正是ItemTemplateSelector. 该Converter方法是我在了解这些模板选择器之前使用的方法,我不再使用它

干杯!

于 2012-09-21T21:08:00.140 回答