只要我不使用编译器FLAGS -Wall -Wextra
。
但当我把它们打开时,我得到了:
invalid cast from 'GtkButton' to 'GtkEntry'
我无法弄清楚问题出在哪里。这种情况发生在用户输入内容并点击按钮后。
以下是代码:
#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
那么,我的问题是如何解决这个问题?
答案 0 :(得分:2)
首先是一般规则:
您不得更改回调函数的签名! 如果它被定义为有2个参数,那么你永远不会放弃一个参数!
如果您不使用其中一个参数,则只需silence your compiler warning。
(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()
这样的库函数来提供仅在第二个参数中将用户数据作为用户数据的信号。
但是再说一遍:如果你提供回调你自己,我就不会使用这些技巧。