如何正确绑定ViewModel(包括分隔符)到WPF的菜单?

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工作符合预期.

快乐的编码!