WPF staticresource 对 DataTemplates 中逻辑资源的引用未在运行时解析

标签 wpf datatemplate dynamicresource staticresource

在从 .net 3.5 升级到 .net 4 的过程中,我是否遗漏了一些东西,因为我看到了似乎与系统目标背道而驰的错误行为。

我正在尝试使用一些示例来构建一个简单的 MVVM 库以用于工作。我在 Twitter 客户端应用程序中使用它进行一些额外的学习,但遇到了一个很大的绊脚石。

场景是这样的。我的根 ViewModel (TwitterClientViewModel) 对象被赋予一个 DialogViewModel 对象的实例以供显示。 DialogViewModel 被添加到一个集合中,并且 bool HasDialogs 被设置为 true。如有必要,将为集合和标志调用 PropertyChanged 事件。这部分效果非常好。

TwitterClientViewModel 的 View 称为 TwitterClientTemplate 并使 Visible 成为 DialogViewTemplate(DialogViewModel 的 View )托管的覆盖层。宿主 ContentControl 的模板引用 DialogViewTemplate 和 DynamicResource 扩展。这在设计器和运行时显示效果很好。

这就是事情变得奇怪的地方。 DialogViewTemplate 的“主体”托管对话框内容,其中包含绑定(bind)到 DialogViewModel.Content(类型对象)的进一步内容控件。希望通过使用 TemplateSelector(我写了一个很好的声明性模板选择器,但出于测试目的注释掉了)我可以同时显示文本和交互元素。例如,在验证 Twitter 帐户时向用户请求详细信息。在这种情况下,是 PIN 码。

此时我有两个用于对话框实现的嵌套内容控件。出于测试目的,DialogViewTemplate 正文中的内容控件使用静态资源扩展来检索 EnterPINDialogTemplate(EnterPINDialogViewModel 的 View )。 EnterPINDialogTemplate 和 DialogViewTemplate 都在同一个文件中(当然前者是先定义的),尽管它们最初是分开的。

在运行时,staticresource 扩展会抛出一个带有消息的 XamlParseException; “为‘System.Windows.Markup.StaticResourceHolder’提供值(value)引发了异常。”

和内部异常消息;

'找不到名为“EnterPINDialogTemplate”的资源。资源名称区分大小写'

使用动态资源会返回 null 并在内容控件中显示 EnterPINDialogViewModel 类型的全名 - 正如资源未解析时所预期的那样。在调用 FrameWorkElement.FindResource() 时闯入我的自定义 TemplateSelector 会引发类似的异常(TryFindResource 返回 null)。

我的第一个想法是在构建数据模板时逻辑树被拆分,我想起了早期项目中该领域的一个问题。我尝试使用 ResourceDictionary 的 MergeDictionaries 属性使资源字典在 DataTemplate 中可用,但设计者不喜欢这一点,错误描述如下: http://connect.microsoft.com/VisualStudio/feedback/details/498844/wpf-designer-throws-invalidcastexception

打消这个念头。我曾尝试在应用程序、窗口和 TwitterClientTemplate 级别合并字典,但没有成功。

下面是 xaml 文件。

DialogTemplates.xaml

<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:VM="clr-namespace:EpicTweet.ViewModel" 
xmlns:ET="clr-namespace:EpicTweet"
xmlns:T="clr-namespace:EpicTweet.Tools"
xmlns:MV="clr-namespace:MVVM;assembly=MVVM"
xmlns:Loc="clr-namespace:EpicTweet.Localization"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d">
<DataTemplate DataType="VM:EnterPINDialogViewModel" x:Key="EnterPINDialogTemplate">
    <Grid d:DesignWidth="453.89" d:DesignHeight="78.92" Loc:ResXManagerProperty.ResourceManager="{x:Static ET:Language.ResourceManager}">
        <Grid.RowDefinitions>
            <RowDefinition Height="Auto"/>
            <RowDefinition/>
        </Grid.RowDefinitions>
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="Auto"/>
            <ColumnDefinition/>
        </Grid.ColumnDefinitions>
        <Label Content="{Loc:ResxExtension ResourceName=String_PIN, FallbackValue='&lt;PIN&gt;'}"/>
        <TextBox Grid.Column="1"/>
        <TextBlock Grid.Row="1" Grid.RowSpan="2"></TextBlock>
    </Grid>
</DataTemplate>
<DataTemplate x:Key="DialogViewTemplate" DataType="MV:DialogViewModel">
    <Border BorderBrush="Black" BorderThickness="1">
        <Grid d:DesignWidth="277.419" d:DesignHeight="74.96" Background="{DynamicResource {x:Static SystemColors.ControlDarkBrushKey}}" Height="Auto" Width="Auto">
            <Grid.RowDefinitions>
                <RowDefinition Height="Auto"/>
                <RowDefinition/>
                <RowDefinition Height="Auto"/>
            </Grid.RowDefinitions>
            <Border d:LayoutOverrides="Width, Height" BorderThickness="0,0,0,1" BorderBrush="Black">
                <Label Content="{Binding DisplayName, FallbackValue=Header}" VerticalAlignment="Center" HorizontalAlignment="Left"/>    
            </Border>
            <ContentControl Content="{Binding Content, FallbackValue=Body}" ContentTemplate="{StaticResource EnterPINDialogTemplate}" HorizontalAlignment="Stretch" d:LayoutOverrides="Height" Grid.Row="1" Margin="5">
                <!--<ContentControl.ContentTemplateSelector>
                    <T:TypeTemplateSelector>
                        <T:TemplateTypeRelationship Type="{x:Type VM:EnterPINDialogViewModel}" ResourceKey="EnterPINDialogTemplate"/>
                    </T:TypeTemplateSelector>
                </ContentControl.ContentTemplateSelector>-->
            </ContentControl>
                <ItemsControl Grid.Row="2" Margin="10" 
                ItemsSource="{Binding Commands, Mode=OneTime, FallbackValue={x:Static VM:TwitterClientViewModel.DEFAULT_DIALOG_COMMANDS}}">
                <ItemsControl.ItemTemplate>
                    <DataTemplate>
                        <Button 
                        Content="{Binding DisplayName, FallbackValue=CommandName, Mode=OneWay}"
                        Command="{Binding}"/>
                    </DataTemplate>
                </ItemsControl.ItemTemplate>
                <ItemsControl.ItemsPanel>
                    <ItemsPanelTemplate>
                        <WrapPanel Orientation="Horizontal"/>
                    </ItemsPanelTemplate>
                </ItemsControl.ItemsPanel>
            </ItemsControl>
        </Grid>
    </Border>
</DataTemplate>

TwitterClientDataTemplate.xaml

<ResourceDictionary
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation" 
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:VM="clr-namespace:EpicTweet.ViewModel" 
xmlns:ET="clr-namespace:EpicTweet"
xmlns:MV="clr-namespace:MVVM;assembly=MVVM"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008" xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d">
<ResourceDictionary.MergedDictionaries>
    <ResourceDictionary Source="DialogTemplates.xaml"/>
</ResourceDictionary.MergedDictionaries>
<BooleanToVisibilityConverter x:Key="BooleanToVisibilityConverter"/>
<DataTemplate x:Key="TwitterClientTemplate" DataType="MV:TwitterClientViewModel">
    <ScrollViewer d:DesignWidth="285.083" d:DesignHeight="119.96">
        <Grid>
            <StackPanel d:LayoutOverrides="Width, Height">
                <StackPanel Orientation="Horizontal">
                    <Button Command="{Binding AddAccountCommand.Command}" Content="{Binding AddAccountCommand.DisplayName, FallbackValue=&lt;Add Account&gt;}"/>
                </StackPanel>
                <ContentControl/>
            </StackPanel>
            <Border BorderThickness="1" Background="#80000000" Visibility="{Binding HasDialogs, Converter={StaticResource BooleanToVisibilityConverter}, FallbackValue=Collapsed, Mode=OneWay}">
                <Grid VerticalAlignment="Stretch" MinWidth="50" MaxWidth="200">
                    <ContentControl Content="{Binding Dialogs[0], Mode=OneWay}" ContentTemplate="{DynamicResource DialogViewTemplate}" HorizontalAlignment="Center" VerticalAlignment="Center"/>
                </Grid>
            </Border>
        </Grid>
    </ScrollViewer>
</DataTemplate>

帮助我的 stackoverflow,你是我唯一的希望!

编辑:在这个问题上做了一些进一步的工作。如果两个模板都在同一个文件中,动态资源和静态资源扩展都可以毫无问题地解析资源。如果它们在单独的文件中,无论我如何合并词典,资源都不会解析;每个扩展返回 null。

显然,解决方案是将两种资源都放入同一个字典中,但据我所知,这是一种 hack,不是逻辑资源查找系统的预期行为。我现在不是一只快乐的兔子。这似乎没有记录......

最佳答案

如果曾经有一个普拉特,那就是我。在周五晚上尝试解决这个问题 4 小时后,我破解了它,不感谢我只能称之为不稳定的错误报告。

这是嗡嗡声。

<DataTemplate x:Key="TwitterClientTemplate" DataType="MV:TwitterClientViewModel">

应该是

<DataTemplate x:Key="TwitterClientTemplate" DataType="{x:Type MV:TwitterClientViewModel}">

砰,它起作用了。

然而,我的大提示仍然存在。为什么不正确的语法在设计器中起作用但在运行时不起作用?我的猜测是因为运行时优化不会费心用编写不当的 xaml 填充字典,但最好能收到错误警告。

关于WPF staticresource 对 DataTemplates 中逻辑资源的引用未在运行时解析,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4413508/

相关文章:

c# - 将 TreeView 绑定(bind)到包含多个对象的单个对象

c# - 有没有办法只用 C# 构建 DataTemplate

c# - WPF:访问同一控件中的两个 DataContext

c# - XAML 标签文本 : Binding + DynamicResource (String Format? )

c# - 如何在 WPF 中的代码中复制资源引用?

c# - Sql、Wpf、Xaml、C#、绑定(bind)数据、动态资源、访问非静态数据、获取对象引用

c# - WPF:选择一个 jpg,调整它的大小然后保存它?最佳选择?

c# - 带圆角的数据网格模板

c# - WPF DVC :Lineseries Style and axes settings using c#

wpf - 水平显示 ItemsControl 内的项目集合