无法使用ItemContainerStyle为菜单项设置图标

akj*_*shi 5 .net wpf xaml menuitem itemcontainerstyle

我试图设置我的菜单项的图标 -

 <Grid>
    <Grid.Resources>
        <Image
              x:Key="ReportIconImage" Height="20" Width="20"
              Source="/Resource/flag.png"/>
        <Image
              x:Key="ReportIconImage1" Height="20" Width="20"
              Source="/Resource/flag.png"/>
    </Grid.Resources>
    <Menu Height="22" Margin="0,9,0,0" Name="menu1" VerticalAlignment="Top">
        <MenuItem Header="Menu">
            <MenuItem Header="Save" ></MenuItem>
            <MenuItem Header="Open"/>
            <MenuItem Header="Exit"/>
            <MenuItem.ItemContainerStyle>
                <Style TargetType="{x:Type MenuItem}">
                    <Setter  
                        Property="Icon" 
                        Value="{StaticResource ReportIconImage}">
                    </Setter>
                </Style>
            </MenuItem.ItemContainerStyle>
        </MenuItem>
        <MenuItem Header="Edit">
            <MenuItem Header="Undo"/>                   
            <MenuItem Header="Redo"/>                    
            <Separator/>
            <MenuItem Header="Cut"/>                    
            <MenuItem Header="Copy"/>                    
            <MenuItem  Header="Paste"/>
            <MenuItem.ItemContainerStyle>
                <Style TargetType="{x:Type MenuItem}">
                    <Setter  
                         Property="Icon" 
                         Value="{StaticResource ReportIconImage1}">
                </Setter>
                </Style>
            </MenuItem.ItemContainerStyle>
        </MenuItem>
    </Menu>
</Grid>
Run Code Online (Sandbox Code Playgroud)

但仅显示最后一个菜单项的图标,而不是前两个.

在此输入图像描述

示例应用程序 - http://weblogs.asp.net/blogs/akjoshi/Samples/WPFMenuItemBugSample.zip

任何人都可以提供此行为的原因和可能的解决方案/解决方法.

Ken*_*art 8

这是因为你Image在资源中使用了一个.一个Image是控制-就像任何其他的控制-只能有一个父.默认情况下,WPF将尝试在所有使用者之间共享资源.因此,最后MenuItem"赢得" Image对其他人MenuItem的监管权利甚至不允许周末访问.

要纠正此问题,您可以将其设置Image为非共享:

<Image x:Shared="False" .../>
Run Code Online (Sandbox Code Playgroud)

或者,更好的是,将您的图像资源显示为适当的ImageSource子类并分享:

<BitmapImage x:Key="ReportIconImage" Uri="/Resource/flag.png"/>
...
<Setter Property="Icon">
    <Setter.Value>
        <Image Source="{StaticResource ReportIconImage}"/>
    </Setter.Value>
</Setter>
Run Code Online (Sandbox Code Playgroud)

  • @akjoshi:感谢repro.将您的项目切换到WPF 4允许我的第一个建议工作,所以3.5中必定存在错误.至于我的第二个建议,这是行不通的,因为WPF试图在所有MenuItem中使用相同的Image,就像我在回答中描述的那样.如果有一个`IconTemplate`属性,你就可以使用它.不幸的是,`MenuItem`的细粒度不足以有一个,所以我只建议覆盖`Template`.至少可以说是沮丧. (2认同)

Aks*_*elK 5

有点晚了,但这是一个适合我的解决方案.

我使用转换器为每个menuitem创建一个新图像:

class PathToImageConverter:IValueConverter
{
    object Convert(object value, Type targetType, object parameter, CultureInfo culture)
    {
        string path = "Data/Icons/" + value + ".png";
        Image img = new Image {Source = new BitmapImage(new Uri(path, UriKind.Relative))};
        return img;
    }

    object ConvertBack(object value, Type targetType, object parameter, CultureInfo culture)
    {
        return "";
    }
}
Run Code Online (Sandbox Code Playgroud)

XAML:

   <MenuItem.ItemContainerStyle>
        <Style TargetType="MenuItem">
            <Setter Property="Icon" Value="{Binding Converter={StaticResource PathToImageConverter1}}"/>
        </Style>
    </MenuItem.ItemContainerStyle>
Run Code Online (Sandbox Code Playgroud)