2010-12-05 2 views
1

У меня есть следующий пример от here, он показывает это под заголовком «Увеличить - Уменьшить».Gtk + Функции обратного вызова и сигналы помогают

#include <gtk/gtk.h> 

gint count = 0; 
char buf[5]; 

void increase(GtkWidget *widget, gpointer label) 
{ 
    count++; 

    sprintf(buf, "%d", count); 
    gtk_label_set_text(label, buf); 
} 

void decrease(GtkWidget *widget, gpointer label) 
{ 
    count--; 

    sprintf(buf, "%d", count); 
    gtk_label_set_text(label, buf); 
} 

int main(int argc, char** argv) { 

    GtkWidget *label; 
    GtkWidget *window; 
    GtkWidget *frame; 
    GtkWidget *plus; 
    GtkWidget *minus; 

    gtk_init(&argc, &argv); 

    window = gtk_window_new(GTK_WINDOW_TOPLEVEL); 
    gtk_window_set_position(GTK_WINDOW(window), GTK_WIN_POS_CENTER); 
    gtk_window_set_default_size(GTK_WINDOW(window), 250, 180); 
    gtk_window_set_title(GTK_WINDOW(window), "+-"); 

    frame = gtk_fixed_new(); 
    gtk_container_add(GTK_CONTAINER(window), frame); 

    plus = gtk_button_new_with_label("+"); 
    gtk_widget_set_size_request(plus, 80, 35); 
    gtk_fixed_put(GTK_FIXED(frame), plus, 50, 20); 

    minus = gtk_button_new_with_label("-"); 
    gtk_widget_set_size_request(minus, 80, 35); 
    gtk_fixed_put(GTK_FIXED(frame), minus, 50, 80); 

    label = gtk_label_new("0"); 
    gtk_fixed_put(GTK_FIXED(frame), label, 190, 58); 

    gtk_widget_show_all(window); 

    g_signal_connect(window, "destroy", 
     G_CALLBACK (gtk_main_quit), NULL); 

    g_signal_connect(plus, "clicked", 
     G_CALLBACK(increase), label); 

    g_signal_connect(minus, "clicked", 
     G_CALLBACK(decrease), label); 

    gtk_main(); 

    return 0; 
} 

то, что мне интересно, функция g_signal_connect(plus, "clicked",G_CALLBACK(increase), label); передает «метку» для увеличения функции, где ее аргументы void increase(GtkWidget *widget, gpointer label). Теперь в функции увеличения функция gtk_label_set_text() требует в качестве первого аргумента типа данных GtkLabel, но я вижу только переменную GtkWidget и указатель void label в качестве аргументов функции увеличения. Если это так, то как работает gtk_label_set_text() ?.

ответ

3

В C (но не C++) вы можете неявным образом нарисовать void* указателю на любой другой тип. Это очень часто наблюдается при выделении памяти с malloc, которая возвращает void*:

int *myIntArray = malloc(10 * sizeof(int)); // allocate array of 10 ints 

Ваш код делает то же самое, только с параметром мимоходом:

void gtk_label_set_text(GtkLabel *label, const char *text); 
void *label = ...; 
gtk_label_set_text(label, "some string"); // label is implicitly cast from 
              // void* to GtkLabel* 
+0

Спасибо, другое дело, почему бы мы включаем виджет `GtkWidget * 'в функции увеличения ?. – silent 2010-12-05 08:39:53

Смежные вопросы