在快捷方式上设置菜单项的状态

时间:2018-09-20 11:37:51

标签: c gtk gtk3

程序具有菜单栏,该菜单栏具有菜单,该菜单具有菜单项。根据某些情况,启用或禁用菜单项。当用户单击包含菜单项的菜单时,将进行情况检查以及菜单项的启用/禁用。

但是,该菜单项也有一个快捷方式。快捷方式仅在启用菜单项时起作用。

当我们要使用快捷方式时,如何在不单击包含菜单项的菜单的情况下设置其状态?

这是一个示例程序,希望可以使我的问题更加清楚:

#include <gtk/gtk.h>

struct check_sensitivity
{
        GtkWidget *menuitem;
        GtkTextBuffer *buffer;
};

void sensitivity(GtkWidget *menu, struct check_sensitivity *sens)
{
        if (gtk_text_buffer_get_modified(sens->buffer))
                gtk_widget_set_sensitive(sens->menuitem, TRUE);
        else
                gtk_widget_set_sensitive(sens->menuitem, FALSE);
}

void menuitem_click(GtkWidget *menuitem, GtkTextBuffer *buffer)
{
        gtk_text_buffer_set_text(buffer, "", -1);
        gtk_text_buffer_set_modified(buffer, FALSE);
}

int main(int argc, char **argv)
{
        gtk_init(&argc, &argv);

        struct check_sensitivity sens;

        GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
        g_signal_connect(window, "delete-event", G_CALLBACK(gtk_main_quit), NULL);

        GtkWidget *view = gtk_text_view_new();
        sens.buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(view));

        GtkWidget *menubar = gtk_menu_bar_new();
        GtkWidget *menu = gtk_menu_new();

        GtkAccelGroup *shortcuts = gtk_accel_group_new();
        gtk_window_add_accel_group(GTK_WINDOW(window), shortcuts);

        GtkWidget *menuitem = gtk_menu_item_new_with_label("Menu");
        gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), menu);
        gtk_menu_shell_append(GTK_MENU_SHELL(menubar), menuitem);
        g_signal_connect(menuitem, "activate", G_CALLBACK(sensitivity), &sens);
        menuitem = gtk_menu_item_new_with_label("Clear");
        gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
        gtk_widget_add_accelerator(menuitem, "activate", shortcuts, GDK_KEY_D, GDK_CONTROL_MASK, GTK_ACCEL_VISIBLE);
        sens.menuitem = menuitem;
        g_signal_connect(sens.menuitem, "activate", G_CALLBACK(menuitem_click), sens.buffer);

        GtkWidget *box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
        gtk_box_pack_start(GTK_BOX(box), menubar, FALSE, FALSE, 0);
        gtk_box_pack_start(GTK_BOX(box), view, TRUE, TRUE, 0);

        gtk_container_add(GTK_CONTAINER(window), box);

        gtk_widget_show_all(window);

        gtk_main();
}

在此程序中,菜单项“清除”的状态取决于缓冲区是否被修改。当然,状态是在用户单击菜单“菜单”时设置的。

还有一个快捷键 Ctrl + D ,其作用与单击“清除”相同。快捷方式的工作方式应取决于菜单项的状态,但是出问题的是菜单项的状态。让我解释一下:

由于当用户单击“菜单”时菜单项被设置为(取消),因此在使用快捷方式之前要更新“清除”的状态,因此用户必须单击该菜单项。如果在更新菜单项的状态之前使用了快捷方式,则它可能无法在应有的状态下发挥作用。

这里是您可以尝试的方法:打开程序并编写一些内容。尝试使用 Ctrl + D 清除它。您会看到什么也不会发生,因为菜单项的状态没有更新。如果单击“菜单”,然后再次尝试使用快捷方式,则文本缓冲区将清除。

我只是想问为什么在按下快捷键时也要更新状态。我尝试了几种基于按下 Ctrl 时设置状态的方式(我无法仅检查 Ctrl + D ,程序中可能还有其他快捷方式) ,但是它们全部失败或产生了错误。您可以通过其中一种方式查看here

2 个答案:

答案 0 :(得分:0)

如果要根据文本缓冲区中是否有文本来更新菜单项灵敏度,请连接到更改或更改了文本缓冲区的信号。由于我不擅长C语言,但通常使用Python:

textbuffer.connect("changed", update_menuitem_sensitivity) 

changed

modified-changed

答案 1 :(得分:0)

最后找到了一种可行的方法。

#include <gtk/gtk.h>

struct check_sensitivity
{
        GtkWidget *menuitem;
        GtkTextBuffer *buffer;
};

void sensitivity(GtkWidget *menu, struct check_sensitivity *sens);

void check_sensitivity_on_ctrl(GtkWidget *window, GdkEventKey *key, struct check_sensitivity *sens)
{
    if(key->keyval == GDK_KEY_Control_L || key->keyval == GDK_KEY_Control_R)
    {
        sensitivity(NULL, sens);
    }
}

void sensitivity(GtkWidget *menu, struct check_sensitivity *sens)
{
        if (gtk_text_buffer_get_modified(sens->buffer))
                gtk_widget_set_sensitive(sens->menuitem, TRUE);
        else
                gtk_widget_set_sensitive(sens->menuitem, FALSE);
}

void menuitem_click(GtkWidget *menuitem, GtkTextBuffer *buffer)
{
        gtk_text_buffer_set_text(buffer, "", -1);
        gtk_text_buffer_set_modified(buffer, FALSE);
}

int main(int argc, char **argv)
{
        gtk_init(&argc, &argv);

        struct check_sensitivity sens;

        GtkWidget *window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
    gtk_window_set_default_size(GTK_WINDOW(window), 400, 400);
        g_signal_connect(window, "delete-event", G_CALLBACK(gtk_main_quit), NULL);

        GtkWidget *view = gtk_text_view_new();
        sens.buffer = gtk_text_view_get_buffer(GTK_TEXT_VIEW(view));
    gtk_text_buffer_set_modified(sens.buffer, FALSE);

        GtkWidget *menubar = gtk_menu_bar_new();
        GtkWidget *menu = gtk_menu_new();

        GtkAccelGroup *shortcuts = gtk_accel_group_new();
        gtk_window_add_accel_group(GTK_WINDOW(window), shortcuts);

        GtkWidget *menuitem = gtk_menu_item_new_with_label("Menu");
        gtk_menu_item_set_submenu(GTK_MENU_ITEM(menuitem), menu);
        gtk_menu_shell_append(GTK_MENU_SHELL(menubar), menuitem);
        g_signal_connect(menuitem, "activate", G_CALLBACK(sensitivity), &sens);
        menuitem = gtk_menu_item_new_with_label("Clear");
        gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
        gtk_widget_add_accelerator(menuitem, "activate", shortcuts, GDK_KEY_D, GDK_CONTROL_MASK, GTK_ACCEL_VISIBLE);
        sens.menuitem = menuitem;
        g_signal_connect(sens.menuitem, "activate", G_CALLBACK(menuitem_click), sens.buffer);
    g_signal_connect_after(window, "key-press-event", G_CALLBACK(check_sensitivity_on_ctrl), &sens);

        GtkWidget *box = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0);
        gtk_box_pack_start(GTK_BOX(box), menubar, FALSE, FALSE, 0);
        gtk_box_pack_start(GTK_BOX(box), view, TRUE, TRUE, 0);

        gtk_container_add(GTK_CONTAINER(window), box);

        gtk_widget_show_all(window);

        gtk_main();
}