手工菜单示例

这些应该差不多了。现在看一个示例来帮你弄明白些。


#include <stdio.h>
#include <gtk/gtk.h>

static gint button_press (GtkWidget *, GdkEvent *);
static void menuitem_response (gchar *);

int main( int   argc,
          char *argv[] )
{

    GtkWidget *window;
    GtkWidget *menu;
    GtkWidget *menu_bar;
    GtkWidget *root_menu;
    GtkWidget *menu_items;
    GtkWidget *vbox;
    GtkWidget *button;
    char buf[128];
    int i;

    gtk_init (&argc, &argv);

    /* 创建一个新窗口 */
    window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
    gtk_widget_set_size_request (GTK_WIDGET (window), 200, 100);
    gtk_window_set_title (GTK_WINDOW (window), "GTK Menu Test");
    g_signal_connect (G_OBJECT (window), "delete_event",
                      G_CALLBACK (gtk_main_quit), NULL);

    /* 初始化菜单构件,记住,永远也不要用
     * gtk_show_widget() 来显示菜单构件。 
     * 这个是包含菜单项的菜单,当你在程序的"Root Menu"上点击时
     * 它会弹出来 */
    menu = gtk_menu_new ();

    /* 接着我们用一个小循环为"test-menu"产生三个菜单条目。
     * 注意对gtk_menu_append的调用。这里我们将一序列的菜单项
     * 加到我们的菜单上。通常,我们也捕获每个菜单项的"clicked"
     * 信号并为它设置一个回调,不过在这里这个被省略了以节省空间。 */

    for (i = 0; i < 3; i++)
        {
            /* 将名称复制到buf. */
            sprintf (buf, "Test-undermenu - %d", i);

            /* 创建一个新的菜单项,名称为... */
            menu_items = gtk_menu_item_new_with_label (buf);

            /* ...并将它加到菜单。 */
            gtk_menu_shell_append (GTK_MENU_SHELL (menu), menu_items);

	    /* 当菜单项被选中时做点有趣的事 */
	    g_signal_connect_swapped (G_OBJECT (menu_items), "activate",
		                      G_CALLBACK (menuitem_response), 
                                      g_strdup (buf));

            /* 显示构件 */
            gtk_widget_show (menu_items);
        }

    /* 这个是根菜单,将成为显示在菜单栏上的标签。
     * 这里不会附上一个信号处理函数,因为它只是在
     * 被按下时弹出其余的菜单。 */
    root_menu = gtk_menu_item_new_with_label ("Root Menu");

    gtk_widget_show (root_menu);

    /* 现在我们指定我们想要让新创建的"menu"成
     * 为"root menu"的菜单 */
    gtk_menu_item_set_submenu (GTK_MENU_ITEM (root_menu), menu);

    /* 将一个菜单和一个按钮放到这个纵向盒子里: */
    vbox = gtk_vbox_new (FALSE, 0);
    gtk_container_add (GTK_CONTAINER (window), vbox);
    gtk_widget_show (vbox);

    /* 创建一个菜单栏以包含菜单,并将它加到主窗口 */
    menu_bar = gtk_menu_bar_new ();
    gtk_box_pack_start (GTK_BOX (vbox), menu_bar, FALSE, FALSE, 2);
    gtk_widget_show (menu_bar);

    /* 创建一个按钮,它带了一个弹出菜单 */
    button = gtk_button_new_with_label ("press me");
    g_signal_connect_swapped (G_OBJECT (button), "event",
	                      G_CALLBACK (button_press), 
                              menu);
    gtk_box_pack_end (GTK_BOX (vbox), button, TRUE, TRUE, 2);
    gtk_widget_show (button);

    /* 最后把菜单项添加到菜单栏上 -- 这就是我
     * 咆哮了多次的“根”菜单项 =) */
    gtk_menu_shell_append (GTK_MENU_SHELL (menu_bar), root_menu);

    /* 总是在最后一步显示窗口,这样它一次性整个地出现在屏幕上。 */
    gtk_widget_show (window);

    gtk_main ();

    return 0;
}

/* 对鼠标按下作出回应,弹出widget变量传递进来的菜单。
 *
 * 注意"widget"参数是被传递进来的菜单,不是
 * 被按下的按钮。
 */

static gint button_press( GtkWidget *widget,
                          GdkEvent *event )
{

    if (event->type == GDK_BUTTON_PRESS) {
        GdkEventButton *bevent = (GdkEventButton *) event; 
        gtk_menu_popup (GTK_MENU (widget), NULL, NULL, NULL, NULL,
                        bevent->button, bevent->time);
        /* 告诉调用代码我们已经处理了这个事件;事件传播(buck)在
         * 这里停止。 */
        return TRUE;
    }

    /* 告诉调用代码我们没有处理这个事件;继续传播它。 */
    return FALSE;
}


/* 当菜单项被选中时打印一个字符串 */

static void menuitem_response( gchar *string )
{
    printf ("%s\n", string);
}

你也可以设置一个菜单项为不敏感(insensitive),也可以使用一个加速表(accelerator table),绑定按键到菜单函数。