Up: [Readme.md](../Readme.md), Prev: [Section 18](sec18.md), Next: [Section 20](sec20.md) # Ui file for menu and action entries ## Ui file for menu You might have thought that building menus is really bothersome. Yes, the program was complicated and it needs lots of time to code it. The situation is similar to building widgets. When we built widgets, using ui file was a good way to avoid such complicated coding. The same goes for menus. The ui file for menus has interface, menu tags. The file starts and ends with interface tag. ~~~xml ~~~ `menu` tag corresponds to GMenu object. `id` attribute defines the name of the object. It will be referred by GtkBuilder. ~~~xml File New win.new ~~~ `item` tag corresponds to item in GMenu which has the same structure as GMenuItem. The item above has a label attribute. Its value is "New". The item also has an action attribute and its value is "win.new". "win" is a prefix and "new" is an action name. `submenu` tag corresponds to both GMenuItem and GMenu. The GMenuItem has a link to GMenu. The ui file above can be described as follows. ~~~xml File New win.new ~~~ `link` tag expresses the link to submenu. And at the same time it also expresses the submenu itself. This file illustrates the relationship between the menus and items better than the prior ui file. But `submenu` tag is simple and easy to understand. So, we usually prefer the former ui file style. The following is a screenshot of the sample program in this section. Its name is `menu3`. ![menu3](../image/menu3.png) The following is the ui file of the menu in `menu3`. ~~~xml 1 2 3 4 5 File 6
7 8 New 9 win.new 10 11 12 Open 13 win.open 14 15
16
17 18 Save 19 win.save 20 21 22 Save As… 23 win.saveas 24 25
26
27 28 Close 29 win.close 30 31
32
33 34 Quit 35 app.quit 36 37
38
39 40 Edit 41
42 43 Cut 44 win.cut 45 46 47 Copy 48 win.copy 49 50 51 Paste 52 win.paste 53 54
55
56 57 Select All 58 win.selectall 59 60
61
62 63 View 64
65 66 Full Screen 67 win.fullscreen 68 69
70
71
72
~~~ The ui file is converted to the resource by the resource compiler `glib-compile-resouces` with xml file below. ~~~xml 1 2 3 4 menu3.ui 5 6 ~~~ GtkBuilder builds menus from the resource. ~~~C GtkBuilder *builder = gtk_builder_new_from_resource ("/com/github/ToshioCP/menu3/menu3.ui"); GMenuModel *menubar = G_MENU_MODEL (gtk_builder_get_object (builder, "menubar")); gtk_application_set_menubar (GTK_APPLICATION (app), menubar); g_object_unref (builder); ~~~ It is important that `builder` is unreferred after the GMenuModel `menubar` is inserted to the application. If you do it before setting, bad thing will happen -- your computer might freeze. ## Action entry The coding for building actions and signal handlers is bothersome work as well. Therefore, it should be automated. You can implement them easily with GActionEntry structure and `g_action_map_add_action_entries` function. GActionEntry contains action name, signal handlers, parameter and state. ~~~C typedef struct _GActionEntry GActionEntry; struct _GActionEntry { /* action name */ const gchar *name; /* activate handler */ void (* activate) (GSimpleAction *action, GVariant *parameter, gpointer user_data); /* the type of the parameter given as a single GVariant type string */ const gchar *parameter_type; /* initial state given in GVariant text format */ const gchar *state; /* change-state handler */ void (* change_state) (GSimpleAction *action, GVariant *value, gpointer user_data); /*< private >*/ gsize padding[3]; }; ~~~ For example, the actions in the previous section are: ~~~C { "fullscreen", NULL, NULL, "false", fullscreen_changed } { "color", color_activated, "s", "red", NULL } { "quit", quit_activated, NULL, NULL, NULL }, ~~~ And `g_action_map_add_action_entries` does all the process instead of the functions you have needed. ~~~C const GActionEntry app_entries[] = { { "quit", quit_activated, NULL, NULL, NULL } }; g_action_map_add_action_entries (G_ACTION_MAP (app), app_entries, G_N_ELEMENTS (app_entries), app); ~~~ The code above does: - Builds the "quit" action - Connects the action and the "activate" signal handler `quit_activate` - Adds the action to the action map `app`. The same goes for the other actions. ~~~C const GActionEntry win_entries[] = { { "fullscreen", NULL, NULL, "false", fullscreen_changed }, { "color", color_activated, "s", "red", NULL } }; g_action_map_add_action_entries (G_ACTION_MAP (win), win_entries, G_N_ELEMENTS (win_entries), win); ~~~ The code above does: - Builds a "fullscreen" action and "color" action. - Connects the "fullscreen" action and the "change-state" signal handler `fullscreen_changed` - Its initial state is set to FALSE. - Connects the "color" action and the "activate" signal handler `color_activate` - Its parameter type is string and the initial value is "red". - Adds the actions to the action map `win`. ## Example code The C source code of `menu3` and `meson.build` is as follows. ~~~C 1 #include 2 3 static void 4 new_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 5 } 6 7 static void 8 open_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 9 } 10 11 static void 12 save_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 13 } 14 15 static void 16 saveas_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 17 } 18 19 static void 20 close_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 21 } 22 23 static void 24 cut_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 25 } 26 27 static void 28 copy_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 29 } 30 31 static void 32 paste_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 33 } 34 35 static void 36 selectall_activated (GSimpleAction *action, GVariant *parameter, gpointer win) { 37 } 38 39 static void 40 fullscreen_changed (GSimpleAction *action, GVariant *state, gpointer win) { 41 if (g_variant_get_boolean (state)) 42 gtk_window_maximize (GTK_WINDOW (win)); 43 else 44 gtk_window_unmaximize (GTK_WINDOW (win)); 45 g_simple_action_set_state (action, state); 46 } 47 48 static void 49 quit_activated (GSimpleAction *action, GVariant *parameter, gpointer app) 50 { 51 g_application_quit (G_APPLICATION(app)); 52 } 53 54 static void 55 on_activate (GApplication *app, gpointer user_data) { 56 GtkWidget *win = gtk_application_window_new (GTK_APPLICATION (app)); 57 58 const GActionEntry win_entries[] = { 59 { "new", new_activated, NULL, NULL, NULL }, 60 { "open", open_activated, NULL, NULL, NULL }, 61 { "save", save_activated, NULL, NULL, NULL }, 62 { "saveas", saveas_activated, NULL, NULL, NULL }, 63 { "close", close_activated, NULL, NULL, NULL }, 64 { "cut", cut_activated, NULL, NULL, NULL }, 65 { "copy", copy_activated, NULL, NULL, NULL }, 66 { "paste", paste_activated, NULL, NULL, NULL }, 67 { "selectall", selectall_activated, NULL, NULL, NULL }, 68 { "fullscreen", NULL, NULL, "false", fullscreen_changed } 69 }; 70 g_action_map_add_action_entries (G_ACTION_MAP (win), win_entries, G_N_ELEMENTS (win_entries), win); 71 72 gtk_application_window_set_show_menubar (GTK_APPLICATION_WINDOW (win), TRUE); 73 74 gtk_window_set_title (GTK_WINDOW (win), "menu3"); 75 gtk_window_set_default_size (GTK_WINDOW (win), 400, 300); 76 gtk_widget_show (win); 77 } 78 79 static void 80 on_startup (GApplication *app, gpointer user_data) { 81 GtkBuilder *builder = gtk_builder_new_from_resource ("/com/github/ToshioCP/menu3/menu3.ui"); 82 GMenuModel *menubar = G_MENU_MODEL (gtk_builder_get_object (builder, "menubar")); 83 84 gtk_application_set_menubar (GTK_APPLICATION (app), menubar); 85 g_object_unref (builder); 86 87 const GActionEntry app_entries[] = { 88 { "quit", quit_activated, NULL, NULL, NULL } 89 }; 90 g_action_map_add_action_entries (G_ACTION_MAP (app), app_entries, G_N_ELEMENTS (app_entries), app); 91 } 92 93 int 94 main (int argc, char **argv) { 95 GtkApplication *app; 96 int stat; 97 98 app = gtk_application_new ("com.github.ToshioCP.menu3", G_APPLICATION_FLAGS_NONE); 99 g_signal_connect (app, "startup", G_CALLBACK (on_startup), NULL); 100 g_signal_connect (app, "activate", G_CALLBACK (on_activate), NULL); 101 102 stat =g_application_run (G_APPLICATION (app), argc, argv); 103 g_object_unref (app); 104 return stat; 105 } 106 ~~~ meson.build ~~~meson 1 project('menu3', 'c') 2 3 gtkdep = dependency('gtk4') 4 5 gnome=import('gnome') 6 resources = gnome.compile_resources('resources','menu3.gresource.xml') 7 8 sourcefiles=files('menu3.c') 9 10 executable('menu3', sourcefiles, resources, dependencies: gtkdep) ~~~ Up: [Readme.md](../Readme.md), Prev: [Section 18](sec18.md), Next: [Section 20](sec20.md)