擷取選取區域是一個非同步的過程。開始這個過程,呼叫:
gboolean gtk_selection_convert( GtkWidget *widget,
GdkAtom selection,
GdkAtom target,
guint32 time );
|
這個將選取區域內容轉換為target指定的目標形式。如果可能,time參數應該為選取區域被觸發事件產生的時間。這對確認事件以用戶要求它們的順序產生有幫助。然而,如果不行(例如,轉換由 “clicked” 信號觸發),你也可以使用GDK_CURRENT_TIME常數。
當選取區域所有者響應要求時,會發送一個 “selection_received” 信號到你的程式。對應的信號處理函式將得到一個指向GtkSelectionData結構的指標,它的定義如下:
struct _GtkSelectionData
{
GdkAtom selection;
GdkAtom target;
GdkAtom type;
gint format;
guchar *data;
gint length;
};
|
selection和target值是你在gtk_selection_convert()呼叫中給定的。type是一個atom用來辯識選取區域所有者所傳回的資料型態。一些可能值是:”STRING”,由拉丁-1(latin-1)字元組成的字串,”ATOM”,一些atom,”INTEGER”,一個整數,等等。大多數目標只能傳回一種類型。format給定其單位(比如字元)的位元數。通常,接收資料時你不必關心這個。data是一個指向傳回資料的指標,length指定傳回資料的長度,以位元組為單位。如果length是負數,則表示發生錯誤且選取區域的訊息無法獲得。這可能在沒有應用程是擁有選取區域時發生,或者你要求了一個應用程式不支持的目標形式。緩衝區實際上總會比length指示的多一個位元組;增加的這一位元組總為0值,這樣就不必為了使字串以NULL結尾而另外複製一份。
在下面的範例裡,我們擷取特定的目標”TARGETS”,它是一個選取區域內容能被轉換的目標列表。
#include <stdlib.h>
#include <gtk/gtk.h>
void selection_received( GtkWidget *widget,
GtkSelectionData *selection_data,
gpointer data );
/* 當用戶點擊"Get Targets"按鈕時呼叫的訊息處理函式 */
void get_targets( GtkWidget *widget,
gpointer data )
{
static GdkAtom targets_atom = GDK_NONE;
GtkWidget *window = (GtkWidget *)data;
/* 得到"TARGETS"字串對應的atom */
if (targets_atom == GDK_NONE)
targets_atom = gdk_atom_intern ("TARGETS", FALSE);
/* 要求擷取主選取區域的"TARGETS"目標 */
gtk_selection_convert (window, GDK_SELECTION_PRIMARY, targets_atom,
GDK_CURRENT_TIME);
}
/* 當選取區域所有者傳回資料時呼叫的信號處理函式 */
void selection_received( GtkWidget *widget,
GtkSelectionData *selection_data,
gpointer data )
{
GdkAtom *atoms;
GList *item_list;
int i;
/* **** 重要 **** 檢測擷取訊息是否成功 */
if (selection_data->length < 0)
{
g_print ("Selection retrieval failedn");
return;
}
/* 確認得到的資料為原來要求的形式 */
if (selection_data->type != GDK_SELECTION_TYPE_ATOM)
{
g_print ("Selection "TARGETS" was not returned as atoms!n");
return;
}
/* 列印接收到的atom */
atoms = (GdkAtom *)selection_data->data;
item_list = NULL;
for (i = 0; i < selection_data->length / sizeof(GdkAtom); i++)
{
char *name;
name = gdk_atom_name (atoms[i]);
if (name != NULL)
g_print ("%sn",name);
else
g_print ("(bad atom)n");
}
return;
}
int main( int argc,
char *argv[] )
{
GtkWidget *window;
GtkWidget *button;
gtk_init (&argc, &argv);
/* 創建頂層視窗 */
window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
gtk_window_set_title (GTK_WINDOW (window), "Event Box");
gtk_container_set_border_width (GTK_CONTAINER (window), 10);
g_signal_connect (G_OBJECT (window), "destroy",
G_CALLBACK (exit), NULL);
/* 創建一個按鈕,用戶按它以擷取目標 */
button = gtk_button_new_with_label ("Get Targets");
gtk_container_add (GTK_CONTAINER (window), button);
g_signal_connect (G_OBJECT (button), "clicked",
G_CALLBACK (get_targets), window);
g_signal_connect (G_OBJECT (window), "selection_received",
G_CALLBACK (selection_received), NULL);
gtk_widget_show (button);
gtk_widget_show (window);
gtk_main ();
return 0;
}
|
1 則留言