/* SPDX-License-Identifier: MIT */ /* * Copyright © 2020 Red Hat, Inc. * * Permission is hereby granted, free of charge, to any person obtaining a * copy of this software and associated documentation files (the "Software"), * to deal in the Software without restriction, including without limitation * the rights to use, copy, modify, merge, publish, distribute, sublicense, * and/or sell copies of the Software, and to permit persons to whom the * Software is furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice (including the next * paragraph) shall be included in all copies or substantial portions of the * Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER * DEALINGS IN THE SOFTWARE. */ #pragma once #ifdef __cplusplus extern "C" { #endif #include #include #include /** * @defgroup libei 🥚 EI - The client API * * libei is the client-side module. This API should be used by processes * that need to emulate devices or capture logical events from existing devices. * * For an example client see the [`tools/` directory in the libei * repository](https://gitlab.freedesktop.org/libinput/libei/-/tree/main/tools). * At its core, a libei client will * - create a context with ei_new_sender() or ei_new_receiver() * - set up a backend with ei_setup_backend_fd() or ei_setup_backend_socket() * - register the ei_get_fd() with its own event loop * - call ei_dispatch() whenever the fd triggers * - call ei_get_event() and process incoming events * * Those events are typically a sequence of * - @ref EI_EVENT_CONNECT - notification that the client was connected * - @ref EI_EVENT_SEAT_ADDED - notification that a @ref ei_seat is * available. The client should respond with ei_seat_bind_capabilities() * - @ref EI_EVENT_DEVICE_ADDED - notification that a @ref ei_device is * available in a seat. * - @ref EI_EVENT_DEVICE_RESUMED - notification that the @ref ei_device is * ready to emulate events. The client should respond with * ei_device_start_emulating() and the required events to emulate input * * libei clients come in @ref ei_new_sender "sender" and @ref ei_new_receiver * "receiver" modes, depending on whether the client sends or receives events * from the EIS implementation. See @ref libei-sender and @ref libei-receiver * for API calls specific to either mode. * * @note A libei context is restricted to either sender or receiver mode, not * both. The EIS implementation however may accept both sender and receiver * clients, and will work as corresponding receiver or sender for this * client. It is up to the implementation to disconnect clients that it does not * want to allow. See eis_client_is_sender() for details. * * @defgroup libei-log The logging API * @ingroup libei * * The API to control logging output. * * @defgroup libei-receiver API for receiver clients * @ingroup libei * * The receiver client API is available only to clients created with * ei_new_receiver(). For those clients the EIS implementation creates * devices and sends events **to** the libei client. The primary use-case * for this is input capturing, e.g. InputLeap. * * It is a client bug to call any of these functions for a client created * with ei_new_sender(). * * @defgroup libei-sender API for sender clients * @ingroup libei * * The sender client API is available only to clients created with * ei_new_sender(). For those clients the EIS implementation creates * devices but it is the libei client that sends events **to** EIS * implementation. * The primary use-case is input emulation from a client, akin to xdotool. * * It is a client bug to call any of these functions for a client created * with ei_new_receiver(). * * @defgroup libei-seat The seat API * @ingroup libei * * The API to query and interact with a struct @ref ei_seat * * @defgroup libei-device The device API * @ingroup libei * * The API to query and interact with a struct @ref ei_device * * @defgroup libei-region The region API * @ingroup libei * * The API to query a struct @ref ei_region for information * * @defgroup libei-keymap The keymap API * @ingroup libei * * The API to query a struct @ref ei_keymap for information * */ /** * @addtogroup libei * @{ */ /** * @struct ei * * The main context to interact with libei. A libei context is a single * connection to an EIS implementation and may contain multiple devices, see * @ref ei_device. * * An @ref ei context is refcounted, see ei_unref(). */ struct ei; /** * @struct ei_device * * A single device to generate input events from. A device may have multiple * capabilities. For example, a single device may be a pointer and a keyboard * and a touch device. It is up to the EIS implementation on how to handle * this case, some implementations may split a single device up into * multiple virtual devices, others may not. * * An @ref ei_device is refcounted, see ei_device_unref(). */ struct ei_device; /** * @struct ei_seat * * A logical seat for a group of devices. Seats are provided by the EIS * implementation, devices may be added to a seat. The hierarchy of objects * looks like this: *
 *    ei ---- ei_seat one ---- ei_device 1
 *       \                \
 *        \                --- ei device 2
 *         --- ei_seat two --- ei device 3
 * 
*/ struct ei_seat; /** * @struct ei_event * * An event received from the EIS implementation. See @ref ei_event_type * for the list of possible event types. * * An @ref ei_event is refcounted, see ei_event_unref(). */ struct ei_event; /** * @struct ei_keymap * * A keymap for a device with the @ref EI_DEVICE_CAP_KEYBOARD capability. * * An @ref ei_keymap is refcounted, see ei_keymap_unref(). */ struct ei_keymap; /** * @struct ei_region * @ingroup libei-region * * A rectangular region, defined by an x/y offset and a width and a height. * A region defines the area on an EIS desktop layout that is accessible by * this device - this region may not be the full area of the desktop. * Input events may only be sent for points within the regions. * * The use of regions is private to the EIS compositor and coordinates may not * match the size of the actual desktop. For example, a compositor may set a * 1920x1080 region to represent a 4K monitor and transparently map input * events into the respective true pixels. * * Absolute devices may have different regions, it is up to the libei client * to send events through the correct device to target the right pixel. For * example, a dual-head setup may have two absolute devices, the first with a * zero offset region spanning the first screen, the second with a nonzero * offset spanning the second screen. */ struct ei_region; /** * @struct ei_touch * * A single touch initiated by a sender context. * * @see ei_device_touch_new * @see ei_touch_down * @see ei_touch_motion * @see ei_touch_up */ struct ei_touch; /** * @struct ei_ping * * A callback struct returned by ei_new_ping() to handle * roundtrips to the EIS implementation. * * @see ei_new_ping */ struct ei_ping; /** * @enum ei_device_type * @ingroup libei-device * * The device type determines what the device represents. * * If the device type is @ref EI_DEVICE_TYPE_VIRTUAL, the device is a * virtual device representing input as applied on the EIS implementation's * screen. A relative virtual device generates input events in logical pixels, * an absolute virtual device generates input events in logical pixels on one * of the device's regions. Virtual devices do not have a size. * * If the device type is @ref EI_DEVICE_TYPE_PHYSICAL, the device is a * representation of a physical device as if connected to the EIS * implementation's host computer. A relative physical device generates input * events in mm, an absolute physical device generates input events in mm * within the device's specified physical size. Physical devices do not have * regions. * * @see ei_device_get_width * @see ei_device_get_height */ enum ei_device_type { EI_DEVICE_TYPE_VIRTUAL = 1, EI_DEVICE_TYPE_PHYSICAL }; /** * @enum ei_device_capability * * The set of supported capabilities. A device may have zero or more * capabilities, a device with perceived zero capabilities is typically a * device with capabilities unsupported by the client - such a device should be * @ref ei_device_close "closed" immediately by the client. * * The EIS implementation decides which capabilities may exist on a seat and * which capabilities exist on any individual seat. However, a device may only * have capabilities the client has bound to, see * ei_seat_bind_capabilities(). * * For example, a client may bind to a seat with the pointer and keyboard * capability but a given device only has the pointer capability. * Keyboard events sent through that device will be treated as client bug. * * If a client calls ei_seat_unbind_capabilities() for a capability, the EIS * implementation may remove any or all devices that have that capability. * * See ei_device_has_capability(). * */ enum ei_device_capability { /** * The device can send relative motion events */ EI_DEVICE_CAP_POINTER = (1 << 0), /** * The device can send absolute motion events */ EI_DEVICE_CAP_POINTER_ABSOLUTE = (1 << 1), /** * The device can send keyboard events */ EI_DEVICE_CAP_KEYBOARD = (1 << 2), /** * The device can send touch events */ EI_DEVICE_CAP_TOUCH = (1 << 3), /** * The device can send scroll events */ EI_DEVICE_CAP_SCROLL = (1 << 4), /** * The device can send button events */ EI_DEVICE_CAP_BUTTON = (1 << 5), /** * The device can send text-like data * * @since 1.6 */ EI_DEVICE_CAP_TEXT = (1 << 6), }; /** * @enum ei_keymap_type * * The set of supported keymap types for a struct @ref ei_keymap. */ enum ei_keymap_type { /** * A libxkbcommon-compatible XKB keymap. */ EI_KEYMAP_TYPE_XKB = 1, }; /** * @enum ei_event_type * * This enum is not exhaustive, future versions of this library may add * new event types. * * Unknown events must be released by the caller with ei_event_unref(), * see ei_get_event(). */ enum ei_event_type { /** * The server has approved the connection to this client. Where the * server does not approve the connection, @ref EI_EVENT_DISCONNECT is * sent instead. * * This event is only sent once after the initial connection * request. */ EI_EVENT_CONNECT = 1, /** * The server has disconnected this client - all resources left to * reference this server are now obsolete. Once this event has been * received, the struct @ref ei and all its associated resources * should be released. * * This event may occur at any time after the connection has been * made and is the last event to be received by this ei instance. * * libei guarantees that a @ref EI_EVENT_DISCONNECT is provided to * the caller even where the server does not send one. */ EI_EVENT_DISCONNECT, /** * The server has added a seat available to this client. * * libei guarantees that any seat added has a corresponding @ref * EI_EVENT_SEAT_REMOVED event before @ref EI_EVENT_DISCONNECT. * libei guarantees that any device in this seat generates a @ref * EI_EVENT_DEVICE_REMOVED event before the @ref * EI_EVENT_SEAT_REMOVED event. */ EI_EVENT_SEAT_ADDED, /** * The server has removed a seat previously available to this * client. The caller should release the struct @ref ei_seat and * all its associated resources. No devices will be added to this seat * anymore. * * libei guarantees that any device in this seat generates a @ref * EI_EVENT_DEVICE_REMOVED event before the @ref * EI_EVENT_SEAT_REMOVED event. */ EI_EVENT_SEAT_REMOVED, /** * The server has added a device for this client. The capabilities * of the device may be a subset of the seat capabilities - it is up * to the client to verify the minimum required capabilities are * indeed set. * * libei guarantees that any device added has a corresponding @ref * EI_EVENT_DEVICE_REMOVED event before @ref EI_EVENT_DISCONNECT. */ EI_EVENT_DEVICE_ADDED, /** * The server has removed a device belonging to this client. The * caller should release the struct @ref ei_device and all its * associated resources. Any events sent through a removed device * are discarded. * * When this event is received, the device is already removed. A * caller does not need to call ei_device_close() even on this * device. */ EI_EVENT_DEVICE_REMOVED, /** * Any events sent from this device will be discarded until the next * resume. * * Pausing a device resets the logical state of the device to neutral. * This includes: * - any buttons or keys logically down are released * - any modifiers logically down are released * - any touches logically down are released * * Sender clients must wait until @ref EI_EVENT_DEVICE_RESUMED * before sending events. */ EI_EVENT_DEVICE_PAUSED, /** * The client may send events. * * Once resumed, a sender client may call ei_device_start_emulating() * and begin emulating events. */ EI_EVENT_DEVICE_RESUMED, /** * The server has changed the modifier state on the device's * keymap. See * ei_event_keyboard_get_xkb_mods_depressed(), * ei_event_keyboard_get_xkb_mods_latched(), * ei_event_keyboard_get_xkb_mods_locked(), and * ei_event_keyboard_get_xkb_group(). * * This event is sent in response to any modifier state or effective * group change, including where the change is triggered by a client * call to ei_device_keyboard_key(). * * For receiver clients, this will always be properly ordered with * EI_EVENT_KEYBOARD_KEY events, so each key event should be * interpreted using the most recently received modifier * state. * * For sender clients, this event is not inherently synchronized * with calls to ei_device_keyboard_key(), but the client may call * ei_ping() when synchronization is required. When the corresponding * EI_EVENT_PONG event is received, all key events sent prior to the * sync request are guaranteed to have been processed, and any * directly-resulting modifiers events are guaranteed to have been * received. Note, however, that it is still possible for * indirectly-triggered state changes, such as via a keyboard * shortcut not encoded in the keymap, to be reported after the done * event. * * This event may arrive while a device is paused. * * Note: It was previously specified that a where a sender client * triggers a modifier state change in response to * ei_device_keyboard_key(), no MODIFIERS event would be sent. * Clients were expected to mix calls to xkb_state_update_key() and * xkb_state_update_mask() to track the state with libxkbcommon, * which could lead to disagreements between the client and server as * to the current state. */ EI_EVENT_KEYBOARD_MODIFIERS, /** * Returned in response to ei_ping(). * * Note that in the ei protocol, the matching client request is called `sync` * but for consistency in the libeis/libei API the C API uses * ei_ping() with `EI_EVENT_PONG` as return. */ EI_EVENT_PONG = 90, /** * This event represents a synchronization request (ping) from the EIS * implementation. The corresponding reply (pong) will be sent when * it is unref'd. It has no other API. * * The caller must ensure that any state changes triggered by messages * received prior to this event have been resolved and communicated to the * EIS implementation prior to calling ei_event_unref(). */ EI_EVENT_SYNC, /** * "Hardware" frame event. This event **must** be sent by the server * and notifies the client that the previous set of events belong to * the same logical hardware event. * * @note These events are only generated on a receiver ei context. * See ei_device_frame() for the sender context API. */ EI_EVENT_FRAME = 100, /** * The server is about to send events for a device. This event should * be used by the client to clear the logical state of the emulated * devices and/or provide UI to the user. * * @note These events are only generated on a receiver ei context. * See ei_device_start_emulating() for the sender context API. * * Note that a server may start multiple emulating sequences * simultaneously, depending on the devices available. * For example, in a synergy-like situation, the server * may start sending pointer and keyboard once the remote device * logically entered the screen. */ EI_EVENT_DEVICE_START_EMULATING = 200, /** * The server stopped emulating events on this device, * see @ref EIS_EVENT_DEVICE_START_EMULATING. * * @note These events are only generated on a receiver ei context. * See ei_device_stop_emulating() for the sender context API. */ EI_EVENT_DEVICE_STOP_EMULATING, /** * A relative motion event with delta coordinates in logical pixels or * mm, depending on the device type. * * @note This event is only generated on a receiver ei context. * See ei_device_pointer_motion() for the sender context API. */ EI_EVENT_POINTER_MOTION = 300, /** * An absolute motion event with absolute position within the device's * regions or size, depending on the device type. * * @note This event is only generated on a receiver ei context. * See ei_device_pointer_motion_absolute() for the sender context API. */ EI_EVENT_POINTER_MOTION_ABSOLUTE = 400, /** * A button press or release event * * @note This event is only generated on a receiver ei context. * See ei_device_button_button() for the sender context API. */ EI_EVENT_BUTTON_BUTTON = 500, /** * A vertical and/or horizontal scroll event with logical-pixels * or mm precision, depending on the device type. * * @note This event is only generated on a receiver ei context. * See ei_device_scroll_delta() for the sender context API. */ EI_EVENT_SCROLL_DELTA = 600, /** * An ongoing scroll sequence stopped. * * @note This event is only generated on a receiver ei context. * See ei_device_scroll_stop() for the sender context API. */ EI_EVENT_SCROLL_STOP, /** * An ongoing scroll sequence was cancelled. * * @note This event is only generated on a receiver ei context. * See ei_device_scroll_cancel() for the sender context API. */ EI_EVENT_SCROLL_CANCEL, /** * A vertical and/or horizontal scroll event with a discrete range in * logical scroll steps, like a scroll wheel. * * @note This event is only generated on a receiver ei context. * See ei_device_scroll_discrete() for the sender context API. */ EI_EVENT_SCROLL_DISCRETE, /** * A key press or release event * * @note This event is only generated on a receiver ei context. * See ei_device_keyboard_key() for the sender context API. */ EI_EVENT_KEYBOARD_KEY = 700, /** * Event for a single touch set down on the device's logical surface. * A touch sequence is always down/up with an optional motion event in * between. On multitouch capable devices, several touch sequences * may be active at any time. * * @note This event is only generated on a receiver ei context. * See ei_device_touch_new() and ei_touch_down() for the sender context API. */ EI_EVENT_TOUCH_DOWN = 800, /** * Event for a single touch released from the device's logical * surface. * * @note This event is only generated on a receiver ei context. * See ei_device_touch_new() and ei_touch_up() for the sender context API. */ EI_EVENT_TOUCH_UP, /** * Event for a single currently-down touch changing position (or other * properties). * * @note This event is only generated on a receiver ei context. * See ei_device_touch_new() and ei_touch_motion() for the sender context API. */ EI_EVENT_TOUCH_MOTION, /** * Event for a single keysym logically pressed/released on this device. * The keysym is an XKB-compatible keysym (not key code!) and may not be * present on any keymap currently active on any device. * * @note This event is only generated on a receiver ei context. * See ei_device_text_keysym() for the sender context API. * * @since 1.6 */ EI_EVENT_TEXT_KEYSYM = 900, /** * Event for a UTF-8 compatible text sequence sent by this device. * * @note This event is only generated on a receiver ei context. * See ei_device_text_utf8() for the sender context API. * * @since 1.6 */ EI_EVENT_TEXT_UTF8, }; /** * This is a debugging helper to return a string of the name of the event type, * or NULL if the event type is invalid. For example, for @ref EI_EVENT_TOUCH_UP this * function returns the string "EI_EVENT_TOUCH_UP". */ const char *ei_event_type_to_string(enum ei_event_type); /** * This is an alias for @ref ei_new_sender. */ struct ei * ei_new(void *user_data); /** * Create a new sender ei context. The context is refcounted and must be * released with ei_unref(). * * A sender ei context sends events to the EIS implementation but cannot * receive events. * * A context supports exactly one backend, set up with one of * ei_setup_backend_fd() or ei_setup_backend_socket(). * * @param user_data An opaque pointer to be returned with ei_get_user_data() * * @see ei_set_user_data * @see ei_get_user_data * @see ei_setup_backend_fd * @see ei_setup_backend_socket */ struct ei * ei_new_sender(void *user_data); /** * Create a new receiver ei context. The context is refcounted and must be * released with ei_unref(). * * A receiver ei context receives events from the EIS implementation but cannot * send events. * * A context supports exactly one backend, set up with one of * ei_setup_backend_fd() or ei_setup_backend_socket(). * * @param user_data An opaque pointer to be returned with ei_get_user_data() * * @see ei_set_user_data * @see ei_get_user_data * @see ei_setup_backend_fd * @see ei_setup_backend_socket */ struct ei * ei_new_receiver(void *user_data); /** * Increase the refcount of this struct by one. Use ei_unref() to decrease * the refcount. * * @return the argument passed into the function */ struct ei * ei_ref(struct ei *ei); /** * Decrease the refcount of this struct by one. When the refcount reaches * zero, the context disconnects from the server and all allocated resources * are released. * * @return always NULL */ struct ei * ei_unref(struct ei *ei); /** * Set a custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_get_user_data() to retrieve a previously set * user data. */ void ei_set_user_data(struct ei *ei, void *user_data); /** * Return the custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_set_user_data() to change the user data. */ void * ei_get_user_data(struct ei *ei); /** * Returns true if the context is was created with ei_new_sender() or false otherwise. */ bool ei_is_sender(struct ei *ei); /** * @ingroup libei-log */ enum ei_log_priority { EI_LOG_PRIORITY_DEBUG = 10, EI_LOG_PRIORITY_INFO = 20, EI_LOG_PRIORITY_WARNING = 30, EI_LOG_PRIORITY_ERROR = 40, }; struct ei_log_context; /** * @ingroup libei-log * @return the line number (``__LINE__``) for a given log message context. */ unsigned int ei_log_context_get_line(struct ei_log_context *ctx); /** * @ingroup libei-log * @return the file name (``__FILE__``) for a given log message context. */ const char * ei_log_context_get_file(struct ei_log_context *ctx); /** * @ingroup libei-log * @return the function name (``__func__``) for a given log message context. */ const char * ei_log_context_get_func(struct ei_log_context *ctx); /** * @ingroup libei-log * * The log handler for library logging. This handler is only called for * messages with a log level equal or greater than than the one set in * ei_log_set_priority(). * * The context passed to this function contains auxiliary information about * this log message such as the line number, file name and function name * this message occurred in. The log context is valid only within the current * invocation of the log handler. * * @param ei The EI context * @param priority The log priority * @param message The log message as a null-terminated string * @param context A log message context for this message */ typedef void (*ei_log_handler)(struct ei *ei, enum ei_log_priority priority, const char *message, struct ei_log_context *context); /** * @ingroup libei-log * * Change the log handler for this context. If the log handler is NULL, the * built-in default log function is used. * * @param ei The EI context * @param log_handler The log handler or NULL to use the default log * handler. */ void ei_log_set_handler(struct ei *ei, ei_log_handler log_handler); /** * @ingroup libei-log */ void ei_log_set_priority(struct ei *ei, enum ei_log_priority priority); /** * @ingroup libei-log */ enum ei_log_priority ei_log_get_priority(const struct ei *ei); /** * Optional override function for ei_now(). * * By default ei_now() returns the current timestamp in CLOCK_MONOTONIC. This * may be overridden by a caller to provide a different timestamp. * * There is rarely a need to override this function. It exists for the libei-internal test suite. */ typedef uint64_t (*ei_clock_now_func)(struct ei *ei); /** * Override the function that returns the current time ei_now(). * * There is rarely a need to override this function. It exists for the libei-internal test suite. */ void ei_clock_set_now_func(struct ei *, ei_clock_now_func func); /** * Set the name for this client. This is a suggestion to the * server only and may not be honored. * * The client name may be used for display to the user, for example in * an authorization dialog that requires the user to approve a connection to * the EIS implementation. * * This function must be called immediately after ei_new() and before * setting up a backend with ei_setup_backend_fd() or * ei_setup_backend_socket(). */ void ei_configure_name(struct ei *ei, const char *name); /** * Initialize the ei context on the given socket file descriptor. * The ei context will initiate the conversation with the EIS server listening * on the other end of this socket. * * This is the preferred entry point for libei and should be the default * used in new clients. It allows for private-by-default socket file descriptors * and the policy on how the socket is created is delegated to the caller. * * If the connection was successful, an event of type @ref EI_EVENT_CONNECT * or @ref EI_EVENT_DISCONNECT will become available after a future call to * ei_dispatch(). * * If the connection failed, use ei_unref() to release the data allocated * for this context. * * This function takes ownership of the file descriptor, and will close it * when tearing down. * * @return zero on success or a negative errno on failure */ int ei_setup_backend_fd(struct ei *ei, int fd); /** * Set this ei context to use the socket backend. The ei context will * connect to the socket at the given path and initiate the conversation * with the EIS server listening on that socket. * * @note This backend requires the socket to be accessible * to connect to and forces the EIS implementation to do identification and * access control for clients. In almost all cases, the EIS implementation * should use pre-created fds per client and the client should instead use * ei_setup_backend_fd(). This backend is primarily useful for testing and * debugging. * * If @a socketpath is `NULL`, the value of the environment variable * `LIBEI_SOCKET` is used. If @a socketpath does not start with '/', it is * relative to `$XDG_RUNTIME_DIR`. If `XDG_RUNTIME_DIR` is not set, this * function fails. * * If the connection was successful, an event of type @ref EI_EVENT_CONNECT * or @ref EI_EVENT_DISCONNECT will become available after a future call to * ei_dispatch(). * * If the connection failed, use ei_unref() to release the data allocated * for this context. * * @return zero on success or a negative errno on failure */ int ei_setup_backend_socket(struct ei *ei, const char *socketpath); /** * libei keeps a single file descriptor for all events. This fd should be * monitored for events by the caller's mainloop, e.g. using select(). When * events are available on this fd, call ei_dispatch() immediately to * process. */ int ei_get_fd(struct ei *ei); /** * Create a new ei_ping object to trigger a round trip to the EIS implementation. * See ei_ping() for details. * * The returned @ref ei_ping is refcounted, use ei_ping_unref() to * drop the reference. * * @since 1.4 */ struct ei_ping * ei_new_ping(struct ei *ei); /** * Return a unique, increasing id for this struct. This ID is assigned at * struct creation and constant for the lifetime of the struct. The ID * does not exist at the protocol level. * * This is a convenience API to make it easier to put this struct into * e.g. a hashtable without having to use the pointer value itself. * * The ID increases by an unspecified amount. * * @since 1.4 */ uint64_t ei_ping_get_id(struct ei_ping *ping); /** * Increase the refcount of this struct by one. Use ei_ping_unref() to decrease * the refcount. * * @return the argument passed into the function * * @since 1.4 */ struct ei_ping * ei_ping_ref(struct ei_ping *ei_ping); /** * Decrease the refcount of this struct by one. When the refcount reaches * zero, all allocated resources for this struct are released. * * @return always NULL * * @since 1.4 */ struct ei_ping * ei_ping_unref(struct ei_ping *ei_ping); /** * Set a custom data pointer for this struct. libei will not look at or * modify the pointer. Use ei_ping_get_user_data() to retrieve a previously set * user data. * * @since 1.4 */ void ei_ping_set_user_data(struct ei_ping *ei_ping, void *user_data); /** * Return the custom data pointer for this struct. libei will not look at or * modify the pointer. Use ei_ping_set_user_data() to change the user data. * * @since 1.4 */ void * ei_ping_get_user_data(struct ei_ping *ei_ping); /** * Issue a roundtrip request to the EIS implementation, resulting in * an @ref EI_EVENT_PONG event when this roundtrip has been processed. Client * requests are processed in-order by the EIS implementation so this * function can be used as synchronization point between requests. * * If the client is disconnected before the roundtrip is complete, * libei will emulate a @ref EI_EVENT_PONG event before @ref * EI_EVENT_DISCONNECT. * * Note that in the ei protocol, the client request is `ei_connection.sync` * followed by `ei_callback.done`. For consistency with the * libeis API the C API uses ei_ping() with and @ref EI_EVENT_PONG as return * event. * * @since 1.4 */ void ei_ping(struct ei_ping *ping); /** * Main event dispatching function. Reads events of the file descriptors * and processes them internally. Use ei_get_event() to retrieve the * events. * * Dispatching does not necessarily queue events. This function * should be called immediately once data is available on the file * descriptor returned by libei_get_fd(). */ void ei_dispatch(struct ei *ei); /** * Return the next event from the event queue, removing it from the queue. * * The returned object must be released by the caller with ei_event_unref(), * even if the event type is unknown to the caller. */ struct ei_event * ei_get_event(struct ei *ei); /** * Returns the next event in the internal event queue (or `NULL`) without * removing that event from the queue; the next call to ei_get_event() * will return that same event. * * This call is useful for checking whether there is an event and/or what * type of event it is. * * Repeated calls to ei_peek_event() return the same event. * * The returned event is refcounted, use ei_event_unref() to drop the * reference. * * A caller must not call ei_get_event() while holding a ref to an event * returned by ei_peek_event(). Doing so is undefined behavior. */ struct ei_event * ei_peek_event(struct ei *ei); /** * @returns a timestamp in microseconds for the current time to pass into * ei_device_frame(). * * By default, the returned timestamp is CLOCK_MONOTONIC for compatibility with * evdev and libinput. This can be overridden with ei_clock_set_now_func(). */ uint64_t ei_now(struct ei *ei); /** * Disconnect the current ei context from the EIS implementation. * * After a call to ei_disconnect(), ei_get_event() will return * events to remove resources (e.g. seats and devices) as if they * had been removed by the EIS implementation. The last event * returned by ei_get_event() is @ref EI_EVENT_DISCONNECT after * which the context should be considered inert and any * remaining resources released with ei_unref(). * * This does not free the resources associated with the ei context, use * ei_unref(). * * @since 1.4 */ void ei_disconnect(struct ei *ei); /** * @ingroup libei-seat * * Set a custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_seat_get_user_data() to retrieve a * previously set user data. */ void ei_seat_set_user_data(struct ei_seat *seat, void *user_data); /** * @ingroup libei-seat * * Return the custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_seat_get_user_data() to change the user data. */ void * ei_seat_get_user_data(struct ei_seat *seat); /** * @ingroup libei-seat */ const char * ei_seat_get_name(struct ei_seat *seat); /** * @ingroup libei-seat * * Return true if the capability is available on this seat or false * otherwise. The return value of this function is not affected by * ei_seat_confirm_capability(). */ bool ei_seat_has_capability(struct ei_seat *seat, enum ei_device_capability cap); /** * @ingroup libei-seat * * Bind this client to the given seat capabilities, terminated by ``NULL``. * Once bound, the server may * create devices for the requested capability and send the respective @ref * EI_EVENT_DEVICE_ADDED events. To undo, call ei_seat_unbind_capabilities(). * * Note that binding to a capability does not guarantee a device for that * capability becomes available. Devices may be added and removed at any time. * * It is an application bug to call this function for a capability already * bound - call ei_seat_unbind_capabilities() first. * * Calling this function for a capability that does not exist on the seat is * permitted (but obviously a noop) */ void ei_seat_bind_capabilities(struct ei_seat *seat, ...) __attribute__((sentinel)); /** * @ingroup libei-seat * * Unbind a seat's capabilities, terminated by ``NULL``. * This function indicates that the application is * no longer interested in devices with the given capability. * * If any devices with the given capability are present, libei automatically * calls ei_device_close() on those devices (and thus the server will send * @ref EI_EVENT_DEVICE_REMOVED for those devices). */ void ei_seat_unbind_capabilities(struct ei_seat *seat, ...) __attribute__((sentinel)); /** * @ingroup libei-seat * * Request a new device with (a subset of) the given capabilities from the EIS * implementation. If the EIS implementation creates a device in response * to this request the device will arrive via an event of type @ref * EI_EVENT_DEVICE_ADDED. * * The device's capabilities may not match the requested capabilities. For * example requesting a pointer + keyboard device may result in the creation * of a pointer-only device or it may result in the creation of a pointer + * keyboard + touch device. It is up to the caller to handle capability * mismatches. * * This function should be used by a caller when the current set of devices * is insufficient for the functionality the client requires. For example * a client that has previously called ei_device_close() on a device may * need a device again with similar capabilities. * * The capabilities must be a subset of the capabilities requested in * ei_seat_bind_capabilities(). * * @since 1.6 */ void ei_seat_request_device_with_capabilities(struct ei_seat *seat, ...) __attribute__((sentinel)); /** * @ingroup libei-seat */ struct ei_seat * ei_seat_ref(struct ei_seat *seat); /** * @ingroup libei-seat */ struct ei_seat * ei_seat_unref(struct ei_seat *seat); /** * @ingroup libei-seat * * Return the struct @ref ei context this seat is associated with. */ struct ei * ei_seat_get_context(struct ei_seat *seat); /** * Increase the refcount of this struct by one. Use ei_event_unref() to decrease * the refcount. This function always returns the same event that the reference * was added to. * * Events should be considered transient data and not be held longer than * required. */ struct ei_event * ei_event_ref(struct ei_event *event); /** * Decrease the refcount of this struct by one. When the refcount reaches * zero, all allocated resources for this struct are released. * * Events should be considered transient data and not be held longer than * required. * * @return always NULL */ struct ei_event * ei_event_unref(struct ei_event *event); /** * @return the type of this event */ enum ei_event_type ei_event_get_type(struct ei_event *event); /** * Return the device from this event. * * For events of type @ref EI_EVENT_CONNECT and @ref EI_EVENT_DISCONNECT, * this function returns NULL. * * This does not increase the refcount of the device. Use eis_device_ref() * to keep a reference beyond the immediate scope. */ struct ei_device * ei_event_get_device(struct ei_event *event); /** * Return the time for the event of type @ref EI_EVENT_FRAME in microseconds. * * @note Only events of type @ref EI_EVENT_FRAME carry a timestamp. For * convenience, the timestamp for other device events is retrofitted by this * library. * * @return the event time in microseconds */ uint64_t ei_event_get_time(struct ei_event *event); /** * @ingroup libei-device * * Increase the refcount of this struct by one. Use ei_device_unref() to * decrease the refcount. * * @return the argument passed into the function */ struct ei_device * ei_device_ref(struct ei_device *device); /** * @ingroup libei-device * * Decrease the refcount of this struct by one. When the refcount reaches * zero, all allocated resources for this struct are released. * * @return always NULL */ struct ei_device * ei_device_unref(struct ei_device *device); /** * @ingroup libei-device */ struct ei_seat * ei_device_get_seat(struct ei_device *device); /** * @ingroup libei-device * * Set a custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_device_get_user_data() to retrieve a * previously set user data. */ void ei_device_set_user_data(struct ei_device *device, void *user_data); /** * @ingroup libei-device * * Return the custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_device_get_user_data() to change the user data. */ void * ei_device_get_user_data(struct ei_device *device); /** * @ingroup libei-device * * Return the width of the device in mm if the device is of type @ref * EI_DEVICE_TYPE_PHYSICAL, otherwise zero. */ uint32_t ei_device_get_width(struct ei_device *device); /** * @ingroup libei-device * * Return the height of the device in mm if the device is of type @ref * EI_DEVICE_TYPE_PHYSICAL, otherwise zero. */ uint32_t ei_device_get_height(struct ei_device *device); /** * @ingroup libei-keymap * * @return the size of the keymap in bytes */ size_t ei_keymap_get_size(struct ei_keymap *keymap); /** * @ingroup libei-keymap * * Returns the type for this keymap. The type specifies how to interpret the * data at the file descriptor returned by ei_keymap_get_fd(). */ enum ei_keymap_type ei_keymap_get_type(struct ei_keymap *keymap); /** * @ingroup libei-keymap * * Return a memmap-able file descriptor pointing to the keymap used by the * device. The keymap is constant for the lifetime of the device and * assigned to this device individually. */ int ei_keymap_get_fd(struct ei_keymap *keymap); /** * @ingroup libei-keymap * * Return the device this keymap belongs to, or `NULL` if it has not yet * been assigned to a device. * * After processing and if the server changed the keymap or set the keymap * to NULL, this keymap may no longer be in use by the device and future * calls to this function return `NULL`. */ struct ei_device * ei_keymap_get_device(struct ei_keymap *keymap); /** * @ingroup libei-keymap * * Increase the refcount of this struct by one. Use ei_keymap_unref() to * decrease the refcount. * * @return the argument passed into the function */ struct ei_keymap * ei_keymap_ref(struct ei_keymap *keymap); /** * @ingroup libei-keymap * * Decrease the refcount of this struct by one. When the refcount reaches * zero, all allocated resources for this struct are released. * * @return always NULL */ struct ei_keymap * ei_keymap_unref(struct ei_keymap *keymap); /** * @ingroup libei-keymap */ void ei_keymap_set_user_data(struct ei_keymap *keymap, void *user_data); /** * @ingroup libei-keymap */ void * ei_keymap_get_user_data(struct ei_keymap *keymap); /** * @ingroup libei-device * * Notify the server that the client is no longer interested in * this device. * * Due to the asynchronous nature of the client-server interaction, * events for this device may still be in transit. The server will send an * @ref EI_EVENT_DEVICE_REMOVED event for this device. After that event, * device is considered removed by the server. * * A client can assume that an @ref EI_EVENT_DEVICE_REMOVED event is sent * for any device for which ei_device_close() was called before the @ref * EI_EVENT_DISCONNECT event. Where a client gets * disconnected libei will emulate that event. * * This does not release any resources associated with this device, use * ei_device_unref() for any references held by the client. */ void ei_device_close(struct ei_device *device); /** * @ingroup libei-device * * @return the name of the device (if any) or NULL */ const char * ei_device_get_name(struct ei_device *device); /** * @ingroup libei-device * * @return the device type */ enum ei_device_type ei_device_get_type(struct ei_device *device); /** * @ingroup libei-device * * Return true if the device has the requested capability. Device * capabilities are constant for the lifetime of the device and always * a subset of the capabilities bound to by ei_seat_bind_capabilities(). */ bool ei_device_has_capability(struct ei_device *device, enum ei_device_capability cap); /** * @ingroup libei-device * * Obtain a region from a device of type @ref EI_DEVICE_TYPE_VIRTUAL. The * number of regions is constant for a device and the indices of any region * remains the same for the lifetime of the device. * * Regions are shared between all capabilities. Where two capabilities need * different regions, the EIS implementation must create multiple devices with * individual capabilities and regions. For example, two touchscreens that are * mapped to two screens would typically show up as two separate devices with * one region each. * * This function returns the given region or NULL if the index is larger than * the number of regions available. * * This does not increase the refcount of the region. Use ei_region_ref() to * keep a reference beyond the immediate scope. * * Devices of type @ref EI_DEVICE_TYPE_PHYSICAL do not have regions. */ struct ei_region * ei_device_get_region(struct ei_device *device, size_t index); /** * @ingroup libei-device * * Return the region that contains the given point x/y (in desktop-wide * coordinates) or NULL if the coordinates are outside all regions. * * @since 1.1 */ struct ei_region * ei_device_get_region_at(struct ei_device *device, double x, double y); /** * @ingroup libei-region */ struct ei_region * ei_region_ref(struct ei_region *region); /** * @ingroup libei-region */ struct ei_region * ei_region_unref(struct ei_region *region); /** * @ingroup libei-region */ void ei_region_set_user_data(struct ei_region *region, void *user_data); /** * @ingroup libei-region */ void * ei_region_get_user_data(struct ei_region *region); /** * @ingroup libei-region */ uint32_t ei_region_get_x(struct ei_region *region); /** * @ingroup libei-region */ uint32_t ei_region_get_y(struct ei_region *region); /** * @ingroup libei-region */ uint32_t ei_region_get_width(struct ei_region *region); /** * @ingroup libei-region */ uint32_t ei_region_get_height(struct ei_region *region); /** * @ingroup libei-region * * Get the unique identifier (representing an external resource) that is * attached to this region, if any. This is only available if the EIS * implementation supports version 2 or later of the ei_device protocol * interface *and* the EIS implementation chooses to attach such an identifier to * the region. * * This ID can be used by the client to identify an external resource that has a * relationship with this region. * * For example the client may receive a data stream with the video * data that this region represents. By attaching the same identifier to the data * stream and this region the EIS implementation can inform the client * that the video data stream and the region represent paired data. * Note that in this example use-case, if the stream is resized * there may be a transition period where two regions have the same identifier - * the old region and the new region with the updated size. A client must be * able to handle the case where two mapping ids are identical. * * libei does not look at or modify the value of the mapping id. Because the ID is * assigned by the caller libei makes no guarantee that the ID is unique * and/or corresponds to any particular format. * * @since 1.1 */ const char * ei_region_get_mapping_id(struct ei_region *region); /** * @ingroup libei-region * * Return true if the point x/y (in desktop-wide coordinates) is within @a * region. */ bool ei_region_contains(struct ei_region *region, double x, double y); /** * @ingroup libei-region * * Convert the point x/y in a desktop-wide coordinate system into the * corresponding point relative to the offset of the given region. * If the point is inside the region, this function returns true and @a x and @a * y are set to the points with the region offset subtracted. * If the point is outside the region, this function returns false and @a x * and @a y are left unmodified. */ bool ei_region_convert_point(struct ei_region *region, double *x, double *y); /** * @ingroup libei-region * * Return the physical scale for this region. The default scale is 1.0. * * The regions' coordinate space is in logical pixels in the EIS range. The * logical pixels may or may not match the physical pixels on the output * range but the mapping from logical pixels to physical pixels is performed * by the EIS implementation. * * In some use-cases though, relative data from a remote input source needs * to be converted by the libei client into an absolute movement on an EIS * region. In that case, the physical scale provides the factor to multiply * the relative logical input to provide the expected physical relative * movement. * * For example consider the following dual-monitor setup comprising a 2k and * a 4k monitor **of the same physical size**: * The physical layout of the monitors appears like this: * @code * 2k 4k * +-------------++-------------+ * | || | * | a b || c d | * | || | * +-------------++-------------+ * @endcode * * The physical distance `ab` is the same as the physical distance `cd`. * Where the EIS implementation supports high-dpi screens, the logical * distance (in pixels) are identical too. * * Where the EIS implementation does not support high-dpi screens, the * logical layout of these two monitors appears like this: * * @code * 2k 4k * +-------------++--------------------------+ * | || | * | a b || | * | || | * +-------------+| c d | * | | * | | * | | * +--------------------------+ * @endcode * * While the two physical distances `ab` and `cd` are still identical, the * logical distance `cd` (in pixels) is twice that of `ab`. * Where a libei client receives relative deltas from an input source and * converts that relative input into an absolute position on the screen, it * needs to take this into account. * * For example, if a remote input source moves by relative 100 logical * pixels, the libei client would convert this as `a + 100 = b` on the * region for the 2k screen and send the absolute events to logically change * the position from `a` to `b`. If the same remote input source moves by * relative 100 logical pixels, the libei client would convert this as * `c + 100 * scale = d` on the region for the 4k screen to logically * change the position from `c` to `d`. While the pixel movement differs, * the physical movement as seen by the user is thus identical. * * A second possible use-case for the physical scale is to match pixels from * one region to their respective counterpart on a different region. * For example, if the bottom-right corner of the 2k screen in the * illustration above has a coordinate of ``(x, y)``, the neighbouring pixel on * the **physical** 4k screen is ``(0, y * scale)``. */ double ei_region_get_physical_scale(struct ei_region *region); /** * @ingroup libei-device * * Return the keymap for this device or `NULL`. The keymap is constant for * the lifetime of the device and applies to this device individually. * * If this function returns `NULL`, this device does not have * an individual keymap assigned. What keymap applies to the device in this * case is a server implementation detail. * * This does not increase the refcount of the keymap. Use ei_keymap_ref() to * keep a reference beyond the immediate scope. * */ /* FIXME: the current API makes it impossible to know when the keymap has * been consumed so the file stays open forever. */ struct ei_keymap * ei_device_keyboard_get_keymap(struct ei_device *device); /** * @ingroup libei-keymap * * Return the struct @ref ei_device this keymap is associated with. */ struct ei_device * ei_keymap_get_context(struct ei_keymap *keymap); /** * @ingroup libei-device * * Return the struct @ref ei context this device is associated with. */ struct ei * ei_device_get_context(struct ei_device *device); /** * @ingroup libei-sender * * Notify the EIS implementation that the given device is about to start * sending events. This should be seen more as a transactional boundary than a * time-based boundary. The primary use-cases for this are to allow for setup on * the EIS implementation side and/or UI updates to indicate that a device is * sending events now and for out-of-band information to sync with a given event * sequence. * * There is no actual requirement that events start immediately once emulation * starts and there is no requirement that a client calls * ei_device_stop_emulating() after the most recent events. * * For example, in a synergy-like use-case the client would call * ei_device_start_emulating() once the pointer moves into the screen and * ei_device_stop_emulating() once the pointer moves out of the screen. * * Sending events before ei_device_start_emulating() or after * ei_device_stop_emulating() is a client bug. * * The sequence number identifies this transaction between start/stop emulating. * It must go up by at least 1 on each call to * ei_device_start_emulating(). Wraparound must be handled by the EIS * implementation but Callers must ensure that detection of wraparound is * reasonable. * * This method is only available on an ei sender context. */ void ei_device_start_emulating(struct ei_device *device, uint32_t sequence); /** * @ingroup libei-sender * * Notify the EIS implementation that the given device is no longer sending * events. See ei_device_start_emulating() for details. * * If the device is not logically in a neutral state, that state is left * as-is. It is the caller's responsibility to release any buttons, keys, touch * sequences, etc. when stopping emulation to avoid adverse side effects. * * This method is only available on an ei sender context. */ void ei_device_stop_emulating(struct ei_device *device); /** * @ingroup libei-sender * * Generate a frame event to group the current set of events * into a logical hardware event. This function **must** be called after any * other event has been generated. * * The given timestamp applies to all events in the current frame. * The timestamp must be in microseconds of CLOCK_MONOTONIC, use the return * value of ei_now() to get a compatible timestamp. * * @note libei does not prevent a caller from passing in a future time but it * is strongly recommended that this is avoided by the caller. * * This method is only available on an ei sender context. */ void ei_device_frame(struct ei_device *device, uint64_t time); /** * @ingroup libei-sender * * Generate a relative motion event on a device with * the @ref EI_DEVICE_CAP_POINTER capability. * * This method is only available on an ei sender context. * * @param device The EI device * @param x The x movement in logical pixels or mm, depending on the device type * @param y The y movement in logical pixels or mm, depending on the device type */ void ei_device_pointer_motion(struct ei_device *device, double x, double y); /** * @ingroup libei-sender * * Generate an absolute motion event on a device with * the @ref EI_DEVICE_CAP_POINTER_ABSOLUTE capability. * * The x/y coordinate must be within the device's regions or the event is * silently discarded. * * This method is only available on an ei sender context. * * @param device The EI device * @param x The x position in logical pixels or mm, depending on the device type * @param y The y position in logical pixels or mm, depending on the device type */ void ei_device_pointer_motion_absolute(struct ei_device *device, double x, double y); /** * @ingroup libei-sender * * Generate a button event on a device with * the @ref EI_DEVICE_CAP_BUTTON capability. * * Button codes must match the defines in ``linux/input-event-codes.h`` * * This method is only available on an ei sender context. * * @param device The EI device * @param button The button code * @param is_press true for button press, false for button release */ void ei_device_button_button(struct ei_device *device, uint32_t button, bool is_press); /** * @ingroup libei-sender * * Generate a smooth (pixel-precise) scroll event on a device with * the @ref EI_DEVICE_CAP_SCROLL capability. * * @note The server is responsible for emulating discrete scrolling based * on the pixel value, do not call ei_device_scroll_discrete() for * the same input event. * * This method is only available on an ei sender context. * * @param device The EI device * @param x The x scroll distance in logical pixels or mm, depending on the device type * @param y The y scroll distance in logical pixels or mm, depending on the device type * * @see ei_device_scroll_discrete */ void ei_device_scroll_delta(struct ei_device *device, double x, double y); /** * @ingroup libei-sender * * Generate a discrete scroll event on a device with * the @ref EI_DEVICE_CAP_SCROLL capability. * * A discrete scroll event is based logical scroll units (equivalent to one * mouse wheel click). The value for one scroll unit is 120, a fraction or * multiple thereof represents a fraction or multiple of a wheel click. * * @note The server is responsible for emulating pixel-based scrolling based * on the discrete value, do not call ei_device_scroll_delta() for the * same input event. * * This method is only available on an ei sender context. * * @param device The EI device * @param x The x scroll distance in fractions or multiples of 120 * @param y The y scroll distance in fractions or multiples of 120 * * @see ei_device_scroll_delta */ void ei_device_scroll_discrete(struct ei_device *device, int32_t x, int32_t y); /** * @ingroup libei-sender * * Generate a scroll stop event on a device with the * @ref EI_DEVICE_CAP_SCROLL capability. * * A scroll stop event notifies the server that the interaction causing a * scroll motion previously triggered with ei_device_scroll_delta() or * ei_device_scroll_discrete() has stopped. For example, if all * fingers are lifted off a touchpad, two-finger scrolling has logically * stopped. * * The server may use this information to e.g. start kinetic scrolling * previously based on the previous finger speed. * * Use ei_device_scroll_cancel() to signal that the scroll motion has * completely stopped. * * Calling ei_device_scroll_stop() after * ei_device_scroll_cancel() without any of ei_device_scroll_delta() * or ei_device_scroll_discrete() in between indicates a client logic bug. * * libei keeps track of the scroll axis and filters duplicate calls to * ei_device_scroll_stop() for the same axis. A nonzero scroll or * scroll-discrete value is required for the given axis to re-start scrolling * for that axis. * * This method is only available on an ei sender context. */ void ei_device_scroll_stop(struct ei_device *device, bool stop_x, bool stop_y); /** * @ingroup libei-sender * * Generate a scroll cancel event on a device with the * @ref EI_DEVICE_CAP_SCROLL capability. * * A scroll cancel event notifies the server that a scroll motion previously * triggered with ei_device_scroll_delta() or * ei_device_scroll_discrete() has ceased and no further events should * be sent. * * This event indicates that the interaction has stopped to the point where * further (server-emulated) scroll events from this device are wrong. * * Use ei_device_scroll_stop() to signal that the interaction has * stopped but a server may emulate further scroll events. * * Calling ei_device_scroll_cancel() after * ei_device_scroll_stop() without any of ei_device_scroll_delta() * or ei_device_scroll_discrete() in between is permitted. * * libei keeps track of the scroll axis and filters duplicate calls to * ei_device_scroll_cancel() for the same axis. A nonzero scroll or * scroll-discrete value is required for the given axis to re-start scrolling * for that axis. * * This method is only available on an ei sender context. */ void ei_device_scroll_cancel(struct ei_device *device, bool cancel_x, bool cancel_y); /** * @ingroup libei-sender * * Generate a key event on a device with * the @ref EI_DEVICE_CAP_KEYBOARD capability. * * Keys use the evdev scan codes as defined in * ``linux/input-event-codes.h``. * * Note that this is a keymap-independent key code, equivalent to the scancode * a physical keyboard would produce. To generate a specific key symbol, a * client must look at the keymap returned by ei_device_keyboard_get_keymap() and * generate the appropriate keycodes. * * This method is only available on an ei sender context. * * @param device The EI device * @param keycode The key code * @param is_press true for key down, false for key up */ void ei_device_keyboard_key(struct ei_device *device, uint32_t keycode, bool is_press); /** * @ingroup libei-sender * * Generate a key event on a device with * the @ref EI_DEVICE_CAP_TEXT capability. * * Keysyms use the XKB-compatible keysyms, see e.g. * [/usr/include/xkbcommon/xkbcommon-keysyms.h](https://xkbcommon.org/doc/current/xkbcommon-keysyms_8h.html) * for a list of key syms * * Note that this keysym is independent of any keymap and the keysym * may not exist on any active keymap on any device. For example, a device * with both @ref EI_DEVICE_CAP_TEXT and @ref EI_DEVICE_CAP_KEYBOARD is not * limited to sending keysyms only present on the keymap. * * This method is only available on an ei sender context. * * @param device The EI device * @param keysym The keysym * @param is_press true for key down, false for key up * * @since 1.6 */ void ei_device_text_keysym(struct ei_device *device, uint32_t keysym, bool is_press); /** * @ingroup libei-sender * * Generate a UTF-8 text event on a device with * the @ref EI_DEVICE_CAP_TEXT capability. * * This method is only available on an ei sender context. * This method is identical to ei_device_text_utf8_with_length() * but calculates the length of the string. * * @param device The EI device * @param text The zero-terminated UTF-8 compatible text * * @see ei_device_text_utf8_with_length * * @since 1.6 */ void ei_device_text_utf8(struct ei_device *device, const char *utf8); /** * @ingroup libei-sender * * Generate a UTF-8 text event on a device with * the @ref EI_DEVICE_CAP_TEXT capability. * * This method is only available on an ei sender context. * * This method is identical to ei_device_text_utf8() * but takes an additional length argument. * * @param device The EI device * @param text The zero-terminated UTF-8 compatible text * @param length The length of text in bytes, excluding the zero bytes * * @see ei_device_text_utf8 * * @since 1.6 */ void ei_device_text_utf8_with_length(struct ei_device *device, const char *text, size_t length); /** * @ingroup libei-sender * * Initiate a new touch on a device with the @ref EI_DEVICE_CAP_TOUCH * capability. This touch does not immediately send events, use * ei_touch_down(), ei_touch_motion(), and ei_touch_up(). * * The returned touch has a refcount of at least 1, use ei_touch_unref() to * release resources associated with this touch * * This method is only available on an ei sender context. */ struct ei_touch * ei_device_touch_new(struct ei_device *device); /** * @ingroup libei-sender * * This function can only be called once on an ei_touch object. Further * calls to ei_touch_down() on the same object are silently ignored. * * The x/y coordinate must be within the device's regions or the event is * silently discarded. * * @param touch A newly created touch * @param x The x position in logical pixels * @param y The y position in logical pixels */ void ei_touch_down(struct ei_touch *touch, double x, double y); /** * @ingroup libei-sender * * Move this touch to the new coordinates. */ void ei_touch_motion(struct ei_touch *touch, double x, double y); /** * @ingroup libei-sender * * Release this touch. After this call, the touch event becomes inert and * no longer responds to either ei_touch_down(), ei_touch_motion() or * ei_touch_up() and the caller should call ei_touch_unref(). */ void ei_touch_up(struct ei_touch *touch); /** * @ingroup libei-sender * * Cancel this touch. After this call, the touch event becomes inert and * no longer responds to either ei_touch_down(), ei_touch_motion() or * ei_touch_up() and the caller should call ei_touch_unref(). * * This is only available if the EIS implementation supports version 2 * or later of the ei_touchscreen protocol interface. Otherwise, * this function is equivalent to calling ei_touch_up(). */ void ei_touch_cancel(struct ei_touch *touch); /** * @ingroup libei-sender * * Increase the refcount of this struct by one. Use ei_touch_unref() to * decrease the refcount. * * @return the argument passed into the function */ struct ei_touch * ei_touch_ref(struct ei_touch *touch); /** * @ingroup libei-sender * * Decrease the refcount of this struct by one. When the refcount reaches * zero, all allocated resources for this struct are released. * * @return always NULL */ struct ei_touch * ei_touch_unref(struct ei_touch *touch); /** * @ingroup libei-sender * * Set a custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_touch_get_user_data() to retrieve a previously * set user data. */ void ei_touch_set_user_data(struct ei_touch *touch, void *user_data); /** * @ingroup libei-sender * * Return the custom data pointer for this context. libei will not look at or * modify the pointer. Use ei_touch_set_user_data() to change the user data. */ void * ei_touch_get_user_data(struct ei_touch *touch); /** * @ingroup libei-sender * * @return the device this touch originates on */ struct ei_device * ei_touch_get_device(struct ei_touch *touch); /** * Return the seat from this event. * * For events of type @ref EI_EVENT_CONNECT and @ref EI_EVENT_DISCONNECT, * this function returns NULL. * * This does not increase the refcount of the seat. Use eis_seat_ref() * to keep a reference beyond the immediate scope. */ struct ei_seat * ei_event_get_seat(struct ei_event *event); /** * Returns the associated @ref ei_ping struct with this event. * * For events of type other than @ref EI_EVENT_PONG this function * returns NULL. * * This does not increase the refcount of the ei_ping. Use ei_ping_ref() * to keep a reference beyond the immediate scope. */ struct ei_ping * ei_event_pong_get_ping(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_DEVICE_START_EMULATING, return the * sequence number set by the EIS implementation. * * See eis_device_start_emulating() for details. */ uint32_t ei_event_emulating_get_sequence(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_KEYBOARD_MODIFIERS, get the * mask of currently logically pressed-down modifiers. * See ei_device_keyboard_get_keymap() for the corresponding keymap. */ uint32_t ei_event_keyboard_get_xkb_mods_depressed(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_KEYBOARD_MODIFIERS, get the * mask of currently logically latched modifiers. * See ei_device_keyboard_get_keymap() for the corresponding keymap. */ uint32_t ei_event_keyboard_get_xkb_mods_latched(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_KEYBOARD_MODIFIERS, get the * mask of currently logically locked modifiers. * See ei_device_keyboard_get_keymap() for the corresponding keymap. */ uint32_t ei_event_keyboard_get_xkb_mods_locked(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_KEYBOARD_MODIFIERS, get the * current effective group. * * This may be passed to xkb_state_update_mask() as either * depressed_layout (effectively pretending the user is holding down some * key for this group at all times) or locked_layout (treating it as a * layout the user has switched to through some mechanism), but never * both at the same time. The other two layout arguments must be set to * zero. * * Note: Because the client only knows the current effective group and * not the combination of state from which it was calculated, any attempt * to predict how future key presses will impact the group state will * necessarily be unreliable. * * See ei_device_keyboard_get_keymap() for the corresponding keymap. */ uint32_t ei_event_keyboard_get_xkb_group(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_POINTER_MOTION return the relative x * movement in logical pixels or mm, depending on the device type. */ double ei_event_pointer_get_dx(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_POINTER_MOTION return the relative y * movement in logical pixels or mm, depending on the device type. */ double ei_event_pointer_get_dy(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_POINTER_MOTION_ABSOLUTE return the x * position in logical pixels or mm, depending on the device type. */ double ei_event_pointer_get_absolute_x(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_POINTER_MOTION_ABSOLUTE return the y * position in logical pixels or mm, depending on the device type. */ double ei_event_pointer_get_absolute_y(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_BUTTON_BUTTON return the button * code as defined in linux/input-event-codes.h */ uint32_t ei_event_button_get_button(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_BUTTON_BUTTON return true if the * event is a button press, false for a release. */ bool ei_event_button_get_is_press(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_SCROLL_DELTA return the x scroll * distance in logical pixels or mm, depending on the device type. */ double ei_event_scroll_get_dx(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_SCROLL_DELTA return the y scroll * distance in logical pixels or mm, depending on the device type. */ double ei_event_scroll_get_dy(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_SCROLL_CANCEL return whether the * x axis has cancelled scrolling. */ bool ei_event_scroll_get_stop_x(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_SCROLL_STOP return whether the * y axis has stopped scrolling. */ bool ei_event_scroll_get_stop_y(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_SCROLL_DISCRETE return the x * scroll distance in fractions or multiples of 120. */ int32_t ei_event_scroll_get_discrete_dx(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_SCROLL_DISCRETE return the y * scroll distance in fractions or multiples of 120. */ int32_t ei_event_scroll_get_discrete_dy(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_KEYBOARD_KEY return the key code (as * defined in include/linux/input-event-codes.h). */ uint32_t ei_event_keyboard_get_key(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_KEYBOARD_KEY return true if the * event is a key down, false for a release. */ bool ei_event_keyboard_get_key_is_press(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_TOUCH_DOWN, @ref * EI_EVENT_TOUCH_MOTION, or @ref EI_EVENT_TOUCH_UP, return the tracking * ID of the touch. * * The tracking ID is a unique identifier for a touch and is valid from * touch down through to touch up but may be re-used in the future. * The tracking ID is randomly assigned to a touch, a client * must not expect any specific value. */ uint32_t ei_event_touch_get_id(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_TOUCH_DOWN, or @ref * EI_EVENT_TOUCH_MOTION, return the x coordinate of the touch * in logical pixels or mm, depending on the device type. */ double ei_event_touch_get_x(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_TOUCH_DOWN, or @ref * EI_EVENT_TOUCH_MOTION, return the y coordinate of the touch * in logical pixels or mm, depending on the device type. */ double ei_event_touch_get_y(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_TOUCH_UP * return true if the event was cancelled instead of * logically released. * * Support for touch cancellation requires the EIS implementation and client to * support version 2 or later of the ei_touchscreen protocol interface. */ bool ei_event_touch_get_is_cancel(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_TEXT_KEYSYM * return the XKB-compatible keysym. * * @since 1.6 */ uint32_t ei_event_text_get_keysym(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_TEXT_KEYSYM * return true if the event is a logical key down for the * keysym. * * @since 1.6 */ bool ei_event_text_get_keysym_is_press(struct ei_event *event); /** * @ingroup libei-receiver * * For an event of type @ref EI_EVENT_TEXT_UTF8 * return the zero-terminated UTF8 string. * * @since 1.6 */ const char * ei_event_text_get_utf8(struct ei_event *event); /** * @} */ #ifdef __cplusplus } #endif