內建 Signal 的發射與停止


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