Выбор DataTemplate на основе свойства DataContext в Avalonia - PullRequest
1 голос
/ 01 августа 2020

Я реализую UserControl, который должен отображать список настроек:

public class SettingPropertyItem {
    string Name { get; }
    Type ValueType { get; }
    object Value { get; set; }
}

На основе каждого типа в ValueType должен использоваться другой DataTemplate. Чтобы облегчить это, UserControl имеет следующий Control с SettingPropertyItem в качестве DataContext:

<UserControl x:Class="AVDump3Gui.Controls.Settings.SettingsView">
    ...
    <ItemsControl Items="{Binding Properties}" Margin="16,0,0,0">
        <ItemsControl.ItemTemplate>
            <DataTemplate>
            ...
                <ContentControl Content="{Binding}"/>
            ...
            </DataTemplate>
        </ItemsControl.ItemTemplate>
    </ItemsControl>
    ...
</UserControl>

Затем в представление, в котором используется Usercontrol, я добавил DataTemplate в его DataTemplates:

<sv:SettingsView.DataTemplates>
  <DataTemplate DataType="{x:Type vm:SettingPropertyItem}">
    ...
  </DataTemplate>
</sv:SettingsView.DataTemplates>

Пока все хорошо, все работает как положено. Но теперь я немного озадачен, поскольку не знаю, как применять различные шаблоны данных на основе свойства в контексте данных. В WPF DataTemplateSelector или триггеры кажутся способом go (без учета дополнительных фреймворков), но в Авалонии их, похоже, нет. Я также пробовал стили, но селектор, похоже, не может получить доступ к свойствам DataContext.

Как это можно сделать?

1 Ответ

2 голосов
/ 01 августа 2020

В Авалонии DataTemplateSelector не требуется, потому что вы можете просто реализовать IDataTemplate самостоятельно и выбрать там шаблон.

т.е.

public class MyTemplateSelector : IDataTemplate
{
    public bool SupportsRecycling => false;
    [Content]
    public Dictionary<string, IDataTemplate> Templates {get;} = new Dictionary<string, IDataTemplate>();

    public IControl Build(object data)
    {
        return Templates[((MyModel) data).Value].Build(data);
    }

    public bool Match(object data)
    {
        return data is MyModel;
    }
}

public class MyModel
{
    public string Value { get; set; }
}
  <ItemsControl>
    <ItemsControl.Items>
      <scg:List x:TypeArguments="local:MyModel">
        <local:MyModel Value="MyKey"/>
        <local:MyModel Value="MyKey2"/>
      </scg:List>
    </ItemsControl.Items>
    <ItemsControl.DataTemplates>
      <local:MyTemplateSelector>
        <DataTemplate x:Key="MyKey">
          <TextBlock Background="Red" Text="{Binding Value}"/>
        </DataTemplate>
        <DataTemplate x:Key="MyKey2">
          <TextBlock Background="Blue" Text="{Binding Value}"/>
        </DataTemplate>
        
      </local:MyTemplateSelector>
    </ItemsControl.DataTemplates>
  </ItemsControl>
...