Kab*_*buo 9 c# wpf mvvm menuitem
我正在使用MVVM,我希望将我的列表数据绑定MenuViewModels到我的maim菜单.其中包含一组菜单项和分隔符.
这是我的MenuItemViewModel代码:
public interface IMenuItemViewModel
{
}
[DebuggerDisplay("---")]
public class SeparatorViewModel : IMenuItemViewModel
{
}
[DebuggerDisplay("{Header}, Children={Children.Count}")]
public class MenuItemViewModel : IMenuItemViewModel, INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
public MenuItemViewModel(string header, ICommand command, ImageSource imageSource)
{
Header = header;
Command = command;
ImageSource = imageSource;
Children = new List<IMenuItemViewModel>();
}
public string Header { get; private set; }
public ICommand Command { get; private set; }
public ImageSource ImageSource { get; private set; }
public IList<IMenuItemViewModel> Children { get; private set; }
}
Run Code Online (Sandbox Code Playgroud)
我的主窗口看起来像这样:
<Window.Resources>
<HierarchicalDataTemplate DataType="{x:Type ViewModel:MenuItemViewModel}"
ItemsSource="{Binding Children}">
<MenuItem Header="{Binding Header}"
Command="{Binding Command}"/>
</HierarchicalDataTemplate>
<DataTemplate DataType="{x:Type ViewModel:SeparatorViewModel}">
<Separator />
</DataTemplate>
</Window.Resources>
<DockPanel>
<Menu DockPanel.Dock="Top"
ItemsSource="{Binding MenuItems}">
</Menu>
</DockPanel>
Run Code Online (Sandbox Code Playgroud)
应该是非常简单的东西.不幸的是,菜单项看起来不对或者分隔符是空的menuItem(取决于我尝试过的).
那么,我Menu如何找到我的两个DataTemplates?
Kab*_*buo 17
解决了我自己的问题
在花了几个小时搜索网络后,我发现很多例子都违背了WPF的自然意图,但没有一个能够解决这个问题.
以下是如何工作用的Menu控制,而不是针对它...
一点背景
WPF的Menu控件通常MenuItem会在使用ItemsSource属性绑定到POCO集合时自动为您创建对象.
但是,可以覆盖此默认行为!这是如何做...
解决方案
首先,您必须创建一个派生自的类ItemContainerTemplateSelector.或者使用我创建的简单类:
public class MenuItemContainerTemplateSelector : ItemContainerTemplateSelector
{
public override DataTemplate SelectTemplate(object item, ItemsControl parentItemsControl)
{
var key = new DataTemplateKey(item.GetType());
return (DataTemplate) parentItemsControl.FindResource(key);
}
}
Run Code Online (Sandbox Code Playgroud)
其次,必须MenuItemContainerTemplateSelector向Windows resources对象添加对类的引用,如下所示:
<Window.Resources>
<Selectors:MenuItemContainerTemplateSelector x:Key="_menuItemContainerTemplateSelector" />
Run Code Online (Sandbox Code Playgroud)
第三,必须在和(在(在)中定义)设置两个属性(UsesItemContainerTemplate,和).ItemContainerTemplateSelectorMenuMenuItemHierarchicalDataTemplate
像这样:
<HierarchicalDataTemplate DataType="{x:Type ViewModel:MenuItemViewModel}"
ItemsSource="{Binding Children}">
<MenuItem Header="{Binding Header}"
Command="{Binding Command}"
UsesItemContainerTemplate ="true"
ItemContainerTemplateSelector=
"{StaticResource _menuItemContainerTemplateSelector}"/>
</HierarchicalDataTemplate>
<Menu DockPanel.Dock="Top"
ItemsSource="{Binding MenuItems}"
UsesItemContainerTemplate="True"
ItemContainerTemplateSelector=
"{StaticResource _menuItemContainerTemplateSelector}">
</Menu>
Run Code Online (Sandbox Code Playgroud)
为什么会这样
出于优化目的,Menu使用UsesItemContainerTemplate标志(其默认值为false)跳过DataTemplate查找并返回普通MenuItem对象.因此,我们需要将此值设置为true然后我们的ItemContainerTemplateSelector工作符合预期.
快乐的编码!