blob: 5914808f45c33c15027a08759834e1828795c557 [file] [log] [blame]
Siarhei Vishniakou6e1e9872022-11-08 17:51:35 -08001/*
2 * Copyright (C) 2022 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
17#include "TouchedWindow.h"
18
Siarhei Vishniakou0836a302023-05-03 13:54:30 -070019#include <android-base/logging.h>
Siarhei Vishniakou6e1e9872022-11-08 17:51:35 -080020#include <android-base/stringprintf.h>
21#include <input/PrintTools.h>
22
23using android::base::StringPrintf;
24
25namespace android {
26
27namespace inputdispatcher {
28
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -080029namespace {
30
31bool hasPointerId(const std::vector<PointerProperties>& pointers, int32_t pointerId) {
32 return std::find_if(pointers.begin(), pointers.end(),
33 [&pointerId](const PointerProperties& properties) {
34 return properties.id == pointerId;
35 }) != pointers.end();
36}
37
38} // namespace
39
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +000040bool TouchedWindow::hasHoveringPointers() const {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -070041 for (const auto& [_, state] : mDeviceStates) {
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -080042 if (!state.hoveringPointers.empty()) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -070043 return true;
44 }
45 }
46 return false;
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +000047}
48
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -070049bool TouchedWindow::hasHoveringPointers(DeviceId deviceId) const {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -070050 const auto stateIt = mDeviceStates.find(deviceId);
51 if (stateIt == mDeviceStates.end()) {
52 return false;
53 }
54 const DeviceState& state = stateIt->second;
55
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -080056 return !state.hoveringPointers.empty();
Siarhei Vishniakoue0431e42023-01-28 17:01:39 -080057}
58
Siarhei Vishniakou2899c552023-07-10 18:20:46 -070059void TouchedWindow::clearHoveringPointers(DeviceId deviceId) {
60 auto stateIt = mDeviceStates.find(deviceId);
61 if (stateIt == mDeviceStates.end()) {
62 return;
Siarhei Vishniakou0836a302023-05-03 13:54:30 -070063 }
Siarhei Vishniakou2899c552023-07-10 18:20:46 -070064 DeviceState& state = stateIt->second;
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -080065 state.hoveringPointers.clear();
Siarhei Vishniakou2899c552023-07-10 18:20:46 -070066 if (!state.hasPointers()) {
67 mDeviceStates.erase(stateIt);
68 }
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +000069}
70
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -070071bool TouchedWindow::hasHoveringPointer(DeviceId deviceId, int32_t pointerId) const {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -070072 const auto stateIt = mDeviceStates.find(deviceId);
73 if (stateIt == mDeviceStates.end()) {
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +000074 return false;
75 }
Siarhei Vishniakou0836a302023-05-03 13:54:30 -070076 const DeviceState& state = stateIt->second;
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -080077 return hasPointerId(state.hoveringPointers, pointerId);
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +000078}
79
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -080080void TouchedWindow::addHoveringPointer(DeviceId deviceId, const PointerProperties& pointer) {
81 std::vector<PointerProperties>& hoveringPointers = mDeviceStates[deviceId].hoveringPointers;
82 const size_t initialSize = hoveringPointers.size();
83 std::erase_if(hoveringPointers, [&pointer](const PointerProperties& properties) {
84 return properties.id == pointer.id;
85 });
86 if (hoveringPointers.size() != initialSize) {
87 LOG(ERROR) << __func__ << ": " << pointer << ", device " << deviceId << " was in " << *this;
88 }
89 hoveringPointers.push_back(pointer);
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +000090}
91
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -070092void TouchedWindow::addTouchingPointers(DeviceId deviceId,
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -080093 const std::vector<PointerProperties>& pointers) {
94 std::vector<PointerProperties>& touchingPointers = mDeviceStates[deviceId].touchingPointers;
95 const size_t initialSize = touchingPointers.size();
96 for (const PointerProperties& pointer : pointers) {
97 std::erase_if(touchingPointers, [&pointer](const PointerProperties& properties) {
98 return properties.id == pointer.id;
99 });
100 }
101 if (touchingPointers.size() != initialSize) {
102 LOG(ERROR) << __func__ << ": " << dumpVector(pointers, streamableToString) << ", device "
103 << deviceId << " already in " << *this;
104 }
105 touchingPointers.insert(touchingPointers.end(), pointers.begin(), pointers.end());
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700106}
107
108bool TouchedWindow::hasTouchingPointers() const {
109 for (const auto& [_, state] : mDeviceStates) {
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800110 if (!state.touchingPointers.empty()) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700111 return true;
112 }
113 }
114 return false;
115}
116
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700117bool TouchedWindow::hasTouchingPointers(DeviceId deviceId) const {
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800118 return !getTouchingPointers(deviceId).empty();
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700119}
120
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700121bool TouchedWindow::hasTouchingPointer(DeviceId deviceId, int32_t pointerId) const {
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800122 const auto stateIt = mDeviceStates.find(deviceId);
123 if (stateIt == mDeviceStates.end()) {
124 return false;
125 }
126 const DeviceState& state = stateIt->second;
127 return hasPointerId(state.touchingPointers, pointerId);
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700128}
129
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800130std::vector<PointerProperties> TouchedWindow::getTouchingPointers(DeviceId deviceId) const {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700131 const auto stateIt = mDeviceStates.find(deviceId);
132 if (stateIt == mDeviceStates.end()) {
133 return {};
134 }
135 const DeviceState& state = stateIt->second;
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800136 return state.touchingPointers;
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700137}
138
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700139void TouchedWindow::removeTouchingPointer(DeviceId deviceId, int32_t pointerId) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700140 std::bitset<MAX_POINTER_ID + 1> pointerIds;
141 pointerIds.set(pointerId, true);
142
143 removeTouchingPointers(deviceId, pointerIds);
144}
145
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700146void TouchedWindow::removeTouchingPointers(DeviceId deviceId,
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700147 std::bitset<MAX_POINTER_ID + 1> pointers) {
148 const auto stateIt = mDeviceStates.find(deviceId);
149 if (stateIt == mDeviceStates.end()) {
150 return;
151 }
152 DeviceState& state = stateIt->second;
153
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800154 std::erase_if(state.touchingPointers, [&pointers](const PointerProperties& properties) {
155 return pointers.test(properties.id);
156 });
157
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700158 state.pilferingPointerIds &= ~pointers;
159
160 if (!state.hasPointers()) {
161 mDeviceStates.erase(stateIt);
Siarhei Vishniakou6464e462023-02-06 18:57:59 -0800162 }
163}
164
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700165std::set<DeviceId> TouchedWindow::getTouchingDeviceIds() const {
166 std::set<DeviceId> deviceIds;
Prabir Pradhan9cd9eb62023-11-22 17:58:06 +0000167 for (const auto& [deviceId, deviceState] : mDeviceStates) {
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800168 if (!deviceState.touchingPointers.empty()) {
Prabir Pradhan9cd9eb62023-11-22 17:58:06 +0000169 deviceIds.insert(deviceId);
170 }
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700171 }
172 return deviceIds;
173}
174
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700175bool TouchedWindow::hasPilferingPointers(DeviceId deviceId) const {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700176 const auto stateIt = mDeviceStates.find(deviceId);
177 if (stateIt == mDeviceStates.end()) {
178 return false;
179 }
180 const DeviceState& state = stateIt->second;
181
182 return state.pilferingPointerIds.any();
183}
184
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700185void TouchedWindow::addPilferingPointers(DeviceId deviceId,
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700186 std::bitset<MAX_POINTER_ID + 1> pointerIds) {
187 mDeviceStates[deviceId].pilferingPointerIds |= pointerIds;
188}
189
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700190void TouchedWindow::addPilferingPointer(DeviceId deviceId, int32_t pointerId) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700191 mDeviceStates[deviceId].pilferingPointerIds.set(pointerId);
192}
193
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700194std::bitset<MAX_POINTER_ID + 1> TouchedWindow::getPilferingPointers(DeviceId deviceId) const {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700195 const auto stateIt = mDeviceStates.find(deviceId);
196 if (stateIt == mDeviceStates.end()) {
197 return {};
198 }
199 const DeviceState& state = stateIt->second;
200
201 return state.pilferingPointerIds;
202}
203
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700204std::map<DeviceId, std::bitset<MAX_POINTER_ID + 1>> TouchedWindow::getPilferingPointers() const {
205 std::map<DeviceId, std::bitset<MAX_POINTER_ID + 1>> out;
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700206 for (const auto& [deviceId, state] : mDeviceStates) {
207 out.emplace(deviceId, state.pilferingPointerIds);
208 }
209 return out;
210}
211
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700212std::optional<nsecs_t> TouchedWindow::getDownTimeInTarget(DeviceId deviceId) const {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700213 const auto stateIt = mDeviceStates.find(deviceId);
214 if (stateIt == mDeviceStates.end()) {
215 return {};
216 }
217 const DeviceState& state = stateIt->second;
218 return state.downTimeInTarget;
219}
220
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700221void TouchedWindow::trySetDownTimeInTarget(DeviceId deviceId, nsecs_t downTime) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700222 auto [stateIt, _] = mDeviceStates.try_emplace(deviceId);
223 DeviceState& state = stateIt->second;
224
225 if (!state.downTimeInTarget) {
226 state.downTimeInTarget = downTime;
227 }
228}
229
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700230void TouchedWindow::removeAllTouchingPointersForDevice(DeviceId deviceId) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700231 const auto stateIt = mDeviceStates.find(deviceId);
232 if (stateIt == mDeviceStates.end()) {
233 return;
234 }
235 DeviceState& state = stateIt->second;
236
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800237 state.touchingPointers.clear();
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700238 state.pilferingPointerIds.reset();
239 state.downTimeInTarget.reset();
240
241 if (!state.hasPointers()) {
242 mDeviceStates.erase(stateIt);
243 }
Siarhei Vishniakou0686f0c2023-05-02 11:56:15 -0700244}
245
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700246void TouchedWindow::removeHoveringPointer(DeviceId deviceId, int32_t pointerId) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700247 const auto stateIt = mDeviceStates.find(deviceId);
248 if (stateIt == mDeviceStates.end()) {
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +0000249 return;
250 }
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700251 DeviceState& state = stateIt->second;
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +0000252
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800253 std::erase_if(state.hoveringPointers, [&pointerId](const PointerProperties& properties) {
254 return properties.id == pointerId;
255 });
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700256
257 if (!state.hasPointers()) {
258 mDeviceStates.erase(stateIt);
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +0000259 }
260}
261
Siarhei Vishniakoud38a1e02023-07-18 11:55:17 -0700262void TouchedWindow::removeAllHoveringPointersForDevice(DeviceId deviceId) {
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700263 const auto stateIt = mDeviceStates.find(deviceId);
264 if (stateIt == mDeviceStates.end()) {
265 return;
266 }
267 DeviceState& state = stateIt->second;
268
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800269 state.hoveringPointers.clear();
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700270
271 if (!state.hasPointers()) {
272 mDeviceStates.erase(stateIt);
273 }
274}
275
276std::string TouchedWindow::deviceStateToString(const TouchedWindow::DeviceState& state) {
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800277 return StringPrintf("[touchingPointers=%s, "
278 "downTimeInTarget=%s, hoveringPointers=%s, pilferingPointerIds=%s]",
279 dumpVector(state.touchingPointers, streamableToString).c_str(),
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700280 toString(state.downTimeInTarget).c_str(),
Siarhei Vishniakou1ff00cc2023-12-13 16:12:13 -0800281 dumpVector(state.hoveringPointers, streamableToString).c_str(),
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700282 bitsetToString(state.pilferingPointerIds).c_str());
Siarhei Vishniakou0686f0c2023-05-02 11:56:15 -0700283}
284
Siarhei Vishniakou6e1e9872022-11-08 17:51:35 -0800285std::string TouchedWindow::dump() const {
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +0000286 std::string out;
Siarhei Vishniakou0836a302023-05-03 13:54:30 -0700287 std::string deviceStates =
288 dumpMap(mDeviceStates, constToString, TouchedWindow::deviceStateToString);
289 out += StringPrintf("name='%s', targetFlags=%s, mDeviceStates=%s\n",
290 windowHandle->getName().c_str(), targetFlags.string().c_str(),
291 deviceStates.c_str());
Siarhei Vishniakoub581f7f2022-12-07 20:23:06 +0000292 return out;
Siarhei Vishniakou6e1e9872022-11-08 17:51:35 -0800293}
294
Siarhei Vishniakou72945a02023-09-18 18:30:25 -0700295std::ostream& operator<<(std::ostream& out, const TouchedWindow& window) {
296 out << window.dump();
297 return out;
298}
299
Siarhei Vishniakou6e1e9872022-11-08 17:51:35 -0800300} // namespace inputdispatcher
301} // namespace android