ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
### 对话框        对话框是一类GtkWindow,它用于提供顶层窗口。它可以为用户提供消息,可以获得用户信息,或者提供一些简短的动作。       对话款构件被一个水平分割线分隔成两半。上半部放置的是用户界面的主要部分。下半部被称为动作区域,它包括一系列的按钮。当点击,每个按钮会发出一个唯一标识的信号,告诉程序员哪个按钮被点击了。       一般来说,对话框构件可以被看做一个窗口,因为它从GtkWindow类继承而来。然而,当您有多个窗口时,父亲-孩子的关系应该建立在对话框和顶层窗口之间,对话框注定是顶层窗口的补充。 ~~~ typedef struct { GtkWidget *vbox; GtkWidget *action_area; } GtkDialog; ~~~ GtkDialog提供了两个公共成员,包括一个横向按钮的盒子,称为动作区域,以及一个纵向盒子。动作区域包括所有的按钮,在对话框下部排列着。通过GtkHbuttonBox,您可以手工添加按钮,但是通常您应该使用GtkDialog提供的函数来添加动作区域构件。 具体看下面一个简单的例子: ~~~ /*File:dialogs.c *Date:2013-12-16 *Author:sjin *Mail:413977243@qq.com */ #include <gtk/gtk.h> //无模式窗口 //#define SCHEMALESS_WINDOW void button_click(GtkWidget *widget,GtkWindow *parent) { GtkWidget *dialog; GtkWidget *label; GtkWidget *image; GtkWidget *hbox; /*GtkWidget * gtk_dialog_new_with_buttons( * const gchar *title,对话框标题 * GtkWindow *parent,对话框父窗口 * GtkDialogFlags flags, * const gchar *first_button_text,动作区域按钮列表 * ...); * falgs:GTK_DIALOG_MODAL:对话框始终保持在父窗口的上部,直到关闭,防止父窗口交互 * GTK_DIALOG_DESTROY_WITH_PARENT:父窗口销毁,对话框也销毁 * GTK_DIALOG_NO_SEPARATOR:不显示分割线 * * */ #ifndef SCHEMALESS_WINDOW dialog = gtk_dialog_new_with_buttons("我的第一个对话框",parent,GTK_DIALOG_MODAL,GTK_STOCK_OK,GTK_RESPONSE_OK,NULL); #else dialog = gtk_dialog_new_with_buttons("我的第一个对话框",parent,GTK_DIALOG_DESTROY_WITH_PARENT,GTK_STOCK_OK,GTK_RESPONSE_OK,NULL); #endif /*设置分割线是否隐藏:FALSE:隐藏。*/ gtk_dialog_set_has_separator(GTK_DIALOG(dialog),FALSE); label = gtk_label_new("the button was clicked!"); /*GtkWidget * gtk_image_new_from_stock(const gchar *stock_id, * GtklconSize size) * 创建一个预制的Gtkimage构件,当载入一个图片时,还需要指定图片的 *大小,如果图片没有找到,GTK会自动寻找一个自制的图标 * size :参数 GTK_ICON_SIZE_INVALID:未指定大小 * GTK_ICON_SIZE_MENU : 16X16像素 * GTK_ICON_SIZE_SMALL_TOOLBAR: 18X18像素 * GTK_ICON_SIZE_LARGE_TOOLBAR: 24X24像素 * GTK_ICON_SIZE_BUTTON: 24X24像素 * GTK_ICON_SIZE_DND: 32X32像素 * GTK_ICON_SIZE_DIALOG: 48X48像素 *还有另外两个加载图片的函数 *GtkWidget*gtk_image_new_from_file(const gchar *filename) *GtkWidget *gtk_image_new_from_pixbuf(GdkPixbuf *pixbuf) * */ image = gtk_image_new_from_stock(GTK_STOCK_DIALOG_INFO,GTK_ICON_SIZE_DIALOG); hbox = gtk_hbox_new(FALSE,5); gtk_container_set_border_width(GTK_CONTAINER(hbox),10); gtk_box_pack_start_defaults(GTK_BOX(hbox),image); gtk_box_pack_start_defaults(GTK_BOX(hbox),label); gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(dialog)->vbox),hbox); gtk_widget_show_all(dialog); #ifndef SCHEMALESS_WINDOW /*显示对话框*/ gtk_dialog_run(GTK_DIALOG(dialog)); gtk_widget_destroy(dialog); #else g_signal_connect(G_OBJECT(dialog),"response",G_CALLBACK(gtk_widget_destroy),NULL); #endif } int main(int argc, char*argv[]) { GtkWidget *window; GtkWidget *button; //初始化GTK gtk_init(&argc,&argv); //创建最上层主窗口 window = gtk_window_new(GTK_WINDOW_TOPLEVEL); //连接信号"delete_event",使得窗口关闭时发生 g_signal_connect(G_OBJECT(window),"delete_event",G_CALLBACK(gtk_main_quit),NULL); //设置窗口标题 gtk_window_set_title(GTK_WINDOW(window),"对话框练习"); //设定窗口的默认宽高 //gtk_window_set_default_size(GTK_WINDOW(window),200,300); //设定窗口的位置,让窗口总在最前端 gtk_window_set_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER); //设定容器边框的宽度 gtk_container_set_border_width(GTK_CONTAINER(window),20); //创建按钮 button = gtk_button_new_with_mnemonic("clicked me"); gtk_container_add(GTK_CONTAINER(window),button); g_signal_connect(G_OBJECT(button),"clicked",G_CALLBACK(button_click),(gpointer)window); gtk_widget_show(button); gtk_widget_show_all(window); gtk_main(); return 0; } ~~~ 编译时有两种模式,-DSCHEMALESS_WINDOW 是无模式窗口,和模式窗口(不能和父窗口交互,永远在父窗口上面)。 运行: ![](https://box.kancloud.cn/2016-08-24_57bd7797b2361.jpg) 另外一个对话框的应用: ~~~ /*File:dialogs3.c *Date:2013-12-16 *Author:sjin *Mail:413977243@qq.com */ #include <gtk/gtk.h> int main(int argc, char*argv[]) { GtkWidget *dialog; GtkWidget *table; GtkWidget *user,*real,*home,*host; GtkWidget *lbl1,*lbl2,*lbl3,*lbl4; gint result; //初始化GTK gtk_init(&argc,&argv); /*GtkWidget * gtk_dialog_new_with_buttons( * const gchar *title,对话框标题 * GtkWindow *parent,对话框父窗口 * GtkDialogFlags flags, * const gchar *first_button_text,动作区域按钮列表 * ...); * falgs:GTK_DIALOG_MODAL:对话框始终保持在父窗口的上部,直到关闭,防止父窗口交互 * GTK_DIALOG_DESTROY_WITH_PARENT:父窗口销毁,对话框也销毁 * GTK_DIALOG_NO_SEPARATOR:不显示分割线 * * */ dialog = gtk_dialog_new_with_buttons("用户信息编辑窗口",NULL, GTK_DIALOG_MODAL, GTK_STOCK_OK,GTK_RESPONSE_OK, GTK_STOCK_CANCEL,GTK_RESPONSE_CANCEL, NULL); //设置缺省按钮 gtk_dialog_set_default_response(GTK_DIALOG(dialog),GTK_RESPONSE_OK); //创建4个条目,告诉用户输入的数据 lbl1 = gtk_label_new("User Name"); lbl2 = gtk_label_new("Real Name"); lbl3 = gtk_label_new("Home Dir"); lbl4 = gtk_label_new("Host Name"); user = gtk_entry_new(); real = gtk_entry_new(); home = gtk_entry_new(); host = gtk_entry_new(); //设置条目的缺省值 gtk_entry_set_text(GTK_ENTRY(user),g_get_user_name()); gtk_entry_set_text(GTK_ENTRY(real),g_get_real_name()); gtk_entry_set_text(GTK_ENTRY(home),g_get_home_dir()); gtk_entry_set_text(GTK_ENTRY(host),g_get_host_name()); //表组装 table = gtk_table_new(4,2,FALSE); gtk_table_attach_defaults(GTK_TABLE(table),lbl1,0,1,0,1); gtk_table_attach_defaults(GTK_TABLE(table),lbl2,0,1,1,2); gtk_table_attach_defaults(GTK_TABLE(table),lbl3,0,1,2,3); gtk_table_attach_defaults(GTK_TABLE(table),lbl4,0,1,3,4); gtk_table_attach_defaults(GTK_TABLE(table),user,1,2,0,1); gtk_table_attach_defaults(GTK_TABLE(table),real,1,2,1,2); gtk_table_attach_defaults(GTK_TABLE(table),home,1,2,2,3); gtk_table_attach_defaults(GTK_TABLE(table),host,1,2,3,4); gtk_table_set_row_spacings(GTK_TABLE(table),5); gtk_table_set_col_spacings(GTK_TABLE(table),5); gtk_container_set_border_width(GTK_CONTAINER(table),5); gtk_box_pack_start_defaults(GTK_BOX(GTK_DIALOG(dialog)->vbox),table); gtk_widget_show_all(dialog); //运行和输出对话框 result = gtk_dialog_run(GTK_DIALOG(dialog)); switch(result){ case GTK_RESPONSE_NONE: /*-1 * 对话框被窗口管理器销毁或通过gtk_widget_destroy()有其他程序销毁。 * 如果构件没有设置一个相应标识,会返回它。 */ break; case GTK_RESPONSE_REJECT: /* -2 * 这个标识没有和对话框内置的任何按钮相关联,可以任意使用它 * */ break; case GTK_RESPONSE_ACCEPT: /* -3 * 这个标识没有和对话框内置的任何按钮相关联,可以任意使用它 * */ break; case GTK_RESPONSE_DELETE_EVENT: /* -4 * 每个对话框都会自动连接到delete-event信号,当gtk_dialog_run() * 运行是,这个标识会被返回,而且delete-event会被终止、不会像 * 往常那样销毁窗口 * */ break; case GTK_RESPONSE_OK: /* -5 * GTK_STOCK_OK 被点击 */ g_print("User Name:\t%s\n",gtk_entry_get_text(GTK_ENTRY(user))); g_print("Real Name:\t%s\n",gtk_entry_get_text(GTK_ENTRY(real))); g_print("Home Dir:\t%s\n",gtk_entry_get_text(GTK_ENTRY(home))); g_print("Host Name:\t%s\n",gtk_entry_get_text(GTK_ENTRY(host))); break; case GTK_RESPONSE_CANCEL: /*-6 * GTK_STOCK_CANCEL被点击 * */ g_print("cancel is press!\n"); break; case GTK_RESPONSE_CLOSE: /* -7 *GTK_STOCK_CLOSE被点击 */ break; case GTK_RESPONSE_YES: /* -8 *GTK_STOCK_YES被点击 */ break; case GTK_RESPONSE_NO: /* -9 *GTK_STOCK_NO被点击 */ break; case GTK_RESPONSE_APPLY: /* -10 *GTK_STOCK_APPLY被点击 */ break; case GTK_RESPONSE_HELP: /* -11 *GTK_STOCK_HELP被点击 */ break; default: g_print("something wrong!\n"); break; } gtk_widget_destroy(dialog); //gtk_main(); return 0; } ~~~ 运行结果如下: ![](https://box.kancloud.cn/2016-08-24_57bd7797c90bf.jpg) **消息对话框**       消息对话框用于下列四种消息:一般信息、错误信息、警告和问题。对话框的类型决定了显示的图表类型、对话框标题和添加的按钮。       还有一种通用类型的对话框,它对内容没有任何假设。在多数情况下,您不需要使用它,因为上述四种类型基本上满足了大部分需求。       重新创建GtkMessageDialog构件是很容易的。前面的两个例子实现了一个简单的消息对话框,但是GtkMessageDialog已经提供了这个功能,因此您不需要重新创建这个构件。使用GtkMessageDialog节省了代码,而且避免您反复重新创建这种构件,其实大部分应用程序反复使用GtkMessageDialog。它还为所有GTK+应用程序提供了一个统一的外观。  具体看下面的示例: ![](https://box.kancloud.cn/2016-08-24_57bd7797df3b7.jpg) 具体代码如下: ~~~ /*File: message.c *Date:2013-12-21 *Author:sjin *Mail:43977243@qq.com */ #include <gtk/gtk.h> static void on_button_clicked(GtkWidget* button,gpointer data) { GtkWidget* dialog; GtkMessageType type; gchar* message; switch((int)data){ case 1: message = "这是个信息提示对话框."; type = GTK_MESSAGE_INFO; break; case 2: message = "这是个错误提示对话框"; type = GTK_MESSAGE_ERROR; break; case 3: message = "这是个问题提示对话框"; type = GTK_MESSAGE_QUESTION; break; case 4: message = "这是个警告提示对话框"; type = GTK_MESSAGE_WARNING; break; default: break; } /*GtkWidget * gtk_message_dialog_new(GtkWindow *parent, * GtkDialogFlags flags, * GtkMessageType type, * GtkButtonsType buttons, * const gchar * message_format, * ...); * type :枚举的五个可能值: * GTK_MESSAGE_INFO:显示用户的一般性消息 * GTK_MESSAGE_WARNING:一个警告信息,表示发生了非致命的错误 * GTK_MESSAGE_QUESTION:询问用户一个问题,用户必须做出选择。 * GTK_MESSAGE_ERROR:警告信息,标识发生了一个致命的错误 * GTK_MESSAGE_OTHER:一般类型的消息 * buttons:显示什么类型的按钮 * GTK_BUTTONS_NONE:不添加任何按钮。 * GTK_BUTTONS_OK:添加GTK_STOCK_OK按钮 * GTK_BUTTONS_CLOSE:添加GTK_STOCK_CLOSE按钮 * GTK_BUTTONS_CANCEL:添加GTK_STOCK_CANCEL按钮 * GTK_BUTTONS_YES_NO:添加GTK_STOCK_YES和GTK_STOCK_NO * GTK_BUTTONS_OK_CANCEL: ... * 最后一个参数是显示对话框中的内容 * * void gtk_message_dialog_set_format_secondary_text(GtkMessageDialog*dialog, * const gchar *message_format, * ...); * 为消息对话框添加第二段文本,他会导致第一个文本被设置成粗体。这个功能 * 非常有用,它允许您在第一段文字中写一个简短的概要,在第二段文本中详细 * 描述。通过gtk_message_set_format_secondary_markup()来设置文本格式 * */ dialog = gtk_message_dialog_new(NULL,GTK_DIALOG_MODAL|GTK_DIALOG_DESTROY_WITH_PARENT,type,GTK_BUTTONS_OK,message); gtk_dialog_run(GTK_DIALOG(dialog)); gtk_widget_destroy(dialog); } int main(int argc,char* argv[]) { GtkWidget* window; GtkWidget* frame; GtkWidget* box; GtkWidget* button1; GtkWidget* button2; GtkWidget* button3; GtkWidget* button4; gtk_init(&argc,&argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title(GTK_WINDOW(window),"创建消息框"); g_signal_connect(G_OBJECT(window),"destroy",G_CALLBACK(gtk_main_quit),NULL); gtk_container_set_border_width(GTK_CONTAINER(window),20); frame = gtk_frame_new("四种消息对话框"); gtk_container_add(GTK_CONTAINER(window),frame); box = gtk_hbox_new(TRUE,0); gtk_container_add(GTK_CONTAINER(frame),box); gtk_container_set_border_width(GTK_CONTAINER(box),20); button1 = gtk_button_new_from_stock(GTK_STOCK_DIALOG_INFO); gtk_box_pack_start(GTK_BOX(box),button1,FALSE,FALSE,5); g_signal_connect(G_OBJECT(button1),"clicked",G_CALLBACK(on_button_clicked),(gpointer)1); /*GtkWidget * gtk_button_new_from_stock(const gchar *stock_id); * GTK内部保存的条目:图片和文字。stock_id:对应的ID */ button2 = gtk_button_new_from_stock(GTK_STOCK_DIALOG_ERROR); gtk_box_pack_start(GTK_BOX(box),button2,FALSE,FALSE,5); g_signal_connect(G_OBJECT(button2),"clicked",G_CALLBACK(on_button_clicked),(gpointer)2); button3 = gtk_button_new_from_stock(GTK_STOCK_DIALOG_QUESTION); gtk_box_pack_start(GTK_BOX(box),button3,FALSE,FALSE,5); g_signal_connect(G_OBJECT(button3),"clicked",G_CALLBACK(on_button_clicked),(gpointer)3); button4 = gtk_button_new_from_stock(GTK_STOCK_DIALOG_WARNING); gtk_box_pack_start(GTK_BOX(box),button4,FALSE,FALSE,5); g_signal_connect(G_OBJECT(button4),"clicked",G_CALLBACK(on_button_clicked),(gpointer)4); gtk_widget_show_all(window); gtk_main(); return FALSE; } ~~~ **关于对话框**        GtkAboutDialog构件为您提供了一个简单方式,来显示程序的“关于“信息。通常,当用户点GTK_STOCK_ABOUT菜单项时,这个对话框会显示出来。然而,我们要到第九章才会谈到菜单,因此我们的例子对话框就是用顶层窗口。        用GtkAboutDialog可以显示不同类型的信息。这包括应用程序的名字、版权、当前版本、授权协议、作者、文档、美工和翻译。因为不是每个程序都有这些内容,每一个属性都是可选的。主窗口只显示基本信息,在Figure 5-4中可以看到,连同作者信息。 看下面示例图片:及代码 ![](https://box.kancloud.cn/2016-08-24_57bd7798218d2.jpg) ~~~ /*File: aboutdialogs.c *Date:2013-12-21 *Author:sjin *Mail:43977243@qq.com */ #include <gtk/gtk.h> int main(int argc,char* argv[]) { GtkWidget *dialog; GdkPixbuf *logo; GError *error = NULL; const gchar *authors[] ={ "Kyle Loudon", "sjin", NULL }; const gchar *documenters[]={ "指针操作", "递归", "算法", "链表", "栈和队列", "集合", "哈希表", "树", NULL }; const gchar * translators = "肖翔\n陈珂"; //初始化环境 gtk_init(&argc,&argv); //创建对话框 dialog = gtk_about_dialog_new(); //创建一个图标 logo = gdk_pixbuf_new_from_file("./logo.jpg",&error); if(error == NULL){ gtk_about_dialog_set_logo(GTK_ABOUT_DIALOG(dialog),logo); } else { if(error->domain == GDK_PIXBUF_ERROR){ g_print("GdkPixbuf Error : %s\n",error->message); } else if(error->domain == G_FILE_ERROR){ g_print("GFileError : %s\n",error->message); } else { g_print("an error in the domain : %d has occured!\n",error->domain); } } //设置显示在主对话框上的数据 gtk_about_dialog_set_name(GTK_ABOUT_DIALOG(dialog),"算法精解-C语言描述"); gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(dialog),"1.0"); gtk_about_dialog_set_copyright(GTK_ABOUT_DIALOG(dialog),"(C)2007 Andrew Krause"); gtk_about_dialog_set_comments(GTK_ABOUT_DIALOG(dialog),"我的第一本算法书籍"); //设置作者,文档,翻译 gtk_about_dialog_set_authors(GTK_ABOUT_DIALOG(dialog),authors); gtk_about_dialog_set_documenters(GTK_ABOUT_DIALOG(dialog),documenters); gtk_about_dialog_set_translator_credits(GTK_ABOUT_DIALOG(dialog),translators); gtk_dialog_run(GTK_DIALOG(dialog)); gtk_widget_destroy(dialog); return FALSE; } ~~~ 下面是另一种实现思路: ![](https://box.kancloud.cn/2016-08-24_57bd779844277.jpg) ~~~ /*File:about.c *Date:2013-12-22 *Author:sjin *Mail:413977243@qq.com * */ #include <gtk/gtk.h> static GtkWidget* credits_window; GtkWidget* create_credits() { GtkWidget* window; GtkWidget* vbox; GtkWidget* notebook; GtkWidget* page; GtkWidget* label; window = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_title(GTK_WINDOW(window),"开发人员"); vbox = gtk_vbox_new(FALSE,0); gtk_container_add(GTK_CONTAINER(window),vbox); //这里使用了notebook notebook = gtk_notebook_new(); gtk_box_pack_start(GTK_BOX(vbox),notebook,FALSE,FALSE,5); page = gtk_vbox_new(FALSE,0); label = gtk_label_new("Kyle Loudon"); gtk_box_pack_start(GTK_BOX(page),label,FALSE,FALSE,5); label = gtk_label_new("sjin"); gtk_box_pack_start(GTK_BOX(page),label,FALSE,FALSE,5); label = gtk_label_new("作者"); gtk_notebook_append_page(GTK_NOTEBOOK(notebook),page,label); page = gtk_vbox_new(FALSE,0); label = gtk_label_new("集合"); gtk_box_pack_start(GTK_BOX(page),label,FALSE,FALSE,5); label = gtk_label_new("文档"); gtk_notebook_append_page(GTK_NOTEBOOK(notebook),page,label); page = gtk_vbox_new(FALSE,0); label = gtk_label_new("肖翔\n陈珂"); gtk_box_pack_start(GTK_BOX(page),label,FALSE,FALSE,5); label = gtk_label_new("翻译"); gtk_notebook_append_page(GTK_NOTEBOOK(notebook),page,label); gtk_widget_show_all(window); return window; } void show_credits() { credits_window = create_credits(); gtk_widget_show(credits_window); } int main(int argc,char* argv[]) { GtkWidget* bbox; GtkWidget* vbox; GtkWidget* label; GtkWidget* window; GtkWidget* sep; GtkWidget* image; GtkWidget* button; gtk_init(&argc,&argv); window = gtk_window_new(GTK_WINDOW_TOPLEVEL); g_signal_connect(G_OBJECT(window),"delete_event",G_CALLBACK(gtk_main_quit),NULL); gtk_window_set_title(GTK_WINDOW(window),"关于对话框"); gtk_window_set_position(GTK_WINDOW(window),GTK_WIN_POS_CENTER); gtk_container_set_border_width(GTK_CONTAINER(window),10); vbox = gtk_vbox_new(FALSE,0); gtk_container_add(GTK_CONTAINER(window),vbox); image = gtk_image_new_from_file("logo.jpg"); gtk_box_pack_start(GTK_BOX(vbox),image,FALSE,FALSE,5); label = gtk_label_new(NULL); gtk_label_set_markup(GTK_LABEL(label),"<span><big>算法精解-C语言描述</big></span>"); gtk_box_pack_start(GTK_BOX(vbox),label,FALSE,FALSE,5); label = gtk_label_new("版权所有:机械工业出版社\n 作者:Kyle Loudon"); gtk_box_pack_start(GTK_BOX(vbox),label,FALSE,FALSE,5); sep = gtk_hseparator_new(); gtk_box_pack_start(GTK_BOX(vbox),sep,FALSE,FALSE,5); bbox = gtk_hbutton_box_new(); gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox),GTK_BUTTONBOX_EDGE); gtk_box_pack_start(GTK_BOX(vbox),bbox,FALSE,FALSE,5); button = gtk_button_new_with_label("credits"); gtk_box_pack_start(GTK_BOX(bbox),button,FALSE,FALSE,25); g_signal_connect(G_OBJECT(button),"clicked",G_CALLBACK(show_credits),NULL); button = gtk_button_new_from_stock(GTK_STOCK_OK); g_signal_connect(GTK_OBJECT(button),"clicked",G_CALLBACK(gtk_main_quit),NULL); gtk_box_pack_start(GTK_BOX(bbox),button,FALSE,FALSE,35); gtk_widget_show_all(window); gtk_main(); return FALSE; } ~~~ 参考资料: [http://guoyinghui2012.blog.163.com/blog/static/20871720020126219391244/](http://guoyinghui2012.blog.163.com/blog/static/20871720020126219391244/)