从 'GtkButton' 到 'GtkEntry' 的无效转换
invalid cast from 'GtkButton' to 'GtkEntry'
我在 GTK3
中写了一段代码并且工作正常:
只要我不使用编译器 FLAGS -Wall -Wextra
。
但是当我打开它们时,我得到:
invalid cast from 'GtkButton' to 'GtkEntry'
而且我无法弄清楚问题 is.This 发生在用户键入内容并点击按钮之后。
这是代码:
#include <gtk/gtk.h>
static void btn_clicked(GtkWidget *widget, gpointer data)
{
g_print("%s\n", gtk_entry_get_text(GTK_ENTRY(data)));
gtk_editable_select_region(GTK_EDITABLE(data), 0,-1);
gtk_editable_copy_clipboard(GTK_EDITABLE(data));
}
int main(int argc, char *argv[])
{
GtkWidget *window, *grid;
GtkWidget *label, *label_fn;
GtkWidget *fn_entry;
GtkWidget *button;
//---------- CSS -------------
GtkCssProvider *provider;
GdkDisplay *display;
GdkScreen *screen;
//---------------------------
gtk_init(&argc, &argv);
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(window), "MyApp");
gtk_window_set_default_size(GTK_WINDOW(window), 370, 155);
gtk_window_set_resizable (GTK_WINDOW(window), TRUE);
gtk_container_set_border_width(GTK_CONTAINER(window), 5);
g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL);
/* Create a Grid */
grid = gtk_grid_new();
gtk_container_set_border_width(GTK_CONTAINER (grid), 15);
gtk_widget_set_name(grid, "myGrid");
gtk_grid_set_column_spacing(GTK_GRID(grid), 5);
gtk_grid_set_row_spacing(GTK_GRID(grid), 5);
gtk_container_add(GTK_CONTAINER(window), grid);
/* Create first Label */
label = gtk_label_new("Please enter your Information:");
gtk_widget_set_margin_top(label, 25);
gtk_widget_set_margin_start(label, 85);
/* Create second Label */
label_fn = gtk_label_new("First Name: ");
gtk_widget_set_margin_start(label_fn, 10);
/* Create an Entry: */
fn_entry = gtk_entry_new();
/* Create a Button */
button = gtk_button_new_with_mnemonic("_Write text");
g_signal_connect(button, "clicked", G_CALLBACK(btn_clicked), fn_entry);
g_signal_connect(fn_entry, "activate", G_CALLBACK(btn_clicked), fn_entry);
/* Putting all together */
gtk_grid_attach(GTK_GRID(grid), label, 0, 0, 2, 1);
gtk_grid_attach(GTK_GRID(grid), label_fn, 0, 1, 1, 1);
gtk_grid_attach(GTK_GRID(grid), fn_entry, 1, 1, 1, 1);
gtk_grid_attach(GTK_GRID(grid), button, 1, 2, 1, 1);
// ---------------------------------------------------- CSS -----------------------------------------------------------
provider = gtk_css_provider_new ();
display = gdk_display_get_default ();
screen = gdk_display_get_default_screen (display);
gtk_style_context_add_provider_for_screen (screen, GTK_STYLE_PROVIDER (provider), GTK_STYLE_PROVIDER_PRIORITY_APPLICATION);
const gchar *myCssFile = "mystyle.css";
GError *error = 0;
gtk_css_provider_load_from_file(provider, g_file_new_for_path(myCssFile), &error);
g_object_unref (provider);
// --------------------------------------------------------------------------------------------------------------------
gtk_widget_show_all(window);
gtk_main();
return 0;
}
mystyle.css:
GtkWindow {
background-color: magenta;
color: black;
border-width: 3px;
border-color: blue;
}
#myGrid {
background-color: red;
border-style: solid;
border-color: black;
border-width: 3px;
border-radius: 15px;
border-color: grey;
}
编译:
gcc myApp.c -o myApp `pkg-config --cflags --libs gtk+-3.0 `
工作正常。
但是函数static void btn_clicked(GtkWidget *widget, gpointer data)
有两个参数,使用的只有gpointer data
。
如果我使用 -Wall -Wextra
我得到:
myApp.c: In function ‘btn_clicked’:
myApp.c:3:36: warning: unused parameter ‘widget’ [-Wunused-parameter]
static void btn_clicked(GtkWidget *widget, gpointer data)
^
问题从这里开始。
当我删除参数 GtkWidget *widget
并编译它(编译正常)时,在我 运行 它之后我在那个字段中输入一些东西然后我点击那个按钮(写文本)我得到以下内容:
(myApp:5778): GLib-GObject-WARNING **: invalid cast from 'GtkButton' to 'GtkEntry'
(myApp:5778): Gtk-CRITICAL **: gtk_entry_get_text: assertion 'GTK_IS_ENTRY (entry)' failed
(null)
(myApp:5778): GLib-GObject-WARNING **: invalid cast from 'GtkButton' to 'GtkEditable'
(myApp:5778): Gtk-CRITICAL **: gtk_editable_select_region: assertion 'GTK_IS_EDITABLE (editable)' failed
(myApp:5778): GLib-GObject-WARNING **: invalid cast from 'GtkButton' to 'GtkEditable'
(myApp:5778): Gtk-CRITICAL **: gtk_editable_copy_clipboard: assertion 'GTK_IS_EDITABLE (editable)' failed
那么,我的问题是如何解决这个问题?
首先是一般规则:
您不得更改回调函数的签名!
如果它被定义为有 2 个参数,那么你 永远不会 删除一个参数!
如果您不使用这些参数之一,您可以 。
(void)widget;
您所做的是使用 GtkWidget*
保存小部件的地址并将其视为您的用户数据指针。
为了更清楚地说明执行此操作时发生的情况:
GTK 调用您的回调函数:
button->event_cb(button, userdata);
但是你的函数与签名不匹配:
my_button_cb(void* mydata)
{
// here mydata holds the value from argument button.
// The value from argument userdata is lost!
}
你提到它编译得很好。这并不奇怪。当您调用 g_signal_connect
添加回调时,函数指针的类型不会被保留。编译器不知道签名应该是什么样子,也无法向您抛出错误消息。
这并不意味着您的代码是正确的。
补充说明:
正如 ptomato 指出的那样,可以在列表末尾删除参数。虽然这在技术上是可能的,但我不鼓励使用这种方式。特别是如果您自己提供回调函数。我认为这是不好的做法。
如果您想使用某个库提供的回调函数,情况会有所不同。签名可能不完全匹配,您需要编写一个包装器来使参数匹配。
在这种情况下,我不认为迫切需要添加这个包装器来获得更清晰的代码。
在浏览了一些 GTK 示例之后,我什至找到了一种在参数顺序不匹配的情况下使用回调的方法:g_signal_connect_swapped
。使用此函数,参数以不同的顺序提供给回调。这将允许使用 gtk_widget_destroy()
之类的库函数来提供 window 仅作为第二个参数中的用户数据的信号。
但是再说一次:如果你自己提供回调,我不会使用这种技巧。
我在 GTK3
中写了一段代码并且工作正常:
只要我不使用编译器 FLAGS -Wall -Wextra
。
但是当我打开它们时,我得到:
invalid cast from 'GtkButton' to 'GtkEntry'
而且我无法弄清楚问题 is.This 发生在用户键入内容并点击按钮之后。
这是代码:
#include <gtk/gtk.h>
static void btn_clicked(GtkWidget *widget, gpointer data)
{
g_print("%s\n", gtk_entry_get_text(GTK_ENTRY(data)));
gtk_editable_select_region(GTK_EDITABLE(data), 0,-1);
gtk_editable_copy_clipboard(GTK_EDITABLE(data));
}
int main(int argc, char *argv[])
{
GtkWidget *window, *grid;
GtkWidget *label, *label_fn;
GtkWidget *fn_entry;
GtkWidget *button;
//---------- CSS -------------
GtkCssProvider *provider;
GdkDisplay *display;
GdkScreen *screen;
//---------------------------
gtk_init(&argc, &argv);
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_title(GTK_WINDOW(window), "MyApp");
gtk_window_set_default_size(GTK_WINDOW(window), 370, 155);
gtk_window_set_resizable (GTK_WINDOW(window), TRUE);
gtk_container_set_border_width(GTK_CONTAINER(window), 5);
g_signal_connect(window, "destroy", G_CALLBACK(gtk_main_quit), NULL);
/* Create a Grid */
grid = gtk_grid_new();
gtk_container_set_border_width(GTK_CONTAINER (grid), 15);
gtk_widget_set_name(grid, "myGrid");
gtk_grid_set_column_spacing(GTK_GRID(grid), 5);
gtk_grid_set_row_spacing(GTK_GRID(grid), 5);
gtk_container_add(GTK_CONTAINER(window), grid);
/* Create first Label */
label = gtk_label_new("Please enter your Information:");
gtk_widget_set_margin_top(label, 25);
gtk_widget_set_margin_start(label, 85);
/* Create second Label */
label_fn = gtk_label_new("First Name: ");
gtk_widget_set_margin_start(label_fn, 10);
/* Create an Entry: */
fn_entry = gtk_entry_new();
/* Create a Button */
button = gtk_button_new_with_mnemonic("_Write text");
g_signal_connect(button, "clicked", G_CALLBACK(btn_clicked), fn_entry);
g_signal_connect(fn_entry, "activate", G_CALLBACK(btn_clicked), fn_entry);
/* Putting all together */
gtk_grid_attach(GTK_GRID(grid), label, 0, 0, 2, 1);
gtk_grid_attach(GTK_GRID(grid), label_fn, 0, 1, 1, 1);
gtk_grid_attach(GTK_GRID(grid), fn_entry, 1, 1, 1, 1);
gtk_grid_attach(GTK_GRID(grid), button, 1, 2, 1, 1);
// ---------------------------------------------------- CSS -----------------------------------------------------------
provider = gtk_css_provider_new ();
display = gdk_display_get_default ();
screen = gdk_display_get_default_screen (display);
gtk_style_context_add_provider_for_screen (screen, GTK_STYLE_PROVIDER (provider), GTK_STYLE_PROVIDER_PRIORITY_APPLICATION);
const gchar *myCssFile = "mystyle.css";
GError *error = 0;
gtk_css_provider_load_from_file(provider, g_file_new_for_path(myCssFile), &error);
g_object_unref (provider);
// --------------------------------------------------------------------------------------------------------------------
gtk_widget_show_all(window);
gtk_main();
return 0;
}
mystyle.css:
GtkWindow {
background-color: magenta;
color: black;
border-width: 3px;
border-color: blue;
}
#myGrid {
background-color: red;
border-style: solid;
border-color: black;
border-width: 3px;
border-radius: 15px;
border-color: grey;
}
编译:
gcc myApp.c -o myApp `pkg-config --cflags --libs gtk+-3.0 `
工作正常。
但是函数static void btn_clicked(GtkWidget *widget, gpointer data)
有两个参数,使用的只有gpointer data
。
如果我使用 -Wall -Wextra
我得到:
myApp.c: In function ‘btn_clicked’:
myApp.c:3:36: warning: unused parameter ‘widget’ [-Wunused-parameter]
static void btn_clicked(GtkWidget *widget, gpointer data)
^
问题从这里开始。
当我删除参数 GtkWidget *widget
并编译它(编译正常)时,在我 运行 它之后我在那个字段中输入一些东西然后我点击那个按钮(写文本)我得到以下内容:
(myApp:5778): GLib-GObject-WARNING **: invalid cast from 'GtkButton' to 'GtkEntry'
(myApp:5778): Gtk-CRITICAL **: gtk_entry_get_text: assertion 'GTK_IS_ENTRY (entry)' failed
(null)
(myApp:5778): GLib-GObject-WARNING **: invalid cast from 'GtkButton' to 'GtkEditable'
(myApp:5778): Gtk-CRITICAL **: gtk_editable_select_region: assertion 'GTK_IS_EDITABLE (editable)' failed
(myApp:5778): GLib-GObject-WARNING **: invalid cast from 'GtkButton' to 'GtkEditable'
(myApp:5778): Gtk-CRITICAL **: gtk_editable_copy_clipboard: assertion 'GTK_IS_EDITABLE (editable)' failed
那么,我的问题是如何解决这个问题?
首先是一般规则:
您不得更改回调函数的签名! 如果它被定义为有 2 个参数,那么你 永远不会 删除一个参数!
如果您不使用这些参数之一,您可以
(void)widget;
您所做的是使用 GtkWidget*
保存小部件的地址并将其视为您的用户数据指针。
为了更清楚地说明执行此操作时发生的情况: GTK 调用您的回调函数:
button->event_cb(button, userdata);
但是你的函数与签名不匹配:
my_button_cb(void* mydata)
{
// here mydata holds the value from argument button.
// The value from argument userdata is lost!
}
你提到它编译得很好。这并不奇怪。当您调用 g_signal_connect
添加回调时,函数指针的类型不会被保留。编译器不知道签名应该是什么样子,也无法向您抛出错误消息。
这并不意味着您的代码是正确的。
补充说明:
正如 ptomato 指出的那样,可以在列表末尾删除参数。虽然这在技术上是可能的,但我不鼓励使用这种方式。特别是如果您自己提供回调函数。我认为这是不好的做法。
如果您想使用某个库提供的回调函数,情况会有所不同。签名可能不完全匹配,您需要编写一个包装器来使参数匹配。
在这种情况下,我不认为迫切需要添加这个包装器来获得更清晰的代码。
在浏览了一些 GTK 示例之后,我什至找到了一种在参数顺序不匹配的情况下使用回调的方法:g_signal_connect_swapped
。使用此函数,参数以不同的顺序提供给回调。这将允许使用 gtk_widget_destroy()
之类的库函数来提供 window 仅作为第二个参数中的用户数据的信号。
但是再说一次:如果你自己提供回调,我不会使用这种技巧。