blob: 34ba62515fd92822b6698694a916823f2471982f [file] [log] [blame]
Prabir Pradhanbaa5c822019-08-30 15:27:05 -07001/*
2 * Copyright (C) 2019 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
Prabir Pradhan48108662022-09-09 21:22:04 +000017#pragma once
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070018
Michael Wright227c5542020-07-02 18:30:52 +010019#include <stdint.h>
Michael Wrighta9cf4192022-12-01 23:46:39 +000020#include <ui/Rotation.h>
Michael Wright227c5542020-07-02 18:30:52 +010021
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070022#include "CursorButtonAccumulator.h"
23#include "CursorScrollAccumulator.h"
24#include "EventHub.h"
25#include "InputMapper.h"
26#include "InputReaderBase.h"
27#include "TouchButtonAccumulator.h"
28
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070029namespace android {
30
Prabir Pradhan7d04c4b2022-10-28 19:23:26 +000031// Maximum amount of latency to add to touch events while waiting for data from an
32// external stylus.
33static constexpr nsecs_t EXTERNAL_STYLUS_DATA_TIMEOUT = ms2ns(72);
34
35// Maximum amount of time to wait on touch data before pushing out new pressure data.
36static constexpr nsecs_t TOUCH_DATA_TIMEOUT = ms2ns(20);
37
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070038/* Raw axis information from the driver. */
39struct RawPointerAxes {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000040 RawAbsoluteAxisInfo x{};
41 RawAbsoluteAxisInfo y{};
42 RawAbsoluteAxisInfo pressure{};
43 RawAbsoluteAxisInfo touchMajor{};
44 RawAbsoluteAxisInfo touchMinor{};
45 RawAbsoluteAxisInfo toolMajor{};
46 RawAbsoluteAxisInfo toolMinor{};
47 RawAbsoluteAxisInfo orientation{};
48 RawAbsoluteAxisInfo distance{};
49 RawAbsoluteAxisInfo tiltX{};
50 RawAbsoluteAxisInfo tiltY{};
51 RawAbsoluteAxisInfo trackingId{};
52 RawAbsoluteAxisInfo slot{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070053
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070054 inline int32_t getRawWidth() const { return x.maxValue - x.minValue + 1; }
55 inline int32_t getRawHeight() const { return y.maxValue - y.minValue + 1; }
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000056 inline void clear() { *this = RawPointerAxes(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070057};
58
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000059using PropertiesArray = std::array<PointerProperties, MAX_POINTERS>;
60using CoordsArray = std::array<PointerCoords, MAX_POINTERS>;
61using IdToIndexArray = std::array<uint32_t, MAX_POINTER_ID + 1>;
62
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070063/* Raw data for a collection of pointers including a pointer id mapping table. */
64struct RawPointerData {
65 struct Pointer {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000066 uint32_t id{0xFFFFFFFF};
67 int32_t x{};
68 int32_t y{};
69 int32_t pressure{};
70 int32_t touchMajor{};
71 int32_t touchMinor{};
72 int32_t toolMajor{};
73 int32_t toolMinor{};
74 int32_t orientation{};
75 int32_t distance{};
76 int32_t tiltX{};
77 int32_t tiltY{};
78 // A fully decoded AMOTION_EVENT_TOOL_TYPE constant.
79 int32_t toolType{AMOTION_EVENT_TOOL_TYPE_UNKNOWN};
80 bool isHovering{false};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070081 };
82
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000083 uint32_t pointerCount{};
84 std::array<Pointer, MAX_POINTERS> pointers{};
85 BitSet32 hoveringIdBits{}, touchingIdBits{}, canceledIdBits{};
86 IdToIndexArray idToIndex{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070087
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000088 inline void clear() { *this = RawPointerData(); }
89
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070090 void getCentroidOfTouchingPointers(float* outX, float* outY) const;
91
92 inline void markIdBit(uint32_t id, bool isHovering) {
93 if (isHovering) {
94 hoveringIdBits.markBit(id);
95 } else {
96 touchingIdBits.markBit(id);
97 }
98 }
99
100 inline void clearIdBits() {
101 hoveringIdBits.clear();
102 touchingIdBits.clear();
arthurhungcc7f9802020-04-30 17:55:40 +0800103 canceledIdBits.clear();
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700104 }
105
106 inline const Pointer& pointerForId(uint32_t id) const { return pointers[idToIndex[id]]; }
107
108 inline bool isHovering(uint32_t pointerIndex) { return pointers[pointerIndex].isHovering; }
109};
110
111/* Cooked data for a collection of pointers including a pointer id mapping table. */
112struct CookedPointerData {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000113 uint32_t pointerCount{};
114 PropertiesArray pointerProperties{};
115 CoordsArray pointerCoords{};
116 BitSet32 hoveringIdBits{}, touchingIdBits{}, canceledIdBits{}, validIdBits{};
117 IdToIndexArray idToIndex{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700118
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000119 inline void clear() { *this = CookedPointerData(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700120
121 inline const PointerCoords& pointerCoordsForId(uint32_t id) const {
122 return pointerCoords[idToIndex[id]];
123 }
124
125 inline PointerCoords& editPointerCoordsWithId(uint32_t id) {
126 return pointerCoords[idToIndex[id]];
127 }
128
129 inline PointerProperties& editPointerPropertiesWithId(uint32_t id) {
130 return pointerProperties[idToIndex[id]];
131 }
132
133 inline bool isHovering(uint32_t pointerIndex) const {
134 return hoveringIdBits.hasBit(pointerProperties[pointerIndex].id);
135 }
136
137 inline bool isTouching(uint32_t pointerIndex) const {
138 return touchingIdBits.hasBit(pointerProperties[pointerIndex].id);
139 }
Nathaniel R. Lewisadb58ea2019-08-21 04:46:29 +0000140
141 inline bool hasPointerCoordsForId(uint32_t id) const { return validIdBits.hasBit(id); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700142};
143
144class TouchInputMapper : public InputMapper {
145public:
Nathaniel R. Lewis26ec2222020-01-10 16:30:54 -0800146 explicit TouchInputMapper(InputDeviceContext& deviceContext);
Michael Wright227c5542020-07-02 18:30:52 +0100147 ~TouchInputMapper() override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700148
Philip Junker4af3b3d2021-12-14 10:36:55 +0100149 uint32_t getSources() const override;
Michael Wright227c5542020-07-02 18:30:52 +0100150 void populateDeviceInfo(InputDeviceInfo* deviceInfo) override;
151 void dump(std::string& dump) override;
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700152 [[nodiscard]] std::list<NotifyArgs> configure(nsecs_t when,
153 const InputReaderConfiguration* config,
154 uint32_t changes) override;
155 [[nodiscard]] std::list<NotifyArgs> reset(nsecs_t when) override;
156 [[nodiscard]] std::list<NotifyArgs> process(const RawEvent* rawEvent) override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700157
Michael Wright227c5542020-07-02 18:30:52 +0100158 int32_t getKeyCodeState(uint32_t sourceMask, int32_t keyCode) override;
159 int32_t getScanCodeState(uint32_t sourceMask, int32_t scanCode) override;
Siarhei Vishniakou74007942022-06-13 13:57:47 -0700160 bool markSupportedKeyCodes(uint32_t sourceMask, const std::vector<int32_t>& keyCodes,
Michael Wright227c5542020-07-02 18:30:52 +0100161 uint8_t* outFlags) override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700162
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700163 [[nodiscard]] std::list<NotifyArgs> cancelTouch(nsecs_t when, nsecs_t readTime) override;
164 [[nodiscard]] std::list<NotifyArgs> timeoutExpired(nsecs_t when) override;
165 [[nodiscard]] std::list<NotifyArgs> updateExternalStylusState(
166 const StylusState& state) override;
Michael Wright227c5542020-07-02 18:30:52 +0100167 std::optional<int32_t> getAssociatedDisplayId() override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700168
169protected:
170 CursorButtonAccumulator mCursorButtonAccumulator;
171 CursorScrollAccumulator mCursorScrollAccumulator;
172 TouchButtonAccumulator mTouchButtonAccumulator;
173
174 struct VirtualKey {
175 int32_t keyCode;
176 int32_t scanCode;
177 uint32_t flags;
178
179 // computed hit box, specified in touch screen coords based on known display size
180 int32_t hitLeft;
181 int32_t hitTop;
182 int32_t hitRight;
183 int32_t hitBottom;
184
185 inline bool isHit(int32_t x, int32_t y) const {
186 return x >= hitLeft && x <= hitRight && y >= hitTop && y <= hitBottom;
187 }
188 };
189
190 // Input sources and device mode.
191 uint32_t mSource;
192
Michael Wright227c5542020-07-02 18:30:52 +0100193 enum class DeviceMode {
194 DISABLED, // input is disabled
195 DIRECT, // direct mapping (touchscreen)
196 UNSCALED, // unscaled mapping (touchpad)
197 NAVIGATION, // unscaled mapping with assist gesture (touch navigation)
198 POINTER, // pointer mapping (pointer)
Dominik Laskowski75788452021-02-09 18:51:25 -0800199
200 ftl_last = POINTER
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700201 };
202 DeviceMode mDeviceMode;
203
204 // The reader's configuration.
205 InputReaderConfiguration mConfig;
206
207 // Immutable configuration parameters.
208 struct Parameters {
Michael Wright227c5542020-07-02 18:30:52 +0100209 enum class DeviceType {
210 TOUCH_SCREEN,
Michael Wright227c5542020-07-02 18:30:52 +0100211 TOUCH_NAVIGATION,
212 POINTER,
Dominik Laskowski75788452021-02-09 18:51:25 -0800213
214 ftl_last = POINTER
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700215 };
216
217 DeviceType deviceType;
218 bool hasAssociatedDisplay;
219 bool associatedDisplayIsExternal;
220 bool orientationAware;
Prabir Pradhanac1c74f2021-08-20 16:09:32 -0700221
Michael Wrighta9cf4192022-12-01 23:46:39 +0000222 ui::Rotation orientation;
Prabir Pradhanac1c74f2021-08-20 16:09:32 -0700223
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700224 bool hasButtonUnderPad;
225 std::string uniqueDisplayId;
226
Michael Wright227c5542020-07-02 18:30:52 +0100227 enum class GestureMode {
228 SINGLE_TOUCH,
229 MULTI_TOUCH,
Dominik Laskowski75788452021-02-09 18:51:25 -0800230
231 ftl_last = MULTI_TOUCH
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700232 };
233 GestureMode gestureMode;
234
235 bool wake;
Prabir Pradhan167c2702022-09-14 00:37:24 +0000236
237 // Whether the device supports the Universal Stylus Initiative (USI) protocol for styluses.
238 bool supportsUsi;
Yuncheol Heo50c19b12022-11-02 20:33:08 -0700239
240 // Allows touches while the display is off.
241 bool enableForInactiveViewport;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700242 } mParameters;
243
244 // Immutable calibration parameters in parsed form.
245 struct Calibration {
246 // Size
Michael Wright227c5542020-07-02 18:30:52 +0100247 enum class SizeCalibration {
248 DEFAULT,
249 NONE,
250 GEOMETRIC,
251 DIAMETER,
252 BOX,
253 AREA,
Siarhei Vishniakou4e837cc2021-12-20 23:24:33 -0800254 ftl_last = AREA
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700255 };
256
257 SizeCalibration sizeCalibration;
258
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700259 std::optional<float> sizeScale;
260 std::optional<float> sizeBias;
261 std::optional<bool> sizeIsSummed;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700262
263 // Pressure
Michael Wright227c5542020-07-02 18:30:52 +0100264 enum class PressureCalibration {
265 DEFAULT,
266 NONE,
267 PHYSICAL,
268 AMPLITUDE,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700269 };
270
271 PressureCalibration pressureCalibration;
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700272 std::optional<float> pressureScale;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700273
274 // Orientation
Michael Wright227c5542020-07-02 18:30:52 +0100275 enum class OrientationCalibration {
276 DEFAULT,
277 NONE,
278 INTERPOLATED,
279 VECTOR,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700280 };
281
282 OrientationCalibration orientationCalibration;
283
284 // Distance
Michael Wright227c5542020-07-02 18:30:52 +0100285 enum class DistanceCalibration {
286 DEFAULT,
287 NONE,
288 SCALED,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700289 };
290
291 DistanceCalibration distanceCalibration;
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700292 std::optional<float> distanceScale;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700293
Michael Wright227c5542020-07-02 18:30:52 +0100294 enum class CoverageCalibration {
295 DEFAULT,
296 NONE,
297 BOX,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700298 };
299
300 CoverageCalibration coverageCalibration;
301
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700302 inline void applySizeScaleAndBias(float& outSize) const {
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700303 if (sizeScale) {
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700304 outSize *= *sizeScale;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700305 }
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700306 if (sizeBias) {
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700307 outSize += *sizeBias;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700308 }
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700309 if (outSize < 0) {
310 outSize = 0;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700311 }
312 }
313 } mCalibration;
314
315 // Affine location transformation/calibration
316 struct TouchAffineTransformation mAffineTransform;
317
318 RawPointerAxes mRawPointerAxes;
319
320 struct RawState {
Prabir Pradhan2f37bcb2022-11-08 20:41:28 +0000321 nsecs_t when{std::numeric_limits<nsecs_t>::min()};
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000322 nsecs_t readTime{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700323
324 // Raw pointer sample data.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000325 RawPointerData rawPointerData{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700326
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000327 int32_t buttonState{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700328
329 // Scroll state.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000330 int32_t rawVScroll{};
331 int32_t rawHScroll{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700332
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000333 inline void clear() { *this = RawState(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700334 };
335
336 struct CookedState {
337 // Cooked pointer sample data.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000338 CookedPointerData cookedPointerData{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700339
340 // Id bits used to differentiate fingers, stylus and mouse tools.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000341 BitSet32 fingerIdBits{};
342 BitSet32 stylusIdBits{};
343 BitSet32 mouseIdBits{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700344
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000345 int32_t buttonState{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700346
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000347 inline void clear() { *this = CookedState(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700348 };
349
350 std::vector<RawState> mRawStatesPending;
351 RawState mCurrentRawState;
352 CookedState mCurrentCookedState;
353 RawState mLastRawState;
354 CookedState mLastCookedState;
355
356 // State provided by an external stylus
357 StylusState mExternalStylusState;
Prabir Pradhan8d9ba912022-11-11 22:26:33 +0000358 // If an external stylus is capable of reporting pointer-specific data like pressure, we will
359 // attempt to fuse the pointer data reported by the stylus to the first touch pointer. This is
360 // the id of the pointer to which the external stylus data is fused.
361 std::optional<uint32_t> mFusedStylusPointerId;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700362 nsecs_t mExternalStylusFusionTimeout;
363 bool mExternalStylusDataPending;
Prabir Pradhan124ea442022-10-28 20:27:44 +0000364 // A subset of the buttons in mCurrentRawState that came from an external stylus.
365 int32_t mExternalStylusButtonsApplied;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700366
367 // True if we sent a HOVER_ENTER event.
368 bool mSentHoverEnter;
369
370 // Have we assigned pointer IDs for this stream
371 bool mHavePointerIds;
372
373 // Is the current stream of direct touch events aborted
374 bool mCurrentMotionAborted;
375
376 // The time the primary pointer last went down.
377 nsecs_t mDownTime;
378
379 // The pointer controller, or null if the device is not a pointer.
Michael Wright17db18e2020-06-26 20:51:44 +0100380 std::shared_ptr<PointerControllerInterface> mPointerController;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700381
382 std::vector<VirtualKey> mVirtualKeys;
383
384 virtual void configureParameters();
385 virtual void dumpParameters(std::string& dump);
386 virtual void configureRawPointerAxes();
387 virtual void dumpRawPointerAxes(std::string& dump);
Prabir Pradhan1728b212021-10-19 16:00:03 -0700388 virtual void configureInputDevice(nsecs_t when, bool* outResetNeeded);
389 virtual void dumpDisplay(std::string& dump);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700390 virtual void configureVirtualKeys();
391 virtual void dumpVirtualKeys(std::string& dump);
392 virtual void parseCalibration();
393 virtual void resolveCalibration();
394 virtual void dumpCalibration(std::string& dump);
395 virtual void updateAffineTransformation();
396 virtual void dumpAffineTransformation(std::string& dump);
397 virtual void resolveExternalStylusPresence();
398 virtual bool hasStylus() const = 0;
399 virtual bool hasExternalStylus() const;
400
401 virtual void syncTouch(nsecs_t when, RawState* outState) = 0;
402
403private:
404 // The current viewport.
405 // The components of the viewport are specified in the display's rotated orientation.
406 DisplayViewport mViewport;
407
Prabir Pradhan2d613f42022-11-10 20:22:06 +0000408 // We refer to the display as being in the "natural orientation" when there is no rotation
409 // applied. The display size obtained from the viewport in the natural orientation.
Prabir Pradhan7ddbc952022-11-09 22:03:40 +0000410 // Always starts at (0, 0).
411 ui::Size mDisplayBounds{ui::kInvalidSize};
Arthur Hung4197f6b2020-03-16 15:39:59 +0800412
Prabir Pradhan2d613f42022-11-10 20:22:06 +0000413 // The physical frame is the rectangle in the natural display's coordinate space that maps to
Prabir Pradhan1728b212021-10-19 16:00:03 -0700414 // the logical display frame.
Prabir Pradhan7ddbc952022-11-09 22:03:40 +0000415 Rect mPhysicalFrameInDisplay{Rect::INVALID_RECT};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700416
Prabir Pradhan1728b212021-10-19 16:00:03 -0700417 // The orientation of the input device relative to that of the display panel. It specifies
418 // the rotation of the input device coordinates required to produce the display panel
419 // orientation, so it will depend on whether the device is orientation aware.
Michael Wrighta9cf4192022-12-01 23:46:39 +0000420 ui::Rotation mInputDeviceOrientation;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700421
422 // Translation and scaling factors, orientation-independent.
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700423 float mXScale;
424 float mXPrecision;
425
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700426 float mYScale;
427 float mYPrecision;
428
429 float mGeometricScale;
430
431 float mPressureScale;
432
433 float mSizeScale;
434
435 float mOrientationScale;
436
437 float mDistanceScale;
438
439 bool mHaveTilt;
440 float mTiltXCenter;
441 float mTiltXScale;
442 float mTiltYCenter;
443 float mTiltYScale;
444
445 bool mExternalStylusConnected;
446
447 // Oriented motion ranges for input device info.
448 struct OrientedRanges {
449 InputDeviceInfo::MotionRange x;
450 InputDeviceInfo::MotionRange y;
451 InputDeviceInfo::MotionRange pressure;
452
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700453 std::optional<InputDeviceInfo::MotionRange> size;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700454
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700455 std::optional<InputDeviceInfo::MotionRange> touchMajor;
456 std::optional<InputDeviceInfo::MotionRange> touchMinor;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700457
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700458 std::optional<InputDeviceInfo::MotionRange> toolMajor;
459 std::optional<InputDeviceInfo::MotionRange> toolMinor;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700460
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700461 std::optional<InputDeviceInfo::MotionRange> orientation;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700462
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700463 std::optional<InputDeviceInfo::MotionRange> distance;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700464
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700465 std::optional<InputDeviceInfo::MotionRange> tilt;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700466
467 void clear() {
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700468 size = std::nullopt;
469 touchMajor = std::nullopt;
470 touchMinor = std::nullopt;
471 toolMajor = std::nullopt;
472 toolMinor = std::nullopt;
473 orientation = std::nullopt;
474 distance = std::nullopt;
475 tilt = std::nullopt;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700476 }
477 } mOrientedRanges;
478
479 // Oriented dimensions and precision.
480 float mOrientedXPrecision;
481 float mOrientedYPrecision;
482
483 struct CurrentVirtualKeyState {
484 bool down;
485 bool ignored;
486 nsecs_t downTime;
487 int32_t keyCode;
488 int32_t scanCode;
489 } mCurrentVirtualKey;
490
491 // Scale factor for gesture or mouse based pointer movements.
492 float mPointerXMovementScale;
493 float mPointerYMovementScale;
494
495 // Scale factor for gesture based zooming and other freeform motions.
496 float mPointerXZoomScale;
497 float mPointerYZoomScale;
498
HQ Liue6983c72022-04-19 22:14:56 +0000499 // The maximum swipe width between pointers to detect a swipe gesture
500 // in the number of pixels.Touches that are wider than this are translated
501 // into freeform gestures.
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700502 float mPointerGestureMaxSwipeWidth;
503
504 struct PointerDistanceHeapElement {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000505 uint32_t currentPointerIndex : 8 {};
506 uint32_t lastPointerIndex : 8 {};
507 uint64_t distance : 48 {}; // squared distance
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700508 };
509
Michael Wright227c5542020-07-02 18:30:52 +0100510 enum class PointerUsage {
511 NONE,
512 GESTURES,
513 STYLUS,
514 MOUSE,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700515 };
516 PointerUsage mPointerUsage;
517
518 struct PointerGesture {
Michael Wright227c5542020-07-02 18:30:52 +0100519 enum class Mode {
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700520 // No fingers, button is not pressed.
521 // Nothing happening.
522 NEUTRAL,
523
524 // No fingers, button is not pressed.
525 // Tap detected.
526 // Emits DOWN and UP events at the pointer location.
527 TAP,
528
529 // Exactly one finger dragging following a tap.
530 // Pointer follows the active finger.
531 // Emits DOWN, MOVE and UP events at the pointer location.
532 //
533 // Detect double-taps when the finger goes up while in TAP_DRAG mode.
534 TAP_DRAG,
535
536 // Button is pressed.
537 // Pointer follows the active finger if there is one. Other fingers are ignored.
538 // Emits DOWN, MOVE and UP events at the pointer location.
539 BUTTON_CLICK_OR_DRAG,
540
541 // Exactly one finger, button is not pressed.
542 // Pointer follows the active finger.
543 // Emits HOVER_MOVE events at the pointer location.
544 //
545 // Detect taps when the finger goes up while in HOVER mode.
546 HOVER,
547
548 // Exactly two fingers but neither have moved enough to clearly indicate
549 // whether a swipe or freeform gesture was intended. We consider the
550 // pointer to be pressed so this enables clicking or long-pressing on buttons.
551 // Pointer does not move.
552 // Emits DOWN, MOVE and UP events with a single stationary pointer coordinate.
553 PRESS,
554
555 // Exactly two fingers moving in the same direction, button is not pressed.
556 // Pointer does not move.
557 // Emits DOWN, MOVE and UP events with a single pointer coordinate that
558 // follows the midpoint between both fingers.
559 SWIPE,
560
561 // Two or more fingers moving in arbitrary directions, button is not pressed.
562 // Pointer does not move.
563 // Emits DOWN, POINTER_DOWN, MOVE, POINTER_UP and UP events that follow
564 // each finger individually relative to the initial centroid of the finger.
565 FREEFORM,
566
567 // Waiting for quiet time to end before starting the next gesture.
568 QUIET,
569 };
570
Prabir Pradhan47cf0a02021-03-11 20:30:57 -0800571 // When a gesture is sent to an unfocused window, return true if it can bring that window
572 // into focus, false otherwise.
573 static bool canGestureAffectWindowFocus(Mode mode) {
574 switch (mode) {
575 case Mode::TAP:
576 case Mode::TAP_DRAG:
577 case Mode::BUTTON_CLICK_OR_DRAG:
578 // Taps can affect window focus.
579 return true;
580 case Mode::FREEFORM:
581 case Mode::HOVER:
582 case Mode::NEUTRAL:
583 case Mode::PRESS:
584 case Mode::QUIET:
585 case Mode::SWIPE:
586 // Most gestures can be performed on an unfocused window, so they should not
587 // not affect window focus.
588 return false;
589 }
590 }
591
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700592 // Time the first finger went down.
593 nsecs_t firstTouchTime;
594
595 // The active pointer id from the raw touch data.
596 int32_t activeTouchId; // -1 if none
597
598 // The active pointer id from the gesture last delivered to the application.
599 int32_t activeGestureId; // -1 if none
600
601 // Pointer coords and ids for the current and previous pointer gesture.
602 Mode currentGestureMode;
603 BitSet32 currentGestureIdBits;
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000604 IdToIndexArray currentGestureIdToIndex{};
605 PropertiesArray currentGestureProperties{};
606 CoordsArray currentGestureCoords{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700607
608 Mode lastGestureMode;
609 BitSet32 lastGestureIdBits;
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000610 IdToIndexArray lastGestureIdToIndex{};
611 PropertiesArray lastGestureProperties{};
612 CoordsArray lastGestureCoords{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700613
614 // Time the pointer gesture last went down.
615 nsecs_t downTime;
616
617 // Time when the pointer went down for a TAP.
618 nsecs_t tapDownTime;
619
620 // Time when the pointer went up for a TAP.
621 nsecs_t tapUpTime;
622
623 // Location of initial tap.
624 float tapX, tapY;
625
626 // Time we started waiting for quiescence.
627 nsecs_t quietTime;
628
629 // Reference points for multitouch gestures.
630 float referenceTouchX; // reference touch X/Y coordinates in surface units
631 float referenceTouchY;
632 float referenceGestureX; // reference gesture X/Y coordinates in pixels
633 float referenceGestureY;
634
635 // Distance that each pointer has traveled which has not yet been
636 // subsumed into the reference gesture position.
637 BitSet32 referenceIdBits;
638 struct Delta {
639 float dx, dy;
640 };
641 Delta referenceDeltas[MAX_POINTER_ID + 1];
642
643 // Describes how touch ids are mapped to gesture ids for freeform gestures.
644 uint32_t freeformTouchToGestureIdMap[MAX_POINTER_ID + 1];
645
646 // A velocity tracker for determining whether to switch active pointers during drags.
647 VelocityTracker velocityTracker;
648
649 void reset() {
650 firstTouchTime = LLONG_MIN;
651 activeTouchId = -1;
652 activeGestureId = -1;
Michael Wright227c5542020-07-02 18:30:52 +0100653 currentGestureMode = Mode::NEUTRAL;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700654 currentGestureIdBits.clear();
Michael Wright227c5542020-07-02 18:30:52 +0100655 lastGestureMode = Mode::NEUTRAL;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700656 lastGestureIdBits.clear();
657 downTime = 0;
658 velocityTracker.clear();
659 resetTap();
660 resetQuietTime();
661 }
662
663 void resetTap() {
664 tapDownTime = LLONG_MIN;
665 tapUpTime = LLONG_MIN;
666 }
667
668 void resetQuietTime() { quietTime = LLONG_MIN; }
669 } mPointerGesture;
670
671 struct PointerSimple {
672 PointerCoords currentCoords;
673 PointerProperties currentProperties;
674 PointerCoords lastCoords;
675 PointerProperties lastProperties;
676
677 // True if the pointer is down.
678 bool down;
679
680 // True if the pointer is hovering.
681 bool hovering;
682
683 // Time the pointer last went down.
684 nsecs_t downTime;
685
Prabir Pradhanb80b6c02022-11-02 20:05:13 +0000686 // Values reported for the last pointer event.
687 uint32_t source;
688 int32_t displayId;
689 float lastCursorX;
690 float lastCursorY;
691
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700692 void reset() {
693 currentCoords.clear();
694 currentProperties.clear();
695 lastCoords.clear();
696 lastProperties.clear();
697 down = false;
698 hovering = false;
699 downTime = 0;
Prabir Pradhanb80b6c02022-11-02 20:05:13 +0000700 source = 0;
701 displayId = ADISPLAY_ID_NONE;
702 lastCursorX = 0.f;
703 lastCursorY = 0.f;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700704 }
705 } mPointerSimple;
706
707 // The pointer and scroll velocity controls.
708 VelocityControl mPointerVelocityControl;
709 VelocityControl mWheelXVelocityControl;
710 VelocityControl mWheelYVelocityControl;
711
712 std::optional<DisplayViewport> findViewport();
713
714 void resetExternalStylus();
715 void clearStylusDataPendingFlags();
716
Siarhei Vishniakou12c0fcb2021-12-17 13:40:44 -0800717 int32_t clampResolution(const char* axisName, int32_t resolution) const;
Siarhei Vishniakou4e837cc2021-12-20 23:24:33 -0800718 void initializeOrientedRanges();
719 void initializeSizeRanges();
720
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700721 [[nodiscard]] std::list<NotifyArgs> sync(nsecs_t when, nsecs_t readTime);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700722
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700723 [[nodiscard]] std::list<NotifyArgs> consumeRawTouches(nsecs_t when, nsecs_t readTime,
724 uint32_t policyFlags, bool& outConsumed);
725 [[nodiscard]] std::list<NotifyArgs> processRawTouches(bool timeout);
726 [[nodiscard]] std::list<NotifyArgs> cookAndDispatch(nsecs_t when, nsecs_t readTime);
727 [[nodiscard]] NotifyKeyArgs dispatchVirtualKey(nsecs_t when, nsecs_t readTime,
728 uint32_t policyFlags, int32_t keyEventAction,
729 int32_t keyEventFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700730
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700731 [[nodiscard]] std::list<NotifyArgs> dispatchTouches(nsecs_t when, nsecs_t readTime,
732 uint32_t policyFlags);
733 [[nodiscard]] std::list<NotifyArgs> dispatchHoverExit(nsecs_t when, nsecs_t readTime,
734 uint32_t policyFlags);
735 [[nodiscard]] std::list<NotifyArgs> dispatchHoverEnterAndMove(nsecs_t when, nsecs_t readTime,
736 uint32_t policyFlags);
737 [[nodiscard]] std::list<NotifyArgs> dispatchButtonRelease(nsecs_t when, nsecs_t readTime,
738 uint32_t policyFlags);
739 [[nodiscard]] std::list<NotifyArgs> dispatchButtonPress(nsecs_t when, nsecs_t readTime,
740 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700741 const BitSet32& findActiveIdBits(const CookedPointerData& cookedPointerData);
742 void cookPointerData();
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700743 [[nodiscard]] std::list<NotifyArgs> abortTouches(nsecs_t when, nsecs_t readTime,
744 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700745
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700746 [[nodiscard]] std::list<NotifyArgs> dispatchPointerUsage(nsecs_t when, nsecs_t readTime,
747 uint32_t policyFlags,
748 PointerUsage pointerUsage);
749 [[nodiscard]] std::list<NotifyArgs> abortPointerUsage(nsecs_t when, nsecs_t readTime,
750 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700751
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700752 [[nodiscard]] std::list<NotifyArgs> dispatchPointerGestures(nsecs_t when, nsecs_t readTime,
753 uint32_t policyFlags,
754 bool isTimeout);
755 [[nodiscard]] std::list<NotifyArgs> abortPointerGestures(nsecs_t when, nsecs_t readTime,
756 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700757 bool preparePointerGestures(nsecs_t when, bool* outCancelPreviousGesture,
758 bool* outFinishPreviousGesture, bool isTimeout);
759
Harry Cuttsbea6ce52022-10-14 15:17:30 +0000760 // Returns true if we're in a period of "quiet time" when touchpad gestures should be ignored.
761 bool checkForTouchpadQuietTime(nsecs_t when);
762
763 std::pair<int32_t, float> getFastestFinger();
764
765 void prepareMultiFingerPointerGestures(nsecs_t when, bool* outCancelPreviousGesture,
766 bool* outFinishPreviousGesture);
767
Harry Cutts714d1ad2022-08-24 16:36:43 +0000768 // Moves the on-screen mouse pointer based on the movement of the pointer of the given ID
769 // between the last and current events. Uses a relative motion.
770 void moveMousePointerFromPointerDelta(nsecs_t when, uint32_t pointerId);
771
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700772 [[nodiscard]] std::list<NotifyArgs> dispatchPointerStylus(nsecs_t when, nsecs_t readTime,
773 uint32_t policyFlags);
774 [[nodiscard]] std::list<NotifyArgs> abortPointerStylus(nsecs_t when, nsecs_t readTime,
775 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700776
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700777 [[nodiscard]] std::list<NotifyArgs> dispatchPointerMouse(nsecs_t when, nsecs_t readTime,
778 uint32_t policyFlags);
779 [[nodiscard]] std::list<NotifyArgs> abortPointerMouse(nsecs_t when, nsecs_t readTime,
780 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700781
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700782 [[nodiscard]] std::list<NotifyArgs> dispatchPointerSimple(nsecs_t when, nsecs_t readTime,
783 uint32_t policyFlags, bool down,
784 bool hovering);
785 [[nodiscard]] std::list<NotifyArgs> abortPointerSimple(nsecs_t when, nsecs_t readTime,
786 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700787
Prabir Pradhan3f7545f2022-10-19 16:56:39 +0000788 // Attempts to assign a pointer id to the external stylus. Returns true if the state should be
789 // withheld from further processing while waiting for data from the stylus.
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700790 bool assignExternalStylusId(const RawState& state, bool timeout);
791 void applyExternalStylusButtonState(nsecs_t when);
792 void applyExternalStylusTouchState(nsecs_t when);
793
794 // Dispatches a motion event.
795 // If the changedId is >= 0 and the action is POINTER_DOWN or POINTER_UP, the
796 // method will take care of setting the index and transmuting the action to DOWN or UP
797 // it is the first / last pointer to go down / up.
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700798 [[nodiscard]] NotifyMotionArgs dispatchMotion(
799 nsecs_t when, nsecs_t readTime, uint32_t policyFlags, uint32_t source, int32_t action,
800 int32_t actionButton, int32_t flags, int32_t metaState, int32_t buttonState,
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000801 int32_t edgeFlags, const PropertiesArray& properties, const CoordsArray& coords,
802 const IdToIndexArray& idToIndex, BitSet32 idBits, int32_t changedId, float xPrecision,
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700803 float yPrecision, nsecs_t downTime, MotionClassification classification);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700804
Garfield Tanc734e4f2021-01-15 20:01:39 -0800805 // Returns if this touch device is a touch screen with an associated display.
806 bool isTouchScreen();
807 // Updates touch spots if they are enabled. Should only be used when this device is a
808 // touchscreen.
809 void updateTouchSpots();
810
Prabir Pradhan1728b212021-10-19 16:00:03 -0700811 bool isPointInsidePhysicalFrame(int32_t x, int32_t y) const;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700812 const VirtualKey* findVirtualKeyHit(int32_t x, int32_t y);
813
Siarhei Vishniakou57479982021-03-03 01:32:21 +0000814 static void assignPointerIds(const RawState& last, RawState& current);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700815
Prabir Pradhan1728b212021-10-19 16:00:03 -0700816 void rotateAndScale(float& x, float& y) const;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700817};
818
819} // namespace android