/* -*- mode: C; c-file-style: "gnu"; indent-tabs-mode: nil; -*- */ /* * Copyright (C) 2014-2015 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 * Jonas Ã…dahl */ /** * SECTION:barrier-x11 * @Title: MetaBarrierImplX11 * @Short_Description: Pointer barriers implementation for X11 */ #include "config.h" #include #include #include #include "backends/x11/meta-barrier-x11.h" #include "core/display-private.h" #include "meta/barrier.h" #include "x11/meta-x11-display-private.h" struct _MetaBarrierImplX11 { MetaBarrierImpl parent; MetaBarrier *barrier; PointerBarrier xbarrier; }; G_DEFINE_TYPE (MetaBarrierImplX11, meta_barrier_impl_x11, META_TYPE_BARRIER_IMPL) static gboolean _meta_barrier_impl_x11_is_active (MetaBarrierImpl *impl) { MetaBarrierImplX11 *self = META_BARRIER_IMPL_X11 (impl); return self->xbarrier != 0; } static void _meta_barrier_impl_x11_release (MetaBarrierImpl *impl, MetaBarrierEvent *event) { MetaBarrierImplX11 *self = META_BARRIER_IMPL_X11 (impl); MetaDisplay *display = self->barrier->priv->display; Display *dpy = meta_x11_display_get_xdisplay (display->x11_display); if (META_X11_DISPLAY_HAS_XINPUT_23 (display->x11_display)) { XIBarrierReleasePointer (dpy, META_VIRTUAL_CORE_POINTER_ID, self->xbarrier, event->event_id); } } static void _meta_barrier_impl_x11_destroy (MetaBarrierImpl *impl) { MetaBarrierImplX11 *self = META_BARRIER_IMPL_X11 (impl); MetaDisplay *display = self->barrier->priv->display; Display *dpy; if (display == NULL) return; dpy = meta_x11_display_get_xdisplay (display->x11_display); if (!meta_barrier_is_active (self->barrier)) return; XFixesDestroyPointerBarrier (dpy, self->xbarrier); g_hash_table_remove (display->x11_display->xids, &self->xbarrier); self->xbarrier = 0; } MetaBarrierImpl * meta_barrier_impl_x11_new (MetaBarrier *barrier) { MetaBarrierImplX11 *self; MetaDisplay *display = barrier->priv->display; Display *dpy; Window root; unsigned int allowed_motion_dirs; if (display == NULL) { g_warning ("A display must be provided when constructing a barrier."); return NULL; } self = g_object_new (META_TYPE_BARRIER_IMPL_X11, NULL); self->barrier = barrier; dpy = meta_x11_display_get_xdisplay (display->x11_display); root = DefaultRootWindow (dpy); allowed_motion_dirs = meta_border_get_allows_directions (&barrier->priv->border); self->xbarrier = XFixesCreatePointerBarrier (dpy, root, barrier->priv->border.line.a.x, barrier->priv->border.line.a.y, barrier->priv->border.line.b.x, barrier->priv->border.line.b.y, allowed_motion_dirs, 0, NULL); g_hash_table_insert (display->x11_display->xids, &self->xbarrier, barrier); return META_BARRIER_IMPL (self); } static void meta_barrier_fire_xevent (MetaBarrier *barrier, XIBarrierEvent *xevent) { MetaBarrierEvent *event = g_new0 (MetaBarrierEvent, 1); event->ref_count = 1; event->event_id = xevent->eventid; event->time = xevent->time; event->dt = xevent->dtime; event->x = xevent->root_x; event->y = xevent->root_y; event->dx = xevent->dx; event->dy = xevent->dy; event->released = (xevent->flags & XIBarrierPointerReleased) != 0; event->grabbed = (xevent->flags & XIBarrierDeviceIsGrabbed) != 0; switch (xevent->evtype) { case XI_BarrierHit: _meta_barrier_emit_hit_signal (barrier, event); break; case XI_BarrierLeave: _meta_barrier_emit_left_signal (barrier, event); break; default: g_assert_not_reached (); } meta_barrier_event_unref (event); } gboolean meta_x11_display_process_barrier_xevent (MetaX11Display *x11_display, XIEvent *event) { MetaBarrier *barrier; XIBarrierEvent *xev; if (event == NULL) return FALSE; switch (event->evtype) { case XI_BarrierHit: case XI_BarrierLeave: break; default: return FALSE; } xev = (XIBarrierEvent *) event; barrier = g_hash_table_lookup (x11_display->xids, &xev->barrier); if (barrier != NULL) { meta_barrier_fire_xevent (barrier, xev); return TRUE; } return FALSE; } static void meta_barrier_impl_x11_class_init (MetaBarrierImplX11Class *klass) { MetaBarrierImplClass *impl_class = META_BARRIER_IMPL_CLASS (klass); impl_class->is_active = _meta_barrier_impl_x11_is_active; impl_class->release = _meta_barrier_impl_x11_release; impl_class->destroy = _meta_barrier_impl_x11_destroy; } static void meta_barrier_impl_x11_init (MetaBarrierImplX11 *self) { }