代码之家  ›  专栏  ›  技术社区  ›  akjoshi HCP

对画布使用wpf命令不起作用

  •  0
  • akjoshi HCP  · 技术社区  · 14 年前

    我有一个定制的画布 Canvas . 它包含很少 ApplicationCommands 喜欢 New / Open / Save 像这样添加的-

    this.CommandBindings.Add(new CommandBinding(ApplicationCommands.New, New_Executed, 
    New_Enabled));
    

    New_Enabled 总是返回true。

    此控件用于具有菜单栏的WPF项目中;新的/打开/保存菜单按钮与 Command 设置为各自 ApplicationCommand 像这样——

    <syncfusion:SimpleMenuButton
        x:Name="NewMenu"
        Icon="Images\New_Large.png"
        Label="New"
        IsEnabled="True"
        Command="{x:Static ApplicationCommands.New}"
        syncfusion:Ribbon.KeyTip="N">
    </syncfusion:SimpleMenuButton>
    

    当焦点位于 帆布 但一旦焦点转移到其他控件,新按钮就会被禁用。我试过设置 CommandTarget 到主窗口,但也不行。

    为什么会发生这种情况?如何确保始终启用新的菜单按钮?

    2 回复  |  直到 10 年前
        1
  •  1
  •   repka    14 年前

    问题是,一旦您的按钮和画布在层次结构的上方某个位置共享逻辑焦点范围(很可能是您的窗口),在某些菜单中启动的命令就永远不会到达画布。

    如果只有一个画布要接收所有命令,只需绑定 CommandTarget 要绘制的按钮:

    ...
    Command="New"
    CommandTarget="{Binding ElementName=TheCanvas}"
    ...
    

    注意 ICommand 标记为 TypeConverterAttribute 它将像“new”这样的字符串转换为 ApplicationCommands.New 所以你不必使用 x:Static 标记扩展。

    你可以在一个地方用 Style 对于菜单/工具栏级别上的所有按钮。

    但是,如果您有多幅画布,并且希望将您的命令指向当前的焦点画布,则必须执行两项操作:

    1. 确保画布(或其上的控件)具有 Focusable="True"
    2. 通过设置,限制工具栏(或用于按钮的任何容器)的逻辑焦点范围 FocusManager.IsFocusScope="True" 关于它。一些容器,如菜单,或 ToolBar 默认情况下打开它。这样,一旦命令路由算法到达作用域,它就会将其重定向到当前具有键盘焦点的元素。
        2
  •  0
  •   akjoshi HCP    10 年前

    @雷普卡-谢谢你的回复;我已经试过用画布名称作为 CommandTarget 但它不起作用;只有当焦点在画布上时,按钮才会被启用,只要我单击窗口中的其他控件,它们就会被禁用。我也试过用 IsFocusScope 但同样的结果。感谢您提供命令名字符串提示。

    虽然我对这个工作不太满意,但我还是不得不解决这个问题。-

        public WindowMain()
        {
            InitializeComponent();
    
            //Add commnad bindings
            //Need to do this to keep New/Open/Save/Run buttons always enabled
            //ToDo:[AJ] Look for better solution then this
            this.CommandBindings.Add(new CommandBinding(ApplicationCommands.New, this.TheCanvas.New_Executed, this.TheCanvas.New_Enabled));
            this.CommandBindings.Add(new CommandBinding(ApplicationCommands.Open, this.TheCanvas.Open_Executed, this.TheCanvas.Open_Enabled));
            this.CommandBindings.Add(new CommandBinding(ApplicationCommands.Save, this.TheCanvas.Save_Executed, this.TheCanvas.Save_Enabled));
            this.CommandBindings.Add(new CommandBinding(RTDesignerCanvas.Run, this.TheCanvas.Run_Executed));
        }