使用MVVM结构处理WPF应用程序。
我的窗口显示菜单和当前的ViewModel。在Menu的MenuItem之一上,我想列出当前ViewModel中的一些命令。菜单中列出的命令将根据ViewModel而改变。
我让这个工作得很好,但风格搞砸了 - Command MenuItems在另一个菜单框里面。我会附上一个截图。
我在ViewViewModel中包装了ViewModel的ICommand对象(在本例中为RelayCommands),它在菜单上公开了我想要的Command和Display字符串。这些CommandViewModel位于列表中:CurrentWorkspace.AdditionalOptionsCommands
。
这是菜单的XAML。就像我说的那样,它起作用,它显示正确的项目并执行命令。显示是不正确的 - 任何人都可以告诉我为什么以及如何解决它?见截图。
<Menu>
<MenuItem Header="_Additional Options..." ItemsSource="{Binding Path=CurrentWorkspace.AdditionalOptionsCommands}">
<MenuItem.ItemTemplate>
<DataTemplate DataType="{x:Type vm:CommandViewModel}">
<MenuItem Header="{Binding Path=DisplayText}" Command="{Binding Path=Command}"/>
</DataTemplate>
</MenuItem.ItemTemplate>
</MenuItem>
<MenuItem Header="_Testing">
<MenuItem Header="This looks right" />
<MenuItem Header="This looks right" />
</MenuItem>
</Menu>
目前的外观:
期望的外观:
答案 0 :(得分:8)
这是因为当您通过ItemsSource
指定菜单项时,每个项目都会自动包装到MenuItem
对象中。这样,DataTemplate
(MenuItem
元素)中定义的内容就会被包含在另外一个MenuItem
中。
您需要做的不是定义DataTemplate
,而是为MenuItem
定义一种样式,您可以在其中设置与视图模型属性的绑定,并将此样式用作ItemContainerStyle
父MenuItem
:
<Window.Resources>
<Style x:Key="CommandMenuItemStyle"
TargetType="{x:Type MenuItem}">
<Setter Property="Header"
Value="{Binding Path=DisplayText}"/>
<Setter Property="Command"
Value="{Binding Path=Command}"/>
</Style>
</Window.Resources>
...
<Menu>
<MenuItem Header="_Additional Options..."
ItemsSource="{Binding Path=CurrentWorkspace.AdditionalOptionsCommands}"
ItemContainerStyle="{StaticResource CommandMenuItemStyle}"/>
<MenuItem Header="_Testing">
<MenuItem Header="This looks right" />
<MenuItem Header="This looks right" />
</MenuItem>
</Menu>
有关项容器如何与ItemsControl
控件配合使用的详细说明,请参见http://drwpf.com/blog/2008/03/25/itemscontrol-i-is-for-item-container/。