blob: 2bb9ecebe4c094a8a434e3e471c88ccfa3f97dd3 [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>
20
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070021#include "CursorButtonAccumulator.h"
22#include "CursorScrollAccumulator.h"
23#include "EventHub.h"
24#include "InputMapper.h"
25#include "InputReaderBase.h"
26#include "TouchButtonAccumulator.h"
27
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070028namespace android {
29
30/* Raw axis information from the driver. */
31struct RawPointerAxes {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000032 RawAbsoluteAxisInfo x{};
33 RawAbsoluteAxisInfo y{};
34 RawAbsoluteAxisInfo pressure{};
35 RawAbsoluteAxisInfo touchMajor{};
36 RawAbsoluteAxisInfo touchMinor{};
37 RawAbsoluteAxisInfo toolMajor{};
38 RawAbsoluteAxisInfo toolMinor{};
39 RawAbsoluteAxisInfo orientation{};
40 RawAbsoluteAxisInfo distance{};
41 RawAbsoluteAxisInfo tiltX{};
42 RawAbsoluteAxisInfo tiltY{};
43 RawAbsoluteAxisInfo trackingId{};
44 RawAbsoluteAxisInfo slot{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070045
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070046 inline int32_t getRawWidth() const { return x.maxValue - x.minValue + 1; }
47 inline int32_t getRawHeight() const { return y.maxValue - y.minValue + 1; }
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000048 inline void clear() { *this = RawPointerAxes(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070049};
50
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000051using PropertiesArray = std::array<PointerProperties, MAX_POINTERS>;
52using CoordsArray = std::array<PointerCoords, MAX_POINTERS>;
53using IdToIndexArray = std::array<uint32_t, MAX_POINTER_ID + 1>;
54
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070055/* Raw data for a collection of pointers including a pointer id mapping table. */
56struct RawPointerData {
57 struct Pointer {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000058 uint32_t id{0xFFFFFFFF};
59 int32_t x{};
60 int32_t y{};
61 int32_t pressure{};
62 int32_t touchMajor{};
63 int32_t touchMinor{};
64 int32_t toolMajor{};
65 int32_t toolMinor{};
66 int32_t orientation{};
67 int32_t distance{};
68 int32_t tiltX{};
69 int32_t tiltY{};
70 // A fully decoded AMOTION_EVENT_TOOL_TYPE constant.
71 int32_t toolType{AMOTION_EVENT_TOOL_TYPE_UNKNOWN};
72 bool isHovering{false};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070073 };
74
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000075 uint32_t pointerCount{};
76 std::array<Pointer, MAX_POINTERS> pointers{};
77 BitSet32 hoveringIdBits{}, touchingIdBits{}, canceledIdBits{};
78 IdToIndexArray idToIndex{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070079
Prabir Pradhand6ccedb2022-09-27 21:04:06 +000080 inline void clear() { *this = RawPointerData(); }
81
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070082 void getCentroidOfTouchingPointers(float* outX, float* outY) const;
83
84 inline void markIdBit(uint32_t id, bool isHovering) {
85 if (isHovering) {
86 hoveringIdBits.markBit(id);
87 } else {
88 touchingIdBits.markBit(id);
89 }
90 }
91
92 inline void clearIdBits() {
93 hoveringIdBits.clear();
94 touchingIdBits.clear();
arthurhungcc7f9802020-04-30 17:55:40 +080095 canceledIdBits.clear();
Prabir Pradhanbaa5c822019-08-30 15:27:05 -070096 }
97
98 inline const Pointer& pointerForId(uint32_t id) const { return pointers[idToIndex[id]]; }
99
100 inline bool isHovering(uint32_t pointerIndex) { return pointers[pointerIndex].isHovering; }
101};
102
103/* Cooked data for a collection of pointers including a pointer id mapping table. */
104struct CookedPointerData {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000105 uint32_t pointerCount{};
106 PropertiesArray pointerProperties{};
107 CoordsArray pointerCoords{};
108 BitSet32 hoveringIdBits{}, touchingIdBits{}, canceledIdBits{}, validIdBits{};
109 IdToIndexArray idToIndex{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700110
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000111 inline void clear() { *this = CookedPointerData(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700112
113 inline const PointerCoords& pointerCoordsForId(uint32_t id) const {
114 return pointerCoords[idToIndex[id]];
115 }
116
117 inline PointerCoords& editPointerCoordsWithId(uint32_t id) {
118 return pointerCoords[idToIndex[id]];
119 }
120
121 inline PointerProperties& editPointerPropertiesWithId(uint32_t id) {
122 return pointerProperties[idToIndex[id]];
123 }
124
125 inline bool isHovering(uint32_t pointerIndex) const {
126 return hoveringIdBits.hasBit(pointerProperties[pointerIndex].id);
127 }
128
129 inline bool isTouching(uint32_t pointerIndex) const {
130 return touchingIdBits.hasBit(pointerProperties[pointerIndex].id);
131 }
Nathaniel R. Lewisadb58ea2019-08-21 04:46:29 +0000132
133 inline bool hasPointerCoordsForId(uint32_t id) const { return validIdBits.hasBit(id); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700134};
135
136class TouchInputMapper : public InputMapper {
137public:
Nathaniel R. Lewis26ec2222020-01-10 16:30:54 -0800138 explicit TouchInputMapper(InputDeviceContext& deviceContext);
Michael Wright227c5542020-07-02 18:30:52 +0100139 ~TouchInputMapper() override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700140
Philip Junker4af3b3d2021-12-14 10:36:55 +0100141 uint32_t getSources() const override;
Michael Wright227c5542020-07-02 18:30:52 +0100142 void populateDeviceInfo(InputDeviceInfo* deviceInfo) override;
143 void dump(std::string& dump) override;
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700144 [[nodiscard]] std::list<NotifyArgs> configure(nsecs_t when,
145 const InputReaderConfiguration* config,
146 uint32_t changes) override;
147 [[nodiscard]] std::list<NotifyArgs> reset(nsecs_t when) override;
148 [[nodiscard]] std::list<NotifyArgs> process(const RawEvent* rawEvent) override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700149
Michael Wright227c5542020-07-02 18:30:52 +0100150 int32_t getKeyCodeState(uint32_t sourceMask, int32_t keyCode) override;
151 int32_t getScanCodeState(uint32_t sourceMask, int32_t scanCode) override;
Siarhei Vishniakou74007942022-06-13 13:57:47 -0700152 bool markSupportedKeyCodes(uint32_t sourceMask, const std::vector<int32_t>& keyCodes,
Michael Wright227c5542020-07-02 18:30:52 +0100153 uint8_t* outFlags) override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700154
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700155 [[nodiscard]] std::list<NotifyArgs> cancelTouch(nsecs_t when, nsecs_t readTime) override;
156 [[nodiscard]] std::list<NotifyArgs> timeoutExpired(nsecs_t when) override;
157 [[nodiscard]] std::list<NotifyArgs> updateExternalStylusState(
158 const StylusState& state) override;
Michael Wright227c5542020-07-02 18:30:52 +0100159 std::optional<int32_t> getAssociatedDisplayId() override;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700160
161protected:
162 CursorButtonAccumulator mCursorButtonAccumulator;
163 CursorScrollAccumulator mCursorScrollAccumulator;
164 TouchButtonAccumulator mTouchButtonAccumulator;
165
166 struct VirtualKey {
167 int32_t keyCode;
168 int32_t scanCode;
169 uint32_t flags;
170
171 // computed hit box, specified in touch screen coords based on known display size
172 int32_t hitLeft;
173 int32_t hitTop;
174 int32_t hitRight;
175 int32_t hitBottom;
176
177 inline bool isHit(int32_t x, int32_t y) const {
178 return x >= hitLeft && x <= hitRight && y >= hitTop && y <= hitBottom;
179 }
180 };
181
182 // Input sources and device mode.
183 uint32_t mSource;
184
Michael Wright227c5542020-07-02 18:30:52 +0100185 enum class DeviceMode {
186 DISABLED, // input is disabled
187 DIRECT, // direct mapping (touchscreen)
188 UNSCALED, // unscaled mapping (touchpad)
189 NAVIGATION, // unscaled mapping with assist gesture (touch navigation)
190 POINTER, // pointer mapping (pointer)
Dominik Laskowski75788452021-02-09 18:51:25 -0800191
192 ftl_last = POINTER
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700193 };
194 DeviceMode mDeviceMode;
195
196 // The reader's configuration.
197 InputReaderConfiguration mConfig;
198
199 // Immutable configuration parameters.
200 struct Parameters {
Michael Wright227c5542020-07-02 18:30:52 +0100201 enum class DeviceType {
202 TOUCH_SCREEN,
Michael Wright227c5542020-07-02 18:30:52 +0100203 TOUCH_NAVIGATION,
204 POINTER,
Dominik Laskowski75788452021-02-09 18:51:25 -0800205
206 ftl_last = POINTER
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700207 };
208
209 DeviceType deviceType;
210 bool hasAssociatedDisplay;
211 bool associatedDisplayIsExternal;
212 bool orientationAware;
Prabir Pradhanac1c74f2021-08-20 16:09:32 -0700213
214 enum class Orientation : int32_t {
215 ORIENTATION_0 = DISPLAY_ORIENTATION_0,
216 ORIENTATION_90 = DISPLAY_ORIENTATION_90,
217 ORIENTATION_180 = DISPLAY_ORIENTATION_180,
218 ORIENTATION_270 = DISPLAY_ORIENTATION_270,
Dominik Laskowski75788452021-02-09 18:51:25 -0800219
220 ftl_last = ORIENTATION_270
Prabir Pradhanac1c74f2021-08-20 16:09:32 -0700221 };
222 Orientation orientation;
223
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;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700239 } mParameters;
240
241 // Immutable calibration parameters in parsed form.
242 struct Calibration {
243 // Size
Michael Wright227c5542020-07-02 18:30:52 +0100244 enum class SizeCalibration {
245 DEFAULT,
246 NONE,
247 GEOMETRIC,
248 DIAMETER,
249 BOX,
250 AREA,
Siarhei Vishniakou4e837cc2021-12-20 23:24:33 -0800251 ftl_last = AREA
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700252 };
253
254 SizeCalibration sizeCalibration;
255
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700256 std::optional<float> sizeScale;
257 std::optional<float> sizeBias;
258 std::optional<bool> sizeIsSummed;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700259
260 // Pressure
Michael Wright227c5542020-07-02 18:30:52 +0100261 enum class PressureCalibration {
262 DEFAULT,
263 NONE,
264 PHYSICAL,
265 AMPLITUDE,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700266 };
267
268 PressureCalibration pressureCalibration;
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700269 std::optional<float> pressureScale;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700270
271 // Orientation
Michael Wright227c5542020-07-02 18:30:52 +0100272 enum class OrientationCalibration {
273 DEFAULT,
274 NONE,
275 INTERPOLATED,
276 VECTOR,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700277 };
278
279 OrientationCalibration orientationCalibration;
280
281 // Distance
Michael Wright227c5542020-07-02 18:30:52 +0100282 enum class DistanceCalibration {
283 DEFAULT,
284 NONE,
285 SCALED,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700286 };
287
288 DistanceCalibration distanceCalibration;
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700289 std::optional<float> distanceScale;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700290
Michael Wright227c5542020-07-02 18:30:52 +0100291 enum class CoverageCalibration {
292 DEFAULT,
293 NONE,
294 BOX,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700295 };
296
297 CoverageCalibration coverageCalibration;
298
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700299 inline void applySizeScaleAndBias(float& outSize) const {
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700300 if (sizeScale) {
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700301 outSize *= *sizeScale;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700302 }
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700303 if (sizeBias) {
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700304 outSize += *sizeBias;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700305 }
Siarhei Vishniakou07247342022-07-15 14:27:37 -0700306 if (outSize < 0) {
307 outSize = 0;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700308 }
309 }
310 } mCalibration;
311
312 // Affine location transformation/calibration
313 struct TouchAffineTransformation mAffineTransform;
314
315 RawPointerAxes mRawPointerAxes;
316
317 struct RawState {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000318 nsecs_t when{};
319 nsecs_t readTime{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700320
321 // Raw pointer sample data.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000322 RawPointerData rawPointerData{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700323
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000324 int32_t buttonState{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700325
326 // Scroll state.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000327 int32_t rawVScroll{};
328 int32_t rawHScroll{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700329
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000330 inline void clear() { *this = RawState(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700331 };
332
333 struct CookedState {
334 // Cooked pointer sample data.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000335 CookedPointerData cookedPointerData{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700336
337 // Id bits used to differentiate fingers, stylus and mouse tools.
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000338 BitSet32 fingerIdBits{};
339 BitSet32 stylusIdBits{};
340 BitSet32 mouseIdBits{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700341
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000342 int32_t buttonState{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700343
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000344 inline void clear() { *this = CookedState(); }
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700345 };
346
347 std::vector<RawState> mRawStatesPending;
348 RawState mCurrentRawState;
349 CookedState mCurrentCookedState;
350 RawState mLastRawState;
351 CookedState mLastCookedState;
352
353 // State provided by an external stylus
354 StylusState mExternalStylusState;
355 int64_t mExternalStylusId;
356 nsecs_t mExternalStylusFusionTimeout;
357 bool mExternalStylusDataPending;
358
359 // True if we sent a HOVER_ENTER event.
360 bool mSentHoverEnter;
361
362 // Have we assigned pointer IDs for this stream
363 bool mHavePointerIds;
364
365 // Is the current stream of direct touch events aborted
366 bool mCurrentMotionAborted;
367
368 // The time the primary pointer last went down.
369 nsecs_t mDownTime;
370
371 // The pointer controller, or null if the device is not a pointer.
Michael Wright17db18e2020-06-26 20:51:44 +0100372 std::shared_ptr<PointerControllerInterface> mPointerController;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700373
374 std::vector<VirtualKey> mVirtualKeys;
375
376 virtual void configureParameters();
377 virtual void dumpParameters(std::string& dump);
378 virtual void configureRawPointerAxes();
379 virtual void dumpRawPointerAxes(std::string& dump);
Prabir Pradhan1728b212021-10-19 16:00:03 -0700380 virtual void configureInputDevice(nsecs_t when, bool* outResetNeeded);
381 virtual void dumpDisplay(std::string& dump);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700382 virtual void configureVirtualKeys();
383 virtual void dumpVirtualKeys(std::string& dump);
384 virtual void parseCalibration();
385 virtual void resolveCalibration();
386 virtual void dumpCalibration(std::string& dump);
387 virtual void updateAffineTransformation();
388 virtual void dumpAffineTransformation(std::string& dump);
389 virtual void resolveExternalStylusPresence();
390 virtual bool hasStylus() const = 0;
391 virtual bool hasExternalStylus() const;
392
393 virtual void syncTouch(nsecs_t when, RawState* outState) = 0;
394
395private:
396 // The current viewport.
397 // The components of the viewport are specified in the display's rotated orientation.
398 DisplayViewport mViewport;
399
Prabir Pradhan1728b212021-10-19 16:00:03 -0700400 // The width and height are obtained from the viewport and are specified
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700401 // in the natural orientation.
Prabir Pradhan1728b212021-10-19 16:00:03 -0700402 int32_t mDisplayWidth;
403 int32_t mDisplayHeight;
Arthur Hung4197f6b2020-03-16 15:39:59 +0800404
Prabir Pradhan1728b212021-10-19 16:00:03 -0700405 // The physical frame is the rectangle in the display's coordinate space that maps to the
406 // the logical display frame.
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700407 int32_t mPhysicalWidth;
408 int32_t mPhysicalHeight;
409 int32_t mPhysicalLeft;
410 int32_t mPhysicalTop;
411
Prabir Pradhan1728b212021-10-19 16:00:03 -0700412 // The orientation of the input device relative to that of the display panel. It specifies
413 // the rotation of the input device coordinates required to produce the display panel
414 // orientation, so it will depend on whether the device is orientation aware.
415 int32_t mInputDeviceOrientation;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700416
417 // Translation and scaling factors, orientation-independent.
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700418 float mXScale;
419 float mXPrecision;
420
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700421 float mYScale;
422 float mYPrecision;
423
424 float mGeometricScale;
425
426 float mPressureScale;
427
428 float mSizeScale;
429
430 float mOrientationScale;
431
432 float mDistanceScale;
433
434 bool mHaveTilt;
435 float mTiltXCenter;
436 float mTiltXScale;
437 float mTiltYCenter;
438 float mTiltYScale;
439
440 bool mExternalStylusConnected;
441
442 // Oriented motion ranges for input device info.
443 struct OrientedRanges {
444 InputDeviceInfo::MotionRange x;
445 InputDeviceInfo::MotionRange y;
446 InputDeviceInfo::MotionRange pressure;
447
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700448 std::optional<InputDeviceInfo::MotionRange> size;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700449
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700450 std::optional<InputDeviceInfo::MotionRange> touchMajor;
451 std::optional<InputDeviceInfo::MotionRange> touchMinor;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700452
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700453 std::optional<InputDeviceInfo::MotionRange> toolMajor;
454 std::optional<InputDeviceInfo::MotionRange> toolMinor;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700455
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700456 std::optional<InputDeviceInfo::MotionRange> orientation;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700457
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700458 std::optional<InputDeviceInfo::MotionRange> distance;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700459
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700460 std::optional<InputDeviceInfo::MotionRange> tilt;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700461
462 void clear() {
Siarhei Vishniakou24210882022-07-15 09:42:04 -0700463 size = std::nullopt;
464 touchMajor = std::nullopt;
465 touchMinor = std::nullopt;
466 toolMajor = std::nullopt;
467 toolMinor = std::nullopt;
468 orientation = std::nullopt;
469 distance = std::nullopt;
470 tilt = std::nullopt;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700471 }
472 } mOrientedRanges;
473
474 // Oriented dimensions and precision.
475 float mOrientedXPrecision;
476 float mOrientedYPrecision;
477
478 struct CurrentVirtualKeyState {
479 bool down;
480 bool ignored;
481 nsecs_t downTime;
482 int32_t keyCode;
483 int32_t scanCode;
484 } mCurrentVirtualKey;
485
486 // Scale factor for gesture or mouse based pointer movements.
487 float mPointerXMovementScale;
488 float mPointerYMovementScale;
489
490 // Scale factor for gesture based zooming and other freeform motions.
491 float mPointerXZoomScale;
492 float mPointerYZoomScale;
493
HQ Liue6983c72022-04-19 22:14:56 +0000494 // The maximum swipe width between pointers to detect a swipe gesture
495 // in the number of pixels.Touches that are wider than this are translated
496 // into freeform gestures.
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700497 float mPointerGestureMaxSwipeWidth;
498
499 struct PointerDistanceHeapElement {
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000500 uint32_t currentPointerIndex : 8 {};
501 uint32_t lastPointerIndex : 8 {};
502 uint64_t distance : 48 {}; // squared distance
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700503 };
504
Michael Wright227c5542020-07-02 18:30:52 +0100505 enum class PointerUsage {
506 NONE,
507 GESTURES,
508 STYLUS,
509 MOUSE,
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700510 };
511 PointerUsage mPointerUsage;
512
513 struct PointerGesture {
Michael Wright227c5542020-07-02 18:30:52 +0100514 enum class Mode {
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700515 // No fingers, button is not pressed.
516 // Nothing happening.
517 NEUTRAL,
518
519 // No fingers, button is not pressed.
520 // Tap detected.
521 // Emits DOWN and UP events at the pointer location.
522 TAP,
523
524 // Exactly one finger dragging following a tap.
525 // Pointer follows the active finger.
526 // Emits DOWN, MOVE and UP events at the pointer location.
527 //
528 // Detect double-taps when the finger goes up while in TAP_DRAG mode.
529 TAP_DRAG,
530
531 // Button is pressed.
532 // Pointer follows the active finger if there is one. Other fingers are ignored.
533 // Emits DOWN, MOVE and UP events at the pointer location.
534 BUTTON_CLICK_OR_DRAG,
535
536 // Exactly one finger, button is not pressed.
537 // Pointer follows the active finger.
538 // Emits HOVER_MOVE events at the pointer location.
539 //
540 // Detect taps when the finger goes up while in HOVER mode.
541 HOVER,
542
543 // Exactly two fingers but neither have moved enough to clearly indicate
544 // whether a swipe or freeform gesture was intended. We consider the
545 // pointer to be pressed so this enables clicking or long-pressing on buttons.
546 // Pointer does not move.
547 // Emits DOWN, MOVE and UP events with a single stationary pointer coordinate.
548 PRESS,
549
550 // Exactly two fingers moving in the same direction, button is not pressed.
551 // Pointer does not move.
552 // Emits DOWN, MOVE and UP events with a single pointer coordinate that
553 // follows the midpoint between both fingers.
554 SWIPE,
555
556 // Two or more fingers moving in arbitrary directions, button is not pressed.
557 // Pointer does not move.
558 // Emits DOWN, POINTER_DOWN, MOVE, POINTER_UP and UP events that follow
559 // each finger individually relative to the initial centroid of the finger.
560 FREEFORM,
561
562 // Waiting for quiet time to end before starting the next gesture.
563 QUIET,
564 };
565
Prabir Pradhan47cf0a02021-03-11 20:30:57 -0800566 // When a gesture is sent to an unfocused window, return true if it can bring that window
567 // into focus, false otherwise.
568 static bool canGestureAffectWindowFocus(Mode mode) {
569 switch (mode) {
570 case Mode::TAP:
571 case Mode::TAP_DRAG:
572 case Mode::BUTTON_CLICK_OR_DRAG:
573 // Taps can affect window focus.
574 return true;
575 case Mode::FREEFORM:
576 case Mode::HOVER:
577 case Mode::NEUTRAL:
578 case Mode::PRESS:
579 case Mode::QUIET:
580 case Mode::SWIPE:
581 // Most gestures can be performed on an unfocused window, so they should not
582 // not affect window focus.
583 return false;
584 }
585 }
586
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700587 // Time the first finger went down.
588 nsecs_t firstTouchTime;
589
590 // The active pointer id from the raw touch data.
591 int32_t activeTouchId; // -1 if none
592
593 // The active pointer id from the gesture last delivered to the application.
594 int32_t activeGestureId; // -1 if none
595
596 // Pointer coords and ids for the current and previous pointer gesture.
597 Mode currentGestureMode;
598 BitSet32 currentGestureIdBits;
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000599 IdToIndexArray currentGestureIdToIndex{};
600 PropertiesArray currentGestureProperties{};
601 CoordsArray currentGestureCoords{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700602
603 Mode lastGestureMode;
604 BitSet32 lastGestureIdBits;
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000605 IdToIndexArray lastGestureIdToIndex{};
606 PropertiesArray lastGestureProperties{};
607 CoordsArray lastGestureCoords{};
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700608
609 // Time the pointer gesture last went down.
610 nsecs_t downTime;
611
612 // Time when the pointer went down for a TAP.
613 nsecs_t tapDownTime;
614
615 // Time when the pointer went up for a TAP.
616 nsecs_t tapUpTime;
617
618 // Location of initial tap.
619 float tapX, tapY;
620
621 // Time we started waiting for quiescence.
622 nsecs_t quietTime;
623
624 // Reference points for multitouch gestures.
625 float referenceTouchX; // reference touch X/Y coordinates in surface units
626 float referenceTouchY;
627 float referenceGestureX; // reference gesture X/Y coordinates in pixels
628 float referenceGestureY;
629
630 // Distance that each pointer has traveled which has not yet been
631 // subsumed into the reference gesture position.
632 BitSet32 referenceIdBits;
633 struct Delta {
634 float dx, dy;
635 };
636 Delta referenceDeltas[MAX_POINTER_ID + 1];
637
638 // Describes how touch ids are mapped to gesture ids for freeform gestures.
639 uint32_t freeformTouchToGestureIdMap[MAX_POINTER_ID + 1];
640
641 // A velocity tracker for determining whether to switch active pointers during drags.
642 VelocityTracker velocityTracker;
643
644 void reset() {
645 firstTouchTime = LLONG_MIN;
646 activeTouchId = -1;
647 activeGestureId = -1;
Michael Wright227c5542020-07-02 18:30:52 +0100648 currentGestureMode = Mode::NEUTRAL;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700649 currentGestureIdBits.clear();
Michael Wright227c5542020-07-02 18:30:52 +0100650 lastGestureMode = Mode::NEUTRAL;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700651 lastGestureIdBits.clear();
652 downTime = 0;
653 velocityTracker.clear();
654 resetTap();
655 resetQuietTime();
656 }
657
658 void resetTap() {
659 tapDownTime = LLONG_MIN;
660 tapUpTime = LLONG_MIN;
661 }
662
663 void resetQuietTime() { quietTime = LLONG_MIN; }
664 } mPointerGesture;
665
666 struct PointerSimple {
667 PointerCoords currentCoords;
668 PointerProperties currentProperties;
669 PointerCoords lastCoords;
670 PointerProperties lastProperties;
671
672 // True if the pointer is down.
673 bool down;
674
675 // True if the pointer is hovering.
676 bool hovering;
677
678 // Time the pointer last went down.
679 nsecs_t downTime;
680
681 void reset() {
682 currentCoords.clear();
683 currentProperties.clear();
684 lastCoords.clear();
685 lastProperties.clear();
686 down = false;
687 hovering = false;
688 downTime = 0;
689 }
690 } mPointerSimple;
691
692 // The pointer and scroll velocity controls.
693 VelocityControl mPointerVelocityControl;
694 VelocityControl mWheelXVelocityControl;
695 VelocityControl mWheelYVelocityControl;
696
697 std::optional<DisplayViewport> findViewport();
698
699 void resetExternalStylus();
700 void clearStylusDataPendingFlags();
701
Siarhei Vishniakou12c0fcb2021-12-17 13:40:44 -0800702 int32_t clampResolution(const char* axisName, int32_t resolution) const;
Siarhei Vishniakou4e837cc2021-12-20 23:24:33 -0800703 void initializeOrientedRanges();
704 void initializeSizeRanges();
705
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700706 [[nodiscard]] std::list<NotifyArgs> sync(nsecs_t when, nsecs_t readTime);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700707
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700708 [[nodiscard]] std::list<NotifyArgs> consumeRawTouches(nsecs_t when, nsecs_t readTime,
709 uint32_t policyFlags, bool& outConsumed);
710 [[nodiscard]] std::list<NotifyArgs> processRawTouches(bool timeout);
711 [[nodiscard]] std::list<NotifyArgs> cookAndDispatch(nsecs_t when, nsecs_t readTime);
712 [[nodiscard]] NotifyKeyArgs dispatchVirtualKey(nsecs_t when, nsecs_t readTime,
713 uint32_t policyFlags, int32_t keyEventAction,
714 int32_t keyEventFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700715
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700716 [[nodiscard]] std::list<NotifyArgs> dispatchTouches(nsecs_t when, nsecs_t readTime,
717 uint32_t policyFlags);
718 [[nodiscard]] std::list<NotifyArgs> dispatchHoverExit(nsecs_t when, nsecs_t readTime,
719 uint32_t policyFlags);
720 [[nodiscard]] std::list<NotifyArgs> dispatchHoverEnterAndMove(nsecs_t when, nsecs_t readTime,
721 uint32_t policyFlags);
722 [[nodiscard]] std::list<NotifyArgs> dispatchButtonRelease(nsecs_t when, nsecs_t readTime,
723 uint32_t policyFlags);
724 [[nodiscard]] std::list<NotifyArgs> dispatchButtonPress(nsecs_t when, nsecs_t readTime,
725 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700726 const BitSet32& findActiveIdBits(const CookedPointerData& cookedPointerData);
727 void cookPointerData();
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700728 [[nodiscard]] std::list<NotifyArgs> abortTouches(nsecs_t when, nsecs_t readTime,
729 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700730
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700731 [[nodiscard]] std::list<NotifyArgs> dispatchPointerUsage(nsecs_t when, nsecs_t readTime,
732 uint32_t policyFlags,
733 PointerUsage pointerUsage);
734 [[nodiscard]] std::list<NotifyArgs> abortPointerUsage(nsecs_t when, nsecs_t readTime,
735 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700736
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700737 [[nodiscard]] std::list<NotifyArgs> dispatchPointerGestures(nsecs_t when, nsecs_t readTime,
738 uint32_t policyFlags,
739 bool isTimeout);
740 [[nodiscard]] std::list<NotifyArgs> abortPointerGestures(nsecs_t when, nsecs_t readTime,
741 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700742 bool preparePointerGestures(nsecs_t when, bool* outCancelPreviousGesture,
743 bool* outFinishPreviousGesture, bool isTimeout);
744
Harry Cuttsbea6ce52022-10-14 15:17:30 +0000745 // Returns true if we're in a period of "quiet time" when touchpad gestures should be ignored.
746 bool checkForTouchpadQuietTime(nsecs_t when);
747
748 std::pair<int32_t, float> getFastestFinger();
749
750 void prepareMultiFingerPointerGestures(nsecs_t when, bool* outCancelPreviousGesture,
751 bool* outFinishPreviousGesture);
752
Harry Cutts714d1ad2022-08-24 16:36:43 +0000753 // Moves the on-screen mouse pointer based on the movement of the pointer of the given ID
754 // between the last and current events. Uses a relative motion.
755 void moveMousePointerFromPointerDelta(nsecs_t when, uint32_t pointerId);
756
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700757 [[nodiscard]] std::list<NotifyArgs> dispatchPointerStylus(nsecs_t when, nsecs_t readTime,
758 uint32_t policyFlags);
759 [[nodiscard]] std::list<NotifyArgs> abortPointerStylus(nsecs_t when, nsecs_t readTime,
760 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700761
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700762 [[nodiscard]] std::list<NotifyArgs> dispatchPointerMouse(nsecs_t when, nsecs_t readTime,
763 uint32_t policyFlags);
764 [[nodiscard]] std::list<NotifyArgs> abortPointerMouse(nsecs_t when, nsecs_t readTime,
765 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700766
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700767 [[nodiscard]] std::list<NotifyArgs> dispatchPointerSimple(nsecs_t when, nsecs_t readTime,
768 uint32_t policyFlags, bool down,
769 bool hovering);
770 [[nodiscard]] std::list<NotifyArgs> abortPointerSimple(nsecs_t when, nsecs_t readTime,
771 uint32_t policyFlags);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700772
773 bool assignExternalStylusId(const RawState& state, bool timeout);
774 void applyExternalStylusButtonState(nsecs_t when);
775 void applyExternalStylusTouchState(nsecs_t when);
776
777 // Dispatches a motion event.
778 // If the changedId is >= 0 and the action is POINTER_DOWN or POINTER_UP, the
779 // method will take care of setting the index and transmuting the action to DOWN or UP
780 // it is the first / last pointer to go down / up.
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700781 [[nodiscard]] NotifyMotionArgs dispatchMotion(
782 nsecs_t when, nsecs_t readTime, uint32_t policyFlags, uint32_t source, int32_t action,
783 int32_t actionButton, int32_t flags, int32_t metaState, int32_t buttonState,
Prabir Pradhand6ccedb2022-09-27 21:04:06 +0000784 int32_t edgeFlags, const PropertiesArray& properties, const CoordsArray& coords,
785 const IdToIndexArray& idToIndex, BitSet32 idBits, int32_t changedId, float xPrecision,
Siarhei Vishniakou2935db72022-09-22 13:35:22 -0700786 float yPrecision, nsecs_t downTime, MotionClassification classification);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700787
Garfield Tanc734e4f2021-01-15 20:01:39 -0800788 // Returns if this touch device is a touch screen with an associated display.
789 bool isTouchScreen();
790 // Updates touch spots if they are enabled. Should only be used when this device is a
791 // touchscreen.
792 void updateTouchSpots();
793
Prabir Pradhan1728b212021-10-19 16:00:03 -0700794 bool isPointInsidePhysicalFrame(int32_t x, int32_t y) const;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700795 const VirtualKey* findVirtualKeyHit(int32_t x, int32_t y);
796
Siarhei Vishniakou57479982021-03-03 01:32:21 +0000797 static void assignPointerIds(const RawState& last, RawState& current);
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700798
Prabir Pradhan1728b212021-10-19 16:00:03 -0700799 void rotateAndScale(float& x, float& y) const;
Prabir Pradhanbaa5c822019-08-30 15:27:05 -0700800};
801
802} // namespace android