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

在本章中,我们将展示如何在窗口或对话框中布置窗口小部件。

在设计应用的 UI 时,我们决定要使用哪些小部件以及如何组织这些小部件。 为了组织窗口小部件,我们使用称为布局容器的专用非可见窗口小部件。 在本章中,我们将提及GtkAlignmentGtkFixedGtkVBoxGtkTable

GtkFixed

GtkFixed容器将子窗口小部件放置在固定位置并具有固定大小。 此容器不执行自动布局管理。 因此,它不适用于翻译,字体更改或主题。 在大多数应用中,我们不使用GtkFixed容器。 可能会有一些专门的区域可以使用容器(例如,定位图或图像)。

fixed.c

  1. #include <gtk/gtk.h>
  2. int main(int argc, char *argv[]) {
  3. GtkWidget *window;
  4. GtkWidget *fixed;
  5. GtkWidget *btn1;
  6. GtkWidget *btn2;
  7. GtkWidget *btn3;
  8. gtk_init(&argc, &argv);
  9. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  10. gtk_window_set_title(GTK_WINDOW(window), "GtkFixed");
  11. gtk_window_set_default_size(GTK_WINDOW(window), 300, 200);
  12. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  13. fixed = gtk_fixed_new();
  14. gtk_container_add(GTK_CONTAINER(window), fixed);
  15. btn1 = gtk_button_new_with_label("Button");
  16. gtk_fixed_put(GTK_FIXED(fixed), btn1, 150, 50);
  17. gtk_widget_set_size_request(btn1, 80, 30);
  18. btn2 = gtk_button_new_with_label("Button");
  19. gtk_fixed_put(GTK_FIXED(fixed), btn2, 15, 15);
  20. gtk_widget_set_size_request(btn2, 80, 30);
  21. btn3 = gtk_button_new_with_label("Button");
  22. gtk_fixed_put(GTK_FIXED(fixed), btn3, 100, 100);
  23. gtk_widget_set_size_request(btn3, 80, 30);
  24. g_signal_connect(G_OBJECT(window), "destroy",
  25. G_CALLBACK(gtk_main_quit), NULL);
  26. gtk_widget_show_all(window);
  27. gtk_main();
  28. return 0;
  29. }

在我们的示例中,我们创建了三个按钮并将它们放置在固定坐标上。 当我们调整应用窗口的大小时,按钮将保持其大小和位置。

  1. fixed = gtk_fixed_new();

get_fixed_new()函数创建一个GtkFixed容器。

  1. gtk_fixed_put(GTK_FIXED(fixed), btn1, 150, 50);

使用gtk_fixed_put()函数将第一个按钮放置在坐标x = 150y = 50处。

  1. gtk_widget_set_size_request(btn1, 80, 30);

gtk_widget_set_size_request()设置小部件的最小大小。 它是小部件可以接受的最小大小,同时仍可以正常运行并正确绘制自身。

GTK  布局管理 - 图1

图:GtkFixed容器

GtkAlignment

GtkAlignment控制小部件的对齐。 此外,它可以管理其缩放比例。

bottomleft.c

  1. #include <gtk/gtk.h>
  2. int main(int argc, char *argv[]) {
  3. GtkWidget *window;
  4. GtkWidget *align;
  5. GtkWidget *lbl;
  6. gtk_init(&argc, &argv);
  7. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  8. gtk_window_set_title(GTK_WINDOW(window), "GtkAlignment");
  9. gtk_window_set_default_size(GTK_WINDOW(window), 300, 200);
  10. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  11. gtk_container_set_border_width(GTK_CONTAINER(window), 5);
  12. align = gtk_alignment_new(0, 1, 0, 0);
  13. lbl = gtk_label_new("bottom-left");
  14. gtk_container_add(GTK_CONTAINER(align), lbl);
  15. gtk_container_add(GTK_CONTAINER(window), align);
  16. g_signal_connect(G_OBJECT(window), "destroy",
  17. G_CALLBACK(gtk_main_quit), NULL);
  18. gtk_widget_show_all(window);
  19. gtk_main();
  20. return 0;
  21. }

在示例中,标签位于窗口的左下角。

  1. align = gtk_alignment_new(0, 1, 0, 0);

gtk_alignment_new()函数创建GtkAlignment容器。 参数的取值范围是 0 到 1。第一个参数是水平对齐方式,其中 0 左,1 右。 第二个参数是垂直对齐方式,其中 0 是顶部,1 是底部。 第三个参数是水平比例尺,它是子窗口小部件水平扩展以填充未使用空间的数量。 值为 0 表示子窗口小部件永远不应扩展。 最后一个参数是垂直刻度。

  1. lbl = gtk_label_new("bottom-left");

使用gtk_label_new()函数创建标签窗口小部件。

  1. gtk_container_add(GTK_CONTAINER(align), lbl);

标签被添加到GtkAlignment容器中。

  1. gtk_container_add(GTK_CONTAINER(window), align);

最后,将对齐容器放入窗口中。

GTK  布局管理 - 图2

图:GtkAlignment

GtkVBox

GtkVBox是一个垂直盒式容器。 它将其子窗口小部件放置在单个列中。 GtkHBox是一个非常相似的容器; 它将其子窗口小部件放在一行中。

vbox.c

  1. #include <gtk/gtk.h>
  2. int main(int argc, char *argv[]) {
  3. GtkWidget *window;
  4. GtkWidget *vbox;
  5. GtkWidget *settings;
  6. GtkWidget *accounts;
  7. GtkWidget *loans;
  8. GtkWidget *cash;
  9. GtkWidget *debts;
  10. gtk_init(&argc, &argv);
  11. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  12. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  13. gtk_window_set_default_size(GTK_WINDOW(window), 230, 250);
  14. gtk_window_set_title(GTK_WINDOW(window), "GtkVBox");
  15. gtk_container_set_border_width(GTK_CONTAINER(window), 5);
  16. vbox = gtk_vbox_new(TRUE, 1);
  17. gtk_container_add(GTK_CONTAINER(window), vbox);
  18. settings = gtk_button_new_with_label("Settings");
  19. accounts = gtk_button_new_with_label("Accounts");
  20. loans = gtk_button_new_with_label("Loans");
  21. cash = gtk_button_new_with_label("Cash");
  22. debts = gtk_button_new_with_label("Debts");
  23. gtk_box_pack_start(GTK_BOX(vbox), settings, TRUE, TRUE, 0);
  24. gtk_box_pack_start(GTK_BOX(vbox), accounts, TRUE, TRUE, 0);
  25. gtk_box_pack_start(GTK_BOX(vbox), loans, TRUE, TRUE, 0);
  26. gtk_box_pack_start(GTK_BOX(vbox), cash, TRUE, TRUE, 0);
  27. gtk_box_pack_start(GTK_BOX(vbox), debts, TRUE, TRUE, 0);
  28. g_signal_connect(G_OBJECT(window), "destroy",
  29. G_CALLBACK(gtk_main_quit), G_OBJECT(window));
  30. gtk_widget_show_all(window);
  31. gtk_main();
  32. return 0;
  33. }

本示例将五个按钮打包到一栏中。 如果我们调整应用窗口的大小,则子窗口小部件也将被调整大小。

  1. vbox = gtk_vbox_new(TRUE, 1);

gtk_vbox_new()函数创建一个GtkVBox容器。 我们将homogeneous参数设置为TRUE。 这意味着我们所有的按钮都将具有相同的大小。 小部件之间的间距设置为 1 像素。

  1. gtk_box_pack_start(GTK_BOX(vbox), settings, TRUE, TRUE, 0);

gtk_box_pack_start()函数将小部件添加到框中。 前两个参数是盒子容器和子窗口小部件。 接下来的三个参数是expandfillpadding。 注意,如果expand参数设置为FALSE,则fill参数无效。 同样,如果我们创建的容器的均质参数设置为TRUE,则expand参数无效。 在我们的情况下,当窗口放大且窗口小部件填充额外区域时,“设置”按钮将获得额外的空间。

GTK  布局管理 - 图3

图:GtkVBox容器

GtkTable

GtkTable小部件按行和列排列小部件。

table.c

  1. #include <gtk/gtk.h>
  2. int main(int argc, char *argv[]) {
  3. GtkWidget *window;
  4. GtkWidget *table;
  5. GtkWidget *button;
  6. gchar *values[16] = { "7", "8", "9", "/",
  7. "4", "5", "6", "*",
  8. "1", "2", "3", "-",
  9. "0", ".", "=", "+"
  10. };
  11. gtk_init(&argc, &argv);
  12. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  13. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  14. gtk_window_set_default_size(GTK_WINDOW(window), 250, 180);
  15. gtk_window_set_title(GTK_WINDOW(window), "GtkTable");
  16. gtk_container_set_border_width(GTK_CONTAINER(window), 5);
  17. table = gtk_table_new(4, 4, TRUE);
  18. gtk_table_set_row_spacings(GTK_TABLE(table), 2);
  19. gtk_table_set_col_spacings(GTK_TABLE(table), 2);
  20. int i = 0;
  21. int j = 0;
  22. int pos = 0;
  23. for (i=0; i < 4; i++) {
  24. for (j=0; j < 4; j++) {
  25. button = gtk_button_new_with_label(values[pos]);
  26. gtk_table_attach_defaults(GTK_TABLE(table), button, j, j+1, i, i+1);
  27. pos++;
  28. }
  29. }
  30. gtk_container_add(GTK_CONTAINER(window), table);
  31. g_signal_connect(G_OBJECT(window), "destroy",
  32. G_CALLBACK(gtk_main_quit), NULL);
  33. gtk_widget_show_all(window);
  34. gtk_main();
  35. return 0;
  36. }

在此示例中,我们创建了一组在计算器中看到的按钮。

  1. table = gtk_table_new(4, 4, TRUE);

我们创建一个具有 4 行 4 列的新GtkTable小部件。 当我们将TRUE传递给第三个参数时,所有表单元格的大小都将调整为包含最大窗口小部件的单元格的大小。

  1. gtk_table_set_row_spacings(GTK_TABLE(table), 2);
  2. gtk_table_set_col_spacings(GTK_TABLE(table), 2);

我们在行和列之间设置一些空间。

  1. for (i=0; i < 4; i++) {
  2. for (j=0; j < 4; j++) {
  3. button = gtk_button_new_with_label(values[pos]);
  4. gtk_table_attach_defaults(GTK_TABLE(table), button, j, j+1, i, i+1 );
  5. pos++;
  6. }
  7. }

这段代码创建了 16 个按钮,并将它们放入容器中。 gtk_table_attach_defaults()将子项添加到具有相同填充和扩展选项的表容器中。

GTK  布局管理 - 图4

图:GtkTable

角按钮

下一个示例在窗口的右下角放置两个按钮。

cornerbuttons.c

  1. #include <gtk/gtk.h>
  2. int main(int argc, char *argv[]) {
  3. GtkWidget *window;
  4. GtkWidget *okBtn;
  5. GtkWidget *clsBtn;
  6. GtkWidget *vbox;
  7. GtkWidget *hbox;
  8. GtkWidget *halign;
  9. GtkWidget *valign;
  10. gtk_init(&argc, &argv);
  11. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  12. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  13. gtk_window_set_default_size(GTK_WINDOW(window), 350, 200);
  14. gtk_window_set_title(GTK_WINDOW(window), "Corner buttons");
  15. gtk_container_set_border_width(GTK_CONTAINER(window), 10);
  16. vbox = gtk_vbox_new(FALSE, 5);
  17. valign = gtk_alignment_new(0, 1, 0, 0);
  18. gtk_container_add(GTK_CONTAINER(vbox), valign);
  19. gtk_container_add(GTK_CONTAINER(window), vbox);
  20. hbox = gtk_hbox_new(TRUE, 3);
  21. okBtn = gtk_button_new_with_label("OK");
  22. gtk_widget_set_size_request(okBtn, 70, 30);
  23. gtk_container_add(GTK_CONTAINER(hbox), okBtn);
  24. clsBtn = gtk_button_new_with_label("Close");
  25. gtk_container_add(GTK_CONTAINER(hbox), clsBtn);
  26. halign = gtk_alignment_new(1, 0, 0, 0);
  27. gtk_container_add(GTK_CONTAINER(halign), hbox);
  28. gtk_box_pack_start(GTK_BOX(vbox), halign, FALSE, FALSE, 0);
  29. g_signal_connect(G_OBJECT(window), "destroy",
  30. G_CALLBACK(gtk_main_quit), G_OBJECT(window));
  31. gtk_widget_show_all(window);
  32. gtk_main();
  33. return 0;
  34. }

在示例中,我们使用一个水平框,一个垂直框和两个对齐容器。

  1. valign = gtk_alignment_new(0, 1, 0, 0);

此对齐容器将其子窗口小部件放在底部。

  1. gtk_container_add(GTK_CONTAINER(vbox), valign);

在这里,我们将对齐小部件放置在垂直框中。

  1. hbox = gtk_hbox_new(TRUE, 3);
  2. okBtn = gtk_button_new_with_label("OK");
  3. gtk_widget_set_size_request(okBtn, 70, 30);
  4. gtk_container_add(GTK_CONTAINER(hbox), okBtn);
  5. clsBtn = gtk_button_new_with_label("Close");
  6. gtk_container_add(GTK_CONTAINER(hbox), clsBtn);

我们创建一个水平框,并在其中放置两个按钮。 gtk_widget_set_size_request()设置小部件的最小大小。 由于我们已经将GtkHBoxhomogeneous参数设置为TRUE,因此另一个按钮也被调整为新的大小。

  1. halign = gtk_alignment_new(1, 0, 0, 0);
  2. gtk_container_add(GTK_CONTAINER(halign), hbox);
  3. gtk_box_pack_start(GTK_BOX(vbox), halign, FALSE, FALSE, 0);

这将创建一个对齐容器,将其子窗口小部件放在右侧。 我们将水平框添加到对齐容器中,然后将对齐容器包装到垂直框中。 对齐容器只能使用一个子窗口小部件; 因此,我们还必须使用盒子。

GTK  布局管理 - 图5

图:角按钮

窗口

接下来,我们将创建一个更高级的示例。 我们显示一个可以在 JDeveloper 中找到的窗口。

GTK  布局管理 - 图6

图:窗口 dialog in JDeveloper

该对话框显示所有打开的窗口,或更确切地说是 JDeveloper 应用中的选项卡。

windows.c

  1. #include <gtk/gtk.h>
  2. int main(int argc, char *argv[]) {
  3. GtkWidget *window;
  4. GtkWidget *table;
  5. GtkWidget *title;
  6. GtkWidget *wins;
  7. GtkWidget *halign;
  8. GtkWidget *halign2;
  9. GtkWidget *valign;
  10. GtkWidget *actBtn;
  11. GtkWidget *clsBtn;
  12. GtkWidget *hlpBtn;
  13. GtkWidget *okBtn;
  14. gtk_init(&argc, &argv);
  15. window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
  16. gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER);
  17. gtk_widget_set_size_request (window, 300, 250);
  18. gtk_window_set_title(GTK_WINDOW(window), "Windows");
  19. gtk_container_set_border_width(GTK_CONTAINER(window), 15);
  20. table = gtk_table_new(6, 4, FALSE);
  21. gtk_table_set_col_spacings(GTK_TABLE(table), 3);
  22. gtk_table_set_row_spacing(GTK_TABLE(table), 0, 3);
  23. title = gtk_label_new("Windows");
  24. halign = gtk_alignment_new(0, 0, 0, 0);
  25. gtk_container_add(GTK_CONTAINER(halign), title);
  26. gtk_table_attach(GTK_TABLE(table), halign, 0, 1, 0, 1,
  27. GTK_FILL, GTK_FILL, 0, 0);
  28. wins = gtk_text_view_new();
  29. gtk_text_view_set_editable(GTK_TEXT_VIEW(wins), FALSE);
  30. gtk_text_view_set_cursor_visible(GTK_TEXT_VIEW(wins), FALSE);
  31. gtk_table_attach(GTK_TABLE(table), wins, 0, 2, 1, 3,
  32. GTK_FILL | GTK_EXPAND, GTK_FILL | GTK_EXPAND, 1, 1);
  33. actBtn = gtk_button_new_with_label("Activate");
  34. gtk_widget_set_size_request(actBtn, 50, 30);
  35. gtk_table_attach(GTK_TABLE(table), actBtn, 3, 4, 1, 2,
  36. GTK_FILL, GTK_SHRINK, 1, 1);
  37. valign = gtk_alignment_new(0, 0, 0, 0);
  38. clsBtn = gtk_button_new_with_label("Close");
  39. gtk_widget_set_size_request(clsBtn, 70, 30);
  40. gtk_container_add(GTK_CONTAINER(valign), clsBtn);
  41. gtk_table_set_row_spacing(GTK_TABLE(table), 1, 3);
  42. gtk_table_attach(GTK_TABLE(table), valign, 3, 4, 2, 3,
  43. GTK_FILL, GTK_FILL | GTK_EXPAND, 1, 1);
  44. halign2 = gtk_alignment_new(0, 1, 0, 0);
  45. hlpBtn = gtk_button_new_with_label("Help");
  46. gtk_container_add(GTK_CONTAINER(halign2), hlpBtn);
  47. gtk_widget_set_size_request(hlpBtn, 70, 30);
  48. gtk_table_set_row_spacing(GTK_TABLE(table), 3, 5);
  49. gtk_table_attach(GTK_TABLE(table), halign2, 0, 1, 4, 5,
  50. GTK_FILL, GTK_FILL, 0, 0);
  51. okBtn = gtk_button_new_with_label("OK");
  52. gtk_widget_set_size_request(okBtn, 70, 30);
  53. gtk_table_attach(GTK_TABLE(table), okBtn, 3, 4, 4, 5,
  54. GTK_FILL, GTK_FILL, 0, 0);
  55. gtk_container_add(GTK_CONTAINER(window), table);
  56. g_signal_connect(G_OBJECT(window), "destroy",
  57. G_CALLBACK(gtk_main_quit), G_OBJECT(window));
  58. gtk_widget_show_all(window);
  59. gtk_main();
  60. return 0;
  61. }

该示例使用一个表容器和三个对齐容器。

  1. table = gtk_table_new(6, 4, FALSE);

创建一个GtkTable容器。 它有六行四列。

  1. gtk_table_set_col_spacings(GTK_TABLE(table), 3);

gtk_table_set_col_spacings()将表中每列之间的间隔设置为 3。

  1. gtk_table_set_row_spacing(GTK_TABLE(table), 0, 3);

gtk_table_row_spacing()设置第一行和第二行之间的空间。

  1. title = gtk_label_new("Windows");
  2. halign = gtk_alignment_new(0, 0, 0, 0);
  3. gtk_container_add(GTK_CONTAINER(halign), title);
  4. gtk_table_attach(GTK_TABLE(table), halign, 0, 1, 0, 1,
  5. GTK_FILL, GTK_FILL, 0, 0);

这段代码创建了一个左对齐的标签。 标签放置在GtkTable容器的第一行和第一列中。

  1. wins = gtk_text_view_new();
  2. gtk_text_view_set_editable(GTK_TEXT_VIEW(wins), FALSE);
  3. gtk_text_view_set_cursor_visible(GTK_TEXT_VIEW(wins), FALSE);
  4. gtk_table_attach(GTK_TABLE(table), wins, 0, 2, 1, 3,
  5. GTK_FILL | GTK_EXPAND, GTK_FILL | GTK_EXPAND, 1, 1);

GtkText小部件跨越两行两列。 我们使用gtk_text_view_set_editable()方法使窗口小部件不可编辑,并使用gtk_text_view_set_cursor_visible()方法隐藏其光标。

  1. valign = gtk_alignment_new(0, 0, 0, 0);
  2. clsBtn = gtk_button_new_with_label("Close");
  3. gtk_widget_set_size_request(clsBtn, 70, 30);
  4. gtk_container_add(GTK_CONTAINER(valign), clsBtn);
  5. gtk_table_set_row_spacing(GTK_TABLE(table), 1, 3);
  6. gtk_table_attach(GTK_TABLE(table), valign, 3, 4, 2, 3,
  7. GTK_FILL, GTK_FILL | GTK_EXPAND, 1, 1);

我们将“关闭”按钮放在文本视图窗口小部件旁边的第四列中。 我们将按钮添加到对齐小部件中,以便我们可以将其对齐到顶部。

  1. halign2 = gtk_alignment_new(0, 1, 0, 0);
  2. hlpBtn = gtk_button_new_with_label("Help");
  3. gtk_container_add(GTK_CONTAINER(halign2), hlpBtn);
  4. gtk_widget_set_size_request(hlpBtn, 70, 30);
  5. gtk_table_set_row_spacing(GTK_TABLE(table), 3, 5);
  6. gtk_table_attach(GTK_TABLE(table), halign2, 0, 1, 4, 5,
  7. GTK_FILL, GTK_FILL, 0, 0);

“帮助”按钮向左对齐。 它位于文本小部件下方。 我们在文本小部件和按钮之间放置一些空间。

  1. okBtn = gtk_button_new_with_label("OK");
  2. gtk_widget_set_size_request(okBtn, 70, 30);
  3. gtk_table_attach(GTK_TABLE(table), okBtn, 3, 4, 4, 5,
  4. GTK_FILL, GTK_FILL, 0, 0);

“确定”按钮转到“激活”和“关闭”按钮下面的第二列。

GTK  布局管理 - 图7

图:窗口

本章专门介绍布局管理。