2023-05-18 19:32:29 +02: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 21:17:36 +02:00
|
|
|
#include "../../lib/Point.h"
|
2023-05-18 19:32:29 +02:00
|
|
|
|
2023-06-16 12:07:02 +02: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 14:55:31 +02:00
|
|
|
enum class MouseButton;
|
2023-05-18 19:32:29 +02:00
|
|
|
struct SDL_TouchFingerEvent;
|
|
|
|
|
2023-05-26 17:55:26 +02: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 12:07:02 +02:00
|
|
|
// on timer -> transition to TAP_DOWN_LONG, emit showPopup() event
|
2023-05-26 17:55:26 +02: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 18:21:09 +02:00
|
|
|
// UP -> transition to TAP_DOWN_LONG_AWAIT
|
2023-05-26 17:55:26 +02:00
|
|
|
TAP_DOWN_LONG,
|
|
|
|
|
2023-05-29 18:21:09 +02:00
|
|
|
// right-click popup is active, waiting for new tap to hide popup
|
|
|
|
// DOWN -> ignored
|
|
|
|
// MOTION -> ignored
|
2023-06-16 12:07:02 +02:00
|
|
|
// UP -> transition to IDLE, generate closePopup() event
|
2023-05-29 18:21:09 +02:00
|
|
|
TAP_DOWN_LONG_AWAIT,
|
2023-05-26 17:55:26 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
struct TouchInputParameters
|
|
|
|
{
|
2023-05-31 12:08:12 +02:00
|
|
|
/// Speed factor of mouse pointer when relative mode is used
|
2023-05-26 17:55:26 +02:00
|
|
|
double relativeModeSpeedFactor = 1.0;
|
2023-05-26 21:17:36 +02:00
|
|
|
|
|
|
|
/// tap for period longer than specified here will be qualified as "long tap", triggering corresponding gesture
|
2023-06-16 11:59:20 +02:00
|
|
|
uint32_t longTouchTimeMilliseconds = 750;
|
2023-05-26 17:55:26 +02:00
|
|
|
|
2023-05-26 21:17:36 +02:00
|
|
|
/// moving finger for distance larger than specified will be qualified as panning gesture instead of long press
|
2023-05-31 12:08:12 +02:00
|
|
|
uint32_t panningSensitivityThreshold = 10;
|
2023-05-26 21:17:36 +02:00
|
|
|
|
2023-05-31 15:15:15 +02:00
|
|
|
/// gesture will be qualified as pinch if distance between fingers is at least specified here
|
|
|
|
uint32_t pinchSensitivityThreshold = 10;
|
|
|
|
|
2023-05-26 17:55:26 +02: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 17:55:26 +02:00
|
|
|
};
|
|
|
|
|
2023-05-20 00:51:10 +02:00
|
|
|
/// Class that handles touchscreen input from SDL events
|
2023-05-18 19:32:29 +02:00
|
|
|
class InputSourceTouch
|
|
|
|
{
|
2023-05-26 17:55:26 +02:00
|
|
|
TouchInputParameters params;
|
|
|
|
TouchState state;
|
|
|
|
uint32_t lastTapTimeTicks;
|
2023-05-26 21:17:36 +02:00
|
|
|
Point lastTapPosition;
|
2023-05-18 19:32:29 +02:00
|
|
|
|
|
|
|
Point convertTouchToMouse(const SDL_TouchFingerEvent & current);
|
2023-05-31 15:15:15 +02:00
|
|
|
Point convertTouchToMouse(float x, float y);
|
2023-05-18 19:32:29 +02:00
|
|
|
|
2023-05-26 21:17:36 +02:00
|
|
|
void emitPanningEvent(const SDL_TouchFingerEvent & tfinger);
|
|
|
|
void emitPinchEvent(const SDL_TouchFingerEvent & tfinger);
|
2023-05-18 19:32:29 +02:00
|
|
|
|
|
|
|
public:
|
|
|
|
InputSourceTouch();
|
|
|
|
|
|
|
|
void handleEventFingerMotion(const SDL_TouchFingerEvent & current);
|
|
|
|
void handleEventFingerDown(const SDL_TouchFingerEvent & current);
|
|
|
|
void handleEventFingerUp(const SDL_TouchFingerEvent & current);
|
2023-05-26 14:55:31 +02:00
|
|
|
|
2023-07-23 15:17:30 +02:00
|
|
|
void hapticFeedback();
|
|
|
|
|
2023-05-26 17:55:26 +02:00
|
|
|
void handleUpdate();
|
|
|
|
|
2023-05-30 19:08:27 +02:00
|
|
|
bool hasTouchInputDevice() const;
|
2023-05-18 19:32:29 +02:00
|
|
|
};
|