forked from AuroraMiddleware/gtk
aa43d97a80
If a grab is held on a toplevel surface tree, and events happen on a different surface tree from another toplevel/window group, we rewrite these events so they look like generated on the window group that holds the grab, but it missed that coordinates would fail to be translated, so these would stay unchanged and "pointing" to random parts of the toplevel that is holding the grab and handling the events. Since off-surface coordinates are not specially meaningful, and in fact impossible to obtain in some backends, just fake the coordinates making it sure that all rewritten events point outside the surface. The grabbing window will still handle the events, but the coordinates in these will be harmlessly moot. Fixes: https://gitlab.gnome.org/GNOME/gtk/-/issues/4760
1975 lines
58 KiB
C
1975 lines
58 KiB
C
/* GTK - The GIMP Toolkit
|
||
* Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
|
||
*
|
||
* This library is free software; you can redistribute it and/or
|
||
* modify it under the terms of the GNU Lesser General Public
|
||
* License as published by the Free Software Foundation; either
|
||
* version 2 of the License, or (at your option) any later version.
|
||
*
|
||
* This library is distributed in the hope that it will be useful,
|
||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||
* Lesser General Public License for more details.
|
||
*
|
||
* You should have received a copy of the GNU Lesser General Public
|
||
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
|
||
/*
|
||
* Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
|
||
* file for a list of people on the GTK+ Team. See the ChangeLog
|
||
* files for a list of changes. These files are distributed with
|
||
* GTK+ at ftp://ftp.gtk.org/pub/gtk/.
|
||
*/
|
||
|
||
#include "config.h"
|
||
|
||
#include "gdk/gdk.h"
|
||
#include "gdk/gdk-private.h"
|
||
#include "gdk/gdkprofilerprivate.h"
|
||
#include "gsk/gskprivate.h"
|
||
#include "gsk/gskrendernodeprivate.h"
|
||
#include "gtknative.h"
|
||
|
||
#include <locale.h>
|
||
|
||
#include <stdio.h>
|
||
#include <stdlib.h>
|
||
#include <string.h>
|
||
#ifdef HAVE_UNISTD_H
|
||
#include <unistd.h>
|
||
#endif
|
||
#include <sys/types.h> /* For uid_t, gid_t */
|
||
|
||
#ifdef G_OS_WIN32
|
||
#define STRICT
|
||
#include <windows.h>
|
||
#undef STRICT
|
||
#endif
|
||
|
||
#include "gtkintl.h"
|
||
|
||
#include "gtkbox.h"
|
||
#include "gtkdebug.h"
|
||
#include "gtkdropprivate.h"
|
||
#include "gtkmain.h"
|
||
#include "gtkmediafileprivate.h"
|
||
#include "gtkmodulesprivate.h"
|
||
#include "gtkprivate.h"
|
||
#include "gtkrecentmanager.h"
|
||
#include "gtksettingsprivate.h"
|
||
#include "gtktooltipprivate.h"
|
||
#include "gtkwidgetprivate.h"
|
||
#include "gtkwindowprivate.h"
|
||
#include "gtkwindowgroup.h"
|
||
#include "gtkprintbackendprivate.h"
|
||
#include "gtkimmoduleprivate.h"
|
||
#include "gtkroot.h"
|
||
#include "gtknative.h"
|
||
#include "gtkpopcountprivate.h"
|
||
|
||
#include "inspector/init.h"
|
||
#include "inspector/window.h"
|
||
|
||
#include "gdk/gdkeventsprivate.h"
|
||
#include "gdk/gdksurfaceprivate.h"
|
||
|
||
#define GDK_ARRAY_ELEMENT_TYPE GtkWidget *
|
||
#define GDK_ARRAY_TYPE_NAME GtkWidgetStack
|
||
#define GDK_ARRAY_NAME gtk_widget_stack
|
||
#define GDK_ARRAY_FREE_FUNC g_object_unref
|
||
#define GDK_ARRAY_PREALLOC 16
|
||
#include "gdk/gdkarrayimpl.c"
|
||
|
||
static GtkWindowGroup *gtk_main_get_window_group (GtkWidget *widget);
|
||
|
||
static int pre_initialized = FALSE;
|
||
static int gtk_initialized = FALSE;
|
||
static GList *current_events = NULL;
|
||
|
||
typedef struct {
|
||
GdkDisplay *display;
|
||
guint flags;
|
||
} DisplayDebugFlags;
|
||
|
||
#define N_DEBUG_DISPLAYS 4
|
||
|
||
DisplayDebugFlags debug_flags[N_DEBUG_DISPLAYS];
|
||
/* This is a flag to speed up development builds. We set it to TRUE when
|
||
* any of the debug displays has debug flags >0, but we never set it back
|
||
* to FALSE. This way we don't need to call gtk_widget_get_display() in
|
||
* hot paths. */
|
||
gboolean any_display_debug_flags_set = FALSE;
|
||
|
||
GtkDebugFlags
|
||
gtk_get_display_debug_flags (GdkDisplay *display)
|
||
{
|
||
int i;
|
||
|
||
if (display == NULL)
|
||
display = gdk_display_get_default ();
|
||
|
||
for (i = 0; i < N_DEBUG_DISPLAYS; i++)
|
||
{
|
||
if (debug_flags[i].display == display)
|
||
return (GtkDebugFlags)debug_flags[i].flags;
|
||
}
|
||
|
||
return 0;
|
||
}
|
||
|
||
gboolean
|
||
gtk_get_any_display_debug_flag_set (void)
|
||
{
|
||
return any_display_debug_flags_set;
|
||
}
|
||
|
||
void
|
||
gtk_set_display_debug_flags (GdkDisplay *display,
|
||
GtkDebugFlags flags)
|
||
{
|
||
int i;
|
||
|
||
for (i = 0; i < N_DEBUG_DISPLAYS; i++)
|
||
{
|
||
if (debug_flags[i].display == NULL)
|
||
debug_flags[i].display = display;
|
||
|
||
if (debug_flags[i].display == display)
|
||
{
|
||
debug_flags[i].flags = flags;
|
||
if (flags > 0)
|
||
any_display_debug_flags_set = TRUE;
|
||
|
||
return;
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* gtk_get_debug_flags:
|
||
*
|
||
* Returns the GTK debug flags that are currently active.
|
||
*
|
||
* This function is intended for GTK modules that want
|
||
* to adjust their debug output based on GTK debug flags.
|
||
*
|
||
* Returns: the GTK debug flags.
|
||
*/
|
||
GtkDebugFlags
|
||
gtk_get_debug_flags (void)
|
||
{
|
||
if (gtk_get_any_display_debug_flag_set ())
|
||
return gtk_get_display_debug_flags (gdk_display_get_default ());
|
||
|
||
return 0;
|
||
}
|
||
|
||
/**
|
||
* gtk_set_debug_flags:
|
||
* @flags: the debug flags to set
|
||
*
|
||
* Sets the GTK debug flags.
|
||
*/
|
||
void
|
||
gtk_set_debug_flags (GtkDebugFlags flags)
|
||
{
|
||
gtk_set_display_debug_flags (gdk_display_get_default (), flags);
|
||
}
|
||
|
||
gboolean
|
||
gtk_simulate_touchscreen (void)
|
||
{
|
||
return (gtk_get_debug_flags () & GTK_DEBUG_TOUCHSCREEN) != 0;
|
||
}
|
||
|
||
static const GdkDebugKey gtk_debug_keys[] = {
|
||
{ "keybindings", GTK_DEBUG_KEYBINDINGS, "Information about keyboard shortcuts" },
|
||
{ "modules", GTK_DEBUG_MODULES, "Information about modules and extensions" },
|
||
{ "icontheme", GTK_DEBUG_ICONTHEME, "Information about icon themes" },
|
||
{ "printing", GTK_DEBUG_PRINTING, "Information about printing" },
|
||
{ "geometry", GTK_DEBUG_GEOMETRY, "Information about size allocation" },
|
||
{ "size-request", GTK_DEBUG_SIZE_REQUEST, "Information about size requests" },
|
||
{ "actions", GTK_DEBUG_ACTIONS, "Information about actions and menu models" },
|
||
{ "constraints", GTK_DEBUG_CONSTRAINTS, "Information from the constraints solver" },
|
||
{ "text", GTK_DEBUG_TEXT, "Information about GtkTextView" },
|
||
{ "tree", GTK_DEBUG_TREE, "Information about GtkTreeView" },
|
||
{ "layout", GTK_DEBUG_LAYOUT, "Information from layout managers" },
|
||
{ "builder", GTK_DEBUG_BUILDER, "Trace GtkBuilder operation" },
|
||
{ "builder-objects", GTK_DEBUG_BUILDER_OBJECTS, "Log unused GtkBuilder objects" },
|
||
{ "no-css-cache", GTK_DEBUG_NO_CSS_CACHE, "Disable style property cache" },
|
||
{ "interactive", GTK_DEBUG_INTERACTIVE, "Enable the GTK inspector", TRUE },
|
||
{ "touchscreen", GTK_DEBUG_TOUCHSCREEN, "Pretend the pointer is a touchscreen" },
|
||
{ "snapshot", GTK_DEBUG_SNAPSHOT, "Generate debug render nodes" },
|
||
{ "accessibility", GTK_DEBUG_A11Y, "Information about accessibility state changes" },
|
||
{ "iconfallback", GTK_DEBUG_ICONFALLBACK, "Information about icon fallback" },
|
||
};
|
||
|
||
/* This checks to see if the process is running suid or sgid
|
||
* at the current time. If so, we don’t allow GTK to be initialized.
|
||
* This is meant to be a mild check - we only error out if we
|
||
* can prove the programmer is doing something wrong, not if
|
||
* they could be doing something wrong. For this reason, we
|
||
* don’t use issetugid() on BSD or prctl (PR_GET_DUMPABLE).
|
||
*/
|
||
static gboolean
|
||
check_setugid (void)
|
||
{
|
||
/* this isn't at all relevant on MS Windows and doesn't compile ... --hb */
|
||
#ifndef G_OS_WIN32
|
||
uid_t ruid, euid, suid; /* Real, effective and saved user ID's */
|
||
gid_t rgid, egid, sgid; /* Real, effective and saved group ID's */
|
||
|
||
#ifdef HAVE_GETRESUID
|
||
if (getresuid (&ruid, &euid, &suid) != 0 ||
|
||
getresgid (&rgid, &egid, &sgid) != 0)
|
||
#endif /* HAVE_GETRESUID */
|
||
{
|
||
suid = ruid = getuid ();
|
||
sgid = rgid = getgid ();
|
||
euid = geteuid ();
|
||
egid = getegid ();
|
||
}
|
||
|
||
if (ruid != euid || ruid != suid ||
|
||
rgid != egid || rgid != sgid)
|
||
{
|
||
g_warning ("This process is currently running setuid or setgid.\n"
|
||
"This is not a supported use of GTK. You must create a helper\n"
|
||
"program instead. For further details, see:\n\n"
|
||
" http://www.gtk.org/setuid.html\n\n"
|
||
"Refusing to initialize GTK.");
|
||
exit (1);
|
||
}
|
||
#endif
|
||
return TRUE;
|
||
}
|
||
|
||
static gboolean do_setlocale = TRUE;
|
||
|
||
/**
|
||
* gtk_disable_setlocale:
|
||
*
|
||
* Prevents [id@gtk_init] and [id@gtk_init_check] from automatically calling
|
||
* `setlocale (LC_ALL, "")`.
|
||
*
|
||
* You would want to use this function if you wanted to set the locale for
|
||
* your program to something other than the user’s locale, or if
|
||
* you wanted to set different values for different locale categories.
|
||
*
|
||
* Most programs should not need to call this function.
|
||
**/
|
||
void
|
||
gtk_disable_setlocale (void)
|
||
{
|
||
if (pre_initialized)
|
||
g_warning ("gtk_disable_setlocale() must be called before gtk_init()");
|
||
|
||
do_setlocale = FALSE;
|
||
}
|
||
|
||
#ifdef G_PLATFORM_WIN32
|
||
#undef gtk_init_check
|
||
#endif
|
||
|
||
#ifdef G_OS_WIN32
|
||
|
||
static char *iso639_to_check = NULL;
|
||
static char *iso3166_to_check = NULL;
|
||
static char *script_to_check = NULL;
|
||
static gboolean setlocale_called = FALSE;
|
||
|
||
static BOOL CALLBACK
|
||
enum_locale_proc (LPTSTR locale)
|
||
{
|
||
LCID lcid;
|
||
char iso639[10];
|
||
char iso3166[10];
|
||
char *endptr;
|
||
|
||
|
||
lcid = strtoul (locale, &endptr, 16);
|
||
if (*endptr == '\0' &&
|
||
GetLocaleInfo (lcid, LOCALE_SISO639LANGNAME, iso639, sizeof (iso639)) &&
|
||
GetLocaleInfo (lcid, LOCALE_SISO3166CTRYNAME, iso3166, sizeof (iso3166)))
|
||
{
|
||
if (strcmp (iso639, iso639_to_check) == 0 &&
|
||
((iso3166_to_check != NULL &&
|
||
strcmp (iso3166, iso3166_to_check) == 0) ||
|
||
(iso3166_to_check == NULL &&
|
||
SUBLANGID (LANGIDFROMLCID (lcid)) == SUBLANG_DEFAULT)))
|
||
{
|
||
char language[100], country[100];
|
||
|
||
if (script_to_check != NULL)
|
||
{
|
||
/* If lcid is the "other" script for this language,
|
||
* return TRUE, i.e. continue looking.
|
||
*/
|
||
if (strcmp (script_to_check, "Latn") == 0)
|
||
{
|
||
switch (LANGIDFROMLCID (lcid))
|
||
{
|
||
case MAKELANGID (LANG_AZERI, SUBLANG_AZERI_CYRILLIC):
|
||
return TRUE;
|
||
case MAKELANGID (LANG_UZBEK, SUBLANG_UZBEK_CYRILLIC):
|
||
return TRUE;
|
||
case MAKELANGID (LANG_SERBIAN, SUBLANG_SERBIAN_CYRILLIC):
|
||
return TRUE;
|
||
case MAKELANGID (LANG_SERBIAN, 0x07):
|
||
/* Serbian in Bosnia and Herzegovina, Cyrillic */
|
||
return TRUE;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
else if (strcmp (script_to_check, "Cyrl") == 0)
|
||
{
|
||
switch (LANGIDFROMLCID (lcid))
|
||
{
|
||
case MAKELANGID (LANG_AZERI, SUBLANG_AZERI_LATIN):
|
||
return TRUE;
|
||
case MAKELANGID (LANG_UZBEK, SUBLANG_UZBEK_LATIN):
|
||
return TRUE;
|
||
case MAKELANGID (LANG_SERBIAN, SUBLANG_SERBIAN_LATIN):
|
||
return TRUE;
|
||
case MAKELANGID (LANG_SERBIAN, 0x06):
|
||
/* Serbian in Bosnia and Herzegovina, Latin */
|
||
return TRUE;
|
||
default:
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
SetThreadLocale (lcid);
|
||
|
||
if (GetLocaleInfo (lcid, LOCALE_SENGLANGUAGE, language, sizeof (language)) &&
|
||
GetLocaleInfo (lcid, LOCALE_SENGCOUNTRY, country, sizeof (country)))
|
||
{
|
||
char str[300];
|
||
|
||
strcpy (str, language);
|
||
strcat (str, "_");
|
||
strcat (str, country);
|
||
|
||
if (setlocale (LC_ALL, str) != NULL)
|
||
setlocale_called = TRUE;
|
||
}
|
||
|
||
return FALSE;
|
||
}
|
||
}
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
#endif
|
||
|
||
void
|
||
setlocale_initialization (void)
|
||
{
|
||
static gboolean initialized = FALSE;
|
||
|
||
if (initialized)
|
||
return;
|
||
initialized = TRUE;
|
||
|
||
if (do_setlocale)
|
||
{
|
||
#ifdef G_OS_WIN32
|
||
/* If some of the POSIXish environment variables are set, set
|
||
* the Win32 thread locale correspondingly.
|
||
*/
|
||
char *p = getenv ("LC_ALL");
|
||
if (p == NULL)
|
||
p = getenv ("LANG");
|
||
|
||
if (p != NULL)
|
||
{
|
||
p = g_strdup (p);
|
||
if (strcmp (p, "C") == 0)
|
||
SetThreadLocale (LOCALE_SYSTEM_DEFAULT);
|
||
else
|
||
{
|
||
/* Check if one of the supported locales match the
|
||
* environment variable. If so, use that locale.
|
||
*/
|
||
iso639_to_check = p;
|
||
iso3166_to_check = strchr (iso639_to_check, '_');
|
||
if (iso3166_to_check != NULL)
|
||
{
|
||
*iso3166_to_check++ = '\0';
|
||
|
||
script_to_check = strchr (iso3166_to_check, '@');
|
||
if (script_to_check != NULL)
|
||
*script_to_check++ = '\0';
|
||
|
||
/* Handle special cases. */
|
||
|
||
/* The standard code for Serbia and Montenegro was
|
||
* "CS", but MSFT uses for some reason "SP". By now
|
||
* (October 2006), SP has split into two, "RS" and
|
||
* "ME", but don't bother trying to handle those
|
||
* yet. Do handle the even older "YU", though.
|
||
*/
|
||
if (strcmp (iso3166_to_check, "CS") == 0 ||
|
||
strcmp (iso3166_to_check, "YU") == 0)
|
||
iso3166_to_check = (char *) "SP";
|
||
}
|
||
else
|
||
{
|
||
script_to_check = strchr (iso639_to_check, '@');
|
||
if (script_to_check != NULL)
|
||
*script_to_check++ = '\0';
|
||
/* LANG_SERBIAN == LANG_CROATIAN, recognize just "sr" */
|
||
if (strcmp (iso639_to_check, "sr") == 0)
|
||
iso3166_to_check = (char *) "SP";
|
||
}
|
||
|
||
EnumSystemLocales (enum_locale_proc, LCID_SUPPORTED);
|
||
}
|
||
g_free (p);
|
||
}
|
||
if (!setlocale_called)
|
||
setlocale (LC_ALL, "");
|
||
#else
|
||
if (!setlocale (LC_ALL, ""))
|
||
g_warning ("Locale not supported by C library.\n\tUsing the fallback 'C' locale.");
|
||
#endif
|
||
}
|
||
}
|
||
|
||
/* Return TRUE if module_to_check causes version conflicts.
|
||
* If module_to_check is NULL, check the main module.
|
||
*/
|
||
static gboolean
|
||
_gtk_module_has_mixed_deps (GModule *module_to_check)
|
||
{
|
||
GModule *module;
|
||
gpointer func;
|
||
gboolean result;
|
||
|
||
if (!module_to_check)
|
||
module = g_module_open (NULL, 0);
|
||
else
|
||
module = module_to_check;
|
||
|
||
if (g_module_symbol (module, "gtk_progress_get_type", &func))
|
||
result = TRUE;
|
||
else if (g_module_symbol (module, "gtk_misc_get_type", &func))
|
||
result = TRUE;
|
||
else
|
||
result = FALSE;
|
||
|
||
if (!module_to_check)
|
||
g_module_close (module);
|
||
|
||
return result;
|
||
}
|
||
|
||
static void
|
||
do_pre_parse_initialization (void)
|
||
{
|
||
const char *env_string;
|
||
double slowdown;
|
||
|
||
if (pre_initialized)
|
||
return;
|
||
|
||
pre_initialized = TRUE;
|
||
|
||
if (_gtk_module_has_mixed_deps (NULL))
|
||
g_error ("GTK 2/3 symbols detected. Using GTK 2/3 and GTK 4 in the same process is not supported");
|
||
|
||
gdk_pre_parse ();
|
||
|
||
debug_flags[0].flags = gdk_parse_debug_var ("GTK_DEBUG",
|
||
gtk_debug_keys,
|
||
G_N_ELEMENTS (gtk_debug_keys));
|
||
any_display_debug_flags_set = debug_flags[0].flags > 0;
|
||
|
||
env_string = g_getenv ("GTK_SLOWDOWN");
|
||
if (env_string)
|
||
{
|
||
slowdown = g_ascii_strtod (env_string, NULL);
|
||
_gtk_set_slowdown (slowdown);
|
||
}
|
||
|
||
/* Trigger fontconfig initialization early */
|
||
pango_cairo_font_map_get_default ();
|
||
}
|
||
|
||
static void
|
||
gettext_initialization (void)
|
||
{
|
||
setlocale_initialization ();
|
||
|
||
#ifdef ENABLE_NLS
|
||
bindtextdomain (GETTEXT_PACKAGE, _gtk_get_localedir ());
|
||
bindtextdomain (GETTEXT_PACKAGE "-properties", _gtk_get_localedir ());
|
||
# ifdef HAVE_BIND_TEXTDOMAIN_CODESET
|
||
bind_textdomain_codeset (GETTEXT_PACKAGE, "UTF-8");
|
||
bind_textdomain_codeset (GETTEXT_PACKAGE "-properties", "UTF-8");
|
||
# endif
|
||
#endif
|
||
}
|
||
|
||
static void
|
||
default_display_notify_cb (GdkDisplayManager *dm)
|
||
{
|
||
debug_flags[0].display = gdk_display_get_default ();
|
||
}
|
||
|
||
static void
|
||
do_post_parse_initialization (void)
|
||
{
|
||
GdkDisplayManager *display_manager;
|
||
gint64 before G_GNUC_UNUSED;
|
||
|
||
if (gtk_initialized)
|
||
return;
|
||
|
||
before = GDK_PROFILER_CURRENT_TIME;
|
||
|
||
gettext_initialization ();
|
||
|
||
#ifdef SIGPIPE
|
||
signal (SIGPIPE, SIG_IGN);
|
||
#endif
|
||
|
||
gtk_widget_set_default_direction (gtk_get_locale_direction ());
|
||
|
||
gdk_event_init_types ();
|
||
|
||
gsk_ensure_resources ();
|
||
gsk_render_node_init_types ();
|
||
_gtk_ensure_resources ();
|
||
|
||
gdk_profiler_end_mark (before, "basic initialization", NULL);
|
||
|
||
gtk_initialized = TRUE;
|
||
|
||
before = GDK_PROFILER_CURRENT_TIME;
|
||
#ifdef G_OS_UNIX
|
||
gtk_print_backends_init ();
|
||
#endif
|
||
gtk_im_modules_init ();
|
||
gtk_media_file_extension_init ();
|
||
gdk_profiler_end_mark (before, "init modules", NULL);
|
||
|
||
before = GDK_PROFILER_CURRENT_TIME;
|
||
display_manager = gdk_display_manager_get ();
|
||
if (gdk_display_manager_get_default_display (display_manager) != NULL)
|
||
default_display_notify_cb (display_manager);
|
||
gdk_profiler_end_mark (before, "create display", NULL);
|
||
|
||
g_signal_connect (display_manager, "notify::default-display",
|
||
G_CALLBACK (default_display_notify_cb),
|
||
NULL);
|
||
|
||
gtk_inspector_register_extension ();
|
||
}
|
||
|
||
#ifdef G_PLATFORM_WIN32
|
||
#undef gtk_init_check
|
||
#endif
|
||
|
||
/**
|
||
* gtk_init_check:
|
||
*
|
||
* This function does the same work as gtk_init() with only a single
|
||
* change: It does not terminate the program if the windowing system
|
||
* can’t be initialized. Instead it returns %FALSE on failure.
|
||
*
|
||
* This way the application can fall back to some other means of
|
||
* communication with the user - for example a curses or command line
|
||
* interface.
|
||
*
|
||
* Returns: %TRUE if the windowing system has been successfully
|
||
* initialized, %FALSE otherwise
|
||
*/
|
||
gboolean
|
||
gtk_init_check (void)
|
||
{
|
||
gboolean ret;
|
||
|
||
if (gtk_initialized)
|
||
return TRUE;
|
||
|
||
if (gdk_profiler_is_running ())
|
||
g_info ("Profiling is active");
|
||
|
||
gettext_initialization ();
|
||
|
||
if (!check_setugid ())
|
||
return FALSE;
|
||
|
||
do_pre_parse_initialization ();
|
||
do_post_parse_initialization ();
|
||
|
||
ret = gdk_display_open_default () != NULL;
|
||
|
||
if (ret && (gtk_get_debug_flags () & GTK_DEBUG_INTERACTIVE))
|
||
gtk_window_set_interactive_debugging (TRUE);
|
||
|
||
return ret;
|
||
}
|
||
|
||
#ifdef G_PLATFORM_WIN32
|
||
#undef gtk_init
|
||
#endif
|
||
|
||
/**
|
||
* gtk_init:
|
||
*
|
||
* Call this function before using any other GTK functions in your GUI
|
||
* applications. It will initialize everything needed to operate the
|
||
* toolkit.
|
||
*
|
||
* If you are using `GtkApplication`, you don't have to call gtk_init()
|
||
* or gtk_init_check(); the `GApplication::startup` handler
|
||
* does it for you.
|
||
*
|
||
* This function will terminate your program if it was unable to
|
||
* initialize the windowing system for some reason. If you want
|
||
* your program to fall back to a textual interface you want to
|
||
* call gtk_init_check() instead.
|
||
*
|
||
* GTK calls `signal (SIGPIPE, SIG_IGN)`
|
||
* during initialization, to ignore SIGPIPE signals, since these are
|
||
* almost never wanted in graphical applications. If you do need to
|
||
* handle SIGPIPE for some reason, reset the handler after gtk_init(),
|
||
* but notice that other libraries (e.g. libdbus or gvfs) might do
|
||
* similar things.
|
||
*/
|
||
void
|
||
gtk_init (void)
|
||
{
|
||
if (!gtk_init_check ())
|
||
{
|
||
const char *display_name_arg = NULL;
|
||
if (display_name_arg == NULL)
|
||
display_name_arg = getenv ("DISPLAY");
|
||
g_warning ("cannot open display: %s", display_name_arg ? display_name_arg : "");
|
||
exit (1);
|
||
}
|
||
}
|
||
|
||
#ifdef G_OS_WIN32
|
||
|
||
/* This is relevant when building with gcc for Windows (MinGW),
|
||
* where we want to be struct packing compatible with MSVC,
|
||
* i.e. use the -mms-bitfields switch.
|
||
* For Cygwin there should be no need to be compatible with MSVC,
|
||
* so no need to use G_PLATFORM_WIN32.
|
||
*/
|
||
|
||
static void
|
||
check_sizeof_GtkWindow (size_t sizeof_GtkWindow)
|
||
{
|
||
if (sizeof_GtkWindow != sizeof (GtkWindow))
|
||
g_error ("Incompatible build!\n"
|
||
"The code using GTK thinks GtkWindow is of different\n"
|
||
"size than it actually is in this build of GTK.\n"
|
||
"On Windows, this probably means that you have compiled\n"
|
||
"your code with gcc without the -mms-bitfields switch,\n"
|
||
"or that you are using an unsupported compiler.");
|
||
}
|
||
|
||
/* In GTK 2.0 the GtkWindow struct actually is the same size in
|
||
* gcc-compiled code on Win32 whether compiled with -fnative-struct or
|
||
* not. Unfortunately this wan’t noticed until after GTK 2.0.1. So,
|
||
* from GTK 2.0.2 on, check some other struct, too, where the use of
|
||
* -fnative-struct still matters. GtkBox is one such.
|
||
*/
|
||
static void
|
||
check_sizeof_GtkBox (size_t sizeof_GtkBox)
|
||
{
|
||
if (sizeof_GtkBox != sizeof (GtkBox))
|
||
g_error ("Incompatible build!\n"
|
||
"The code using GTK thinks GtkBox is of different\n"
|
||
"size than it actually is in this build of GTK.\n"
|
||
"On Windows, this probably means that you have compiled\n"
|
||
"your code with gcc without the -mms-bitfields switch,\n"
|
||
"or that you are using an unsupported compiler.");
|
||
}
|
||
|
||
/* These two functions might get more checks added later, thus pass
|
||
* in the number of extra args.
|
||
*/
|
||
void
|
||
gtk_init_abi_check (int num_checks, size_t sizeof_GtkWindow, size_t sizeof_GtkBox)
|
||
{
|
||
check_sizeof_GtkWindow (sizeof_GtkWindow);
|
||
if (num_checks >= 2)
|
||
check_sizeof_GtkBox (sizeof_GtkBox);
|
||
gtk_init ();
|
||
}
|
||
|
||
gboolean
|
||
gtk_init_check_abi_check (int num_checks, size_t sizeof_GtkWindow, size_t sizeof_GtkBox)
|
||
{
|
||
check_sizeof_GtkWindow (sizeof_GtkWindow);
|
||
if (num_checks >= 2)
|
||
check_sizeof_GtkBox (sizeof_GtkBox);
|
||
return gtk_init_check ();
|
||
}
|
||
|
||
#endif
|
||
|
||
/**
|
||
* gtk_is_initialized:
|
||
*
|
||
* Use this function to check if GTK has been initialized with gtk_init()
|
||
* or gtk_init_check().
|
||
*
|
||
* Returns: the initialization status
|
||
*/
|
||
gboolean
|
||
gtk_is_initialized (void)
|
||
{
|
||
return gtk_initialized;
|
||
}
|
||
|
||
|
||
/**
|
||
* gtk_get_locale_direction:
|
||
*
|
||
* Get the direction of the current locale. This is the expected
|
||
* reading direction for text and UI.
|
||
*
|
||
* This function depends on the current locale being set with
|
||
* setlocale() and will default to setting the %GTK_TEXT_DIR_LTR
|
||
* direction otherwise. %GTK_TEXT_DIR_NONE will never be returned.
|
||
*
|
||
* GTK sets the default text direction according to the locale
|
||
* during gtk_init(), and you should normally use
|
||
* gtk_widget_get_direction() or gtk_widget_get_default_direction()
|
||
* to obtain the current direction.
|
||
*
|
||
* This function is only needed rare cases when the locale is
|
||
* changed after GTK has already been initialized. In this case,
|
||
* you can use it to update the default text direction as follows:
|
||
*
|
||
* |[<!-- language="C" -->
|
||
* #include <locale.h>
|
||
*
|
||
* static void
|
||
* update_locale (const char *new_locale)
|
||
* {
|
||
* setlocale (LC_ALL, new_locale);
|
||
* GtkTextDirection direction = gtk_get_locale_direction ();
|
||
* gtk_widget_set_default_direction (direction);
|
||
* }
|
||
* ]|
|
||
*
|
||
* Returns: the `GtkTextDirection` of the current locale
|
||
*/
|
||
GtkTextDirection
|
||
gtk_get_locale_direction (void)
|
||
{
|
||
/* Translate to default:RTL if you want your widgets
|
||
* to be RTL, otherwise translate to default:LTR.
|
||
* Do *not* translate it to "predefinito:LTR", if it
|
||
* it isn't default:LTR or default:RTL it will not work
|
||
*/
|
||
char *e = _("default:LTR");
|
||
GtkTextDirection dir = GTK_TEXT_DIR_LTR;
|
||
|
||
if (g_strcmp0 (e, "default:RTL") == 0)
|
||
dir = GTK_TEXT_DIR_RTL;
|
||
else if (g_strcmp0 (e, "default:LTR") != 0)
|
||
g_warning ("Whoever translated default:LTR did so wrongly. Defaulting to LTR.");
|
||
|
||
return dir;
|
||
}
|
||
|
||
/**
|
||
* gtk_get_default_language:
|
||
*
|
||
* Returns the `PangoLanguage` for the default language
|
||
* currently in effect.
|
||
*
|
||
* Note that this can change over the life of an
|
||
* application.
|
||
*
|
||
* The default language is derived from the current
|
||
* locale. It determines, for example, whether GTK uses
|
||
* the right-to-left or left-to-right text direction.
|
||
*
|
||
* This function is equivalent to
|
||
* [func@Pango.Language.get_default].
|
||
* See that function for details.
|
||
*
|
||
* Returns: (transfer none): the default language as a
|
||
* `PangoLanguage`
|
||
*/
|
||
PangoLanguage *
|
||
gtk_get_default_language (void)
|
||
{
|
||
return pango_language_get_default ();
|
||
}
|
||
|
||
typedef struct {
|
||
GMainLoop *store_loop;
|
||
guint n_clipboards;
|
||
guint timeout_id;
|
||
} ClipboardStore;
|
||
|
||
static void
|
||
clipboard_store_finished (GObject *source,
|
||
GAsyncResult *result,
|
||
gpointer data)
|
||
{
|
||
ClipboardStore *store;
|
||
GError *error = NULL;
|
||
|
||
if (!gdk_clipboard_store_finish (GDK_CLIPBOARD (source), result, &error))
|
||
{
|
||
if (g_error_matches (error, G_IO_ERROR, G_IO_ERROR_CANCELLED))
|
||
{
|
||
g_error_free (error);
|
||
return;
|
||
}
|
||
|
||
g_error_free (error);
|
||
}
|
||
|
||
store = data;
|
||
store->n_clipboards--;
|
||
if (store->n_clipboards == 0)
|
||
g_main_loop_quit (store->store_loop);
|
||
}
|
||
|
||
static gboolean
|
||
sync_timed_out_cb (ClipboardStore *store)
|
||
{
|
||
store->timeout_id = 0;
|
||
g_main_loop_quit (store->store_loop);
|
||
return G_SOURCE_REMOVE;
|
||
}
|
||
|
||
void
|
||
gtk_main_sync (void)
|
||
{
|
||
ClipboardStore store = { NULL, };
|
||
GSList *displays, *l;
|
||
GCancellable *cancel;
|
||
|
||
/* Try storing all clipboard data we have */
|
||
displays = gdk_display_manager_list_displays (gdk_display_manager_get ());
|
||
if (displays == NULL)
|
||
return;
|
||
|
||
cancel = g_cancellable_new ();
|
||
|
||
for (l = displays; l; l = l->next)
|
||
{
|
||
GdkDisplay *display = l->data;
|
||
GdkClipboard *clipboard = gdk_display_get_clipboard (display);
|
||
|
||
gdk_clipboard_store_async (clipboard,
|
||
G_PRIORITY_HIGH,
|
||
cancel,
|
||
clipboard_store_finished,
|
||
&store);
|
||
store.n_clipboards++;
|
||
}
|
||
g_slist_free (displays);
|
||
|
||
store.store_loop = g_main_loop_new (NULL, TRUE);
|
||
store.timeout_id = g_timeout_add_seconds (10, (GSourceFunc) sync_timed_out_cb, &store);
|
||
gdk_source_set_static_name_by_id (store.timeout_id, "[gtk] gtk_main_sync clipboard store timeout");
|
||
|
||
if (g_main_loop_is_running (store.store_loop))
|
||
g_main_loop_run (store.store_loop);
|
||
|
||
g_cancellable_cancel (cancel);
|
||
g_object_unref (cancel);
|
||
g_clear_handle_id (&store.timeout_id, g_source_remove);
|
||
g_clear_pointer (&store.store_loop, g_main_loop_unref);
|
||
|
||
/* Synchronize the recent manager singleton */
|
||
_gtk_recent_manager_sync ();
|
||
}
|
||
|
||
static GdkEvent *
|
||
rewrite_event_for_surface (GdkEvent *event,
|
||
GdkSurface *new_surface)
|
||
{
|
||
GdkEventType type;
|
||
double x = -G_MAXDOUBLE, y = -G_MAXDOUBLE;
|
||
double dx, dy;
|
||
|
||
type = gdk_event_get_event_type (event);
|
||
|
||
switch ((guint) type)
|
||
{
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_BUTTON_RELEASE:
|
||
return gdk_button_event_new (type,
|
||
new_surface,
|
||
gdk_event_get_device (event),
|
||
gdk_event_get_device_tool (event),
|
||
gdk_event_get_time (event),
|
||
gdk_event_get_modifier_state (event),
|
||
gdk_button_event_get_button (event),
|
||
x, y,
|
||
gdk_event_dup_axes (event));
|
||
case GDK_MOTION_NOTIFY:
|
||
return gdk_motion_event_new (new_surface,
|
||
gdk_event_get_device (event),
|
||
gdk_event_get_device_tool (event),
|
||
gdk_event_get_time (event),
|
||
gdk_event_get_modifier_state (event),
|
||
x, y,
|
||
gdk_event_dup_axes (event));
|
||
case GDK_TOUCH_BEGIN:
|
||
case GDK_TOUCH_UPDATE:
|
||
case GDK_TOUCH_END:
|
||
case GDK_TOUCH_CANCEL:
|
||
return gdk_touch_event_new (type,
|
||
gdk_event_get_event_sequence (event),
|
||
new_surface,
|
||
gdk_event_get_device (event),
|
||
gdk_event_get_time (event),
|
||
gdk_event_get_modifier_state (event),
|
||
x, y,
|
||
gdk_event_dup_axes (event),
|
||
gdk_touch_event_get_emulating_pointer (event));
|
||
case GDK_TOUCHPAD_SWIPE:
|
||
gdk_touchpad_event_get_deltas (event, &dx, &dy);
|
||
return gdk_touchpad_event_new_swipe (new_surface,
|
||
gdk_event_get_event_sequence (event),
|
||
gdk_event_get_device (event),
|
||
gdk_event_get_time (event),
|
||
gdk_event_get_modifier_state (event),
|
||
gdk_touchpad_event_get_gesture_phase (event),
|
||
x, y,
|
||
gdk_touchpad_event_get_n_fingers (event),
|
||
dx, dy);
|
||
case GDK_TOUCHPAD_PINCH:
|
||
gdk_touchpad_event_get_deltas (event, &dx, &dy);
|
||
return gdk_touchpad_event_new_pinch (new_surface,
|
||
gdk_event_get_event_sequence (event),
|
||
gdk_event_get_device (event),
|
||
gdk_event_get_time (event),
|
||
gdk_event_get_modifier_state (event),
|
||
gdk_touchpad_event_get_gesture_phase (event),
|
||
x, y,
|
||
gdk_touchpad_event_get_n_fingers (event),
|
||
dx, dy,
|
||
gdk_touchpad_event_get_pinch_scale (event),
|
||
gdk_touchpad_event_get_pinch_angle_delta (event));
|
||
case GDK_TOUCHPAD_HOLD:
|
||
return gdk_touchpad_event_new_hold (new_surface,
|
||
gdk_event_get_device (event),
|
||
gdk_event_get_time (event),
|
||
gdk_event_get_modifier_state (event),
|
||
gdk_touchpad_event_get_gesture_phase (event),
|
||
x, y,
|
||
gdk_touchpad_event_get_n_fingers (event));
|
||
default:
|
||
break;
|
||
}
|
||
|
||
return NULL;
|
||
}
|
||
|
||
/* If there is a pointer or keyboard grab in effect with owner_events = TRUE,
|
||
* then what X11 does is deliver the event normally if it was going to this
|
||
* client, otherwise, delivers it in terms of the grab surface. This function
|
||
* rewrites events to the effect that events going to the same window group
|
||
* are delivered normally, otherwise, the event is delivered in terms of the
|
||
* grab window.
|
||
*/
|
||
static GdkEvent *
|
||
rewrite_event_for_grabs (GdkEvent *event)
|
||
{
|
||
GdkSurface *grab_surface;
|
||
GtkWidget *event_widget, *grab_widget;
|
||
gboolean owner_events;
|
||
GdkDisplay *display;
|
||
GdkDevice *device;
|
||
|
||
switch ((guint) gdk_event_get_event_type (event))
|
||
{
|
||
case GDK_SCROLL:
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_BUTTON_RELEASE:
|
||
case GDK_MOTION_NOTIFY:
|
||
case GDK_PROXIMITY_IN:
|
||
case GDK_PROXIMITY_OUT:
|
||
case GDK_KEY_PRESS:
|
||
case GDK_KEY_RELEASE:
|
||
case GDK_TOUCH_BEGIN:
|
||
case GDK_TOUCH_UPDATE:
|
||
case GDK_TOUCH_END:
|
||
case GDK_TOUCH_CANCEL:
|
||
case GDK_TOUCHPAD_SWIPE:
|
||
case GDK_TOUCHPAD_PINCH:
|
||
case GDK_TOUCHPAD_HOLD:
|
||
display = gdk_event_get_display (event);
|
||
device = gdk_event_get_device (event);
|
||
|
||
if (!gdk_device_grab_info (display, device, &grab_surface, &owner_events) ||
|
||
!owner_events)
|
||
return NULL;
|
||
break;
|
||
default:
|
||
return NULL;
|
||
}
|
||
|
||
event_widget = gtk_get_event_widget (event);
|
||
grab_widget = GTK_WIDGET (gtk_native_get_for_surface (grab_surface));
|
||
|
||
if (grab_widget &&
|
||
gtk_main_get_window_group (grab_widget) != gtk_main_get_window_group (event_widget))
|
||
return rewrite_event_for_surface (event, grab_surface);
|
||
else
|
||
return NULL;
|
||
}
|
||
|
||
static GdkEvent *
|
||
rewrite_event_for_toplevel (GdkEvent *event)
|
||
{
|
||
GdkSurface *surface;
|
||
GdkEventType event_type;
|
||
GdkTranslatedKey *key, *key_no_lock;
|
||
|
||
surface = gdk_event_get_surface (event);
|
||
if (!surface->parent)
|
||
return NULL;
|
||
|
||
event_type = gdk_event_get_event_type (event);
|
||
if (event_type != GDK_KEY_PRESS &&
|
||
event_type != GDK_KEY_RELEASE)
|
||
return NULL;
|
||
|
||
while (surface->parent)
|
||
surface = surface->parent;
|
||
|
||
key = gdk_key_event_get_translated_key (event, FALSE);
|
||
key_no_lock = gdk_key_event_get_translated_key (event, TRUE);
|
||
|
||
return gdk_key_event_new (gdk_event_get_event_type (event),
|
||
surface,
|
||
gdk_event_get_device (event),
|
||
gdk_event_get_time (event),
|
||
gdk_key_event_get_keycode (event),
|
||
gdk_event_get_modifier_state (event),
|
||
gdk_key_event_is_modifier (event),
|
||
key, key_no_lock);
|
||
}
|
||
|
||
static gboolean
|
||
translate_event_coordinates (GdkEvent *event,
|
||
double *x,
|
||
double *y,
|
||
GtkWidget *widget)
|
||
{
|
||
GtkWidget *event_widget;
|
||
GtkNative *native;
|
||
graphene_point_t p;
|
||
double event_x, event_y;
|
||
double native_x, native_y;
|
||
|
||
*x = *y = 0;
|
||
|
||
if (!gdk_event_get_position (event, &event_x, &event_y))
|
||
return FALSE;
|
||
|
||
event_widget = gtk_get_event_widget (event);
|
||
native = gtk_widget_get_native (event_widget);
|
||
|
||
gtk_native_get_surface_transform (GTK_NATIVE (native), &native_x, &native_y);
|
||
event_x -= native_x;
|
||
event_y -= native_y;
|
||
|
||
if (!gtk_widget_compute_point (event_widget,
|
||
widget,
|
||
&GRAPHENE_POINT_INIT (event_x, event_y),
|
||
&p))
|
||
return FALSE;
|
||
|
||
*x = p.x;
|
||
*y = p.y;
|
||
|
||
return TRUE;
|
||
}
|
||
|
||
static void
|
||
gtk_synthesize_crossing_events (GtkRoot *toplevel,
|
||
GtkCrossingType crossing_type,
|
||
GtkWidget *old_target,
|
||
GtkWidget *new_target,
|
||
GdkEvent *event,
|
||
GdkCrossingMode mode,
|
||
GdkDrop *drop)
|
||
{
|
||
GtkCrossingData crossing;
|
||
GtkWidget *ancestor;
|
||
GtkWidget *widget;
|
||
double x, y;
|
||
GtkWidget *prev;
|
||
gboolean seen_ancestor;
|
||
GtkWidgetStack target_array;
|
||
int i;
|
||
|
||
if (old_target == new_target)
|
||
return;
|
||
|
||
if (old_target && new_target)
|
||
ancestor = gtk_widget_common_ancestor (old_target, new_target);
|
||
else
|
||
ancestor = NULL;
|
||
|
||
crossing.type = crossing_type;
|
||
crossing.mode = mode;
|
||
crossing.old_target = old_target ? g_object_ref (old_target) : NULL;
|
||
crossing.old_descendent = NULL;
|
||
crossing.new_target = new_target ? g_object_ref (new_target) : NULL;
|
||
crossing.new_descendent = NULL;
|
||
crossing.drop = drop;
|
||
|
||
crossing.direction = GTK_CROSSING_OUT;
|
||
|
||
prev = NULL;
|
||
seen_ancestor = FALSE;
|
||
widget = old_target;
|
||
while (widget)
|
||
{
|
||
crossing.old_descendent = prev;
|
||
if (seen_ancestor)
|
||
{
|
||
crossing.new_descendent = new_target ? prev : NULL;
|
||
}
|
||
else if (widget == ancestor)
|
||
{
|
||
GtkWidget *w;
|
||
|
||
crossing.new_descendent = NULL;
|
||
for (w = new_target; w != ancestor; w = _gtk_widget_get_parent (w))
|
||
crossing.new_descendent = w;
|
||
|
||
seen_ancestor = TRUE;
|
||
}
|
||
else
|
||
{
|
||
crossing.new_descendent = NULL;
|
||
}
|
||
check_crossing_invariants (widget, &crossing);
|
||
translate_event_coordinates (event, &x, &y, widget);
|
||
gtk_widget_handle_crossing (widget, &crossing, x, y);
|
||
if (crossing_type == GTK_CROSSING_POINTER)
|
||
gtk_widget_unset_state_flags (widget, GTK_STATE_FLAG_PRELIGHT);
|
||
prev = widget;
|
||
widget = _gtk_widget_get_parent (widget);
|
||
}
|
||
|
||
gtk_widget_stack_init (&target_array);
|
||
for (widget = new_target; widget; widget = _gtk_widget_get_parent (widget))
|
||
gtk_widget_stack_append (&target_array, g_object_ref (widget));
|
||
|
||
crossing.direction = GTK_CROSSING_IN;
|
||
|
||
seen_ancestor = FALSE;
|
||
for (i = gtk_widget_stack_get_size (&target_array) - 1; i >= 0; i--)
|
||
{
|
||
widget = gtk_widget_stack_get (&target_array, i);
|
||
|
||
if (i < gtk_widget_stack_get_size (&target_array) - 1)
|
||
crossing.new_descendent = gtk_widget_stack_get (&target_array, i + 1);
|
||
else
|
||
crossing.new_descendent = NULL;
|
||
|
||
if (seen_ancestor)
|
||
{
|
||
crossing.old_descendent = NULL;
|
||
}
|
||
else if (widget == ancestor)
|
||
{
|
||
GtkWidget *w;
|
||
|
||
crossing.old_descendent = NULL;
|
||
for (w = old_target; w != ancestor; w = _gtk_widget_get_parent (w))
|
||
crossing.old_descendent = w;
|
||
|
||
seen_ancestor = TRUE;
|
||
}
|
||
else
|
||
{
|
||
crossing.old_descendent = old_target ? crossing.new_descendent : NULL;
|
||
}
|
||
|
||
translate_event_coordinates (event, &x, &y, widget);
|
||
gtk_widget_handle_crossing (widget, &crossing, x, y);
|
||
if (crossing_type == GTK_CROSSING_POINTER)
|
||
gtk_widget_set_state_flags (widget, GTK_STATE_FLAG_PRELIGHT, FALSE);
|
||
}
|
||
|
||
g_clear_object (&crossing.old_target);
|
||
g_clear_object (&crossing.new_target);
|
||
|
||
gtk_widget_stack_clear (&target_array);
|
||
}
|
||
|
||
static GtkWidget *
|
||
update_pointer_focus_state (GtkWindow *toplevel,
|
||
GdkEvent *event,
|
||
GtkWidget *new_target)
|
||
{
|
||
GtkWidget *old_target = NULL;
|
||
GdkEventSequence *sequence;
|
||
GdkDevice *device;
|
||
double x, y;
|
||
double nx, ny;
|
||
|
||
device = gdk_event_get_device (event);
|
||
sequence = gdk_event_get_event_sequence (event);
|
||
old_target = gtk_window_lookup_pointer_focus_widget (toplevel, device, sequence);
|
||
if (old_target == new_target)
|
||
return old_target;
|
||
|
||
gdk_event_get_position (event, &x, &y);
|
||
gtk_native_get_surface_transform (GTK_NATIVE (toplevel), &nx, &ny);
|
||
x -= nx;
|
||
y -= ny;
|
||
|
||
gtk_window_update_pointer_focus (toplevel, device, sequence,
|
||
new_target, x, y);
|
||
|
||
return old_target;
|
||
}
|
||
|
||
static gboolean
|
||
is_pointing_event (GdkEvent *event)
|
||
{
|
||
switch ((guint) gdk_event_get_event_type (event))
|
||
{
|
||
case GDK_MOTION_NOTIFY:
|
||
case GDK_ENTER_NOTIFY:
|
||
case GDK_LEAVE_NOTIFY:
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_BUTTON_RELEASE:
|
||
case GDK_SCROLL:
|
||
case GDK_TOUCH_BEGIN:
|
||
case GDK_TOUCH_UPDATE:
|
||
case GDK_TOUCH_END:
|
||
case GDK_TOUCH_CANCEL:
|
||
case GDK_TOUCHPAD_PINCH:
|
||
case GDK_TOUCHPAD_SWIPE:
|
||
case GDK_TOUCHPAD_HOLD:
|
||
case GDK_DRAG_ENTER:
|
||
case GDK_DRAG_LEAVE:
|
||
case GDK_DRAG_MOTION:
|
||
case GDK_DROP_START:
|
||
return TRUE;
|
||
|
||
case GDK_GRAB_BROKEN:
|
||
return gdk_device_get_source (gdk_event_get_device (event)) != GDK_SOURCE_KEYBOARD;
|
||
|
||
default:
|
||
return FALSE;
|
||
}
|
||
}
|
||
|
||
static gboolean
|
||
is_key_event (GdkEvent *event)
|
||
{
|
||
switch ((guint) gdk_event_get_event_type (event))
|
||
{
|
||
case GDK_KEY_PRESS:
|
||
case GDK_KEY_RELEASE:
|
||
return TRUE;
|
||
break;
|
||
case GDK_GRAB_BROKEN:
|
||
return gdk_device_get_source (gdk_event_get_device (event)) == GDK_SOURCE_KEYBOARD;
|
||
default:
|
||
return FALSE;
|
||
}
|
||
}
|
||
|
||
static inline void
|
||
set_widget_active_state (GtkWidget *target,
|
||
const gboolean is_active)
|
||
{
|
||
GtkWidget *w;
|
||
|
||
w = target;
|
||
while (w)
|
||
{
|
||
gtk_widget_set_active_state (w, is_active);
|
||
w = _gtk_widget_get_parent (w);
|
||
}
|
||
}
|
||
|
||
static GtkWidget *
|
||
handle_pointing_event (GdkEvent *event)
|
||
{
|
||
GtkWidget *target = NULL, *old_target = NULL, *event_widget;
|
||
GtkWindow *toplevel;
|
||
GdkEventSequence *sequence;
|
||
GdkDevice *device;
|
||
double x, y;
|
||
double native_x, native_y;
|
||
GtkWidget *native;
|
||
GdkEventType type;
|
||
gboolean has_implicit;
|
||
GdkModifierType modifiers;
|
||
|
||
event_widget = gtk_get_event_widget (event);
|
||
device = gdk_event_get_device (event);
|
||
gdk_event_get_position (event, &x, &y);
|
||
|
||
toplevel = GTK_WINDOW (gtk_widget_get_root (event_widget));
|
||
native = GTK_WIDGET (gtk_widget_get_native (event_widget));
|
||
|
||
gtk_native_get_surface_transform (GTK_NATIVE (native), &native_x, &native_y);
|
||
x -= native_x;
|
||
y -= native_y;
|
||
|
||
type = gdk_event_get_event_type (event);
|
||
sequence = gdk_event_get_event_sequence (event);
|
||
|
||
if (type == GDK_SCROLL &&
|
||
(gdk_device_get_source (device) == GDK_SOURCE_TOUCHPAD ||
|
||
gdk_device_get_source (device) == GDK_SOURCE_TRACKPOINT ||
|
||
gdk_device_get_source (device) == GDK_SOURCE_MOUSE))
|
||
{
|
||
/* A bit of a kludge, resolve target lookups for scrolling devices
|
||
* on the seat pointer.
|
||
*/
|
||
device = gdk_seat_get_pointer (gdk_event_get_seat (event));
|
||
}
|
||
else if (type == GDK_TOUCHPAD_PINCH ||
|
||
type == GDK_TOUCHPAD_SWIPE ||
|
||
type == GDK_TOUCHPAD_HOLD)
|
||
{
|
||
/* Another bit of a kludge, touchpad gesture sequences do not
|
||
* reflect on the pointer focus lookup.
|
||
*/
|
||
sequence = NULL;
|
||
}
|
||
|
||
switch ((guint) type)
|
||
{
|
||
case GDK_LEAVE_NOTIFY:
|
||
if (gdk_crossing_event_get_mode (event) == GDK_CROSSING_GRAB)
|
||
{
|
||
GtkWidget *grab_widget;
|
||
|
||
grab_widget =
|
||
gtk_window_lookup_pointer_focus_implicit_grab (toplevel,
|
||
device,
|
||
sequence);
|
||
if (grab_widget)
|
||
set_widget_active_state (grab_widget, FALSE);
|
||
}
|
||
|
||
old_target = update_pointer_focus_state (toplevel, event, NULL);
|
||
gtk_synthesize_crossing_events (GTK_ROOT (toplevel), GTK_CROSSING_POINTER, old_target, NULL,
|
||
event, gdk_crossing_event_get_mode (event), NULL);
|
||
break;
|
||
case GDK_TOUCH_END:
|
||
case GDK_TOUCH_CANCEL:
|
||
old_target = update_pointer_focus_state (toplevel, event, NULL);
|
||
set_widget_active_state (old_target, FALSE);
|
||
break;
|
||
case GDK_DRAG_LEAVE:
|
||
{
|
||
GdkDrop *drop = gdk_dnd_event_get_drop (event);
|
||
old_target = update_pointer_focus_state (toplevel, event, NULL);
|
||
gtk_drop_begin_event (drop, GDK_DRAG_LEAVE);
|
||
gtk_synthesize_crossing_events (GTK_ROOT (toplevel), GTK_CROSSING_DROP, old_target, NULL,
|
||
event, GDK_CROSSING_NORMAL, drop);
|
||
gtk_drop_end_event (drop);
|
||
}
|
||
break;
|
||
case GDK_ENTER_NOTIFY:
|
||
case GDK_DRAG_ENTER:
|
||
case GDK_DRAG_MOTION:
|
||
case GDK_DROP_START:
|
||
case GDK_TOUCH_BEGIN:
|
||
case GDK_TOUCH_UPDATE:
|
||
case GDK_MOTION_NOTIFY:
|
||
target = gtk_window_lookup_pointer_focus_implicit_grab (toplevel, device, sequence);
|
||
|
||
if (!target)
|
||
target = gtk_widget_pick (native, x, y, GTK_PICK_DEFAULT);
|
||
|
||
if (!target)
|
||
target = GTK_WIDGET (native);
|
||
|
||
old_target = update_pointer_focus_state (toplevel, event, target);
|
||
|
||
if (type == GDK_MOTION_NOTIFY || type == GDK_ENTER_NOTIFY)
|
||
{
|
||
if (!gtk_window_lookup_pointer_focus_implicit_grab (toplevel, device,
|
||
sequence))
|
||
{
|
||
gtk_synthesize_crossing_events (GTK_ROOT (toplevel), GTK_CROSSING_POINTER, old_target, target,
|
||
event, GDK_CROSSING_NORMAL, NULL);
|
||
}
|
||
|
||
gtk_window_maybe_update_cursor (toplevel, NULL, device);
|
||
}
|
||
else if ((old_target != target) &&
|
||
(type == GDK_DRAG_ENTER || type == GDK_DRAG_MOTION || type == GDK_DROP_START))
|
||
{
|
||
GdkDrop *drop = gdk_dnd_event_get_drop (event);
|
||
gtk_drop_begin_event (drop, type);
|
||
gtk_synthesize_crossing_events (GTK_ROOT (toplevel), GTK_CROSSING_DROP, old_target, target,
|
||
event, GDK_CROSSING_NORMAL, gdk_dnd_event_get_drop (event));
|
||
gtk_drop_end_event (drop);
|
||
}
|
||
else if (type == GDK_TOUCH_BEGIN)
|
||
{
|
||
gtk_window_set_pointer_focus_grab (toplevel, device, sequence, target);
|
||
set_widget_active_state (target, TRUE);
|
||
}
|
||
|
||
/* Let it take the effective pointer focus anyway, as it may change due
|
||
* to implicit grabs.
|
||
*/
|
||
target = NULL;
|
||
break;
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_BUTTON_RELEASE:
|
||
target = gtk_window_lookup_effective_pointer_focus_widget (toplevel,
|
||
device,
|
||
sequence);
|
||
has_implicit =
|
||
gtk_window_lookup_pointer_focus_implicit_grab (toplevel,
|
||
device,
|
||
sequence) != NULL;
|
||
modifiers = gdk_event_get_modifier_state (event);
|
||
|
||
if (type == GDK_BUTTON_RELEASE &&
|
||
gtk_popcount (modifiers & (GDK_BUTTON1_MASK |
|
||
GDK_BUTTON2_MASK |
|
||
GDK_BUTTON3_MASK |
|
||
GDK_BUTTON4_MASK |
|
||
GDK_BUTTON5_MASK)) == 1)
|
||
{
|
||
GtkWidget *new_target = gtk_widget_pick (native, x, y, GTK_PICK_DEFAULT);
|
||
|
||
gtk_window_set_pointer_focus_grab (toplevel, device, sequence, NULL);
|
||
|
||
if (new_target == NULL)
|
||
new_target = GTK_WIDGET (toplevel);
|
||
|
||
gtk_synthesize_crossing_events (GTK_ROOT (toplevel), GTK_CROSSING_POINTER, target, new_target,
|
||
event, GDK_CROSSING_UNGRAB, NULL);
|
||
gtk_window_maybe_update_cursor (toplevel, NULL, device);
|
||
update_pointer_focus_state (toplevel, event, new_target);
|
||
}
|
||
else if (type == GDK_BUTTON_PRESS)
|
||
{
|
||
gtk_window_set_pointer_focus_grab (toplevel, device,
|
||
sequence, target);
|
||
}
|
||
|
||
if (type == GDK_BUTTON_PRESS)
|
||
set_widget_active_state (target, TRUE);
|
||
else if (has_implicit)
|
||
set_widget_active_state (target, FALSE);
|
||
|
||
break;
|
||
case GDK_SCROLL:
|
||
case GDK_TOUCHPAD_PINCH:
|
||
case GDK_TOUCHPAD_SWIPE:
|
||
case GDK_TOUCHPAD_HOLD:
|
||
break;
|
||
case GDK_GRAB_BROKEN:
|
||
if (gdk_grab_broken_event_get_implicit (event))
|
||
{
|
||
target = gtk_window_lookup_effective_pointer_focus_widget (toplevel,
|
||
device,
|
||
sequence);
|
||
set_widget_active_state (target, FALSE);
|
||
}
|
||
break;
|
||
default:
|
||
g_assert_not_reached ();
|
||
}
|
||
|
||
if (!target)
|
||
target = gtk_window_lookup_effective_pointer_focus_widget (toplevel,
|
||
device,
|
||
sequence);
|
||
return target ? target : old_target;
|
||
}
|
||
|
||
static GtkWidget *
|
||
handle_key_event (GdkEvent *event)
|
||
{
|
||
GtkWidget *event_widget;
|
||
GtkWidget *focus_widget;
|
||
|
||
event_widget = gtk_get_event_widget (event);
|
||
|
||
focus_widget = gtk_root_get_focus (gtk_widget_get_root (event_widget));
|
||
return focus_widget ? focus_widget : event_widget;
|
||
}
|
||
|
||
static gboolean
|
||
is_transient_for (GtkWindow *child,
|
||
GtkWindow *parent)
|
||
{
|
||
GtkWindow *transient_for;
|
||
|
||
transient_for = gtk_window_get_transient_for (child);
|
||
|
||
while (transient_for)
|
||
{
|
||
if (transient_for == parent)
|
||
return TRUE;
|
||
|
||
transient_for = gtk_window_get_transient_for (transient_for);
|
||
}
|
||
|
||
return FALSE;
|
||
}
|
||
|
||
void
|
||
gtk_main_do_event (GdkEvent *event)
|
||
{
|
||
GtkWidget *event_widget;
|
||
GtkWidget *target_widget;
|
||
GtkWidget *grab_widget = NULL;
|
||
GtkWindowGroup *window_group;
|
||
GdkEvent *rewritten_event = NULL;
|
||
GList *tmp_list;
|
||
|
||
if (gtk_inspector_handle_event (event))
|
||
return;
|
||
|
||
/* Find the widget which got the event. We store the widget
|
||
* in the user_data field of GdkSurface's. Ignore the event
|
||
* if we don't have a widget for it.
|
||
*/
|
||
event_widget = gtk_get_event_widget (event);
|
||
if (!event_widget)
|
||
return;
|
||
|
||
target_widget = event_widget;
|
||
|
||
/* We propagate key events from the root, even if they are
|
||
* delivered to a popup surface.
|
||
*
|
||
* If pointer or keyboard grabs are in effect, munge the events
|
||
* so that each window group looks like a separate app.
|
||
*/
|
||
if (is_key_event (event))
|
||
rewritten_event = rewrite_event_for_toplevel (event);
|
||
else
|
||
rewritten_event = rewrite_event_for_grabs (event);
|
||
if (rewritten_event)
|
||
{
|
||
event = rewritten_event;
|
||
target_widget = gtk_get_event_widget (event);
|
||
}
|
||
|
||
/* Push the event onto a stack of current events for
|
||
* gtk_current_event_get().
|
||
*/
|
||
current_events = g_list_prepend (current_events, event);
|
||
|
||
if (is_pointing_event (event))
|
||
{
|
||
target_widget = handle_pointing_event (event);
|
||
}
|
||
else if (is_key_event (event))
|
||
{
|
||
target_widget = handle_key_event (event);
|
||
}
|
||
|
||
if (!target_widget)
|
||
goto cleanup;
|
||
|
||
window_group = gtk_main_get_window_group (target_widget);
|
||
|
||
/* check whether there is a grab in effect... */
|
||
grab_widget = gtk_window_group_get_current_grab (window_group);
|
||
|
||
/* If the grab widget is an ancestor of the event widget
|
||
* then we send the event to the original event widget.
|
||
* This is the key to implementing modality. This also applies
|
||
* across windows that are directly or indirectly transient-for
|
||
* the modal one.
|
||
*/
|
||
if (!grab_widget ||
|
||
((gtk_widget_is_sensitive (target_widget) || gdk_event_get_event_type (event) == GDK_SCROLL) &&
|
||
gtk_widget_is_ancestor (target_widget, grab_widget)) ||
|
||
(GTK_IS_WINDOW (grab_widget) &&
|
||
GTK_IS_WINDOW (event_widget) &&
|
||
grab_widget != event_widget &&
|
||
is_transient_for (GTK_WINDOW (event_widget), GTK_WINDOW (grab_widget))))
|
||
grab_widget = target_widget;
|
||
|
||
g_object_ref (target_widget);
|
||
|
||
/* Not all events get sent to the grabbing widget.
|
||
* The delete, destroy, expose, focus change and resize
|
||
* events still get sent to the event widget because
|
||
* 1) these events have no meaning for the grabbing widget
|
||
* and 2) redirecting these events to the grabbing widget
|
||
* could cause the display to be messed up.
|
||
*
|
||
* Drag events are also not redirected, since it isn't
|
||
* clear what the semantics of that would be.
|
||
*/
|
||
switch ((guint)gdk_event_get_event_type (event))
|
||
{
|
||
case GDK_DELETE:
|
||
if (!gtk_window_group_get_current_grab (window_group) ||
|
||
GTK_WIDGET (gtk_widget_get_root (gtk_window_group_get_current_grab (window_group))) == target_widget)
|
||
{
|
||
if (GTK_IS_WINDOW (target_widget) &&
|
||
!gtk_window_emit_close_request (GTK_WINDOW (target_widget)))
|
||
gtk_window_destroy (GTK_WINDOW (target_widget));
|
||
}
|
||
break;
|
||
|
||
case GDK_FOCUS_CHANGE:
|
||
{
|
||
GtkWidget *root = GTK_WIDGET (gtk_widget_get_root (target_widget));
|
||
if (!_gtk_widget_captured_event (root, event, root))
|
||
gtk_widget_event (root, event, root);
|
||
}
|
||
break;
|
||
|
||
case GDK_KEY_PRESS:
|
||
case GDK_KEY_RELEASE:
|
||
case GDK_SCROLL:
|
||
case GDK_BUTTON_PRESS:
|
||
case GDK_TOUCH_BEGIN:
|
||
case GDK_MOTION_NOTIFY:
|
||
case GDK_BUTTON_RELEASE:
|
||
case GDK_PROXIMITY_IN:
|
||
case GDK_PROXIMITY_OUT:
|
||
case GDK_TOUCH_UPDATE:
|
||
case GDK_TOUCH_END:
|
||
case GDK_TOUCH_CANCEL:
|
||
case GDK_TOUCHPAD_SWIPE:
|
||
case GDK_TOUCHPAD_PINCH:
|
||
case GDK_TOUCHPAD_HOLD:
|
||
case GDK_PAD_BUTTON_PRESS:
|
||
case GDK_PAD_BUTTON_RELEASE:
|
||
case GDK_PAD_RING:
|
||
case GDK_PAD_STRIP:
|
||
case GDK_PAD_GROUP_MODE:
|
||
case GDK_GRAB_BROKEN:
|
||
gtk_propagate_event (grab_widget, event);
|
||
break;
|
||
|
||
case GDK_ENTER_NOTIFY:
|
||
case GDK_LEAVE_NOTIFY:
|
||
case GDK_DRAG_ENTER:
|
||
case GDK_DRAG_LEAVE:
|
||
/* Crossing event propagation happens during picking */
|
||
break;
|
||
|
||
case GDK_DRAG_MOTION:
|
||
case GDK_DROP_START:
|
||
{
|
||
GdkDrop *drop = gdk_dnd_event_get_drop (event);
|
||
gtk_drop_begin_event (drop, gdk_event_get_event_type (event));
|
||
gtk_propagate_event (target_widget, event);
|
||
gtk_drop_end_event (drop);
|
||
}
|
||
break;
|
||
|
||
case GDK_EVENT_LAST:
|
||
default:
|
||
g_assert_not_reached ();
|
||
break;
|
||
}
|
||
|
||
_gtk_tooltip_handle_event (target_widget, event);
|
||
|
||
g_object_unref (target_widget);
|
||
|
||
cleanup:
|
||
tmp_list = current_events;
|
||
current_events = g_list_remove_link (current_events, tmp_list);
|
||
g_list_free_1 (tmp_list);
|
||
|
||
if (rewritten_event)
|
||
gdk_event_unref (rewritten_event);
|
||
}
|
||
|
||
static GtkWindowGroup *
|
||
gtk_main_get_window_group (GtkWidget *widget)
|
||
{
|
||
GtkWidget *toplevel = NULL;
|
||
|
||
if (widget)
|
||
toplevel = GTK_WIDGET (gtk_widget_get_root (widget));
|
||
|
||
if (GTK_IS_WINDOW (toplevel))
|
||
return gtk_window_get_group (GTK_WINDOW (toplevel));
|
||
else
|
||
return gtk_window_get_group (NULL);
|
||
}
|
||
|
||
static void
|
||
gtk_grab_notify (GtkWindowGroup *group,
|
||
GtkWidget *old_grab_widget,
|
||
GtkWidget *new_grab_widget,
|
||
gboolean from_grab)
|
||
{
|
||
GList *toplevels;
|
||
|
||
if (old_grab_widget == new_grab_widget)
|
||
return;
|
||
|
||
g_object_ref (group);
|
||
|
||
toplevels = gtk_window_list_toplevels ();
|
||
g_list_foreach (toplevels, (GFunc)g_object_ref, NULL);
|
||
|
||
while (toplevels)
|
||
{
|
||
GtkWindow *toplevel = toplevels->data;
|
||
toplevels = g_list_delete_link (toplevels, toplevels);
|
||
|
||
gtk_window_grab_notify (toplevel,
|
||
old_grab_widget,
|
||
new_grab_widget,
|
||
from_grab);
|
||
g_object_unref (toplevel);
|
||
}
|
||
|
||
g_object_unref (group);
|
||
}
|
||
|
||
/**
|
||
* gtk_grab_add: (method)
|
||
* @widget: The widget that grabs keyboard and pointer events
|
||
*
|
||
* Makes @widget the current grabbed widget.
|
||
*
|
||
* This means that interaction with other widgets in the same
|
||
* application is blocked and mouse as well as keyboard events
|
||
* are delivered to this widget.
|
||
*
|
||
* If @widget is not sensitive, it is not set as the current
|
||
* grabbed widget and this function does nothing.
|
||
*/
|
||
void
|
||
gtk_grab_add (GtkWidget *widget)
|
||
{
|
||
GtkWindowGroup *group;
|
||
GtkWidget *old_grab_widget;
|
||
|
||
g_return_if_fail (widget != NULL);
|
||
|
||
if (!gtk_widget_has_grab (widget) && gtk_widget_is_sensitive (widget))
|
||
{
|
||
_gtk_widget_set_has_grab (widget, TRUE);
|
||
|
||
group = gtk_main_get_window_group (widget);
|
||
|
||
old_grab_widget = gtk_window_group_get_current_grab (group);
|
||
|
||
g_object_ref (widget);
|
||
_gtk_window_group_add_grab (group, widget);
|
||
|
||
gtk_grab_notify (group, old_grab_widget, widget, TRUE);
|
||
}
|
||
}
|
||
|
||
/**
|
||
* gtk_grab_remove: (method)
|
||
* @widget: The widget which gives up the grab
|
||
*
|
||
* Removes the grab from the given widget.
|
||
*
|
||
* You have to pair calls to gtk_grab_add() and gtk_grab_remove().
|
||
*
|
||
* If @widget does not have the grab, this function does nothing.
|
||
*/
|
||
void
|
||
gtk_grab_remove (GtkWidget *widget)
|
||
{
|
||
GtkWindowGroup *group;
|
||
GtkWidget *new_grab_widget;
|
||
|
||
g_return_if_fail (widget != NULL);
|
||
|
||
if (gtk_widget_has_grab (widget))
|
||
{
|
||
_gtk_widget_set_has_grab (widget, FALSE);
|
||
|
||
group = gtk_main_get_window_group (widget);
|
||
_gtk_window_group_remove_grab (group, widget);
|
||
new_grab_widget = gtk_window_group_get_current_grab (group);
|
||
|
||
gtk_grab_notify (group, widget, new_grab_widget, FALSE);
|
||
|
||
g_object_unref (widget);
|
||
}
|
||
}
|
||
|
||
guint32
|
||
gtk_get_current_event_time (void)
|
||
{
|
||
if (current_events)
|
||
return gdk_event_get_time (current_events->data);
|
||
else
|
||
return GDK_CURRENT_TIME;
|
||
}
|
||
|
||
/**
|
||
* gtk_get_event_widget:
|
||
* @event: a `GdkEvent`
|
||
*
|
||
* If @event is %NULL or the event was not associated with any widget,
|
||
* returns %NULL, otherwise returns the widget that received the event
|
||
* originally.
|
||
*
|
||
* Returns: (transfer none) (nullable): the widget that originally
|
||
* received @event
|
||
*/
|
||
GtkWidget *
|
||
gtk_get_event_widget (GdkEvent *event)
|
||
{
|
||
GdkSurface *surface;
|
||
|
||
surface = gdk_event_get_surface (event);
|
||
if (surface && !gdk_surface_is_destroyed (surface))
|
||
return GTK_WIDGET (gtk_native_get_for_surface (surface));
|
||
|
||
return NULL;
|
||
}
|
||
|
||
gboolean
|
||
gtk_propagate_event_internal (GtkWidget *widget,
|
||
GdkEvent *event,
|
||
GtkWidget *topmost)
|
||
{
|
||
int handled_event = FALSE;
|
||
GtkWidget *target = widget;
|
||
GtkWidgetStack widget_array;
|
||
int i;
|
||
|
||
/* First, propagate event down */
|
||
gtk_widget_stack_init (&widget_array);
|
||
gtk_widget_stack_append (&widget_array, g_object_ref (widget));
|
||
|
||
for (;;)
|
||
{
|
||
widget = _gtk_widget_get_parent (widget);
|
||
if (!widget)
|
||
break;
|
||
|
||
gtk_widget_stack_append (&widget_array, g_object_ref (widget));
|
||
|
||
if (widget == topmost)
|
||
break;
|
||
}
|
||
|
||
i = gtk_widget_stack_get_size (&widget_array) - 1;
|
||
for (;;)
|
||
{
|
||
widget = gtk_widget_stack_get (&widget_array, i);
|
||
|
||
if (!_gtk_widget_is_sensitive (widget))
|
||
{
|
||
/* stop propagating on SCROLL, but don't handle the event, so it
|
||
* can propagate up again and reach its handling widget
|
||
*/
|
||
if (gdk_event_get_event_type (event) == GDK_SCROLL)
|
||
break;
|
||
else
|
||
handled_event = TRUE;
|
||
}
|
||
else if (_gtk_widget_get_realized (widget))
|
||
handled_event = _gtk_widget_captured_event (widget, event, target);
|
||
|
||
handled_event |= !_gtk_widget_get_realized (widget);
|
||
|
||
if (handled_event)
|
||
break;
|
||
|
||
if (i == 0)
|
||
break;
|
||
|
||
i--;
|
||
}
|
||
|
||
/* If not yet handled, also propagate down */
|
||
if (!handled_event)
|
||
{
|
||
/* Propagate event up the widget tree so that
|
||
* parents can see the button and motion
|
||
* events of the children.
|
||
*/
|
||
for (i = 0; i < gtk_widget_stack_get_size (&widget_array); i++)
|
||
{
|
||
widget = gtk_widget_stack_get (&widget_array, i);
|
||
|
||
/* Scroll events are special cased here because it
|
||
* feels wrong when scrolling a GtkViewport, say,
|
||
* to have children of the viewport eat the scroll
|
||
* event
|
||
*/
|
||
if (!_gtk_widget_is_sensitive (widget))
|
||
handled_event = gdk_event_get_event_type (event) != GDK_SCROLL;
|
||
else if (_gtk_widget_get_realized (widget))
|
||
handled_event = gtk_widget_event (widget, event, target);
|
||
|
||
handled_event |= !_gtk_widget_get_realized (widget);
|
||
|
||
if (handled_event)
|
||
break;
|
||
}
|
||
}
|
||
|
||
gtk_widget_stack_clear (&widget_array);
|
||
return handled_event;
|
||
}
|
||
|
||
/**
|
||
* gtk_propagate_event:
|
||
* @widget: a `GtkWidget`
|
||
* @event: an event
|
||
*
|
||
* Sends an event to a widget, propagating the event to parent widgets
|
||
* if the event remains unhandled. This function will emit the event
|
||
* through all the hierarchy of @widget through all propagation phases.
|
||
*
|
||
* Events received by GTK from GDK normally begin at a `GtkRoot` widget.
|
||
* Depending on the type of event, existence of modal dialogs, grabs, etc.,
|
||
* the event may be propagated; if so, this function is used.
|
||
*
|
||
* All that said, you most likely don’t want to use any of these
|
||
* functions; synthesizing events is rarely needed. There are almost
|
||
* certainly better ways to achieve your goals. For example, use
|
||
* gtk_widget_queue_draw() instead
|
||
* of making up expose events.
|
||
*
|
||
* Returns: %TRUE if the event was handled
|
||
*/
|
||
gboolean
|
||
gtk_propagate_event (GtkWidget *widget,
|
||
GdkEvent *event)
|
||
{
|
||
GtkWindowGroup *window_group;
|
||
GtkWidget *event_widget, *topmost = NULL;
|
||
|
||
g_return_val_if_fail (GTK_IS_WIDGET (widget), FALSE);
|
||
g_return_val_if_fail (event != NULL, FALSE);
|
||
|
||
event_widget = gtk_get_event_widget (event);
|
||
window_group = gtk_main_get_window_group (event_widget);
|
||
|
||
/* check whether there is a grab in effect... */
|
||
topmost = gtk_window_group_get_current_grab (window_group);
|
||
|
||
return gtk_propagate_event_internal (widget, event, topmost);
|
||
}
|