`我有一个列表框,其中包含代表各种任务结果的项目。它们每个都扩展了通用的基类“ ResultsViewModel”,因此共享了显示器所需的某些属性。
我要在资源字典中定义的是每种不同类型任务结果的数据模板,例如,任务A将具有一个实现类的ResultsAViewModel,任务B将具有一个ResultsBViewModel等。我想为每个任务定义一个数据模板子类,将列表框的ItemsSource绑定到ObservableCollection(父类),并使用WPF多态性确定运行时要使用的数据模板。复杂之处在于,取决于结果是处理中的,完成的还是失败的,它将是为每个结果类型选择的三个数据模板之一,这取决于各种触发器。因此,每个派生类将具有模板。
到目前为止,我已经将通用样式应用于列表框的样式,如下所示
<ListBox Background="{StaticResource AppBackground}" Grid.Row="1" Grid.Column="1" Grid.ColumnSpan="2" Padding="10" Style="{StaticResource ResultsItemTemplate}" ItemsSource="{Binding Results}" MouseDoubleClick="ListBox_MouseDoubleClick" />
那个样式如下
<!-- Region General Results styles -->
<Style TargetType="{x:Type ListBox}" x:Key="ResultsItemTemplate" >
<Setter Property="Background">
<Setter.Value>
Tan
</Setter.Value>
</Setter>
<Setter Property="ItemTemplate">
<Setter.Value>
<DataTemplate>
<ContentControl Content="{Binding}">
<ContentControl.Style>
<Style TargetType="{x:Type ContentControl}">
<Setter Property="ContentTemplate" Value="{DynamicResource CalculatingResultsTemplate}"/>
<Style.Triggers>
<DataTrigger Binding="{Binding Path=ProcessingResult}" Value="1">
<Setter Property="ContentTemplate" Value="{DynamicResource ProcessedResultsTemplate}"/>
</DataTrigger>
<DataTrigger Binding="{Binding Path=ProcessingResult}" Value="-1">
<Setter Property="ContentTemplate" Value="{DynamicResource ErroredResultsTemplate}"/>
</DataTrigger>
<DataTrigger Binding="{Binding RelativeSource={RelativeSource Mode=FindAncestor, AncestorType={x:Type ListBoxItem}}, Path=IsSelected}" Value="True">
<Setter Property="Background" Value="White"/>
</DataTrigger>
</Style.Triggers>
</Style>
</ContentControl.Style>
</ContentControl>
</DataTemplate>
</Setter.Value>
</Setter>
<Setter Property="ItemContainerStyle">
<Setter.Value>
<Style TargetType="{x:Type ListBoxItem}">
<Setter Property="Background" Value="{StaticResource AppBackground}" />
<Style.Triggers>
<Trigger Property="IsSelected" Value="True" >
<Setter Property="Background" Value="{StaticResource AppBackground}" />
</Trigger>
</Style.Triggers>
<Style.Resources>
<SolidColorBrush x:Key="{x:Static SystemColors.HighlightBrushKey}" Color="Transparent"/>
<SolidColorBrush x:Key="{x:Static SystemColors.ControlBrushKey}" Color="Transparent" />
</Style.Resources>
</Style>
</Setter.Value>
</Setter>
<Setter Property="ItemsPanel">
<Setter.Value>
<ItemsPanelTemplate>
<WrapPanel />
</ItemsPanelTemplate>
</Setter.Value>
</Setter>
<Setter Property="ScrollViewer.HorizontalScrollBarVisibility" Value="Disabled"/>
</Style>
<!-- EndRegion -->
重要的是,您可以看到此样式根据触发器将ItemTemplate设置为三个值之一。这些模板键是
ProcessedResultsTemplate
CalculatingResultsTemplate
ErroredResultsTemplate
每个都是我的资源字典中的数据模板。
我的问题是,我需要以上三个数据模板中的每一个用于EACH派生的viewmodel类型。我是按键引用它们的。并且您的词典中不能有两个具有相同键的项。例如,如果我用键“ ProcessedResultsTemplate”创建两个数据模板,一个数据类型为x:Type CalculationResultsViewModel,第二个数据模板具有相同的键,但数据类型为x:Type SpotStressResultsViewModel,则由于两个键具有相同的键,它将无法正常工作。
因此,我不确定实现此功能的正确方法,以获得此处发生的一些WPF多态性乐趣。我做错了什么基本的事情吗?
更新:我尝试过使用datatemplate选择器,并使用下面的类,该逻辑实现良好,但是问题是datatemplateselector仅在首次渲染对象时才调用。当某些依赖项属性发生更改时,原始XAML会触发更改数据模板的操作。如何使用触发器或依赖项属性来要求datatemplateselector从新数据值中重新选择数据模板?
class ResultsDataTemplateSelector : DataTemplateSelector
{
public DataTemplate CalculateOnlyProcessedTemplate { get; set; }
public DataTemplate CalculateOnlyCalculatingTemplate { get; set; }
public DataTemplate CalculateOnlyErroredTemplate { get; set; }
public DataTemplate SpotStressProcessedTemplate { get; set; }
public DataTemplate SpotStressCalculatingTemplate { get; set; }
public DataTemplate SpotStressErroredTemplate { get; set; }
public override DataTemplate SelectTemplate(object item, DependencyObject container)
{
if (item is CalculationResultsViewModel)
{
var vm = item as CalculationResultsViewModel;
if (vm.ProcessingResult == 1)
return CalculateOnlyProcessedTemplate;
if (vm.ProcessingResult == -1)
return CalculateOnlyErroredTemplate;
return CalculateOnlyCalculatingTemplate;
}
if (item is SpotStressResultsViewModel)
{
var vm = item as SpotStressResultsViewModel;
if (vm.ProcessingResult == 1)
return SpotStressProcessedTemplate;
if (vm.ProcessingResult == -1)
return SpotStressErroredTemplate;
return SpotStressCalculatingTemplate;
}
return null;
}
}
最佳答案
显而易见的解决方案是将DataTemplateSelector用于选择正确模板的任意复杂决定。将实际逻辑转移到XAML中会导致痛苦。
我个人认为ViewModels应该是视图,其组成需求及其交互的直接,可测试和简洁的表示。您要演示的ViewModel似乎更像是业务模型对象,而不是表示视图。您的ViewModel层应负责生成隐含的内容:
结果ACalculatingViewModel
结果AProcessingViewModel
结果AErrorViewModel
结果BCalculatingViewModel
结果BProcessingViewModel
ResultsBErrorViewModel
而且,您的视图代码应该只是在设置样式。多态是一种更容易实现这些隐式ViewModel的好方法,但无论创建UI的人如何,都不必担心。
关于c# - 列表框中的WPF数据模板多态,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/17885566/