1
0
mirror of https://github.com/RIOT-OS/RIOT.git synced 2025-01-17 10:32:44 +01:00
RIOT/drivers/include/touch_dev_gestures.h
2023-08-31 20:12:33 +02:00

250 lines
9.1 KiB
C

/*
* Copyright (C) 2023 Gunar Schorcht
*
* This file is subject to the terms and conditions of the GNU Lesser
* General Public License v2.1. See the file LICENSE in the top level
* directory for more details.
*/
/**
* @defgroup drivers_touch_dev_gestures Touch device gesture recognition
* @ingroup drivers_misc
*
* @brief Gesture recognition for touch devices
* @{
*
* This driver implements a simple gesture recognition with a maximum of two
* touches for touch devices that use the generic touch device API.
*
* The application that receives the events from the touch device via the
* callback function registered with @ref touch_dev_set_touch_event_callback
* must first create and initialize a touch device gesture context of type
* @ref touch_dev_gesture_ctx_t. For each touch event received from the
* touch device, it then calls @ref touch_dev_recognize_gesture function
* with this context so that the gesture recognition fetches the data from
* the touch device to detect the gestures, for example:
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~{.c}
* static void _touch_event_cb(void *arg)
* {
* // indicate that a touch event occurred
* mutex_unlock(arg);
* }
*
* void *_input_task(void *arg)
* {
* ...
* mutex_t lock = MUTEX_INIT_LOCKED;
*
* touch_dev_t *dev = touch_dev_reg_find_screen(0).dev;
* touch_dev_gesture_ctx_t ctx;
*
* // set the event callback function and initialize the touch device gesture context
* touch_dev_set_touch_event_callback(dev, _touch_event_cb, &lock);
* touch_dev_init_gesture(dev, &ctx);
*
* while (1) {
* // wait for the indication of a touch event
* mutex_lock(&lock);
*
* // call the gesture recognition
* touch_t pos;
* touch_dev_gesture_t gesture = touch_dev_recognize_gesture(&ctx, &pos);
*
* // process recognized gestures
* switch (gesture) {
* case TOUCH_DEV_GEST_SINGLE_TAP:
* if ((pos.x == 50) && (pos.y == 75)) {
* ...
* case TOUCH_DEV_GEST_DOUBLE_TAP:
* ...
* ...
* }
* }
* return NULL;
* }
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* @note To use this event-driven approach the driver for the touch device
* has to report the following touch events by interrupt:
* - a new touch is detected,
* - a touch is released, and
* - regularly the current touch positions as long as there are touches.
*
* If the event-driven approach cannot be used because either the touch
* device does not support all these touch events or the application wants
* to use the touch device in polling mode, the application must call the
* @ref touch_dev_recognize_gesture function with the gesture context
* of the touch device at regular intervals, for example:
*
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~{.c}
* #ifndef TOUCH_DEV_POLLING_PERIOD
* #define TOUCH_DEV_POLLING_PERIOD 50
* #endif
* ...
*
* void *_input_task(void *arg)
* {
* ...
*
* touch_dev_t *dev = touch_dev_reg_find_screen(0).dev;
* touch_dev_gesture_ctx_t ctx;
*
* // initialize the touch device gesture context
* touch_dev_init_gesture(dev, &ctx);
*
* while (1) {
* // call the gesture recognition
* touch_t pos;
* touch_dev_gesture_t gesture = touch_dev_recognize_gesture(&ctx, &pos);
*
* // process recognized gestures
* switch (gesture) {
* case TOUCH_DEV_GEST_SINGLE_TAP:
* if ((pos.x == 50) && (pos.y == 75)) {
* ...
* case TOUCH_DEV_GEST_DOUBLE_TAP:
* ...
* ...
* }
*
* // wait the period time for polling
* ztimer_sleep(ZTIMER_MSEC, TOUCH_DEV_POLLING_PERIOD);
* }
* return NULL;
* }
* ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
*
* The following gestures are supported by the driver:
*
* - @ref TOUCH_DEV_GEST_SINGLE_TAP : a single tap with one touch at the given position
* - @ref TOUCH_DEV_GEST_DOUBLE_TAP : a double tap with one touch at the given position
* - @ref TOUCH_DEV_GEST_PRESSED : a long press with one touch at the given position
* - @ref TOUCH_DEV_GEST_RELEASED : one touch released after a long press at given position
* - @ref TOUCH_DEV_GEST_MOVE : moving while pressed with current position
* - @ref TOUCH_DEV_GEST_SWIPE_LEFT : swiping left with one touch
* - @ref TOUCH_DEV_GEST_SWIPE_RIGHT : swiping right with one touch
* - @ref TOUCH_DEV_GEST_SWIPE_UP : swiping up with one touch
* - @ref TOUCH_DEV_GEST_SWIPE_DOWN : swiping down with one touch
* - @ref TOUCH_DEV_GEST_ZOOM_IN : zooming in (spreading) with two touches
* - @ref TOUCH_DEV_GEST_ZOOM_OUT : zooming out (pinching) with two touches
*
* @note
* - For technical reasons, a double-tap event is always preceded by a
* single-tap event, i.e. for a double-tap, the application always
* receives a single-tap event first and then a double-tap event if
* the second tap follows.
* - Zooming gestures are only available if the touch device supports two
* touches.
*
* @author Gunar Schorcht <gunar@schorcht.net>
*/
#ifndef TOUCH_DEV_GESTURES_H
#define TOUCH_DEV_GESTURES_H
#ifdef __cplusplus
extern "C" {
#endif
#include <stddef.h>
#include <stdint.h>
#include <stdbool.h>
#include "touch_dev.h"
/**
* @brief Maximum number of touches supported by gesture recognition
*/
#define TOUCH_DEV_TOUCHES_MAX_NUMOF 2
/**
* @brief Minimum distance in one direction to recognize a swipe gesture
*/
#ifndef CONFIG_TOUCH_DEV_SWIPE_TRESH
#define CONFIG_TOUCH_DEV_SWIPE_TRESH 5
#endif
/**
* @brief Minimum touch time in milliseconds to detect a long press gesture
*/
#ifndef CONFIG_TOUCH_DEV_PRESS_TIME_MS
#define CONFIG_TOUCH_DEV_PRESS_TIME_MS 600
#endif
/**
* @brief Maximum time in milliseconds between two taps to detect a double tap
*/
#ifndef CONFIG_TOUCH_DEV_DOUBLE_TIME_MS
#define CONFIG_TOUCH_DEV_DOUBLE_TIME_MS 400
#endif
/**
* @brief Touch device states used for gesture recognition
*/
typedef enum {
TOUCH_DEV_STATE_RELEASED, /**< no touches detected, default state */
TOUCH_DEV_STATE_TAPPED_SINGLE, /**< a single touch is detected */
TOUCH_DEV_STATE_TAPPED_MULTIPLE, /**< a second touch is detected */
TOUCH_DEV_STATE_PRESSED, /**< a long press is detected */
TOUCH_DEV_STATE_WAIT_FOR_RELEASE, /**< gesture detected, waiting for releasing touches */
} touch_dev_state_t;
/**
* @brief Touch gesture events
*/
typedef enum {
TOUCH_DEV_GEST_NONE, /**< No gesture recognized */
TOUCH_DEV_GEST_SINGLE_TAP, /**< Single tap recognized at the given position */
TOUCH_DEV_GEST_DOUBLE_TAP, /**< Double tap recognized at the given position */
TOUCH_DEV_GEST_PRESSED, /**< Long press recognized at the given position */
TOUCH_DEV_GEST_RELEASED, /**< Release after a long press at given position */
TOUCH_DEV_GEST_MOVE, /**< Moving while pressed recognized, current position is given */
TOUCH_DEV_GEST_SWIPE_LEFT, /**< Swipe left recognized, no position is given */
TOUCH_DEV_GEST_SWIPE_RIGHT, /**< Swipe right recognized, no position is given */
TOUCH_DEV_GEST_SWIPE_UP, /**< Swipe up recognized, no position is given */
TOUCH_DEV_GEST_SWIPE_DOWN, /**< Swipe down recognized, no position is given */
TOUCH_DEV_GEST_ZOOM_IN, /**< Zoom in (spread) recognized, no position is given */
TOUCH_DEV_GEST_ZOOM_OUT, /**< Zoom out (pinch) recognized, no position is given */
} touch_dev_gesture_t;
/**
* @brief Context information for a touch device needed for gesture recognition
*/
typedef struct {
touch_dev_t *dev; /**< Pointer to the touch device */
uint32_t t_changed; /**< Time of last state change in ms */
uint32_t t_prev_tap; /**< Time of previous tap */
touch_t prev[TOUCH_DEV_TOUCHES_MAX_NUMOF]; /**< Previous set of touches */
uint8_t prev_num; /**< Previous number of touches */
touch_dev_state_t state; /**< State of touch device */
} touch_dev_gesture_ctx_t;
/**
* @brief Initialize gesture recognition
*
* @param[in] dev Pointer to the touch device
* @param[in] ctx Pointer to the context information for the touch device
*/
void touch_dev_init_gesture(touch_dev_t *dev,
touch_dev_gesture_ctx_t *ctx);
/**
* @brief Recognize gestures by handling next touch device event
*
* @param[in] ctx Pointer to the context information for the touch device
* @param[out] pos Position of the gesture if interested in it (can be NULL)
*
* return the gesture of type @ref touch_dev_gesture_t if one was
* recognized or @ref TOUCH_DEV_GEST_NONE
*/
touch_dev_gesture_t touch_dev_recognize_gesture(touch_dev_gesture_ctx_t *ctx,
touch_t *pos);
#ifdef __cplusplus
}
#endif
#endif /* TOUCH_DEV_GESTURES_H */
/** @} */