从 '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 仅作为第二个参数中的用户数据的信号。

但是再说一次:如果你自己提供回调,我不会使用这种技巧。