gpt4 book ai didi

c - [GTK]将 g_signal_connect 中的循环计数器传递给 g_callback

转载 作者:太空宇宙 更新时间:2023-11-04 04:42:33 26 4
gpt4 key购买 nike

刚开始接触GTK+编程,一个星期以来我主要遇到了一个大问题,这里代码解释一下:

for(j=1;j<5;j++)
{
for(i=1;i<6;i++)
{
box[i][j] = gtk_hbox_new(TRUE, 0);
check[i][j]=gtk_check_button_new();
event[i][j]= gtk_event_box_new ();
gtk_container_add (GTK_CONTAINER (event[i][j]),color[i][j]);
gtk_box_pack_start(box[i][j],check[i][j], TRUE, FALSE, 0);
gtk_box_pack_start(box[i][j],event[i][j], TRUE, FALSE, 0);
gtk_widget_set_events (event[i][j], GDK_BUTTON_PRESS_MASK);
gtk_table_attach_defaults(table,box[i][j],j-1, j, i, i+1);
}
}

我有一个表,对于每个 (i,j) 我都有一个包含可点击图像和复选框的框,我想在单击事件时将 check[i][j] 选中/不选中[i][j](可点击的图像)所以信号会像这样

g_signal_connect (GTK_OBJECT(event[i][j]), "button-press-event",G_CALLBACK(events),NULL);

回调函数是这样的

void events(void)
{
void* k;
k= gtk_toggle_button_get_active(check[i][j]);
if(k==FALSE)
{
gtk_toggle_button_set_active(check[i][j],TRUE);
}
else
{
gtk_toggle_button_set_active(check[i][j],FALSE);
}
}

我没有找到如何将 i 和 j 传递给回调,我已经用包含 i 和 j 的结构/选项卡更改了 g_signal_connect 中的 NULL,(还在回调中添加了 gpointer 数据和所有需要的东西)但是没用...我花了很多时间在互联网上搜索,但没有找到我真正需要的东西。到目前为止我找到的唯一解决方案是为每个 i、j 创建 20 个回调函数和 20 个 g_signal_connect,但代码大约需要 270 行......非常感谢您的帮助 :D

更多解释:这是结构:

typedef struct {
int i;
int j;
} t_ig ;
t_ig a;

循环:

for(j=1;j<5;j++)
{
for(i=1;i<6;i++)
{
box[i][j] = gtk_hbox_new(TRUE, 0);
check[i][j]=gtk_check_button_new();
event[i][j]= gtk_event_box_new ();
gtk_container_add (GTK_CONTAINER (event[i][j]),color[i][j]);
tr=event[i][j];
gtk_box_pack_start(box[i][j],check[i][j], TRUE, FALSE, 0);
gtk_box_pack_start(box[i][j],event[i][j], TRUE, FALSE, 0);
gtk_widget_set_events (event[i][j], GDK_BUTTON_PRESS_MASK);
a.i=i;
a.j=j;
g_signal_connect (GTK_OBJECT(tr)), "button-press-event",G_CALLBACK(events),&a);
}

回调

void events(GtkWidget * tr, gpointer userdata)
{
t_ig * d = (t_ig *) userdata ;
printf ("%d",d->i);
}

我通常应该得到我用 a.i=i 初始化的 i;因为我没有得到 i 和 j 的 requiered 值,所以我无法编辑 check[i][j]

编辑:我的问题已经解决了,对于那些可能感兴趣的人来说,这里是代码:

     for(j=1;j<5;j++)
{
for(i=1;i<6;i++)
{
box[i][j] = gtk_hbox_new(TRUE, 0);
check[i][j]=gtk_check_button_new();
event[i][j]= gtk_event_box_new ();
gtk_container_add (GTK_CONTAINER (event[i][j]),color[i][j]);
tr=event[i][j];
gtk_box_pack_start(box[i][j],check[i][j], TRUE, FALSE, 0);
gtk_box_pack_start(box[i][j],event[i][j], TRUE, FALSE, 0);
gtk_widget_set_events (event[i][j], GDK_BUTTON_PRESS_MASK);
g_signal_connect ((GTK_OBJECT(event[i][j])), "button-press-event",G_CALLBACK(events),check[i][j]);
}
}

回调:

void events(GtkWidget * wid,GdkEvent  *event, gpointer userdata)
{

GtkWidget * d=userdata;
void* k;
k= gtk_toggle_button_get_active(d);
if(k==FALSE)
{
gtk_toggle_button_set_active(d,TRUE);
}
else
{
gtk_toggle_button_set_active(d,FALSE);
}
(void) wid;
}

谢谢大家的回答:D

最佳答案

按钮按下事件定义如下:

gboolean
user_function (GtkWidget *widget,
GdkEvent *event,
gpointer user_data)

您的回调必须具有此签名。

第二个问题是您正在重用相同的数据结构。那行不通:您需要为每个不同的结构分配一个结构:例如ti_g *data = g_slice_new0 (ti_g);

但是...如果您只想修改回调中的特定复选框,为什么不直接将指向该复选框的指针作为用户数据指针,如下所示:

gboolean
button_press_event_cb (GtkWidget *widget,
GdkEvent *event,
gpointer user_data)
{
GtkToggleButton *btn = GTK_TOGGLE_BUTTON (user_data);
gboolean active = gtk_toggle_button_get_active (btn);

gtk_toggle_button_set_active (btn, !active);

return TRUE;
}

/* ... in construction code ... */
g_signal_connect (event[i][j], "button-press-event",
G_CALLBACK(button_press_event_cb), check[i][j]);

关于c - [GTK]将 g_signal_connect 中的循环计数器传递给 g_callback,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24675518/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com