GTK+ Использовать один обработчик для нескольких виджетов

У меня есть функция обратного вызова следующим образом:

void handle(GtkWidget *widget, gpointer data) {...}

Поскольку у меня есть много виджетов для этого окна, я хотел бы использовать этот обратный вызов в качестве единственного обработчика, чтобы избежать написания множества небольших функций. Первоначально я думал об использовании enum, который хранится в классе пользовательского интерфейса, который оборачивается вокруг окна, а затем я протестировал бы его следующим образом:

UIClass::Signal signal = (UIClass::Signal) data;
switch (signal) {
  case UIClass::main_button:
    // handle
  case UIClass::check_box:
  ...
}

Но компилятор отклоняет приведение в первой строке этого фрагмента.

Есть ли стандартный способ сделать это? Или GTK+ был разработан, чтобы иметь один обработчик для каждого виджета?

1 ответ

Решение

Сохраните указатель на виджет в классе и передайте весь объект обработчику:

#include <gtk/gtk.h>
#include <iostream>

struct UIClass
{
    GtkWidget* widget1, *widget2, *box;
    UIClass()
    {
        widget1 = gtk_button_new_with_label("button1");
        widget2 = gtk_button_new_with_label("button2");

        box = gtk_hbox_new(true, 10);
        gtk_box_pack_start(GTK_BOX(box), widget1, 0 ,0, 1);
        gtk_box_pack_start(GTK_BOX(box), widget2, 0 ,0, 1);
    }

    static void handle(GtkWidget* sender, UIClass* uiClass)
    {
        if(sender == uiClass->widget1)
            std::cout<<"widget1"<<std::endl;
        else if(sender == uiClass->widget2)
            std::cout<<"widget2"<<std::endl;
        else
            std::cout<<"something else"<<std::endl;
    }

    void connect()
    {
        g_signal_connect(widget1, "clicked", G_CALLBACK(UIClass::handle), this);
        g_signal_connect(widget2, "clicked", G_CALLBACK(UIClass::handle), this);
    }
};

int main(int argc, char *argv[])
{
  gtk_init (&argc, &argv);

  GtkWidget* window = gtk_window_new(GTK_WINDOW_TOPLEVEL);

  UIClass ui;
  ui.connect();

  gtk_container_add(GTK_CONTAINER(window), ui.box);

  gtk_widget_show_all(window);
  gtk_main();

  return 0;
}
Другие вопросы по тегам