diff --git a/src/Makefile.am b/src/Makefile.am index f60158fa1..e2b342e65 100644 --- a/src/Makefile.am +++ b/src/Makefile.am @@ -82,6 +82,8 @@ libmutter_la_SOURCES = \ backends/native/meta-launcher.h \ backends/x11/meta-backend-x11.c \ backends/x11/meta-backend-x11.h \ + backends/x11/meta-cursor-renderer-x11.c \ + backends/x11/meta-cursor-renderer-x11.h \ backends/x11/meta-idle-monitor-xsync.c \ backends/x11/meta-idle-monitor-xsync.h \ backends/x11/meta-monitor-manager-xrandr.c \ diff --git a/src/backends/meta-cursor-tracker.c b/src/backends/meta-cursor-tracker.c index 6005e5965..16e876d45 100644 --- a/src/backends/meta-cursor-tracker.c +++ b/src/backends/meta-cursor-tracker.c @@ -97,11 +97,6 @@ sync_cursor (MetaCursorTracker *tracker) static void meta_cursor_tracker_init (MetaCursorTracker *self) { - /* (JS) Best (?) that can be assumed since XFixes doesn't provide a way of - detecting if the system mouse cursor is showing or not. - - On wayland we start with the cursor showing - */ self->is_showing = TRUE; } @@ -473,19 +468,7 @@ meta_cursor_tracker_set_pointer_visible (MetaCursorTracker *tracker, return; tracker->is_showing = visible; - if (meta_is_wayland_compositor ()) - { - sync_cursor (tracker); - } - else - { - if (visible) - XFixesShowCursor (tracker->screen->display->xdisplay, - tracker->screen->xroot); - else - XFixesHideCursor (tracker->screen->display->xdisplay, - tracker->screen->xroot); - } + sync_cursor (tracker); } MetaCursorReference * diff --git a/src/backends/x11/meta-backend-x11.c b/src/backends/x11/meta-backend-x11.c index 7716780f6..923963bef 100644 --- a/src/backends/x11/meta-backend-x11.c +++ b/src/backends/x11/meta-backend-x11.c @@ -33,6 +33,7 @@ #include "meta-idle-monitor-xsync.h" #include "meta-monitor-manager-xrandr.h" #include "backends/meta-monitor-manager-dummy.h" +#include "meta-cursor-renderer-x11.h" #include #include "display-private.h" @@ -66,21 +67,11 @@ handle_alarm_notify (MetaBackend *backend, meta_idle_monitor_xsync_handle_xevent (backend->device_monitors[i], (XSyncAlarmNotifyEvent*) event); } -static Window -get_stage_window (MetaBackendX11 *x11) -{ - MetaDisplay *display = meta_get_display (); - MetaCompositor *compositor = display->compositor; - ClutterStage *stage = CLUTTER_STAGE (compositor->stage); - - return clutter_x11_get_stage_window (stage); -} - static void translate_device_event (MetaBackendX11 *x11, XIDeviceEvent *device_event) { - Window stage_window = get_stage_window (x11); + Window stage_window = meta_backend_x11_get_xwindow (x11); if (device_event->event != stage_window) { @@ -305,6 +296,12 @@ meta_backend_x11_create_monitor_manager (MetaBackend *backend) return g_object_new (META_TYPE_MONITOR_MANAGER_XRANDR, NULL); } +static MetaCursorRenderer * +meta_backend_x11_create_cursor_renderer (MetaBackend *backend) +{ + return g_object_new (META_TYPE_CURSOR_RENDERER_X11, NULL); +} + static gboolean meta_backend_x11_grab_device (MetaBackend *backend, int device_id, @@ -325,7 +322,7 @@ meta_backend_x11_grab_device (MetaBackend *backend, XISetMask (mask.mask, XI_KeyRelease); ret = XIGrabDevice (priv->xdisplay, device_id, - get_stage_window (x11), + meta_backend_x11_get_xwindow (x11), timestamp, None, XIGrabModeAsync, XIGrabModeAsync, @@ -357,6 +354,7 @@ meta_backend_x11_class_init (MetaBackendX11Class *klass) backend_class->post_init = meta_backend_x11_post_init; backend_class->create_idle_monitor = meta_backend_x11_create_idle_monitor; backend_class->create_monitor_manager = meta_backend_x11_create_monitor_manager; + backend_class->create_cursor_renderer = meta_backend_x11_create_cursor_renderer; backend_class->grab_device = meta_backend_x11_grab_device; backend_class->ungrab_device = meta_backend_x11_ungrab_device; @@ -377,3 +375,15 @@ meta_backend_x11_get_xdisplay (MetaBackendX11 *x11) return priv->xdisplay; } +Window +meta_backend_x11_get_xwindow (MetaBackendX11 *x11) +{ + MetaDisplay *display = meta_get_display (); + MetaCompositor *compositor = display->compositor; + + if (compositor == NULL) + return None; + + ClutterStage *stage = CLUTTER_STAGE (compositor->stage); + return clutter_x11_get_stage_window (stage); +} diff --git a/src/backends/x11/meta-backend-x11.h b/src/backends/x11/meta-backend-x11.h index 204654ea5..040f927d3 100644 --- a/src/backends/x11/meta-backend-x11.h +++ b/src/backends/x11/meta-backend-x11.h @@ -53,4 +53,6 @@ GType meta_backend_x11_get_type (void) G_GNUC_CONST; Display * meta_backend_x11_get_xdisplay (MetaBackendX11 *backend); +Window meta_backend_x11_get_xwindow (MetaBackendX11 *backend); + #endif /* META_BACKEND_X11_H */ diff --git a/src/backends/x11/meta-cursor-renderer-x11.c b/src/backends/x11/meta-cursor-renderer-x11.c new file mode 100644 index 000000000..051afb52e --- /dev/null +++ b/src/backends/x11/meta-cursor-renderer-x11.c @@ -0,0 +1,99 @@ +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ + +/* + * Copyright (C) 2014 Red Hat + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or (at your option) any later version. + * + * This program 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 + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA + * 02111-1307, USA. + * + * Written by: + * Jasper St. Pierre + */ + +#include "config.h" + +#include "meta-cursor-renderer-x11.h" + +#include "meta-backend-x11.h" +#include "meta-stage.h" + +struct _MetaCursorRendererX11Private +{ + gboolean server_cursor_visible; +}; +typedef struct _MetaCursorRendererX11Private MetaCursorRendererX11Private; + +G_DEFINE_TYPE_WITH_PRIVATE (MetaCursorRendererX11, meta_cursor_renderer_x11, META_TYPE_CURSOR_RENDERER); + +static gboolean +meta_cursor_renderer_x11_update_cursor (MetaCursorRenderer *renderer) +{ + MetaCursorRendererX11 *x11 = META_CURSOR_RENDERER_X11 (renderer); + MetaCursorRendererX11Private *priv = meta_cursor_renderer_x11_get_instance_private (x11); + + MetaBackendX11 *backend = META_BACKEND_X11 (meta_get_backend ()); + Window xwindow = meta_backend_x11_get_xwindow (backend); + + if (xwindow == None) + return FALSE; + + Display *xdisplay = meta_backend_x11_get_xdisplay (backend); + + MetaCursorReference *cursor_ref = meta_cursor_renderer_get_cursor (renderer); + gboolean has_server_cursor = FALSE; + + if (cursor_ref) + { + MetaCursor cursor = meta_cursor_reference_get_meta_cursor (cursor_ref); + if (cursor != META_CURSOR_NONE) + { + Cursor xcursor = meta_cursor_create_x_cursor (xdisplay, cursor); + XDefineCursor (xdisplay, xwindow, xcursor); + XFlush (xdisplay); + XFreeCursor (xdisplay, xcursor); + + has_server_cursor = TRUE; + } + } + + if (has_server_cursor != priv->server_cursor_visible) + { + if (has_server_cursor) + XFixesShowCursor (xdisplay, xwindow); + else + XFixesHideCursor (xdisplay, xwindow); + + priv->server_cursor_visible = has_server_cursor; + } + + return priv->server_cursor_visible; +} + +static void +meta_cursor_renderer_x11_class_init (MetaCursorRendererX11Class *klass) +{ + MetaCursorRendererClass *renderer_class = META_CURSOR_RENDERER_CLASS (klass); + + renderer_class->update_cursor = meta_cursor_renderer_x11_update_cursor; +} + +static void +meta_cursor_renderer_x11_init (MetaCursorRendererX11 *x11) +{ + MetaCursorRendererX11Private *priv = meta_cursor_renderer_x11_get_instance_private (x11); + + /* XFixes has no way to retrieve the current cursor visibility. */ + priv->server_cursor_visible = TRUE; +} diff --git a/src/backends/x11/meta-cursor-renderer-x11.h b/src/backends/x11/meta-cursor-renderer-x11.h new file mode 100644 index 000000000..398a32ec8 --- /dev/null +++ b/src/backends/x11/meta-cursor-renderer-x11.h @@ -0,0 +1,52 @@ +/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ + +/* + * Copyright (C) 2014 Red Hat + * + * This program is free software; you can redistribute it and/or + * modify it under the terms of the GNU General Public License as + * published by the Free Software Foundation; either version 2 of the + * License, or (at your option) any later version. + * + * This program 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 + * General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program; if not, write to the Free Software + * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA + * 02111-1307, USA. + * + * Written by: + * Jasper St. Pierre + */ + +#ifndef META_CURSOR_RENDERER_X11_H +#define META_CURSOR_RENDERER_X11_H + +#include "meta-cursor-renderer.h" + +#define META_TYPE_CURSOR_RENDERER_X11 (meta_cursor_renderer_x11_get_type ()) +#define META_CURSOR_RENDERER_X11(obj) (G_TYPE_CHECK_INSTANCE_CAST ((obj), META_TYPE_CURSOR_RENDERER_X11, MetaCursorRendererX11)) +#define META_CURSOR_RENDERER_X11_CLASS(klass) (G_TYPE_CHECK_CLASS_CAST ((klass), META_TYPE_CURSOR_RENDERER_X11, MetaCursorRendererX11Class)) +#define META_IS_CURSOR_RENDERER_X11(obj) (G_TYPE_CHECK_INSTANCE_TYPE ((obj), META_TYPE_CURSOR_RENDERER_X11)) +#define META_IS_CURSOR_RENDERER_X11_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), META_TYPE_CURSOR_RENDERER_X11)) +#define META_CURSOR_RENDERER_X11_GET_CLASS(obj) (G_TYPE_INSTANCE_GET_CLASS ((obj), META_TYPE_CURSOR_RENDERER_X11, MetaCursorRendererX11Class)) + +typedef struct _MetaCursorRendererX11 MetaCursorRendererX11; +typedef struct _MetaCursorRendererX11Class MetaCursorRendererX11Class; + +struct _MetaCursorRendererX11 +{ + MetaCursorRenderer parent; +}; + +struct _MetaCursorRendererX11Class +{ + MetaCursorRendererClass parent_class; +}; + +GType meta_cursor_renderer_x11_get_type (void) G_GNUC_CONST; + +#endif /* META_CURSOR_RENDERER_X11_H */