如何在 Gtk# TreeView 中管理特殊用途的按键?

How to manage key pressings for special purposes in Gtk# TreeView?

我在 Gtk#/mono C# 中有一个 KeyPressed 信号用于两个不同的目的,默认情况下不存在 TreeView: a) 按 TAB 键转到下一个单元格,b) 按任意键开始编辑。

TreeView 很简单,它有一个 ListStore 只显示行和列,即它包含表格数据。

我的代码如下。

[GLib.ConnectBefore]
protected void OnTableKeyPressed(object o, Gtk.KeyPressEventArgs args)
{
    int rowIndex;
    int colIndex;

    // Do not "eat" the key, by default
    args.RetVal = false;

    // Get the current position, needed in both cases.
    this.GetCurrentCell( out rowIndex, out colIndex );

    // Adapt the column
    colIndex += NumFixedColumns;

    if ( args.Event.Key != Gdk.Key.ISO_Enter ) {
        if ( args.Event.Key == Gdk.Key.Tab
          || args.Event.Key == Gdk.Key.ISO_Left_Tab )
        {
            if( args.Event.State == Gdk.ModifierType.ShiftMask ) {
                // Back
                colIndex -= 1;
                if ( colIndex < 1 ) {
                    colIndex = document.Columns;
                    --rowIndex;
                }

                rowIndex = Math.Max( 0, rowIndex );
            } else {
                // Advance
                colIndex += 1;
                if ( colIndex > document.Columns ) {
                    colIndex = 1;
                    ++rowIndex;
                }

                rowIndex = Math.Min( rowIndex, document.Rows );
            }

            this.SetCurrentCell( rowIndex, colIndex );
            args.RetVal = true;                              // Eat the TAB
        } else {
            this.SetCurrentCell( rowIndex, colIndex, true );
        }
    }

    return;
}

我有两个问题:

  1. 如何向 TreeView 发出单元格已完成编辑的信号?问题是,如果在没有单元格被编辑时按 TAB 键,一切正常。但是,如果用户正在编辑一个单元格,那么到目前为止输入的内容将会丢失。因此,如果用户正在编辑单元格,我想向 TreeView 发出信号以完成编辑,并继续当前的行为。

  2. 如何在编辑单元格时避免丢失第一个键?假设你在一个单元格上。您按 1、2、3 和 4 键。我的处理程序正确地进行了干预,并将当前单元格置于编辑模式。但是,尽管我将 arg.RetVal 设置为 false

  3. ,但该单元格仅获得 2、3 和 4

关于我的功能的信息

我不是 GTK 方面的专家,事实上我从未使用过它。但我玩过标准控件,以欺骗它们进入非默认行为。我特别修改了菜单栏,当按下 Alt 键时它会捕获所有输入,但我需要 Alt 键作为各种交互的修饰符。因此,我可以为您提供一些关于将 TreeView 欺骗到您需要的东西的一般性建议。

问题一:

根据您的描述,我想默认行为是按 Enter 键成功编辑,离开 Cell 取消编辑。这在许多应用程序中可能是可以接受的。其他人(例如 Microsoft Excel)甚至在离开单元格时也倾向于接受编辑。所以我可以理解你想要那种行为。

如果没有这样的内置行为,您可以模拟用户必须执行的操作以指示 TreeView 完成编辑,例如按回车。您可以使用 here or if GTK builds on WPF like here 中描述的方法发送伪造的 Key 事件。第二种方法甚至更底层,因为它实际上是在 windows 事件队列中植入了伪造的按键事件。我想这在任何情况下都应该有效,只要您的平台是 Windows。但我敢肯定在其他操作系统中也有类似的机制。

然后仅在此之后,转换到下一个单元格,TreeView 获得失去焦点事件,但它不再处于编辑模式,应该不会发生任何事情。

问题二:

我认为会发生以下情况:按下一个键,TreeView 不处于编辑模式,因此忽略该事件。您获得事件,并将其设置为编辑模式。但随后事件将不会return到TreeView,所以不再进行任何输入。

您可以试试上面的方法,手动重新发送按键事件。另一种方法是更早地捕获事件,然后在 TreeView 处理它时。在 WPF 中通常有 PreviewOn* 事件(例如参见 [​​=21=])。所以也许有这样的事件供您控制?

你还可以把自己钩得更深。在 WPF 中,InputManager.Current.PreProcessInput 事件位于 windows 消息循环的正上方,可让您过滤和处理各种输入。

这是我的代码的一部分,可能对您有所帮助:

InputManager.Current.PreProcessInput += (sender, e) =>
{
    if (e.StagingItem.Input is MouseButtonEventArgs)
    {
        var earg = (MouseButtonEventArgs)e.StagingItem.Input;
        if (earg.RoutedEvent == Mouse.PreviewMouseDownOutsideCapturedElementEvent)
            OnPreviewMouseDownOutsideCapturedElement(sender, earg);
    }
};

有关更多低级钩子,请参见例如此 question

祝你好运,如果您有更具体的问题,请发表评论。