在没有ViewModel的情况下将静态ICommand绑定到MenuItem

时间:2013-11-09 07:08:13

标签: c# wpf xaml contextmenu commandbinding

我正在尝试将命令绑定到MenuItems中的各个ContextMenu,我将使用Button进行连接。但为此,我在我static中导入的类中将所有命令定义为ResourceDictionary

public class DesignerCanvas{
    ....
    public static RoutedCommand MyCommand = new RoutedCommand();
    ....
}

在我的MainWindow.xaml中,我正在使用MainWindow.xaml.cs中的实现来挂钩此命令:

<CommandBinding Command="{x:Static Designer:DesignerCanvas.MyCommand}"
                    Executed="DoStuff"
                    CanExecute="CanDoStuff" /> 

在我的ResourceDictionary.xaml中,我正在Button使用ContextMenu挂钩Triggers

    <Button x:Name="btnMyButton" Content="Click this">
      <Button.Style>
        <Style TargetType="{x:Type Button}">
          <Style.Triggers>
            <EventTrigger RoutedEvent="Click">
              <EventTrigger.Actions>
                <BeginStoryboard>
                  <Storyboard>
                    <BooleanAnimationUsingKeyFrames Storyboard.TargetProperty="ContextMenu.IsOpen">
                      <DiscreteBooleanKeyFrame KeyTime="0:0:0" Value="True"/>
                    </BooleanAnimationUsingKeyFrames>
                  </Storyboard>
                </BeginStoryboard>
              </EventTrigger.Actions>
            </EventTrigger>
          </Style.Triggers>
          <Setter Property="ContextMenu">
            <Setter.Value>
              <ContextMenu>
                <MenuItem x:Name="myMenu" Header="MyMenuItem 1">
                  <MenuItem x:Name="menuItem1" Header="MySubMenuItem 1"
Command="{x:Static DesignerItems:DesignerCanvas.MyCommand}">     <<<=== Command Binding
                    <MenuItem.Icon>
                      <Image Source="myImage.png" Width="20"/>
                    </MenuItem.Icon>
                  </MenuItem>
                </MenuItem>
              </ContextMenu>
            </Setter.Value>
          </Setter>
        </Style>
      </Button.Style>
    </Button>

但是这似乎不起作用,因为在XAML中指定Command的菜单项显示为disabledCanDoStuff()和{{ 1}}被调试器击中。此外,由于我没有使用ViewModel,我无法写出类似的内容:

DoStuff()

我怎么能这样做,任何帮助将不胜感激。提前谢谢。

2 个答案:

答案 0 :(得分:3)

您实际上在代码中做得很好但是在WPF中几乎没有关于ContextMenus的隐藏错误/功能。那些坏男孩缺乏WPF通常的功能。 ContextMenu实际上只是WPF中每个控件可用的转储属性,一旦用户右键单击,它就会将其项目列表注入VisualTree。

要解决您的问题,请不要在Style中设置ContextMenu,也不要在Window级别设置CommandBinding,而是在ContextMenu级别设置它。

虽然最重要的是没有在Style中设置ContextMenu!

修改

将ContextMenu远离Style并将CommandBinding设置为:

<ContextMenu>
    <ContextMenu.CommandBindings>
        <CommandBinding Command="foo:MyCommands.CmdFoo" 
                        CanExecute="CanExecuteRerollCommand" 
                        Executed="ExecuteRerollCommand" />
    </ContextMenu.CommandBindings>
    <MenuItem Header="Reroll"  Command="foo:MyCommands.CmdFoo"/>
</ContextMenu>

答案 1 :(得分:3)

终于搞定了,做了一些愚蠢的错误!

必须更改static命令初始化以将ContextMenu作为其所有者

public static RoutedCommand MyCommand = new RoutedCommand("MyCommand", typeof(ContextMenu));

MainWindow.xaml.cs中,添加了一个注册命令绑定的方法,并从MainWindow的构造函数中调用它:

private void InitializeMenuItemsCommands()
    {
        CommandManager.RegisterClassCommandBinding(typeof(ContextMenu), new CommandBinding(DesignerCanvas.TestDialog, OpenTestDialog, CanOpenTestDialog));
    }

最后是CanExecuteExecuted事件的处理程序:

private void CanOpenTestDialog(object sender, CanExecuteRoutedEventArgs e)
{
    e.CanExecute = true; // set if MenuItem is enabled
}

private void OpenTestDialog(object sender, ExecutedRoutedEventArgs e)
{
    // handle the MenuItem click here
}

DID不必更改ResourceDictionary MenuItem中的任何内容,所以这里再次是MenuItem:

<MenuItem x:Name="menuItem1" Header="MySubMenuItem 1"
Command="{x:Static DesignerItems:DesignerCanvas.MyCommand}">     <<<=== Command Binding
                <MenuItem.Icon>
                  <Image Source="myImage.png" Width="20"/>
                </MenuItem.Icon>
              </MenuItem>