У меня есть функция обратного вызова следующим образом:
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 + был разработан, чтобы иметь один обработчик для каждого виджета?
Сохраните указатель на виджет в классе и передайте весь объект обработчику:
#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;
}
Других решений пока нет …