GTK のプログラミングはウィジェットと呼ばれる GUI パーツを使って作っていきます。ウィンドウもウィジェットの一つです。そして、さらにその上に(中に)ウィジェットを配置できるコンテナーと呼ばれるウィジェットです。 目に見えるアプリケーションの場合、ウィンドウがその土台になります。
GUI ライブラリである GTK4 にとってウィンドウの存在は非常に大きく、ウィンドウの存在がアプリケーションの起動と終了にリンクしています。
順を追って説明していきます。例として app.c というファイルで作業していきます。
#include <gtk/gtk.h>
int
main (int argc, char **argv)
{
int stat;
g_print ("main\n");
return stat;
}
// ビルド
cc app.c -o app `pkg-config --cflags --libs gtk4`
// 実行
./app
// 結果
main
通常の C プログラミングと変わりありません。g_print 関数を使うために <gtk/gtk.h> を読み込んでいます。 <gtk/gtk.h> には GTK に必要な 関数、定数、マクロが宣言されています。
GTK4 プログラミングでの main 関数の役割は、GUI アプリケーションの起動と終了を実質管理している gtk_application のオブジェクトを作成して、そのオブジェクトを実行することです。
#include <gtk/gtk.h>
int
main (int argc, char **argv)
{
GtkApplication *app;
int stat;
// アプリケーションオブジェクトの作成
app = gtk_application_new ("com.usodesu.app", G_APPLICATION_DEFAULT_FLAGS);
// アプリケーションオブジェクトの実行
stat = g_application_run (G_APPLICATION (app), argc, argv);
g_print ("main\n");
return stat;
}
// ビルド
cc app.c -o app `pkg-config --cflags --libs gtk4`
// 実行
./app
// 結果
Your application does not implement g_application_activate() and has no handlers connected to the 'activate' signal.
It should do one of these.
アプリケーションには g_application_activate 関数がなく、activate 信号に接続されたハンドルもありません。
これらの一つを行うべきです。
指示のとおりに activate 関数を作って、信号を接続します。
なお、ここで重要なのは g_application_run 関数の第一引数です。ここにはアプリケーションの識別子となる文字列を与えますが、必ず、 逆 URL 方式にしなければなりません。逆 URL 方式でない場合は、ビルドエラーになります。
#include <gtk/gtk.h>
static void
activate (GApplication *app, gpointer data)
{
g_print ("activate\n");
}
int
main (int argc, char **argv)
{
g_print ("main\n");
GtkApplication *app;
int stat;
app = gtk_application_new ("com.usodesu.app", G_APPLICATION_DEFAULT_FLAGS);
g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
stat = g_application_run (G_APPLICATION (app), argc, argv);
g_print ("main\n");
return stat;
}
// ビルド
cc app.c -o app `pkg-config --cflags --libs gtk4`
// 実行
./app
// 結果
main
activate
main
main 関数から activate 関数を呼び出して、main 関数に戻り、プログラムを終了していることがわかります。
activate 関数には、アプリケーションの振る舞いや見た目を設定していきます。
#include <gtk/gtk.h>
static void
activate (GApplication *app, gpointer data)
{
g_print ("activate\n");
GtkWidget * win;
win = gtk_window_new ();
g_print ("activate\n");
}
int
main (int argc, char **argv)
{
g_print ("main\n");
GtkApplication *app;
int stat;
app = gtk_application_new ("com.usodesu.app", G_APPLICATION_DEFAULT_FLAGS);
g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
stat = g_application_run (G_APPLICATION (app), argc, argv);
g_print ("main\n");
return stat;
}
// ビルド
cc app.c -o app `pkg-config --cflags --libs gtk4`
// 実行
./app
// 結果
main
activate
activate
main
activate 関数内でウィンドウは作られたみたいですが、すぐに main に戻ってプログラムを終了しています。 アプリケーションを起動し続けるには、ウィンドウと application オブジェクトをリンクさせる必要があります。
#include <gtk/gtk.h>
static void
activate (GApplication *app, gpointer data)
{
g_print ("activate\n");
GtkWidget * win;
win = gtk_window_new ();
gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app));
g_print ("activate\n");
}
int
main (int argc, char **argv)
{
g_print ("main\n");
GtkApplication *app;
int stat;
app = gtk_application_new ("com.usodesu.app", G_APPLICATION_DEFAULT_FLAGS);
g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
stat = g_application_run (G_APPLICATION (app), argc, argv);
g_print ("main\n");
return stat;
}
// ビルド
cc app.c -o app `pkg-config --cflags --libs gtk4`
// 実行
./app
// 結果
main
activate
activate
// 何も表示されずに起動したままになるので、Control + C とキー入力して終了させる
^C
ウィンドウと application オブジェクトがリンクされ、アプリケーションは起動し続けますが、ウィンドウの表示を指定しなければ、 ウインドウは表示されません。
#include <gtk/gtk.h>
static void
activate (GApplication *app, gpointer data)
{
g_print ("activate\n");
GtkWidget * win;
win = gtk_window_new ();
gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app));
gtk_window_present (GTK_WINDOW (win));
g_print ("activate\n");
}
int
main (int argc, char **argv)
{
g_print ("main\n");
GtkApplication *app;
int stat;
app = gtk_application_new ("com.usodesu.app", G_APPLICATION_DEFAULT_FLAGS);
g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
stat = g_application_run (G_APPLICATION (app), argc, argv);
g_print ("main\n");
return stat;
}
// ビルド
cc app.c -o app `pkg-config --cflags --libs gtk4`
// 実行
./app
// 結果
main
activate
activate
// ウィンドウが表示され続けます。ウィンドウをクローズさせると、main 関数に戻り、アプリケーションは終了します
main
ウィンドウと application オブジェクトがリンクしているので、ウィンドウを閉じると、application オブジェクトも終了します。 そして main 関数に戻りプログラムは終了します。
ウィンドウのタイトルを設定していない場合は次のようになります
なお、ウィンドウの作成と、作成したウィンドウを application オブジェクトとリンクさせるのは一つの関数でできます。 通常はこちらを使います。
win = gtk_window_new ();
gtk_window_set_application (GTK_WINDOW (win), GTK_APPLICATION (app));
↓
win = gtk_application_window_new (GTK_APPLICATION (app));
以下が完成した、ミニマルなアプリケーションです。
#include <gtk/gtk.h>
static void
activate (GApplication *app, gpointer data)
{
GtkWidget *win = gtk_application_window_new (GTK_APPLICATION (app));
gtk_window_present (GTK_WINDOW (win));
}
int
main (int argc, char **argv)
{
GtkApplication *app;
int stat;
app = gtk_application_new ("jp.vivacocoa.window", G_APPLICATION_DEFAULT_FLAGS);
g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
stat = g_application_run (G_APPLICATION (app), argc, argv);
g_object_unref (app);
return stat;
}
ウィンドウのサイズを指定していない場合は次のようになります
ウィンドウのタイトルとサイズを指定してみました
以下のファイルを window.c という名前で記述してください。
#include <gtk/gtk.h>
static void
activate(GtkApplication *app,
gpointer user_data)
{
GtkWidget *window;
window = gtk_application_window_new(app);
gtk_window_set_title(GTK_WINDOW(window),"window");
gtk_window_set_default_size(GTK_WINDOW(window),600,400);
gtk_window_present(GTK_WINDOW(window));
}
int
main(int argc,
char **argv)
{
GtkApplication *app;
int status;
app = gtk_application_new("jp.vivacocoa.window", G_APPLICATION_DEFAULT_FLAGS);
g_signal_connect(app, "activate",G_CALLBACK(activate),NULL);
status = g_application_run(G_APPLICATION(app),argc,argv);
g_object_unref(app);
return status;
}
以下のとおりです。Windows の場合は通常版でも ARM64版 でも mingw64 で行ってください。
// コンパイル
cc `pkg-config --cflags gtk4` window.c -o window `pkg-config --libs gtk4`
もしくは
gcc $(pkg-config --cflags gtk4) -o window window.c $(pkg-config --libs gtk4)
もしくは
cc window.c -o window `pkg-config --cflags --libs gtk4`
// 実行
./window