Gtk + Callback функции и сигналы помогают - PullRequest
1 голос
/ 05 декабря 2010

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

#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 и пустой указатель label в качестве аргументов функции увеличения. Если это так, как работает gtk_label_set_text ()?.

1 Ответ

3 голосов
/ 05 декабря 2010

В 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*
...