2023-05-18 20:32:29 +03:00
|
|
|
/*
|
|
|
|
* InputSourceTouch.h, part of VCMI engine
|
|
|
|
*
|
|
|
|
* Authors: listed in file AUTHORS in main folder
|
|
|
|
*
|
|
|
|
* License: GNU General Public License v2.0 or later
|
|
|
|
* Full text of license available in license.txt file, in main folder
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
2023-05-26 22:17:36 +03:00
|
|
|
#include "../../lib/Point.h"
|
2023-05-18 20:32:29 +03:00
|
|
|
|
2023-06-16 13:07:02 +03:00
|
|
|
// Debug option. If defined, mouse events will instead generate touch events, allowing testing of touch input on desktop
|
|
|
|
// #define VCMI_EMULATE_TOUCHSCREEN_WITH_MOUSE
|
|
|
|
|
2023-05-26 15:55:31 +03:00
|
|
|
enum class MouseButton;
|
2023-05-18 20:32:29 +03:00
|
|
|
struct SDL_TouchFingerEvent;
|
|
|
|
|
2023-05-26 18:55:26 +03:00
|
|
|
/// Enumeration that describes current state of gesture recognition
|
|
|
|
enum class TouchState
|
|
|
|
{
|
|
|
|
// special state that allows no transitions
|
|
|
|
// used when player selects "relative mode" in Launcher
|
|
|
|
// in this mode touchscreen acts like touchpad, moving cursor at certains speed
|
|
|
|
// and generates events for positions below cursor instead of positions below touch events
|
|
|
|
RELATIVE_MODE,
|
|
|
|
|
|
|
|
// no active touch events
|
|
|
|
// DOWN -> transition to TAP_DOWN_SHORT
|
|
|
|
// MOTION / UP -> not expected
|
|
|
|
IDLE,
|
|
|
|
|
|
|
|
// single finger is touching the screen for a short time
|
|
|
|
// DOWN -> transition to TAP_DOWN_DOUBLE
|
|
|
|
// MOTION -> transition to TAP_DOWN_PANNING
|
|
|
|
// UP -> transition to IDLE, emit onLeftClickDown and onLeftClickUp
|
2023-06-16 13:07:02 +03:00
|
|
|
// on timer -> transition to TAP_DOWN_LONG, emit showPopup() event
|
2023-05-26 18:55:26 +03:00
|
|
|
TAP_DOWN_SHORT,
|
|
|
|
|
|
|
|
// single finger is moving across screen
|
|
|
|
// DOWN -> transition to TAP_DOWN_DOUBLE
|
|
|
|
// MOTION -> emit panning event
|
|
|
|
// UP -> transition to IDLE
|
|
|
|
TAP_DOWN_PANNING,
|
|
|
|
|
|
|
|
// two fingers are touching the screen
|
|
|
|
// DOWN -> ??? how to handle 3rd finger? Ignore?
|
|
|
|
// MOTION -> emit pinch event
|
|
|
|
// UP -> transition to TAP_DOWN
|
|
|
|
TAP_DOWN_DOUBLE,
|
|
|
|
|
|
|
|
// single finger is down for long period of time
|
|
|
|
// DOWN -> ignored
|
|
|
|
// MOTION -> ignored
|
2023-05-29 19:21:09 +03:00
|
|
|
// UP -> transition to TAP_DOWN_LONG_AWAIT
|
2023-05-26 18:55:26 +03:00
|
|
|
TAP_DOWN_LONG,
|
|
|
|
|
2023-05-29 19:21:09 +03:00
|
|
|
// right-click popup is active, waiting for new tap to hide popup
|
|
|
|
// DOWN -> ignored
|
|
|
|
// MOTION -> ignored
|
2023-06-16 13:07:02 +03:00
|
|
|
// UP -> transition to IDLE, generate closePopup() event
|
2023-05-29 19:21:09 +03:00
|
|
|
TAP_DOWN_LONG_AWAIT,
|
2023-05-26 18:55:26 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
struct TouchInputParameters
|
|
|
|
{
|
2023-05-31 13:08:12 +03:00
|
|
|
/// Speed factor of mouse pointer when relative mode is used
|
2023-05-26 18:55:26 +03:00
|
|
|
double relativeModeSpeedFactor = 1.0;
|
2023-05-26 22:17:36 +03:00
|
|
|
|
|
|
|
/// tap for period longer than specified here will be qualified as "long tap", triggering corresponding gesture
|
2023-06-16 12:59:20 +03:00
|
|
|
uint32_t longTouchTimeMilliseconds = 750;
|
2023-05-26 18:55:26 +03:00
|
|
|
|
2023-09-18 16:05:39 +02:00
|
|
|
/// time span in where the second tap has to happen for qualifing as "double click"
|
|
|
|
uint32_t doubleTouchTimeMilliseconds = 500;
|
|
|
|
|
|
|
|
/// max distance in where the second tap has to happen for qualifing as "double click"
|
|
|
|
uint32_t doubleTouchToleranceDistance = 50;
|
|
|
|
|
2023-05-26 22:17:36 +03:00
|
|
|
/// moving finger for distance larger than specified will be qualified as panning gesture instead of long press
|
2023-05-31 13:08:12 +03:00
|
|
|
uint32_t panningSensitivityThreshold = 10;
|
2023-05-26 22:17:36 +03:00
|
|
|
|
2023-05-31 16:15:15 +03:00
|
|
|
/// gesture will be qualified as pinch if distance between fingers is at least specified here
|
|
|
|
uint32_t pinchSensitivityThreshold = 10;
|
|
|
|
|
2023-09-06 01:06:01 +03:00
|
|
|
/// touch event will trigger clicking of elements up to X pixels away from actual touch position
|
|
|
|
uint32_t touchToleranceDistance = 20;
|
|
|
|
|
2023-05-26 18:55:26 +03:00
|
|
|
bool useRelativeMode = false;
|
2023-07-08 18:47:38 +02:00
|
|
|
|
2023-07-08 19:11:26 +02:00
|
|
|
bool hapticFeedbackEnabled = false;
|
2023-05-26 18:55:26 +03:00
|
|
|
};
|
|
|
|
|
2023-05-20 01:51:10 +03:00
|
|
|
/// Class that handles touchscreen input from SDL events
|
2023-05-18 20:32:29 +03:00
|
|
|
class InputSourceTouch
|
|
|
|
{
|
2023-05-26 18:55:26 +03:00
|
|
|
TouchInputParameters params;
|
|
|
|
TouchState state;
|
|
|
|
uint32_t lastTapTimeTicks;
|
2023-05-26 22:17:36 +03:00
|
|
|
Point lastTapPosition;
|
2023-05-18 20:32:29 +03:00
|
|
|
|
2023-09-18 16:05:39 +02:00
|
|
|
uint32_t lastLeftClickTimeTicks;
|
|
|
|
Point lastLeftClickPosition;
|
|
|
|
|
2023-05-18 20:32:29 +03:00
|
|
|
Point convertTouchToMouse(const SDL_TouchFingerEvent & current);
|
2023-05-31 16:15:15 +03:00
|
|
|
Point convertTouchToMouse(float x, float y);
|
2023-05-18 20:32:29 +03:00
|
|
|
|
2023-05-26 22:17:36 +03:00
|
|
|
void emitPanningEvent(const SDL_TouchFingerEvent & tfinger);
|
|
|
|
void emitPinchEvent(const SDL_TouchFingerEvent & tfinger);
|
2023-05-18 20:32:29 +03:00
|
|
|
|
|
|
|
public:
|
|
|
|
InputSourceTouch();
|
|
|
|
|
|
|
|
void handleEventFingerMotion(const SDL_TouchFingerEvent & current);
|
|
|
|
void handleEventFingerDown(const SDL_TouchFingerEvent & current);
|
|
|
|
void handleEventFingerUp(const SDL_TouchFingerEvent & current);
|
2023-05-26 15:55:31 +03:00
|
|
|
|
2023-07-23 15:17:30 +02:00
|
|
|
void hapticFeedback();
|
|
|
|
|
2023-05-26 18:55:26 +03:00
|
|
|
void handleUpdate();
|
|
|
|
|
2023-05-30 20:08:27 +03:00
|
|
|
bool hasTouchInputDevice() const;
|
2023-05-18 20:32:29 +03:00
|
|
|
};
|