如何在 C# 中为自定义 DataTemplateSelector 获取 DataTemplate 的 x:DataType

Posted

技术标签:

【中文标题】如何在 C# 中为自定义 DataTemplateSelector 获取 DataTemplate 的 x:DataType【英文标题】:How to get x:DataType for a DataTemplate in C# for custom DataTemplateSelector如何在 C# 中为自定义 DataTemplateSelector 获取 DataTemplate 的 x:DataType 【发布时间】:2017-09-10 17:35:45 【问题描述】:

我正在为ComboBox 控件编写自定义DataTemplateSelector,我需要使用它为不同类型的对象显示不同的DateTemplates,在ComboBox 的关闭和打开模式下.

这是我想出的DataTemplateSelector

public class ComboBoxTypedDataTemplateSelector : DataTemplateSelector

    public IEnumerable<DataTemplate> SelectedTemplates  get; set; 

    public IEnumerable<DataTemplate> DropDownTemplates  get; set; 

    protected override DataTemplate SelectTemplateCore(object item, DependencyObject container)
    
        IEnumerable<DataTemplate> source = container.FindParent<ComboBoxItem>() == null
            ? SelectedTemplates // Get the template for the closed mode
            : DropDownTemplates; // Get the template for the open UI mode
        Type type = item.GetType();
        return null; // Some LINQ to get the first DataTemplate in source with the x:DataType that equals type
    


public sealed class DataTemplatesCollection : List<DataTemplate>  

下面是我在 XAML 中的使用方式:

<ComboBox>
    <mvvm:ComboBoxTypedDataTemplateSelector>
        <mvvm:ComboBoxTypedDataTemplateSelector.SelectedTemplates>
            <mvvm:DataTemplatesCollection>
                <DataTemplate x:DataType="models:SomeType">
                    <TextBlock Text="x:Bind ..."/>
                </DataTemplate>
                <DataTemplate x:DataType="models:SomeOtherType">
                    <TextBlock Text="x:Bind ..."/>
                </DataTemplate>
            </mvvm:DataTemplatesCollection>
        </mvvm:ComboBoxTypedDataTemplateSelector.SelectedTemplates>
        <mvvm:ComboBoxTypedDataTemplateSelector.DropDownTemplates>
            <mvvm:DataTemplatesCollection>
                <DataTemplate x:DataType="models:SomeType">
                    <TextBlock Text="x:Bind ..."/>
                </DataTemplate>
                <DataTemplate x:DataType="models:SomeOtherType">
                    <TextBlock Text="x:Bind ..."/>
                </DataTemplate>
            </mvvm:DataTemplatesCollection>
        </mvvm:ComboBoxTypedDataTemplateSelector.DropDownTemplates>
    </mvvm:ComboBoxTypedDataTemplateSelector>
</ComboBox>

现在,我唯一遗漏的难题是,我不知道如何在 C# 中获取 x:DataType 属性(我知道它实际上不是真正的属性,但我希望有办法通过代码检索它)。 我需要类似的东西才能从正确的模板组中为每个对象获得正确的DataTemplate。 有什么方法可以实现吗?

注意:我知道我可以编写一个特定的DataTemplateSelector,其中包含要为每种项目类型返回的模板的硬编码名称,我可以使用该方法作为备用选项。但是,我想知道是否可以使用这种方法编写更通用的选择器,以使其更加模块化并能够在将来重用它。

感谢您的帮助!

编辑:按照文森特的建议,我编写了一个附加属性来将给定的Type 存储在DataTemplate 中:

public class DataTypeHelper

    public static Type GetAttachedDataType(DataTemplate element)
    
        return (Type)element.GetValue(AttachedDataTypeProperty);
    

    public static void SetAttachedDataType(DataTemplate element, Type value)
    
        element.SetValue(AttachedDataTypeProperty, value);
    

    public static readonly DependencyProperty AttachedDataTypeProperty =
        DependencyProperty.RegisterAttached("AttachedDataType", typeof(Type), typeof(DataTypeHelper), new PropertyMetadata(default(Type)));

我试过这样使用它:

...
 <DataTemplate x:DataType="someXlmns:SomeClass"
               mvvm:DataTypeHelper.AttachedDataType="someXlmns:SomeClass">
     ...
 </DataTemplate>

但是我在将附加属性设置为我的类型的那一行得到了一个XamlParseException。我尝试将该属性设置为“Grid”(仅作为测试)并且它没有崩溃,我不明白为什么它不能与我的自定义类型一起使用。

EDIT #2:看起来 x:Type 标记扩展在 UWP 中不可用,我找不到其他方法(我认为根本不可能)获取 Type直接来自 XAML 的实例,因此我只需要使用 XAML 中的类型名称,然后将其与模板选择器中的 item.GetType().Name 进行比较。

直接从 XAML 分配 Type 属性的能力会更好,因为它也会在 XAML 设计器中进行语法/拼写检查,但至少这种方法可以正常工作。

【问题讨论】:

【参考方案1】:

您无法检索此值。这只是提示编译器允许它为绑定生成适当的代码。

您可以创建自定义附加属性来存储您需要的内容,也可以使用 Name 属性。

<local:Selector x:Key="selector" >
    <local:Selector.Template1>
        <DataTemplate x:DataType="local:Item" x:Name="template1" >
            <.../>
        </DataTemplate>
    </local:Selector.Template1>
</local:Selector>

然后在选择器实现中

Template1.GetValue(FrameworkElement.NameProperty);

【讨论】:

您好,感谢您的建议!我已经用附加属性的实现更新了我的问题,你能看一下吗?当我设法让这个建议在我的情况下发挥作用时,会将其标记为答案。 没关系,我认为无法在 UWP 上从 XAML 分配 Type 属性,因此我只是使用字符串与模板选择器中的类型名称进行比较(请参阅更新的问题)。再次感谢!【参考方案2】:

这是我发现根据数据类型使用不同 DataTemplate 的示例。代码:

public class MyDataTemplateSelector : DataTemplateSelector 
 
    public Dictionary<Type, DataTemplate> TemplateDictionary  get; set;  
    protected override DataTemplate SelectTemplateCore(object item) 
     
        // select datatemplate depending on item type 
        Type itemType = item.GetType(); 
        if (!TemplateDictionary.Keys.Contains(itemType)) 
         
            return null; 
         
        return TemplateDictionary[itemType]; 
     
    protected override DataTemplate SelectTemplateCore(object item, DependencyObject container) 
     
        // select datatemplate depending on item type 
        Type itemType = item.GetType(); 
        if (!TemplateDictionary.Keys.Contains(itemType)) 
         
            return null; 
         
        return TemplateDictionary[itemType]; 
     

这是链接https://code.msdn.microsoft.com/How-to-bind-data-to-04dcf26d。您可以下载源代码。

【讨论】:

这样做的问题是很难在 XAML 中定义字典。【参考方案3】:

这是我的 2 美分:

[ContentProperty(Name = nameof(Templates))]
public class TypedDataTemplateSelector : DataTemplateSelector

  public IList<TypedDataTemplate> Templates  get;  
    = new ObservableCollection<TypedDataTemplate>();

  public TypedDataTemplateSelector()
  
    var incc = (INotifyCollectionChanged)Templates;
    incc.CollectionChanged += (sender, e) =>
    
      if (e?.NewItems.Cast<TypedDataTemplate>()
          .Any(tdt => tdt?.DataType == null || tdt?.Template == null) == true)
        throw new InvalidOperationException("All items must have all properties set.");
    ;
  

  protected override DataTemplate SelectTemplateCore(object item, 
      DependencyObject container)
  
    if (item == null) return null;
    if (!Templates.Any()) throw new InvalidOperationException("No DataTemplates found.");

    var result =
      Templates.FirstOrDefault(t => t.DataType.IsAssignableFrom(item.GetType()));
    if (result == null)
      throw new ArgumentOutOfRangeException(
        $"Could not find a matching template for type 'item.GetType()'.");

    return result.Template;
  


[ContentProperty(Name = nameof(Template))]
public class TypedDataTemplate

  public Type DataType  get; set; 
  public DataTemplate Template  get; set; 

用法:

<ContentControl Content="Binding">
  <ContentControl.ContentTemplateSelector>
    <v:TypedDataTemplateSelector>
      <v:TypedDataTemplate DataType="data:Person">
        <DataTemplate>
          <StackPanel>
            <TextBox Header="First name" Text="Binding FirstName" />
            <TextBox Header="Last name" Text="Binding LastName" />
          </StackPanel>
        </DataTemplate>
      </v:TypedDataTemplate>
      <v:TypedDataTemplate DataType="data:Company">
        <DataTemplate>
          <StackPanel>
            <TextBox Header="Company name" Text="Binding CompanyName" />
          </StackPanel>
        </DataTemplate>
      </v:TypedDataTemplate>
    </v:TypedDataTemplateSelector>
  </ContentControl.ContentTemplateSelector>
</ContentControl>

【讨论】:

【参考方案4】:

您实际上可以做的是创建 AttachedProperty 并在 DataTemplate 上使用它,幸运的是,在 UWP DataTemplate 中从树上的 DependecyObject 继承,因此我们可以在其上使用 AP。示例如下:

AP 定义

public class SomeAttachedProperty : DependencyObject
    
        public static readonly DependencyProperty TypeProperty =
        DependencyProperty.RegisterAttached(
          "SomeAttachedProperty",
          typeof(string),
          typeof(SomeAttachedProperty),
          new PropertyMetadata(null)
        );
        public static void SetType(DependencyObject element, string value)
        
            element.SetValue(TypeProperty, value);
        
        public static string GetType(DependencyObject element)
        
            return (string)element.GetValue(TypeProperty);
        
    

XAML

<DataTemplate x:Key="SomeKey" local:SomeAttachedProperty.Type="SomeValue">

在代码中

DataTemplate dataTemplate = resourceDictionary["SomeKey"] as DataTemplate;

    string myValue = dataTemplate.GetValue(SomeAttachedProperty.TypeProperty) as string;

【讨论】:

以上是关于如何在 C# 中为自定义 DataTemplateSelector 获取 DataTemplate 的 x:DataType的主要内容,如果未能解决你的问题,请参考以下文章

如何在 JavaScript 中为自定义对象创建方法?

如何在 iOS 中为自定义属性设置动画

如何在 Angular 中为自定义组件实现伪事件?

如何在 .NET 中为自定义配置部分启用 configSource 属性?

如何在 WooCommerce 中为自定义产品类型启用价格和库存

如何在 app.config 中为自定义部分获取智能感知?