我正在写一个小程序:它是一个单独的下拉菜单,小写字母作为menuitem标签:
如果您持有Shift
,标签会变为大写(我为此写了一个“按键事件”和“键释放事件”处理程序)。问题是,在按下Shift
时,我仍然想要导航菜单并选择Enter
按下的项目。如果按下某个修改器,则不会触发默认处理程序,因此我按以下方式处理它:
static gboolean menu_key_event(GtkWidget *menu, GdkEvent *event, gpointer data) {
(void)data;
GdkEventKey *key_event = (GdkEventKey*)event;
switch (key_event->keyval) {
case GDK_KEY_Shift_L:
case GDK_KEY_Shift_R: ;
bool b = (key_event->type == GDK_KEY_PRESS) ? true : false;
gtk_container_foreach(GTK_CONTAINER(menu), menuitem_capitalize_label, &b);
return TRUE;
break;
case GDK_KEY_Return:
if ((key_event->type == GDK_KEY_PRESS) &&
(key_event->state & GDK_SHIFT_MASK)) {
// I want default callback to handle this
g_signal_emit_by_name(menu, "activate-current");
return TRUE;
}
break;
case GDK_KEY_Up:
case GDK_KEY_Down:
if ((key_event->type == GDK_KEY_PRESS) &&
(key_event->state & GDK_SHIFT_MASK)) {
// Some function I wrote to fiddle with menu items,
// simulating default selection behavior
menu_rotate_selection(GTK_MENU_SHELL(menu), key_event->keyval);
return TRUE;
}
break;
}
return FALSE;
}
这可以以更优雅的方式完成吗?简而言之,我希望我的应用程序句柄Enter
,箭头键和Shift+Enter
,Shift
+箭头键的方式相同,而无需手动处理。
答案 0 :(得分:0)
我终于找到了选择menuitems所需的信号(“move-current”),因此不再需要我自己的menu_rotate_selection
功能。那个信号名称令人困惑,我宁愿认为它的目的是在菜单中实际移动菜单项(首先我认为另一个模糊命名的信号“cycle-focus”用于改变选择)。现在可以重写如下:
...
case GDK_KEY_Up:
case GDK_KEY_Down:
if ((key_event->type == GDK_KEY_PRESS) &&
(key_event->state & GDK_SHIFT_MASK)) {
GtkMenuDirectionType dir = (key_event->keyval == GDK_KEY_Up) ?
GTK_MENU_DIR_PREV : GTK_MENU_DIR_NEXT;
g_signal_emit_by_name(menu, "move-current", dir);
return TRUE;
}
break;
...
这几乎回答了我的问题。