微調按鈕(Spin Button)元件通常用於讓用戶從一個取值範圍裡選擇一個值。它由一個文字輸入框和旁邊的向上和向下兩個按鈕組成。點擊某一個按鈕會讓文字輸入框裡的數值大小在一定範圍裡改變。文字輸入框裡也可以直接輸入一個特定值。
微調按鈕元件允許其中的數值沒有小數位或具有指定的小數位,並且數值可以按一種可配置的方式增加或減小。在按鈕較長時間呈按下狀態時,元件的數值會根據工具按下時間的長短加速變化。
微調按鈕用一個調整物件來維護該按鈕能夠取值的範圍。微調按鈕元件因此而具有了很強大的功能。
下面是創建調整物件的函式。這裡的用意是展示其中所包含的數值的意義:
GtkObject *gtk_adjustment_new( gdouble value, gdouble lower, gdouble upper, gdouble step_increment, gdouble page_increment, gdouble page_size ); |
調整物件的這些屬性在微調按鈕元件中有如下用處:
- value:微調按鈕元件的初始值
- lower:元件允許的最小值
- upper:元件允許的最大值
- step_increment:當滑鼠左鍵按下時元件一次增加/減小的值
- page_increment:當滑鼠右鍵按下時元件一次增加/減小的值
- page_size:沒有用到
另外,當用滑鼠中間鍵點擊按鈕時,可以直接跳到元件的upper或lower值。下面看看怎樣創建一個微調按鈕元件:
GtkWidget *gtk_spin_button_new( GtkAdjustment *adjustment, gdouble climb_rate, guint digits ); |
其中的climb_rate參數是介於0.0和1.0間的值,指明元件數值變化的加速度(長時間按住按鈕,數值會加速變化)。digits參數指定要顯示的值的小數位數。
創建微調按鈕元件之後,還可以用下面的函式對其重新配置:
void gtk_spin_button_configure( GtkSpinButton *spin_button, GtkAdjustment *adjustment, gdouble climb_rate, guint digits ); |
其中spin_button參數就是要重新配置的元件。其它的參數與創建時的意思相同。
使用下面的兩個函數可以設置或獲取元件內部使用的調整物件:
void gtk_spin_button_set_adjustment( GtkSpinButton *spin_button, GtkAdjustment *adjustment ); GtkAdjustment *gtk_spin_button_get_adjustment( GtkSpinButton *spin_button ); |
顯示數值的小數位數可以用下面的函式改變:
void gtk_spin_button_set_digits( GtkSpinButton *spin_button, guint digits) ; |
微調按鈕上當前顯示的數值可以用下面的函式改變:
void gtk_spin_button_set_value( GtkSpinButton *spin_button, gdouble value ); |
微調按鈕元件的當前值可以以浮點數或整數的形式獲得。使用下面的函式:
gdouble gtk_spin_button_get_value ( GtkSpinButton *spin_button ); gint gtk_spin_button_get_value_as_int( GtkSpinButton *spin_button ); |
如果想以當前值為基數改變微調按鈕的值,可以使用下面的函式:
void gtk_spin_button_spin( GtkSpinButton *spin_button, GtkSpinType direction, gdouble increment ); |
其中,direction參數可以取下面的值:
GTK_SPIN_STEP_FORWARD GTK_SPIN_STEP_BACKWARD GTK_SPIN_PAGE_FORWARD GTK_SPIN_PAGE_BACKWARD GTK_SPIN_HOME GTK_SPIN_END GTK_SPIN_USER_DEFINED |
這個函式中包含的一些功能將在下面詳細介紹。其中的許多設置都使用了與微調按鈕元件相關聯的調整物件的值。
GTK_SPIN_STEP_FORWARD和GTK_SPIN_STEP_BACKWARD將元件的值按increment參數指定的數值增大或減小,除非increment參數是0。這種情況下,元件的值將按與其相關聯的調整物件的step_increment值改變。
GTK_SPIN_PAGE_FORWARD和GTK_SPIN_PAGE_BACKWARD只是簡單地按increment參數改變微調按鈕元件的值。
GTK_SPIN_HOME將元件的值設置為相關聯調整物件的範圍的最小值。
GTK_SPIN_END將元件的值設置為相關聯調整物件的範圍的最大值。
GTK_SPIN_USER_DEFINED簡單地按指定的數值改變元件的數值。
介紹了設置和獲取微調按鈕的範圍屬性的函式,下面再介紹影響微調按鈕元件的外觀和行為的函式。
要介紹的第一個函式就是限制微調按鈕元件的文字框只能輸入數值。這樣就阻止了用戶輸入任何非法的字元:
void gtk_spin_button_set_numeric( GtkSpinButton *spin_button, gboolean numeric ); |
可以設置讓微調按鈕元件在upper和lower之間循環。也就是當達到最大值後再向上調整回到最小值,當達到最小值後再向下調整變為最大值。可以用下面的函式實現:
void gtk_spin_button_set_wrap( GtkSpinButton *spin_button, gboolean wrap ); |
可以設置讓微調按鈕元件將其值四捨五入到最接近step_increment的值(在該微調按鈕元件使用的調整物件中設置的)。用下面的函式實現:
void gtk_spin_button_set_snap_to_ticks( GtkSpinButton *spin_button, gboolean snap_to_ticks ); |
微調按鈕元件的更新方式可以用下面的函式改變:
void gtk_spin_button_set_update_policy( GtkSpinButton *spin_button, GtkSpinButtonUpdatePolicy policy ); |
其中policy參數可以取GTK_UPDATE_ALWAYS或GTK_UPDATE_IF_VALID。
這些更新方式影響微調按鈕元件在解析插入文字並將其值與調整物件的值同步時的行為。
在GTK_UPDATE_IF_VALID方式下,微調按鈕元件只有在輸入文字是其調整物件指定範圍裡合法的值時才進行更新,否則文字會被重置為當前的值。
在GTK_UPDATE_ALWAYS方式下,我們將忽略在文本轉換為數值時的錯誤。
最後,可以強行要求微調按鈕元件更新自己:
void gtk_spin_button_update( GtkSpinButton *spin_button ); |
下面是一個使用微調按鈕元件的範例。
#include <stdio.h> #include <gtk/gtk.h> static GtkWidget *spinner1; static void toggle_snap( GtkWidget *widget, GtkSpinButton *spin ) { gtk_spin_button_set_snap_to_ticks (spin, GTK_TOGGLE_BUTTON (widget)->active); } static void toggle_numeric( GtkWidget *widget, GtkSpinButton *spin ) { gtk_spin_button_set_numeric (spin, GTK_TOGGLE_BUTTON (widget)->active); } static void change_digits( GtkWidget *widget, GtkSpinButton *spin ) { gtk_spin_button_set_digits (GTK_SPIN_BUTTON (spinner1), gtk_spin_button_get_value_as_int (spin)); } static void get_value( GtkWidget *widget, gpointer data ) { gchar *buf; GtkLabel *label; GtkSpinButton *spin; spin = GTK_SPIN_BUTTON (spinner1); label = GTK_LABEL (g_object_get_data (G_OBJECT (widget), "user_data")); if (GPOINTER_TO_INT (data) == 1) buf = g_strdup_printf ("%d", gtk_spin_button_get_value_as_int (spin)); else buf = g_strdup_printf ("%0.*f", spin->digits, gtk_spin_button_get_value (spin)); gtk_label_set_text (label, buf); g_free (buf); } int main( int argc, char *argv[] ) { GtkWidget *window; GtkWidget *frame; GtkWidget *hbox; GtkWidget *main_vbox; GtkWidget *vbox; GtkWidget *vbox2; GtkWidget *spinner2; GtkWidget *spinner; GtkWidget *button; GtkWidget *label; GtkWidget *val_label; GtkAdjustment *adj; /* 初始化 */ gtk_init (&argc, &argv); window = gtk_window_new (GTK_WINDOW_TOPLEVEL); g_signal_connect (window, "destroy", G_CALLBACK (gtk_main_quit), NULL); gtk_window_set_title (GTK_WINDOW (window), "Spin Button"); main_vbox = gtk_vbox_new (FALSE, 5); gtk_container_set_border_width (GTK_CONTAINER (main_vbox), 10); gtk_container_add (GTK_CONTAINER (window), main_vbox); frame = gtk_frame_new ("Not accelerated"); gtk_box_pack_start (GTK_BOX (main_vbox), frame, TRUE, TRUE, 0); vbox = gtk_vbox_new (FALSE, 0); gtk_container_set_border_width (GTK_CONTAINER (vbox), 5); gtk_container_add (GTK_CONTAINER (frame), vbox); /* 日、月、年微調器 */ hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (vbox), hbox, TRUE, TRUE, 5); vbox2 = gtk_vbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), vbox2, TRUE, TRUE, 5); label = gtk_label_new ("Day :"); gtk_misc_set_alignment (GTK_MISC (label), 0, 0.5); gtk_box_pack_start (GTK_BOX (vbox2), label, FALSE, TRUE, 0); adj = (GtkAdjustment *) gtk_adjustment_new (1.0, 1.0, 31.0, 1.0, 5.0, 0.0); spinner = gtk_spin_button_new (adj, 0, 0); gtk_spin_button_set_wrap (GTK_SPIN_BUTTON (spinner), TRUE); gtk_box_pack_start (GTK_BOX (vbox2), spinner, FALSE, TRUE, 0); vbox2 = gtk_vbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), vbox2, TRUE, TRUE, 5); label = gtk_label_new ("Month :"); gtk_misc_set_alignment (GTK_MISC (label), 0, 0.5); gtk_box_pack_start (GTK_BOX (vbox2), label, FALSE, TRUE, 0); adj = (GtkAdjustment *) gtk_adjustment_new (1.0, 1.0, 12.0, 1.0, 5.0, 0.0); spinner = gtk_spin_button_new (adj, 0, 0); gtk_spin_button_set_wrap (GTK_SPIN_BUTTON (spinner), TRUE); gtk_box_pack_start (GTK_BOX (vbox2), spinner, FALSE, TRUE, 0); vbox2 = gtk_vbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), vbox2, TRUE, TRUE, 5); label = gtk_label_new ("Year :"); gtk_misc_set_alignment (GTK_MISC (label), 0, 0.5); gtk_box_pack_start (GTK_BOX (vbox2), label, FALSE, TRUE, 0); adj = (GtkAdjustment *) gtk_adjustment_new (1998.0, 0.0, 2100.0, 1.0, 100.0, 0.0); spinner = gtk_spin_button_new (adj, 0, 0); gtk_spin_button_set_wrap (GTK_SPIN_BUTTON (spinner), FALSE); gtk_widget_set_size_request (spinner, 55, -1); gtk_box_pack_start (GTK_BOX (vbox2), spinner, FALSE, TRUE, 0); frame = gtk_frame_new ("Accelerated"); gtk_box_pack_start (GTK_BOX (main_vbox), frame, TRUE, TRUE, 0); vbox = gtk_vbox_new (FALSE, 0); gtk_container_set_border_width (GTK_CONTAINER (vbox), 5); gtk_container_add (GTK_CONTAINER (frame), vbox); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 5); vbox2 = gtk_vbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), vbox2, TRUE, TRUE, 5); label = gtk_label_new ("Value :"); gtk_misc_set_alignment (GTK_MISC (label), 0, 0.5); gtk_box_pack_start (GTK_BOX (vbox2), label, FALSE, TRUE, 0); adj = (GtkAdjustment *) gtk_adjustment_new (0.0, -10000.0, 10000.0, 0.5, 100.0, 0.0); spinner1 = gtk_spin_button_new (adj, 1.0, 2); gtk_spin_button_set_wrap (GTK_SPIN_BUTTON (spinner1), TRUE); gtk_widget_set_size_request (spinner1, 100, -1); gtk_box_pack_start (GTK_BOX (vbox2), spinner1, FALSE, TRUE, 0); vbox2 = gtk_vbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (hbox), vbox2, TRUE, TRUE, 5); label = gtk_label_new ("Digits :"); gtk_misc_set_alignment (GTK_MISC (label), 0, 0.5); gtk_box_pack_start (GTK_BOX (vbox2), label, FALSE, TRUE, 0); adj = (GtkAdjustment *) gtk_adjustment_new (2, 1, 5, 1, 1, 0); spinner2 = gtk_spin_button_new (adj, 0.0, 0); gtk_spin_button_set_wrap (GTK_SPIN_BUTTON (spinner2), TRUE); g_signal_connect (adj, "value_changed", G_CALLBACK (change_digits), spinner2); gtk_box_pack_start (GTK_BOX (vbox2), spinner2, FALSE, TRUE, 0); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 5); button = gtk_check_button_new_with_label ("Snap to 0.5-ticks"); g_signal_connect (button, "clicked", G_CALLBACK (toggle_snap), spinner1); gtk_box_pack_start (GTK_BOX (vbox), button, TRUE, TRUE, 0); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), TRUE); button = gtk_check_button_new_with_label ("Numeric only input mode"); g_signal_connect (button, "clicked", G_CALLBACK (toggle_numeric), spinner1); gtk_box_pack_start (GTK_BOX (vbox), button, TRUE, TRUE, 0); gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (button), TRUE); val_label = gtk_label_new (""); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 5); button = gtk_button_new_with_label ("Value as Int"); g_object_set_data (G_OBJECT (button), "user_data", val_label); g_signal_connect (button, "clicked", G_CALLBACK (get_value), GINT_TO_POINTER (1)); gtk_box_pack_start (GTK_BOX (hbox), button, TRUE, TRUE, 5); button = gtk_button_new_with_label ("Value as Float"); g_object_set_data (G_OBJECT (button), "user_data", val_label); g_signal_connect (button, "clicked", G_CALLBACK (get_value), GINT_TO_POINTER (2)); gtk_box_pack_start (GTK_BOX (hbox), button, TRUE, TRUE, 5); gtk_box_pack_start (GTK_BOX (vbox), val_label, TRUE, TRUE, 0); gtk_label_set_text (GTK_LABEL (val_label), "0"); hbox = gtk_hbox_new (FALSE, 0); gtk_box_pack_start (GTK_BOX (main_vbox), hbox, FALSE, TRUE, 0); button = gtk_button_new_with_label ("Close"); g_signal_connect_swapped (button, "clicked", G_CALLBACK (gtk_widget_destroy), window); gtk_box_pack_start (GTK_BOX (hbox), button, TRUE, TRUE, 5); gtk_widget_show_all (window); /* 進入事件循環 */ gtk_main (); return 0; } |
1 則留言