C# 绑定到';它不在暗箱里

C# 绑定到';它不在暗箱里,c#,wpf,data-binding,xaml,command,C#,Wpf,Data Binding,Xaml,Command,我有一个静态类,它包含我想在绑定中使用的RouteDuicCommand public static class CommandLibrary { public static ProjectViewModel Project { get; set; } public static RoutedUICommand AddPage { get; private set; } static CommandLibrary() { AddPage = n

我有一个静态类,它包含我想在绑定中使用的RouteDuicCommand

public static class CommandLibrary
{
    public static ProjectViewModel Project { get; set; }

    public static RoutedUICommand AddPage { get; private set; }

    static CommandLibrary()
    {
        AddPage = new RoutedUICommand("AddPage", "AddPage", typeof(CommandLibrary));

    }

    public static void AddPage_Executed(object sender, ExecutedRoutedEventArgs args)
    {
        Project.AddPage();
    }

    public static void AddPage_CanExecute(object sender, CanExecuteRoutedEventArgs args)
    {
        // We need a project before we can add pages.
        if (Project != null)
        {
            args.CanExecute = true;
        }
        else
        {
            // Did not find project, turning Add Page off.
            args.CanExecute = false;
        }
    }
}
当我试图为这个AddPage命令创建CommandBinding时,VS发出了一声怒吼,抱怨它在Window1中找不到AddPage_CanExecute。。。考虑到我所看到的所有示例都表明,考虑到我现有的代码,这个XAML应该很好,这是没有意义的:

<Window x:Class="MyProject.Window1"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:local="clr-namespace:MyProject">
    <Menu>
        <Menu.CommandBindings>
            <CommandBinding Command="local:CommandLibrary.AddPage" 
                            Executed="AddPage_Executed" CanExecute="AddPage_CanExecute" />
        </Menu.CommandBindings>
        <MenuItem Header="_Page">
            <MenuItem Header="_New" Command="local:CommandLibrary.AddPage" />
        </MenuItem>
    </Menu>
</Window>

我也尝试过不包括Menu.CommandBindings部分,而只是简单地使用它(根据建议,但不是具体的):


这阻止了错误的流动,但是它生成的菜单项总是被禁用的!CanExecute似乎从未接到过电话。我假设在这种情况下绑定也会失败,尽管会更加安静


为什么VS讨厌我的命令,拒绝在正确的位置查找已执行和CanExecute方法?我已经看到了许多示例(在Matthew McDonald的Pro WPF和一些在线自定义命令教程中)正如我所做的。

A
CommandBinding
与可视化树中的任何其他元素一样。在其上指定的任何事件都将由可视化树的根(在本例中为
窗口
)处理。这意味着,如果您将执行的
AddPage\u
AddPage\u CanExecute
移动到窗口的代码后面,它就会工作。这允许您在许多UI组件中使用相同的命令,但具有不同的处理程序

然而,我看到您的命令对视图模型执行了一些逻辑。为了节省时间和挫折感,请理解路由命令是错误的解决方案。相反,在视图模型中封装命令,如下所示:

public class ProjectViewModel
{
    private readonly ICollection<PageViewModel> _pages;
    private readonly ICommand _addPageCommand;

    public ProjectViewModel()
    {
        _pages = new ObservableCollection<PageViewModel>();
        _addPageCommand = new DelegateCommand(AddPage);
    }

    public ICommand AddPageCommand
    {
        get { return _addPageCommand; }
    }

    private void AddPage(object state)
    {
        _pages.Add(new PageViewModel());
    }
}
我建议你通读这一系列的文章,给你更多的背景知识:


回答得很好。我得到了我所需要的。我已经阅读了一些MVVM教程,以前见过这种命令的使用,但不记得怎么做,所以我求助于我在书中找到的解决方案,并通过搜索。谢谢你的帮助。我的头撞到了这个。。。学习wpf+mvvm并不像我希望的那样简单。常规命令对于我想要的太复杂了,我知道应该有一个更简单的方法@肯特,谢谢你的例子和博客。
public class ProjectViewModel
{
    private readonly ICollection<PageViewModel> _pages;
    private readonly ICommand _addPageCommand;

    public ProjectViewModel()
    {
        _pages = new ObservableCollection<PageViewModel>();
        _addPageCommand = new DelegateCommand(AddPage);
    }

    public ICommand AddPageCommand
    {
        get { return _addPageCommand; }
    }

    private void AddPage(object state)
    {
        _pages.Add(new PageViewModel());
    }
}
<MenuItem Header="_New" Command="{Binding AddPageCommand}"/>