Невозможно изменить курсор с помощью GTK3

Я пытаюсь динамически изменить курсор окна с помощью GTK3, но gtk_widget_get_parent_window, похоже, не работает.

Может ли кто-нибудь указать, что я делаю неправильно? Спасибо!

// https://developer.gnome.org/gtk3/stable/gtk-getting-started.html
// minimal example
#include <gtk/gtk.h>

static void
activate (GtkApplication* app,
          gpointer        user_data)
{
  GtkWidget *window;

  window = gtk_application_window_new (app);
  gtk_window_set_title (GTK_WINDOW (window), "Window");
  gtk_window_set_default_size (GTK_WINDOW (window), 200, 200);

  // Here \/\/\/\/\/ .
  GdkWindow* w = gtk_widget_get_parent_window(window);
  GdkCursor* c = gdk_cursor_new_for_display(gdk_display_get_default(), GDK_WATCH);
  gdk_window_set_cursor(w, c);
  //      /\/\/\/\/\ .

  gtk_widget_show_all (window);
}

int
main (int    argc,
      char **argv)
{
  GtkApplication *app;
  int status;

  app = gtk_application_new ("org.gtk.example", G_APPLICATION_FLAGS_NONE);
  g_signal_connect (app, "activate", G_CALLBACK (activate), NULL);
  status = g_application_run (G_APPLICATION (app), argc, argv);
  g_object_unref (app);

  return status;
}

(main.exe:16508): Gdk-CRITICAL **: gdk_window_set_cursor: утверждение «GDK_IS_WINDOW (окно)» не удалось

Я использую GTK 3.16 с msys2.

Спасибо заранее.


person I .    schedule 14.02.2016    source источник
comment
GtkWindow вряд ли будет иметь родителя GdkWindow. Однако у него будет собственный GdkWindow, который вы захотите использовать вместо него. Вместо этого используйте gtk_widget_get_window().   -  person andlabs    schedule 14.02.2016
comment
@andlabs Если я переведу код в обратный вызов события реализации и воспользуюсь вашим советом, он будет работать безупречно. Не могли бы вы опубликовать это как ответ, чтобы другие люди могли видеть? Большое спасибо!   -  person I .    schedule 14.02.2016


Ответы (1)


Расширение комментария @andlabs

Любая попытка изменить курсор должна быть сделана после того, как виджет был добавлен в иерархию виджетов или в терминах GTK realized.

Вызов gtk_widget_get_parent_window() или даже gtk_widget_get_window() до того, как для виджета будет запущено событие realize, в обоих случаях приведет к нулевому указателю.

Как и @andlabs, безопаснее использовать gtk_widget_get_window() в сочетании с GtkWindow.

Решение.

static GdkWindow* G_WINDOW = 0;
static GdkCursor* G_CURSOR = 0;

// call after WindowRealize()
void changecursor()
{
     assert(G_WINDOW != NULL);
     gdk_window_set_cursor(G_WINDOW, G_CURSOR);
}

static void WindowRealize(GtkWidget *window, gpointer data)
{
    G_CURSOR_HAND = gdk_cursor_new_for_display(gdk_display_get_default(), GDK_HAND2);
    G_WINDOW = gtk_widget_get_window(window);
}

static void activate(GtkApplication* app,gpointer user_data)
{
    GtkWidget *window = gtk_application_window_new(app);
    ...
    g_signal_connect(window, "realize", G_CALLBACK(WindowRealize), NULL);
    gtk_widget_show_all (window);
}
person I .    schedule 16.02.2016