从资源字典加载DataTemplate

时间:2012-09-10 10:42:33

标签: wpf image data-binding menuitem hierarchicaldatatemplate

在WPF客户端应用程序中,我在使用分层数据模板加载数据绑定菜单项时遇到了一些困难。

首先,我创建了ViewModel,其中包含菜单项的基本属性:Title,Command,ImageSource(用于Icon属性的图像的路径)和子项。

然后,我在XAML窗口中创建了视图以显示我的菜单。为了绑定我的集合,考虑子项,我使用分层数据模板作为菜单项模板。

以下是ItemTemplate的XAML代码:

<HierarchicalDataTemplate DataType="{x:Type vm:MenuItemViewModel}" ItemsSource="{Binding Path=Items}">
    <HierarchicalDataTemplate.ItemContainerStyle>
        <Style TargetType="MenuItem">
            <Style.Resources>
                <Image x:Key="ItemImage" Source="{Binding ImageSource}" Width="16" Height="16" x:Shared="false" />
            </Style.Resources>
            <Style.Setters>
                <Setter Property="Command" Value="{Binding Command}" />
                <Setter Property="CommandParameter" Value="{Binding CommandParameter}" />
                <Setter Property="Icon" Value="{StaticResource ItemImage}" />
            </Style.Setters>
        </Style>
    </HierarchicalDataTemplate.ItemContainerStyle>
    <StackPanel Orientation="Horizontal">
        <TextBlock Text="{Binding Title}" />
    </StackPanel>
</HierarchicalDataTemplate>

当我初始化菜单并显示窗口时,一切看起来都很好。

之后,我尝试将菜单项模板放入资源字典中,以便能够从我的应用程序中的任何位置重用它作为默认模板。 当我这样做时,我抛出异常:

  

命名空间中的共享属性   'http://schemas.microsoft.com/winfx/2006/xaml'只能用于   编译资源词典。

在花了这么多时间寻找解决方案后,我终于制作了一个测试项目(可用here)来证明这个问题。

我不知道如何让我的资源字典成为编译资源字典...... 有谁可以帮助我?

2 个答案:

答案 0 :(得分:0)

尝试从x:Shared="false"元素

中删除<Image>

答案 1 :(得分:0)

找到解决方案,实施转换器:

public class MenuIconConverter : IValueConverter
{
    #region IValueConverter Members

    public object Convert(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        if (value == null) return Binding.DoNothing;

        string imageUri = value.ToString();

        if (string.IsNullOrEmpty(imageUri)) return Binding.DoNothing;

        BitmapImage bitmapImage = new BitmapImage(new Uri(imageUri, UriKind.RelativeOrAbsolute)) { DecodePixelHeight = 16, DecodePixelWidth = 16 };

        return new Image() { Height = 16, Width = 16, Source = bitmapImage, SnapsToDevicePixels = true, UseLayoutRounding = true };
    }

    public object ConvertBack(object value, Type targetType, object parameter, System.Globalization.CultureInfo culture)
    {
        return Binding.DoNothing;
    }

    #endregion
}

要使用转换器,应修改DataTemplate:

<converter:MenuIconConverter x:Key="MenuIconConverter" />

<HierarchicalDataTemplate x:Key="MenuItemTemplate" DataType="{x:Type vm:MenuItemViewModel}" ItemsSource="{Binding Path=Items}">
    <HierarchicalDataTemplate.ItemContainerStyle>
        <Style TargetType="MenuItem">
            <Style.Setters>
                <Setter Property="Command" Value="{Binding Command}" />
                <Setter Property="CommandParameter" Value="{Binding CommandParameter}" />
                <Setter Property="Icon" Value="{Binding ImageSource, Converter={StaticResource MenuIconConverter}, Mode=OneWay}" />
            </Style.Setters>
        </Style>
    </HierarchicalDataTemplate.ItemContainerStyle>
    <TextBlock Text="{Binding Title}" />
</HierarchicalDataTemplate>

有了这个,一切正常。 我期待没有代码的解决方案,但似乎不可能:(