Usually people write programming code to make an application. What are applications? Applications are software that runs using libraries, which includes the OS, frameworks and so on. In Gtk4 programming, the GtkApplication is a program (or executable) that runs using Gtk libraries.
The basic way to write a GtkApplication is as follows.
That’s all. Very simple. The following is the C code representing the scenario above.
#include <gtk/gtk.h>
int
main (int argc, char **argv) {
GtkApplication *app;
int stat;
app = gtk_application_new ("com.github.ToshioCP.pr1", G_APPLICATION_FLAGS_NONE);
stat =g_application_run (G_APPLICATION (app), argc, argv);
g_object_unref (app);
return stat;
}
The first line says that this program includes the header files of the Gtk libraries. The function main
above is a startup function in C language. The variable app
is defined as a pointer to a GtkApplication instance. The function gtk_application_new
creates a GtkApplication instance and returns a pointer to the instance. The GtkApplication instance is a C structure data in which the information about the application is stored. The meaning of the arguments will be explained later. The function g_application_run
runs an application that the instance defined. (We often say that the function runs app
. Actually, app
is not an application but a pointer to the instance of the application. However, it is simple and short, and probably no confusion occurs.)
To compile this, the following command needs to be run. The string pr1.c
is the filename of the C source code above.
$ gcc `pkg-config --cflags gtk4` pr1.c `pkg-config --libs gtk4`
The C compiler gcc generates an executable file, a.out
. Let’s run it.
$ ./a.out
(a.out:13533): GLib-GIO-WARNING **: 15:30:17.449: Your application does not implement
g_application_activate() and has no handlers connected to the "activate" signal.
It should do one of these.
$
Oh, it just produces an error message. This error message means that the GtkApplication object ran, without a doubt. Now, let’s think about what this message means.
The message tells us that:
g_application_activate()
,These two causes of the error are related to signals. So, I will explain that to you first.
A signal is emitted when something happens. For example, a window is created, a window is destroyed and so on. The signal “activate” is emitted when the application is activated, or started. If the signal is connected to a function, which is called a signal handler or simply handler, then the function is invoked when the signal emits.
The flow is like this:
Signals are defined in objects. For example, the “activate” signal belongs to the GApplication object, which is a parent object of GtkApplication object.
The GApplication object is a child object of the GObject object. GObject is the top object in the hierarchy of all the objects.
GObject -- GApplication -- GtkApplication
<---parent --->child
A child object inherits signals, functions, properties and so on from its parent object. So, GtkApplication also has the “activate” signal.
Now we can solve the problem in pr1.c
. We need to connect the “activate” signal to a handler. We use a function g_signal_connect
which connects a signal to a handler.
#include <gtk/gtk.h>
static void
app_activate (GApplication *app, gpointer *user_data) {
g_print ("GtkApplication is activated.\n");
}
int
main (int argc, char **argv) {
GtkApplication *app;
int stat;
app = gtk_application_new ("com.github.ToshioCP.pr2", G_APPLICATION_FLAGS_NONE);
g_signal_connect (app, "activate", G_CALLBACK (app_activate), NULL);
stat =g_application_run (G_APPLICATION (app), argc, argv);
g_object_unref (app);
return stat;
}
First, we define the handler app_activate
which simply displays a message. In the function main
, we add g_signal_connect
before g_application_run
. The function g_signal_connect
has four arguments.
G_CALLBACK
.You can find the description of each signal in the API reference manual. For example, “activate” signal is in GApplication section in GIO API Reference. The handler function is described in it.
In addition, g_signal_connect
is described in GObject API Reference. API reference manual is very important. You should see and understand it to write Gtk applications. They are located in ‘GTK Documentation’.
Let’s compile the source file above (pr2.c
) and run it.
$ gcc `pkg-config --cflags gtk4` pr2.c `pkg-config --libs gtk4`
$ ./a.out
GtkApplication is activated.
$
OK, well done. However, you may have noticed that it’s painful to type such a long line to compile. It is a good idea to use shell script to solve this problem. Make a text file which contains the following line.
gcc `pkg-config --cflags gtk4` $1.c `pkg-config --libs gtk4`
Then, save it under the directory $HOME/bin, which is usually /home/(username)/bin. (If your user name is James, then the directory is /home/james/bin). And turn on the execute bit of the file. If the filename is comp
, do like this:
$ chmod 755 $HOME/bin/comp
$ ls -log $HOME/bin
... ... ...
-rwxr-xr-x 1 62 May 23 08:21 comp
... ... ...
If this is the first time that you make a $HOME/bin directory and save a file in it, then you need to logout and login again.
$ comp pr2
$ ./a.out
GtkApplication is activated.
$
A message “GtkApplication is activated.” was printed out in the previous subsection. It was good in terms of a test of GtkApplication. However, it is insufficient because Gtk is a framework for graphical user interface (GUI). Now we go ahead with adding a window into this program. What we need to do is:
Now rewrite the function app_activate
.
static void
app_activate (GApplication *app, gpointer user_data) {
GtkWidget *win;
win = gtk_window_new ();
gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app));
gtk_widget_show (win);
}
Widget is an abstract concept that includes all the GUI interfaces such as windows, dialogs, buttons, multi-line text, containers and so on. And GtkWidget is a base object from which all the GUI objects derive.
parent <-----> child
GtkWidget -- GtkWindow
GtkWindow includes GtkWidget at the top of its object.
The function gtk_window_new
is defined as follows.
GtkWidget *void); gtk_window_new (
By this definition, it returns a pointer to GtkWidget, not GtkWindow. It actually creates a new GtkWindow instance (not GtkWidget) but returns a pointer to GtkWidget. However,the pointer points the GtkWidget and at the same time it also points GtkWindow that contains GtkWidget in it.
If you want to use win
as a pointer to the GtkWindow, you need to cast it.
(GtkWindow *) win
Or you can use GTK_WINDOW
macro that performs a similar function.
GTK_WINDOW (win)
This is a recommended way.
The function gtk_window_set_application
is used to connect GtkWindow to GtkApplication.
gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app));
You need to cast win
to GtkWindow and app
to GtkApplication. GTK_WINDOW
and GTK_APPLICATION
macro is appropriate for that.
GtkApplication continues to run until the related window is destroyed. If you didn’t connect GtkWindow and GtkApplication, GtkApplication destroys itself immediately. Because no window is connected to GtkApplication, GtkApplication doesn’t need to wait anything. As it destroys itself, the GtkWindow is also destroyed.
The function gtk_widget_show
is used to show the window.
Gtk4 changes the default widget visibility to on, so every widget doesn’t need this function to show itself. But, there’s an exception. Top window (this term will be explained later) isn’t visible when it is created. So you need to use the function above to show the window.
Save the program as pr3.c
and compile and run it.
$ comp pr3
$ ./a.out
A small window appears.
Click on the close button then the window disappears and the program finishes.
GtkApplicationWindow is a child object of GtkWindow. It has some extra functionality for better integration with GtkApplication. It is recommended to use it instead of GtkWindow when you use GtkApplication.
Now rewrite the program and use GtkAppliction Window.
static void
app_activate (GApplication *app, gpointer user_data) {
GtkWidget *win;
win = gtk_application_window_new (GTK_APPLICATION (app));
gtk_window_set_title (GTK_WINDOW (win), "pr4");
gtk_window_set_default_size (GTK_WINDOW (win), 400, 300);
gtk_widget_show (win);
}
When you create GtkApplicationWindow, you need to give GtkApplication instance as an argument. Then it automatically connect these two instances. So you don’t need to call gtk_window_set_application
any more.
The program sets the title and the default size of the window. Compile it and run a.out
, then you will see a bigger window with its title “pr4”.