/* GDK - The GIMP Drawing Kit * Copyright (C) 2015 Red Hat * * 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/>. * * Author: Carlos Garnacho <carlosg@gnome.org> */ #include "config.h" #include <glib-object.h> #include "gdkdisplay.h" #include "gdkdevice.h" #include "gdkdevicetoolprivate.h" #include "gdkseatprivate.h" #include "gdkdeviceprivate.h" #include "gdkintl.h" #include "gdkinternals.h" /** * SECTION:gdkseat * @Short_description: Object representing a user seat * @Title: GdkSeat * @See_also: #GdkDisplay, #GdkDevice * * The #GdkSeat object represents a collection of input devices * that belong to a user. */ /** * GdkSeat: * * The GdkSeat struct contains only private fields and * should not be accessed directly. */ typedef struct _GdkSeatPrivate GdkSeatPrivate; struct _GdkSeatPrivate { GdkDisplay *display; }; enum { DEVICE_ADDED, DEVICE_REMOVED, TOOL_ADDED, TOOL_REMOVED, N_SIGNALS }; enum { PROP_0, PROP_DISPLAY, N_PROPS }; static guint signals[N_SIGNALS] = { 0 }; static GParamSpec *props[N_PROPS] = { NULL }; G_DEFINE_ABSTRACT_TYPE_WITH_PRIVATE (GdkSeat, gdk_seat, G_TYPE_OBJECT) static void gdk_seat_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GdkSeatPrivate *priv = gdk_seat_get_instance_private (GDK_SEAT (object)); switch (prop_id) { case PROP_DISPLAY: priv->display = g_value_get_object (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gdk_seat_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { GdkSeatPrivate *priv = gdk_seat_get_instance_private (GDK_SEAT (object)); switch (prop_id) { case PROP_DISPLAY: g_value_set_object (value, priv->display); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void gdk_seat_class_init (GdkSeatClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->set_property = gdk_seat_set_property; object_class->get_property = gdk_seat_get_property; /** * GdkSeat::device-added: * @seat: the object on which the signal is emitted * @device: the newly added #GdkDevice. * * The ::device-added signal is emitted when a new input * device is related to this seat. */ signals [DEVICE_ADDED] = g_signal_new (g_intern_static_string ("device-added"), G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GdkSeatClass, device_added), NULL, NULL, NULL, G_TYPE_NONE, 1, GDK_TYPE_DEVICE); /** * GdkSeat::device-removed: * @seat: the object on which the signal is emitted * @device: the just removed #GdkDevice. * * The ::device-removed signal is emitted when an * input device is removed (e.g. unplugged). */ signals [DEVICE_REMOVED] = g_signal_new (g_intern_static_string ("device-removed"), G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (GdkSeatClass, device_removed), NULL, NULL, NULL, G_TYPE_NONE, 1, GDK_TYPE_DEVICE); /** * GdkSeat::tool-added: * @seat: the object on which the signal is emitted * @tool: the new #GdkDeviceTool known to the seat * * The ::tool-added signal is emitted whenever a new tool * is made known to the seat. The tool may later be assigned * to a device (i.e. on proximity with a tablet). The device * will emit the #GdkDevice::tool-changed signal accordingly. * * A same tool may be used by several devices. */ signals [TOOL_ADDED] = g_signal_new (g_intern_static_string ("tool-added"), G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 1, GDK_TYPE_DEVICE_TOOL); /** * GdkSeat::tool-removed: * @seat: the object on which the signal is emitted * @tool: the just removed #GdkDeviceTool * * This signal is emitted whenever a tool is no longer known * to this @seat. */ signals [TOOL_REMOVED] = g_signal_new (g_intern_static_string ("tool-removed"), G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, 0, NULL, NULL, NULL, G_TYPE_NONE, 1, GDK_TYPE_DEVICE_TOOL); /** * GdkSeat:display: * * #GdkDisplay of this seat. */ props[PROP_DISPLAY] = g_param_spec_object ("display", P_("Display"), P_("Display"), GDK_TYPE_DISPLAY, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_STRINGS); g_object_class_install_properties (object_class, N_PROPS, props); } static void gdk_seat_init (GdkSeat *seat) { } /** * gdk_seat_get_capabilities: * @seat: a #GdkSeat * * Returns the capabilities this #GdkSeat currently has. * * Returns: the seat capabilities **/ GdkSeatCapabilities gdk_seat_get_capabilities (GdkSeat *seat) { GdkSeatClass *seat_class; g_return_val_if_fail (GDK_IS_SEAT (seat), GDK_SEAT_CAPABILITY_NONE); seat_class = GDK_SEAT_GET_CLASS (seat); return seat_class->get_capabilities (seat); } /* * gdk_seat_grab: * @seat: a #GdkSeat * @surface: the #GdkSurface which will own the grab * @capabilities: capabilities that will be grabbed * @owner_events: if %FALSE then all device events are reported with respect to * @surface and are only reported if selected by @event_mask. If * %TRUE then pointer events for this application are reported * as normal, but pointer events outside this application are * reported with respect to @surface and only if selected by * @event_mask. In either mode, unreported events are discarded. * @cursor: (nullable): the cursor to display while the grab is active. If * this is %NULL then the normal cursors are used for * @surface and its descendants, and the cursor for @surface is used * elsewhere. * @event: (nullable): the event that is triggering the grab, or %NULL if none * is available. * @prepare_func: (nullable) (scope call): function to * prepare the surface to be grabbed, it can be %NULL if @surface is * visible before this call. * @prepare_func_data: (closure): user data to pass to @prepare_func * * Grabs the seat so that all events corresponding to the given @capabilities * are passed to this application until the seat is ungrabbed with gdk_seat_ungrab(), * or the surface becomes hidden. This overrides any previous grab on the * seat by this client. * * As a rule of thumb, if a grab is desired over %GDK_SEAT_CAPABILITY_POINTER, * all other "pointing" capabilities (eg. %GDK_SEAT_CAPABILITY_TOUCH) should * be grabbed too, so the user is able to interact with all of those while * the grab holds, you should thus use %GDK_SEAT_CAPABILITY_ALL_POINTING most * commonly. * * Grabs are used for operations which need complete control over the * events corresponding to the given capabilities. For example in GTK this * is used for Drag and Drop operations, popup menus and such. * * Note that if the event mask of a #GdkSurface has selected both button press * and button release events, or touch begin and touch end, then a press event * will cause an automatic grab until the button is released, equivalent to a * grab on the surface with @owner_events set to %TRUE. This is done because most * applications expect to receive paired press and release events. * * If you set up anything at the time you take the grab that needs to be * cleaned up when the grab ends, you should handle the #GdkEventGrabBroken * events that are emitted when the grab ends unvoluntarily. * * Returns: %GDK_GRAB_SUCCESS if the grab was successful. **/ GdkGrabStatus gdk_seat_grab (GdkSeat *seat, GdkSurface *surface, GdkSeatCapabilities capabilities, gboolean owner_events, GdkCursor *cursor, GdkEvent *event, GdkSeatGrabPrepareFunc prepare_func, gpointer prepare_func_data) { GdkSeatClass *seat_class; g_return_val_if_fail (GDK_IS_SEAT (seat), GDK_GRAB_FAILED); g_return_val_if_fail (GDK_IS_SURFACE (surface), GDK_GRAB_FAILED); g_return_val_if_fail (gdk_surface_get_display (surface) == gdk_seat_get_display (seat), GDK_GRAB_FAILED); capabilities &= GDK_SEAT_CAPABILITY_ALL; g_return_val_if_fail (capabilities != GDK_SEAT_CAPABILITY_NONE, GDK_GRAB_FAILED); seat_class = GDK_SEAT_GET_CLASS (seat); return seat_class->grab (seat, surface, capabilities, owner_events, cursor, event, prepare_func, prepare_func_data); } /* * gdk_seat_ungrab: * @seat: a #GdkSeat * * Releases a grab added through gdk_seat_grab(). **/ void gdk_seat_ungrab (GdkSeat *seat) { GdkSeatClass *seat_class; g_return_if_fail (GDK_IS_SEAT (seat)); seat_class = GDK_SEAT_GET_CLASS (seat); seat_class->ungrab (seat); } /** * gdk_seat_get_devices: * @seat: a #GdkSeat * @capabilities: capabilities to get devices for * * Returns the devices that match the given capabilities. * * Returns: (transfer container) (element-type GdkDevice): A list of #GdkDevices. * The list must be freed with g_list_free(), the elements are owned * by GTK and must not be freed. **/ GList * gdk_seat_get_devices (GdkSeat *seat, GdkSeatCapabilities capabilities) { GdkSeatClass *seat_class; g_return_val_if_fail (GDK_IS_SEAT (seat), NULL); seat_class = GDK_SEAT_GET_CLASS (seat); return seat_class->get_devices (seat, capabilities); } /** * gdk_seat_get_pointer: * @seat: a #GdkSeat * * Returns the device that routes pointer events. * * Returns: (transfer none) (nullable): a #GdkDevice with pointer * capabilities. This object is owned by GTK and must not be freed. **/ GdkDevice * gdk_seat_get_pointer (GdkSeat *seat) { GdkSeatClass *seat_class; g_return_val_if_fail (GDK_IS_SEAT (seat), NULL); seat_class = GDK_SEAT_GET_CLASS (seat); return seat_class->get_logical_device (seat, GDK_SEAT_CAPABILITY_POINTER); } /** * gdk_seat_get_keyboard: * @seat: a #GdkSeat * * Returns the device that routes keyboard events. * * Returns: (transfer none) (nullable): a #GdkDevice with keyboard * capabilities. This object is owned by GTK and must not be freed. **/ GdkDevice * gdk_seat_get_keyboard (GdkSeat *seat) { GdkSeatClass *seat_class; g_return_val_if_fail (GDK_IS_SEAT (seat), NULL); seat_class = GDK_SEAT_GET_CLASS (seat); return seat_class->get_logical_device (seat, GDK_SEAT_CAPABILITY_KEYBOARD); } void gdk_seat_device_added (GdkSeat *seat, GdkDevice *device) { gdk_device_set_seat (device, seat); g_signal_emit (seat, signals[DEVICE_ADDED], 0, device); } void gdk_seat_device_removed (GdkSeat *seat, GdkDevice *device) { gdk_device_set_seat (device, NULL); g_signal_emit (seat, signals[DEVICE_REMOVED], 0, device); } /** * gdk_seat_get_display: * @seat: a #GdkSeat * * Returns the #GdkDisplay this seat belongs to. * * Returns: (transfer none): a #GdkDisplay. This object is owned by GTK * and must not be freed. **/ GdkDisplay * gdk_seat_get_display (GdkSeat *seat) { GdkSeatPrivate *priv = gdk_seat_get_instance_private (seat); g_return_val_if_fail (GDK_IS_SEAT (seat), NULL); return priv->display; } void gdk_seat_tool_added (GdkSeat *seat, GdkDeviceTool *tool) { g_signal_emit (seat, signals[TOOL_ADDED], 0, tool); } void gdk_seat_tool_removed (GdkSeat *seat, GdkDeviceTool *tool) { g_signal_emit (seat, signals[TOOL_REMOVED], 0, tool); } GdkDeviceTool * gdk_seat_get_tool (GdkSeat *seat, guint64 serial, guint64 hw_id) { GdkDeviceTool *match = NULL; GList *tools, *l; tools = gdk_seat_get_tools (seat); for (l = tools; l; l = l->next) { GdkDeviceTool *tool = l->data; if (tool->serial == serial && tool->hw_id == hw_id) { match = tool; break; } } g_list_free (tools); return match; } /** * gdk_seat_get_tools: * @seat: A #GdkSeat * * Returns all #GdkDeviceTools that are known to the * application. * * Returns: (transfer container) (element-type Gdk.DeviceTool): * A list of tools. Free with g_list_free(). **/ GList * gdk_seat_get_tools (GdkSeat *seat) { GdkSeatClass *seat_class; g_return_val_if_fail (GDK_IS_SEAT (seat), NULL); seat_class = GDK_SEAT_GET_CLASS (seat); return seat_class->get_tools (seat); }