原文: http://zetcode.com/gui/gtk2/gtkdialogs/

在 GTK+ 编程教程的这一部分中,我们使用对话框。

对话框窗口或对话框是大多数现代 GUI 应用必不可少的部分。 对话被定义为两个或更多人之间的对话。 在计算机应用中,对话框是一个窗口,用于与应用“对话”。 对话框用于输入数据,修改数据,更改应用设置等。

MessageDialog

消息对话框是方便的对话框,可向应用的用户提供消息。 该消息包含文本和图像数据。

messagedialogs.c

  1. #include <gtk/gtk.h>
  2. void show_info(GtkWidget *widget, gpointer window) {
  3. GtkWidget *dialog;
  4. dialog = gtk_message_dialog_new(GTK_WINDOW(window),
  5. GTK_DIALOG_DESTROY_WITH_PARENT,
  6. GTK_MESSAGE_INFO,
  7. GTK_BUTTONS_OK,
  8. "Download Completed");
  9. gtk_window_set_title(GTK_WINDOW(dialog), "Information");
  10. gtk_dialog_run(GTK_DIALOG(dialog));
  11. gtk_widget_destroy(dialog);
  12. }
  13. void show_error(GtkWidget *widget, gpointer window) {
  14. GtkWidget *dialog;
  15. dialog = gtk_message_dialog_new(GTK_WINDOW(window),
  16. GTK_DIALOG_DESTROY_WITH_PARENT,
  17. GTK_MESSAGE_ERROR,
  18. GTK_BUTTONS_OK,
  19. "Error loading file");
  20. gtk_window_set_title(GTK_WINDOW(dialog), "Error");
  21. gtk_dialog_run(GTK_DIALOG(dialog));
  22. gtk_widget_destroy(dialog);
  23. }
  24. void show_question(GtkWidget *widget, gpointer window) {
  25. GtkWidget *dialog;
  26. dialog = gtk_message_dialog_new(GTK_WINDOW(window),
  27. GTK_DIALOG_DESTROY_WITH_PARENT,
  28. GTK_MESSAGE_QUESTION,
  29. GTK_BUTTONS_YES_NO,
  30. "Are you sure to quit?");
  31. gtk_window_set_title(GTK_WINDOW(dialog), "Question");
  32. gtk_dialog_run(GTK_DIALOG(dialog));
  33. gtk_widget_destroy(dialog);
  34. }
  35. void show_warning(GtkWidget *widget, gpointer window) {
  36. GtkWidget *dialog;
  37. dialog = gtk_message_dialog_new(GTK_WINDOW(window),
  38. GTK_DIALOG_DESTROY_WITH_PARENT,
  39. GTK_MESSAGE_WARNING,
  40. GTK_BUTTONS_OK,
  41. "Unallowed operation");
  42. gtk_window_set_title(GTK_WINDOW(dialog), "Warning");
  43. gtk_dialog_run(GTK_DIALOG(dialog));
  44. gtk_widget_destroy(dialog);
  45. }
  46. int main(int argc, char *argv[]) {
  47. GtkWidget *window;
  48. GtkWidget *table;
  49. GtkWidget *info;
  50. GtkWidget *warn;
  51. GtkWidget *que;
  52. GtkWidget *err;
  53. gtk_init(&argc, &argv);
  54. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  55. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  56. gtk_window_set_default_size(GTK_WINDOW(window), 220, 150);
  57. gtk_window_set_title(GTK_WINDOW(window), "Message dialogs");
  58. table = gtk_table_new(2, 2, TRUE);
  59. gtk_table_set_row_spacings(GTK_TABLE(table), 2);
  60. gtk_table_set_col_spacings(GTK_TABLE(table), 2);
  61. info = gtk_button_new_with_label("Info");
  62. warn = gtk_button_new_with_label("Warning");
  63. que = gtk_button_new_with_label("Question");
  64. err = gtk_button_new_with_label("Error");
  65. gtk_table_attach(GTK_TABLE(table), info, 0, 1, 0, 1,
  66. GTK_FILL, GTK_FILL, 3, 3);
  67. gtk_table_attach(GTK_TABLE(table), warn, 1, 2, 0, 1,
  68. GTK_FILL, GTK_FILL, 3, 3);
  69. gtk_table_attach(GTK_TABLE(table), que, 0, 1, 1, 2,
  70. GTK_FILL, GTK_FILL, 3, 3);
  71. gtk_table_attach(GTK_TABLE(table), err, 1, 2, 1, 2,
  72. GTK_FILL, GTK_FILL, 3, 3);
  73. gtk_container_add(GTK_CONTAINER(window), table);
  74. gtk_container_set_border_width(GTK_CONTAINER(window), 15);
  75. g_signal_connect(G_OBJECT(info), "clicked",
  76. G_CALLBACK(show_info), (gpointer) window);
  77. g_signal_connect(G_OBJECT(warn), "clicked",
  78. G_CALLBACK(show_warning), (gpointer) window);
  79. g_signal_connect(G_OBJECT(que), "clicked",
  80. G_CALLBACK(show_question), (gpointer) window);
  81. g_signal_connect(G_OBJECT(err), "clicked",
  82. G_CALLBACK(show_error), (gpointer) window);
  83. g_signal_connect(G_OBJECT(window), "destroy",
  84. G_CALLBACK(gtk_main_quit), G_OBJECT(window));
  85. gtk_widget_show_all(window);
  86. gtk_main();
  87. return 0;
  88. }

在我们的示例中,我们显示了四种消息对话框:信息,警告,问题和错误消息对话框。

  1. void show_question(GtkWidget *widget, gpointer window) {
  2. GtkWidget *dialog;
  3. dialog = gtk_message_dialog_new(GTK_WINDOW(window),
  4. GTK_DIALOG_DESTROY_WITH_PARENT,
  5. GTK_MESSAGE_QUESTION,
  6. GTK_BUTTONS_YES_NO,
  7. "Are you sure to quit?");
  8. gtk_window_set_title(GTK_WINDOW(dialog), "Question");
  9. gtk_dialog_run(GTK_DIALOG(dialog));
  10. gtk_widget_destroy(dialog);
  11. }

show_question()函数中,我们弹出消息对话框。 消息对话框是使用gtk_message_dialog_new()调用创建的。 函数的参数指定我们创建哪种消息对话框。 GTK_MESSAGE_QUESTION常量创建一个问题类型对话框。 GTK_BUTTONS_YES_NO常量将在对话框中添加“是”和“否”按钮。 最后一个参数是我们在对话框中显示的文本。 gtk_dialog_run()函数显示对话框并阻塞主循环,直到对话框响应或被破坏为止。 该对话框必须使用gtk_widget_destroy()函数销毁。

GTK  对话框 - 图1

GTK  对话框 - 图2

GTK  对话框 - 图3

GTK  对话框 - 图4

图:消息对话框

GtkAboutDialog

GtkAboutDialog是一个对话框,其目的是显示有关应用的信息。 它可以显示徽标,应用名称,版本,版权,网站和许可证信息。 也有可能对作者,文档撰写者,翻译者和艺术家予以赞扬。

aboutdialog.c

  1. #include <gtk/gtk.h>
  2. void show_about(GtkWidget *widget, gpointer data) {
  3. GdkPixbuf *pixbuf = gdk_pixbuf_new_from_file("battery.png", NULL);
  4. GtkWidget *dialog = gtk_about_dialog_new();
  5. gtk_about_dialog_set_name(GTK_ABOUT_DIALOG(dialog), "Battery");
  6. gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(dialog), "0.9");
  7. gtk_about_dialog_set_copyright(GTK_ABOUT_DIALOG(dialog),"(c) Jan Bodnar");
  8. gtk_about_dialog_set_comments(GTK_ABOUT_DIALOG(dialog),
  9. "Battery is a simple tool for battery checking.");
  10. gtk_about_dialog_set_website(GTK_ABOUT_DIALOG(dialog),
  11. "http://www.batteryhq.net");
  12. gtk_about_dialog_set_logo(GTK_ABOUT_DIALOG(dialog), pixbuf);
  13. g_object_unref(pixbuf), pixbuf = NULL;
  14. gtk_dialog_run(GTK_DIALOG (dialog));
  15. gtk_widget_destroy(dialog);
  16. }
  17. int main(int argc, char *argv[]) {
  18. GtkWidget *window;
  19. GtkWidget *about;
  20. GdkPixbuf *battery;
  21. gtk_init(&argc, &argv);
  22. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  23. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  24. gtk_window_set_default_size(GTK_WINDOW(window), 220, 150);
  25. gtk_window_set_title(GTK_WINDOW(window), "Battery");
  26. gtk_container_set_border_width(GTK_CONTAINER(window), 15);
  27. gtk_widget_add_events(window, GDK_BUTTON_PRESS_MASK);
  28. g_signal_connect(G_OBJECT(window), "button-press-event",
  29. G_CALLBACK(show_about), (gpointer) window);
  30. g_signal_connect(G_OBJECT(window), "destroy",
  31. G_CALLBACK(gtk_main_quit), G_OBJECT(window));
  32. gtk_widget_show_all(window);
  33. gtk_main();
  34. return 0;
  35. }

该代码示例使用具有某些功能的GtkAboutDialog。 单击窗口的客户区域会弹出对话框。

  1. GtkWidget *dialog = gtk_about_dialog_new();

GtkAboutDialog是使用gtk_about_dialog_new()函数创建的。

  1. gtk_about_dialog_set_name(GTK_ABOUT_DIALOG(dialog), "Battery");
  2. gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(dialog), "0.9");
  3. gtk_about_dialog_set_copyright(GTK_ABOUT_DIALOG(dialog),
  4. "(c) Jan Bodnar");

这些函数调用设置应用的名称,版本和版权。

  1. gtk_about_dialog_set_comments(GTK_ABOUT_DIALOG(dialog),
  2. "Battery is a simple tool for battery checking.");
  3. gtk_about_dialog_set_website(GTK_ABOUT_DIALOG(dialog),
  4. "http://www.batteryhq.net");

这些行设置了描述性注释和应用的网站。

  1. GdkPixbuf *pixbuf = gdk_pixbuf_new_from_file("battery.png", NULL);
  2. ...
  3. gtk_about_dialog_set_logo(GTK_ABOUT_DIALOG(dialog), pixbuf);
  4. g_object_unref(pixbuf), pixbuf = NULL;

此代码创建应用的徽标。

GTK  对话框 - 图5

图:GtkAboutDialog

GtkFontSelectionDialog

GtkFontSelectionDialog是用于选择字体的对话框。 它通常用于进行一些文本编辑或格式化的应用中。

fontdialog.c

  1. #include <gtk/gtk.h>
  2. void select_font(GtkWidget *widget, gpointer label) {
  3. GtkResponseType result;
  4. GtkWidget *dialog = gtk_font_selection_dialog_new("Select Font");
  5. result = gtk_dialog_run(GTK_DIALOG(dialog));
  6. if (result == GTK_RESPONSE_OK || result == GTK_RESPONSE_APPLY) {
  7. PangoFontDescription *font_desc;
  8. gchar *fontname = gtk_font_selection_dialog_get_font_name(
  9. GTK_FONT_SELECTION_DIALOG(dialog));
  10. font_desc = pango_font_description_from_string(fontname);
  11. gtk_widget_modify_font(GTK_WIDGET(label), font_desc);
  12. g_free(fontname);
  13. }
  14. gtk_widget_destroy(dialog);
  15. }
  16. int main(int argc, char *argv[]) {
  17. GtkWidget *window;
  18. GtkWidget *label;
  19. GtkWidget *vbox;
  20. GtkWidget *toolbar;
  21. GtkToolItem *font;
  22. gtk_init(&argc, &argv);
  23. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  24. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  25. gtk_window_set_default_size(GTK_WINDOW(window), 280, 200);
  26. gtk_window_set_title(GTK_WINDOW(window), "Font Selection Dialog");
  27. vbox = gtk_vbox_new(FALSE, 0);
  28. gtk_container_add(GTK_CONTAINER(window), vbox);
  29. toolbar = gtk_toolbar_new();
  30. gtk_toolbar_set_style(GTK_TOOLBAR(toolbar), GTK_TOOLBAR_ICONS);
  31. gtk_container_set_border_width(GTK_CONTAINER(toolbar), 2);
  32. font = gtk_tool_button_new_from_stock(GTK_STOCK_SELECT_FONT);
  33. gtk_toolbar_insert(GTK_TOOLBAR(toolbar), font, -1);
  34. gtk_box_pack_start(GTK_BOX(vbox), toolbar, FALSE, FALSE, 5);
  35. label = gtk_label_new("ZetCode");
  36. gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_CENTER);
  37. gtk_box_pack_start(GTK_BOX(vbox), label, TRUE, FALSE, 5);
  38. g_signal_connect(G_OBJECT(font), "clicked",
  39. G_CALLBACK(select_font), label);
  40. g_signal_connect(G_OBJECT(window), "destroy",
  41. G_CALLBACK(gtk_main_quit), NULL);
  42. gtk_widget_show_all(window);
  43. gtk_main();
  44. return 0;
  45. }

在代码示例中,我们在窗口中心放置了一个简单标签。 我们通过单击工具栏按钮显示一个字体选择对话框。

  1. GtkWidget *dialog = gtk_font_selection_dialog_new("Select Font");
  2. result = gtk_dialog_run(GTK_DIALOG(dialog));

我们使用gtk_font_selection_dialog_new()函数创建GtkFontSelectionDialog,然后使用gtk_dialog_run()函数运行它。

  1. if (result == GTK_RESPONSE_OK || result == GTK_RESPONSE_APPLY) {
  2. PangoFontDescription *font_desc;
  3. gchar *fontname = gtk_font_selection_dialog_get_font_name(
  4. GTK_FONT_SELECTION_DIALOG(dialog));
  5. font_desc = pango_font_description_from_string(fontname);
  6. gtk_widget_modify_font(GTK_WIDGET(label), font_desc);
  7. g_free(fontname);
  8. }

如果用户单击“确定”或“应用”按钮,则继续。 我们使用gtk_font_selection_dialog_get_font_name()函数获得选定的字体名称。 然后,将标签的字体更改为选定的字体名称。

GTK  对话框 - 图6

图:GtkFontSelectionDialog

GtkColorSelectionDialog

GtkColorSelectionDialog是用于选择颜色的对话框。

colordialog.c

  1. #include <gtk/gtk.h>
  2. void select_font(GtkWidget *widget, gpointer label) {
  3. GtkResponseType result;
  4. GtkColorSelection *colorsel;
  5. GtkWidget *dialog = gtk_color_selection_dialog_new("Font Color");
  6. result = gtk_dialog_run(GTK_DIALOG(dialog));
  7. if (result == GTK_RESPONSE_OK) {
  8. GdkColor color;
  9. colorsel = GTK_COLOR_SELECTION(
  10. GTK_COLOR_SELECTION_DIALOG(dialog)->colorsel);
  11. gtk_color_selection_get_current_color(colorsel,
  12. &color);
  13. gtk_widget_modify_fg(GTK_WIDGET(label),
  14. GTK_STATE_NORMAL,
  15. &color);
  16. }
  17. gtk_widget_destroy(dialog);
  18. }
  19. int main(int argc, char *argv[]) {
  20. GtkWidget *window;
  21. GtkWidget *widget;
  22. GtkWidget *label;
  23. GtkWidget *vbox;
  24. GtkWidget *toolbar;
  25. GtkToolItem *font;
  26. gtk_init(&argc, &argv);
  27. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  28. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  29. gtk_window_set_default_size(GTK_WINDOW(window), 280, 200);
  30. gtk_window_set_title(GTK_WINDOW(window), "Color Selection Dialog");
  31. vbox = gtk_vbox_new(FALSE, 0);
  32. gtk_container_add(GTK_CONTAINER(window), vbox);
  33. toolbar = gtk_toolbar_new();
  34. gtk_toolbar_set_style(GTK_TOOLBAR(toolbar), GTK_TOOLBAR_ICONS);
  35. gtk_container_set_border_width(GTK_CONTAINER(toolbar), 2);
  36. font = gtk_tool_button_new_from_stock(GTK_STOCK_SELECT_COLOR);
  37. gtk_toolbar_insert(GTK_TOOLBAR(toolbar), font, -1);
  38. gtk_box_pack_start(GTK_BOX(vbox), toolbar, FALSE, FALSE, 5);
  39. label = gtk_label_new("ZetCode");
  40. gtk_label_set_justify(GTK_LABEL(label), GTK_JUSTIFY_CENTER);
  41. gtk_box_pack_start(GTK_BOX(vbox), label, TRUE, FALSE, 5);
  42. g_signal_connect(G_OBJECT(font), "clicked",
  43. G_CALLBACK(select_font), label);
  44. g_signal_connect(G_OBJECT(window), "destroy",
  45. G_CALLBACK(gtk_main_quit), NULL);
  46. gtk_widget_show_all(window);
  47. gtk_main();
  48. return 0;
  49. }

该示例与上一个示例非常相似。 这次我们更改标签的颜色。

  1. GtkWidget *dialog = gtk_color_selection_dialog_new("Font Color");
  2. result = gtk_dialog_run(GTK_DIALOG(dialog));

我们创建并显示GtkColorSelectionDialog

  1. if (result == GTK_RESPONSE_OK) {
  2. GdkColor color;
  3. colorsel = GTK_COLOR_SELECTION(
  4. GTK_COLOR_SELECTION_DIALOG(dialog)->colorsel);
  5. gtk_color_selection_get_current_color(colorsel,
  6. &color);
  7. gtk_widget_modify_fg(GTK_WIDGET(label),
  8. GTK_STATE_NORMAL,
  9. &color);
  10. }

如果按“确定”按钮,我们将获得颜色并修改标签的颜色。 颜色值通过gtk_color_selection_get_current_color()函数返回。

GTK  对话框 - 图7

图:GtkColorSelectionDialog

本章是关于 GTK+ 中的对话框的。