GTK的Signal不一定得由事件來發出,您可以主動發出Signal,利用g_signal_emit_by_name(),您可以指定一個物件已建立的Signal名稱來發出該Signal。
下面這個程式利用POSIX執行緒(GLib 亦有提供 GThread 來啟用多執行緒),改寫 自訂 callback 函式 中的範例,每秒發出一個GtkButton的"clicked" Signal,程式開始後即使您沒有按下按鈕,也會在終端機下顯示"按鈕按下:哈囉!按鈕!"的訊息:
- signal_emit_demo.c
#include <gtk/gtk.h>
#include <pthread.h>
void *signal_thread(void *arg) {
int i;
for(i = 0; i < 5; i++) {
sleep(1);
g_signal_emit_by_name(arg, "clicked");
}
pthread_exit("Thread exit");
}
// 自訂Callback函式
void button_clicked(GtkWidget *button, gpointer data) {
g_print("按鈕按下:%s\n", (char *) data);
}
int main(int argc, char *argv[]) {
pthread_t a_thread;
GtkWidget *window;
GtkWidget *button;
gtk_init(&argc, &argv);
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(window), "哈囉!GTK+!");
button = gtk_button_new_with_label("按我");
gtk_container_add(GTK_CONTAINER(window), button);
g_signal_connect(GTK_OBJECT(window), "destroy",
G_CALLBACK(gtk_main_quit), NULL);
g_signal_connect(GTK_OBJECT(button), "clicked",
G_CALLBACK(button_clicked), "哈囉!按鈕!");
gtk_widget_show(window);
gtk_widget_show(button);
pthread_create(&a_thread, NULL, signal_thread, button);
gtk_main();
return 0;
}
程式執行後,會使用另一個執行緒,每秒發出一個"clicked" Signal,為了使用POSIX執行緒,編譯這個程式時需要定義_REENTRANT及使用pthread程式庫:
\$ gcc signal_emit_demo.c -o signal_emit_demo -D_REENTRANT -lpthread `pkg-config --cflags --libs gtk+-2.0`
g_signal_emit_by_name()可以發出Signal,如果您想要中止Signal的傳播,則可以使用g_signal_stop_by_name(),例如在某個Signal處理函式處理完畢後,若不想讓其它的Signal處理函式繼續處理了,則可以使用g_signal_stop_by_name()來停止Signal。
若不想使用pthread來撰寫這個程式,則可以考慮使用GLib的 Timeout 。