ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
### 菜单控件 创建菜单栏和子菜单时要用到三种构件: • 一个菜单项(menu item),就是用户要选择的东西,比如,"Save" • 一个菜单(menu),作为菜单项的容器。 • 一个菜单栏(menubar),是各个单独菜单的容器。 ### 下面是创建菜单控件的一般步骤: ### • 用 gtk_menu_new() 创建一个新的菜单 • 多次调用 gtk_menu_item_new() 创建每个你想在你的菜单上出现的菜单项。并使用 gtk_menu_append() 将每个新的菜单项放到 菜单上。 • 用 gtk_menu_item_new() 创建一个菜单项。这将是菜单的根(root),上面显示的文本将自己出现在菜单栏上。 • 用 gtk_menu_item_set_submenu() 将菜单绑定到根菜单项(就是上一步创建的那个)。 • 用 gtk_menu_bar_new 创建一个新的菜单栏。在一个菜单栏上创建一系列菜单时这步只要做一次就行了。 • 用 gtk_menu_bar_append() 将根菜单项放到菜单栏上。 创建一个弹出菜单和上面的几乎是一样,只是在设置回调函数时,如下: ~~~ g_signal_connect_swapped (G_OBJECT (widget), "event", G_CALLBACK (handler), G_OBJECT (menu)); ~~~ 其中widget是你要绑定到的构件,handler是处理函数,而menu是一个用 gtk_menu_new() 创建的菜单。它可以是一个也被菜单栏弹出的菜单。 ### 看下面具体的例子: ### ![](https://box.kancloud.cn/2016-08-24_57bd779a35f97.jpg) ~~~ /*File:menu_item.c *Date:2013-11-23 *Author:sjin *Mail:413977243@qq.com */ #include <gtk/gtk.h> #include <stdio.h> static void menuitem_response(gchar *string) { printf("%s\n",string); } static int 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); /*告诉调用代码我们已经处理这个事件,*/ return TRUE; } /*else 未处理*/ return FALSE; } gint delete_event( GtkWidget *widget,GdkEvent *event,gpointer data ) { /* 如果你的 "delete_event" 信号处理函数返回 FALSE,GTK 会发出 "destroy" 信号。 * 返回 TRUE,你不希望关闭窗口。 * 当你想弹出“你确定要退出吗?”对话框时它很有用。*/ g_print ("delete event occurred\n"); /* 改 TRUE 为 FALSE 程序会关闭,关闭时调用destroy()。*/ return FALSE; } /* 另一个回调函数 */ void destroy( GtkWidget *widget,gpointer data ) { gtk_main_quit (); } int main( int argc, char *argv[] ) { /* GtkWidget 是构件的存储类型 */ GtkWidget *window; GtkWidget *menu; GtkWidget *menu_bar; GtkWidget *root_menu; GtkWidget *menu_items; GtkWidget *vbox; GtkWidget *button; char buf[128]; int i; /* 这个函数在所有的 GTK 程序都要调用。参数由命令行中解析出来并且送到该程序中*/ gtk_init (&argc, &argv); /* 创建一个新窗口 */ window = gtk_window_new (GTK_WINDOW_TOPLEVEL); /*设置窗口标题*/ gtk_window_set_title(GTK_WINDOW(window),"My first program helloworld!"); /* 当窗口收到 "delete_event" 信号 (这个信号由窗口管理器发出,通常是“关闭” * 选项或是标题栏上的关闭按钮发出的),我们让它调用在前面定义的 delete_event() 函数。 * 传给回调函数的 data 参数值是 NULL,它会被回调函数忽略。*/ g_signal_connect (G_OBJECT (window), "delete_event",G_CALLBACK (delete_event), NULL); /* 在这里我们连接 "destroy" 事件到一个信号处理函数。 * 对这个窗口调用 gtk_widget_destroy() 函数或在 "delete_event" 回调函数中返回 FALSE 值 * 都会触发这个事件。*/ g_signal_connect (G_OBJECT (window), "destroy",G_CALLBACK (destroy), NULL); /* 设置窗口边框的宽度。*/ gtk_container_set_border_width (GTK_CONTAINER (window), 10); /*创建窗口宽度*/ gtk_widget_set_size_request(GTK_WIDGET(window),200,100); /************* * 初始化菜单构件 * 记住这里永远不要用gtk_widget_show()函数来显示菜单控件 * 这个是包含菜单项的菜单,运行程序时点击会弹出来 ************/ menu = gtk_menu_new(); /*********×××××××××× * * * ***********/ for(i = 0; i < 3; i++){ /*将名称复制到buf*/ sprintf(buf,"undetmenu-%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("根菜单"); gtk_widget_show(root_menu); /************** *指定想要穿件的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("点击"); 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; } ~~~ ### 下面是利用gtk_item_factory来实现菜单选项: ~~~ /***************** *使用套件生成菜单方式 *gtk_item_factory ************/ /*File:menu_item2.c *Date:2014-03-23 *Author:sjin *Mail:413977243@qq.com */ #include <gtk/gtk.h> #include <stdio.h> /*必要的回调函数*/ static void print_hello(GtkWidget *w,gpointer data) { g_message("Hello,World!\n"); } /******************** * 这是用来生成新惨淡的GtkItemFactoryEntry结构 * 第一项 菜单路径 下划线后字母指出惨淡打开的快捷键 * 第二项:这个条目的快捷键 * 第三项:回调函数 * 第四项,回调动作 * 第五项:项类型用 * NULL -> "Item" * "" "Item" * "<Title>" 标题 * "<Item>" 创建一个简单的项 * "<CheckItem>" 创建一个检查项 * "<ToggleItem>" 创建一个开关项 * "<RadioItem>" 创建一个选择项 * <path> 选择项链接到的路径 * "<Separator>" 分割线 * "Branch" 创建一个包含子项的项 * "LastBranch" 创建一个右对齐的分支 *******************/ static GtkItemFactoryEntry menu_items[] = { {"/_File", NULL, NULL,0,"<Branch>"}, {"/File/_New", "<control>N", print_hello,0,NULL}, {"/File/_Open", "<control>O", print_hello,0,NULL}, {"/File/_Save", "<control>S" ,print_hello,0,NULL}, {"/File/Save _As", NULL,NULL,0,NULL}, {"/File/sepl", NULL,NULL,0,"<Separator>"}, {"/File/Quit", "<control>Q", gtk_main_quit,0,NULL}, {"/_Options", NULL,NULL,0,"<Branch>"}, {"/Options/Test", NULL,NULL,0,NULL}, {"/_Help", NULL,NULL,0,"<LastBranch>"}, {"/Help/About", NULL,NULL,0,NULL}, }; void get_main_menu(GtkWidget *window,GtkWidget **menu_bar) { GtkItemFactory *item_factory; GtkAccelGroup *accel_group; gint nmenu_items = sizeof(menu_items)/sizeof(menu_items[0]); accel_group = gtk_accel_group_new(); /************** *这个函数初始化套件 *参数1:菜单类型- *参数2:菜单路径 *参数3:指向一个gtk_accel_group 的指针 * ***************/ item_factory = gtk_item_factory_new(GTK_TYPE_MENU_BAR,"<main>",accel_group); /*生成菜单项,把数组里想的数量,数组自身,和菜单项的任意 * 回调函数以此传递给套件*/ gtk_item_factory_create_items(item_factory,nmenu_items,menu_items,NULL); /*把新的加速组绑定到窗口*/ gtk_window_add_accel_group(GTK_WINDOW(window),accel_group); if(menu_bar){ /*返回套件已经创建的菜单栏*/ *menu_bar = gtk_item_factory_get_widget(item_factory,"<main>"); } } gint delete_event( GtkWidget *widget,GdkEvent *event,gpointer data ) { /* 如果你的 "delete_event" 信号处理函数返回 FALSE,GTK 会发出 "destroy" 信号。 * 返回 TRUE,你不希望关闭窗口。 * 当你想弹出“你确定要退出吗?”对话框时它很有用。*/ g_print ("delete event occurred\n"); /* 改 TRUE 为 FALSE 程序会关闭,关闭时调用destroy()。*/ return TRUE; } /* 另一个回调函数 */ void destroy( GtkWidget *widget,gpointer data ) { gtk_main_quit (); } int main( int argc, char *argv[] ) { /* GtkWidget 是构件的存储类型 */ GtkWidget *window; GtkWidget *menu_bar; GtkWidget *vbox; /* 这个函数在所有的 GTK 程序都要调用。参数由命令行中解析出来并且送到该程序中*/ gtk_init (&argc, &argv); /* 创建一个新窗口 */ window = gtk_window_new (GTK_WINDOW_TOPLEVEL); /*设置窗口标题*/ gtk_window_set_title(GTK_WINDOW(window),"My first program helloworld!"); /* 当窗口收到 "delete_event" 信号 (这个信号由窗口管理器发出,通常是“关闭” * 选项或是标题栏上的关闭按钮发出的),我们让它调用在前面定义的 delete_event() 函数。 * 传给回调函数的 data 参数值是 NULL,它会被回调函数忽略。*/ g_signal_connect (G_OBJECT (window), "delete_event",G_CALLBACK (delete_event), NULL); /* 在这里我们连接 "destroy" 事件到一个信号处理函数。 * 对这个窗口调用 gtk_widget_destroy() 函数或在 "delete_event" 回调函数中返回 FALSE 值 * 都会触发这个事件。*/ g_signal_connect (G_OBJECT (window), "destroy",G_CALLBACK (destroy), NULL); /* 设置窗口边框的宽度。*/ gtk_container_set_border_width (GTK_CONTAINER (window), 10); /*创建窗口宽度*/ gtk_widget_set_size_request(GTK_WIDGET(window),200,100); vbox = gtk_vbox_new(FALSE,1); gtk_container_add(GTK_CONTAINER(window),vbox); gtk_widget_show(vbox); get_main_menu(window,&menu_bar); gtk_box_pack_start(GTK_BOX(vbox),menu_bar,FALSE,TRUE,0); gtk_widget_show(menu_bar); gtk_widget_show(window); gtk_main (); return 0; } ~~~ 问题:在运行后无法将菜单栏显示出来的的问题? 运行程序需要ROOT权限。具体原因不清楚。。