GTK的Signal不一定得由事件來發出,您可以主動發出Signal,利用g_signal_emit_by_name(),您可以指定一個物件已建立的Signal名稱來發出該Signal。
下面這個程式利用POSIX執行緒(GLib 亦有提供 GThread 來啟用多執行緒),改寫 自訂 callback 函式 中的範例,每秒發出一個GtkButton的"clicked" Signal,程式開始後即使您沒有按下按鈕,也會在終端機下顯示"按鈕按下:哈囉!按鈕!"的訊息:
#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 。
|
|