Up: [Readme.md](Readme.md), Prev: [Section 5](sec5.md), Next: [Section 7](sec7.md) # Ui file and GtkBuiler ## New, open and save button We made the simplest editor in the previous section. It reads the files in `on_open` funciton at start-up and writes it at closing window. It works but is not good. It is better to make "New", "Open", "Save" and "Close" buttons. This section describes how to put those buttons into the window. Signals and handlers will be explained later. ![Screenshot of the file editor](image/screenshot_tfe2.png) The screenshot above shows the layout. The function `on_open` in the source code `tfe2.c` is as follows. 1 static void 2 on_open (GApplication *app, GFile ** files, gint n_files, gchar *hint, gpointer user_data) { 3 GtkWidget *win; 4 GtkWidget *nb; 5 GtkWidget *lab; 6 GtkNotebookPage *nbp; 7 GtkWidget *scr; 8 GtkWidget *tv; 9 GtkTextBuffer *tb; 10 char *contents; 11 gsize length; 12 char *filename; 13 int i; 14 15 GtkWidget *boxv; 16 GtkWidget *boxh; 17 GtkWidget *dmy1; 18 GtkWidget *dmy2; 19 GtkWidget *dmy3; 20 GtkWidget *btnn; /* button for new */ 21 GtkWidget *btno; /* button for open */ 22 GtkWidget *btns; /* button for save */ 23 GtkWidget *btnc; /* button for close */ 24 25 26 win = gtk_application_window_new (GTK_APPLICATION (app)); 27 gtk_window_set_title (GTK_WINDOW (win), "file editor"); 28 gtk_window_set_default_size (GTK_WINDOW (win), 600, 400); 29 30 boxv = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); 31 gtk_window_set_child (GTK_WINDOW (win), boxv); 32 33 boxh = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0); 34 gtk_box_append (GTK_BOX (boxv), boxh); 35 36 dmy1 = gtk_label_new(NULL); /* dummy label for left space */ 37 gtk_label_set_width_chars (GTK_LABEL (dmy1), 10); 38 dmy2 = gtk_label_new(NULL); /* dummy label for center space */ 39 gtk_widget_set_hexpand (dmy2, TRUE); 40 dmy3 = gtk_label_new(NULL); /* dummy label for right space */ 41 gtk_label_set_width_chars (GTK_LABEL (dmy3), 10); 42 btnn = gtk_button_new_with_label ("New"); 43 btno = gtk_button_new_with_label ("Open"); 44 btns = gtk_button_new_with_label ("Save"); 45 btnc = gtk_button_new_with_label ("Close"); 46 47 gtk_box_append (GTK_BOX (boxh), dmy1); 48 gtk_box_append (GTK_BOX (boxh), btnn); 49 gtk_box_append (GTK_BOX (boxh), btno); 50 gtk_box_append (GTK_BOX (boxh), dmy2); 51 gtk_box_append (GTK_BOX (boxh), btns); 52 gtk_box_append (GTK_BOX (boxh), btnc); 53 gtk_box_append (GTK_BOX (boxh), dmy3); 54 55 nb = gtk_notebook_new (); 56 gtk_widget_set_hexpand (nb, TRUE); 57 gtk_widget_set_vexpand (nb, TRUE); 58 gtk_box_append (GTK_BOX (boxv), nb); 59 60 for (i = 0; i < n_files; i++) { 61 if (g_file_load_contents (files[i], NULL, &contents, &length, NULL, NULL)) { 62 scr = gtk_scrolled_window_new (); 63 tv = tfe_text_view_new (); 64 tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv)); 65 gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (tv), GTK_WRAP_WORD_CHAR); 66 gtk_scrolled_window_set_child (GTK_SCROLLED_WINDOW (scr), tv); 67 68 tfe_text_view_set_file (TFE_TEXT_VIEW (tv), g_file_dup (files[i])); 69 gtk_text_buffer_set_text (tb, contents, length); 70 g_free (contents); 71 filename = g_file_get_basename (files[i]); 72 lab = gtk_label_new (filename); 73 gtk_notebook_append_page (GTK_NOTEBOOK (nb), scr, lab); 74 nbp = gtk_notebook_get_page (GTK_NOTEBOOK (nb), scr); 75 g_object_set (nbp, "tab-expand", TRUE, NULL); 76 g_free (filename); 77 } else { 78 filename = g_file_get_path (files[i]); 79 g_print ("No such file: %s.\n", filename); 80 g_free (filename); 81 } 82 } 83 if (gtk_notebook_get_n_pages (GTK_NOTEBOOK (nb)) > 0) { 84 gtk_widget_show (win); 85 } else 86 gtk_window_destroy (GTK_WINDOW (win)); 87 } The point is how to build the window. - 26-28: Generate GtkApplicationWindow and set its title and default size. - 30-31: Generate GtkBox `boxv`. It is a vertical box and a child of GtkApplicationWindow. It has two children. The first child is a horizontal box includes buttons. The second child is GtkNotebook. - 33-34: Generate GtkBox `boxh` and append it to 'boxv' as a first child. - 36-41: Generate three dummy labels. The labels `dmy1` and `dmy3` has a character width of ten. The other label `dmy2` is set hexpand property TRUE. This makes the label expands horizontally as long as possible. - 42-45: Generate four buttons. - 47-53: Append these GtkLabel and GtkButton to `boxh`. - 55-58: Generate GtkNotebook and set hexpand and vexpand properties TRUE. This makes it expands horizontally and vertically as big as possible. It is appended to `boxv` as the second child. The number of lines is 33(=58-26+1) to build the widgets. And we needed many variables (boxv, boxh, dmy1 ...). Most of them aren't necessary except building the widgets. Are there any good solution to reduce these work? Gtk provides GtkBuilder. It reads ui data and builds a window. It reduces the cumbersom work. ## Ui file First, let's look at the ui file `tfe3.ui` that defines a structure of the widgets. 1 2 3 file editor 4 600 5 400 6 7 8 GTK_ORIENTATION_VERTICAL 9 10 11 GTK_ORIENTATION_HORIZONTAL 12 13 14 10 15 16 17 18 19 New 20 21 22 23 24 Open 25 26 27 28 29 TRUE 30 31 32 33 34 Save 35 36 37 38 39 Close 40 41 42 43 44 10 45 46 47 48 49 50 51 TRUE 52 TRUE 53 54 55 56 57 58 59 This is coded with XML structure. Constructs begin with `<` and end with `>` is called tags. And it is divided into two parts, start tag and end tag. For example, `` is a start tag and `` is an end tag. Ui file begins and ends with interface tags. Some tags, for example, object tags can have a class and id attributes inside the start tag. - 2-5: An object with `GtkApplicationWindow` class and `win` id is defined. This is the top level window. And the three properties of the window are defined. `title` property is "file editor", `default-width` property is 400 and `default-height` property is 300. - 6: child tag means a child of the object above. For example, line 7 tells us that GtkBox object which id is "boxv" is a child of `win`. Compare this ui file and the lines 26-58 in the source code of `on_open`. Those two decribe the same structure of widgets. ## GtkBuilder GtkBuilder builds widgets based on the ui file. GtkBuilder *build; build = gtk_builder_new_from_file ("tfe3.ui"); win = GTK_WIDGET (gtk_builder_get_object (build, "win")); gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app)); nb = GTK_WIDGET (gtk_builder_get_object (build, "nb")); The function `gtk_builder_new_from_file` reads the file given as an argument, build the widgets, generate GtkBuilder object and set pointers to the widgets in it. The function `gtk_builder_get_object (build, "win")` returns the pointer to the widget `win`, which is the id in the ui file. All the widgets are connected based on the parent-children relationship described in the ui file. We only need `win` and `nb` for the program after this, so we don't need to take out any other widgets. This reduces lines in the C source file. $ diff tfe2.c tfe3.c 58a59 > GtkBuilder *build; 60,104c61,65 < GtkWidget *boxv; < GtkWidget *boxh; < GtkWidget *dmy1; < GtkWidget *dmy2; < GtkWidget *dmy3; < GtkWidget *btnn; /* button for new */ < GtkWidget *btno; /* button for open */ < GtkWidget *btns; /* button for save */ < GtkWidget *btnc; /* button for close */ < < < win = gtk_application_window_new (GTK_APPLICATION (app)); < gtk_window_set_title (GTK_WINDOW (win), "file editor"); < gtk_window_set_default_size (GTK_WINDOW (win), 600, 400); < < boxv = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0); < gtk_window_set_child (GTK_WINDOW (win), boxv); < < boxh = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0); < gtk_box_append (GTK_BOX (boxv), boxh); < < dmy1 = gtk_label_new(NULL); /* dummy label for left space */ < gtk_label_set_width_chars (GTK_LABEL (dmy1), 10); < dmy2 = gtk_label_new(NULL); /* dummy label for center space */ < gtk_widget_set_hexpand (dmy2, TRUE); < dmy3 = gtk_label_new(NULL); /* dummy label for right space */ < gtk_label_set_width_chars (GTK_LABEL (dmy3), 10); < btnn = gtk_button_new_with_label ("New"); < btno = gtk_button_new_with_label ("Open"); < btns = gtk_button_new_with_label ("Save"); < btnc = gtk_button_new_with_label ("Close"); < < gtk_box_append (GTK_BOX (boxh), dmy1); < gtk_box_append (GTK_BOX (boxh), btnn); < gtk_box_append (GTK_BOX (boxh), btno); < gtk_box_append (GTK_BOX (boxh), dmy2); < gtk_box_append (GTK_BOX (boxh), btns); < gtk_box_append (GTK_BOX (boxh), btnc); < gtk_box_append (GTK_BOX (boxh), dmy3); < < nb = gtk_notebook_new (); < gtk_widget_set_hexpand (nb, TRUE); < gtk_widget_set_vexpand (nb, TRUE); < gtk_box_append (GTK_BOX (boxv), nb); < --- > build = gtk_builder_new_from_file ("tfe3.ui"); > win = GTK_WIDGET (gtk_builder_get_object (build, "win")); > gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app)); > nb = GTK_WIDGET (gtk_builder_get_object (build, "nb")); > g_object_unref(build); 139c100 < app = gtk_application_new ("com.github.ToshioCP.tfe2", G_APPLICATION_HANDLES_OPEN); --- > app = gtk_application_new ("com.github.ToshioCP.tfe3", G_APPLICATION_HANDLES_OPEN); `65,104c61,65` means 40 (=104-65+1) lines change to 5 (=65-61+1) lines. Therefore 35 lines are reduced. Using ui file not only shortens C source files, but also makes the widgets' structure clear. Now I'll show you the C source code `tfe3.c`. Only functions `on_open` are shown as follows. 1 static void 2 on_open (GApplication *app, GFile ** files, gint n_files, gchar *hint, gpointer user_data) { 3 GtkWidget *win; 4 GtkWidget *nb; 5 GtkWidget *lab; 6 GtkNotebookPage *nbp; 7 GtkWidget *scr; 8 GtkWidget *tv; 9 GtkTextBuffer *tb; 10 char *contents; 11 gsize length; 12 char *filename; 13 int i; 14 GtkBuilder *build; 15 16 build = gtk_builder_new_from_file ("tfe3.ui"); 17 win = GTK_WIDGET (gtk_builder_get_object (build, "win")); 18 gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app)); 19 nb = GTK_WIDGET (gtk_builder_get_object (build, "nb")); 20 g_object_unref(build); 21 for (i = 0; i < n_files; i++) { 22 if (g_file_load_contents (files[i], NULL, &contents, &length, NULL, NULL)) { 23 scr = gtk_scrolled_window_new (); 24 tv = tfe_text_view_new (); 25 tb = gtk_text_view_get_buffer (GTK_TEXT_VIEW (tv)); 26 gtk_text_view_set_wrap_mode (GTK_TEXT_VIEW (tv), GTK_WRAP_WORD_CHAR); 27 gtk_scrolled_window_set_child (GTK_SCROLLED_WINDOW (scr), tv); 28 29 tfe_text_view_set_file (TFE_TEXT_VIEW (tv), g_file_dup (files[i])); 30 gtk_text_buffer_set_text (tb, contents, length); 31 g_free (contents); 32 filename = g_file_get_basename (files[i]); 33 lab = gtk_label_new (filename); 34 gtk_notebook_append_page (GTK_NOTEBOOK (nb), scr, lab); 35 nbp = gtk_notebook_get_page (GTK_NOTEBOOK (nb), scr); 36 g_object_set (nbp, "tab-expand", TRUE, NULL); 37 g_free (filename); 38 } else { 39 filename = g_file_get_path (files[i]); 40 g_print ("No such file: %s.\n", filename); 41 g_free (filename); 42 } 43 } 44 if (gtk_notebook_get_n_pages (GTK_NOTEBOOK (nb)) > 0) { 45 gtk_widget_show (win); 46 } else 47 gtk_window_destroy (GTK_WINDOW (win)); 48 } ### Using ui string GtkBuilder can build widgets using string. Use the function gtk\_builder\_new\_from\_string instead of gtk\_builder\_new\_from\_file. char *uistring; uistring = "" "" "file editor" "600" "400" "" "" "GTK_ORIENTATION_VERTICAL" ... ... ... ... ... ... ""; build = gtk_builder_new_from_stringfile (uistring); This method has an advantage and disadvantage. The advantage is that the ui string is written in the source code. So ui file is not necessary on runtime. The disadvantage is that writing C string is a bit bothersome because of the double quotes. If you want to use this method, you should write a script that transforms ui file into C-string. - add backslash before each double quote. - add double quote at the left and right. ### Using Gresource Using Gresource is similar to using string. But Gresource is compressed binary data, not text data. And there's a compiler that compiles ui file into Gresource. It can compile not only text files but also binary files such as images, sounds and so on. And after compilation, it bundles them up into one Gresource object. An xml file is necessary for the resource compiler `glib-compile-resources`. It describes resource files. 1 2 3 4 tfe3.ui 5 6 - 2: gresources tag can include mulitple gresources (gresource tags). However, this xml has only one gresource. - 3: The gresource has a prefix `/com/github/ToshioCP/tfe3`. - 4: The gresource has tfe3.ui. And it is pointed by `/com/github/ToshioCP/tfe3/tfe3.ui` because it needs prefix. If you want to add more files, then insert them between line 4 and 5. Save this xml text to `tfe3.gresource.xml`. The gresource compiler `glib-compile-resources` shows its ussage with the argument `--help`. $ LANG=C glib-compile-resources --help Usage: glib-compile-resources [OPTION?] FILE Compile a resource specification into a resource file. Resource specification files have the extension .gresource.xml, and the resource file have the extension called .gresource. Help Options: -h, --help Show help options Application Options: --version Show program version and exit --target=FILE Name of the output file --sourcedir=DIRECTORY The directories to load files referenced in FILE from (default: current directory) --generate Generate output in the format selected for by the target filename extension --generate-header Generate source header --generate-source Generate source code used to link in the resource file into your code --generate-dependencies Generate dependency list --dependency-file=FILE Name of the dependency file to generate --generate-phony-targets Include phony targets in the generated dependency file --manual-register Don?t automatically create and register resource --internal Don?t export functions; declare them G_GNUC_INTERNAL --external-data Don?t embed resource data in the C file; assume it's linked externally instead --c-name C identifier name used for the generated source code Now run the compiler. $ glib-compile-resources tfe3.gresource.xml --target=resources.c --generate-source Then a C source file `resources.c` is generated. Modify tfe3.c and save it as tfe3_r.c # include "resources.c" ... ... ... ... ... ... build = gtk_builder_new_from_resource ("/com/github/ToshioCP/tfe3/tfe3.ui"); ... ... ... ... ... ... Then, compile and run it. The window appears and it is the same as the screenshot at the beginning of this page. Up: [Readme.md](Readme.md), Prev: [Section 5](sec5.md), Next: [Section 7](sec7.md)