Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(633)

Side by Side Diff: content/browser/renderer_host/input/synthetic_pointer_action_controller_unittest.cc

Issue 1884883005: Prepare SyntheticPointerAction to handle touch actions for multiple fingers (Closed) Base URL: https://chromium.googlesource.com/chromium/src.git@master
Patch Set: Move synthetic pointer action tests in a seperate file Created 4 years, 6 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch
OLDNEW
(Empty)
1 // Copyright 2016 The Chromium Authors. All rights reserved.
2 // Use of this source code is governed by a BSD-style license that can be
3 // found in the LICENSE file.
4
5 #include "base/bind.h"
6 #include "base/time/time.h"
7 #include "content/browser/renderer_host/input/synthetic_gesture.h"
8 #include "content/browser/renderer_host/input/synthetic_gesture_controller.h"
9 #include "content/browser/renderer_host/input/synthetic_gesture_target.h"
10 #include "content/browser/renderer_host/input/synthetic_pointer_action.h"
11 #include "content/browser/renderer_host/input/synthetic_touch_pointer.h"
12 #include "testing/gtest/include/gtest/gtest.h"
13 #include "third_party/WebKit/public/web/WebInputEvent.h"
14 #include "ui/gfx/geometry/point.h"
15 #include "ui/gfx/geometry/point_f.h"
16
17 using blink::WebInputEvent;
18 using blink::WebTouchEvent;
19 using blink::WebTouchPoint;
20
21 namespace content {
22
23 namespace {
24
25 const int kFlushInputRateInMs = 16;
26
27 class MockSyntheticGestureTarget : public SyntheticGestureTarget {
28 public:
29 MockSyntheticGestureTarget() : flush_requested_(false) {}
30 ~MockSyntheticGestureTarget() override {}
31
32 // SyntheticGestureTarget:
33 void DispatchInputEventToPlatform(const WebInputEvent& event) override {}
34
35 void SetNeedsFlush() override { flush_requested_ = true; }
36
37 SyntheticGestureParams::GestureSourceType
38 GetDefaultSyntheticGestureSourceType() const override {
39 return SyntheticGestureParams::TOUCH_INPUT;
40 }
41
42 base::TimeDelta PointerAssumedStoppedTime() const override {
43 NOTIMPLEMENTED();
44 return base::TimeDelta();
45 }
46
47 void set_pointer_assumed_stopped_time_ms(int time_ms) { NOTIMPLEMENTED(); }
tdresser 2016/06/06 18:36:08 Why is this here? It isn't overriding anything...
lanwei 2016/06/08 14:48:37 SyntheticGestureTarget class has these virtual met
lanwei 2016/06/09 14:37:03 Done.
48
49 float GetTouchSlopInDips() const override {
50 NOTIMPLEMENTED();
51 return 0.0f;
52 }
53
54 float GetMinScalingSpanInDips() const override {
55 NOTIMPLEMENTED();
56 return 0.0f;
57 }
58
59 bool flush_requested() const { return flush_requested_; }
60 void ClearFlushRequest() { flush_requested_ = false; }
61
62 private:
63 bool flush_requested_;
64 };
65
66 class MockSyntheticPointerActionTarget : public MockSyntheticGestureTarget {
67 public:
68 MockSyntheticPointerActionTarget() {}
69 ~MockSyntheticPointerActionTarget() override {}
70
71 gfx::PointF positions(int index) const { return positions_[index]; }
72 int indexes(int index) const { return indexes_[index]; }
73 WebTouchPoint::State states(int index) { return states_[index]; }
74 unsigned touch_length() const { return touch_length_; }
75 WebInputEvent::Type type() const { return type_; }
76
77 protected:
78 gfx::PointF positions_[WebTouchEvent::touchesLengthCap];
79 unsigned touch_length_;
80 int indexes_[WebTouchEvent::touchesLengthCap];
81 WebTouchPoint::State states_[WebTouchEvent::touchesLengthCap];
82 WebInputEvent::Type type_;
83 };
84
85 class MockSyntheticPointerTouchActionTarget
86 : public MockSyntheticPointerActionTarget {
87 public:
88 MockSyntheticPointerTouchActionTarget() {}
89 ~MockSyntheticPointerTouchActionTarget() override {}
90
91 void DispatchInputEventToPlatform(const WebInputEvent& event) override {
92 ASSERT_TRUE(WebInputEvent::isTouchEventType(event.type));
93 const WebTouchEvent& touch_event = static_cast<const WebTouchEvent&>(event);
94 type_ = touch_event.type;
95 for (size_t i = 0; i < touch_event.touchesLength; ++i) {
96 indexes_[i] = touch_event.touches[i].id;
97 positions_[i] = gfx::PointF(touch_event.touches[i].position);
98 states_[i] = touch_event.touches[i].state;
99 }
100 touch_length_ = touch_event.touchesLength;
101 }
102 };
103
104 class SyntheticPointerActionControllerTest : public testing::Test {
tdresser 2016/06/06 18:36:08 Shouldn't this just be SyntheticPointerActionTest?
lanwei 2016/06/09 14:37:03 Done.
105 public:
106 SyntheticPointerActionControllerTest() {}
107 ~SyntheticPointerActionControllerTest() override {}
108
109 protected:
110 template <typename MockGestureTarget>
111 void CreateControllerAndTarget() {
112 target_ = new MockGestureTarget();
113 controller_.reset(new SyntheticGestureController(
114 std::unique_ptr<SyntheticGestureTarget>(target_)));
115 }
116
117 void SetUp() override {
118 num_success_ = 0;
119 num_failure_ = 0;
120 time_ = base::TimeTicks::Now();
121 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
122 std::fill(index_map_.begin(), index_map_.end(), -1);
123 }
124
125 void TearDown() override {
126 controller_.reset();
127 target_ = nullptr;
128 time_ = base::TimeTicks();
129 action_param_list_.reset();
130 }
131
132 void QueueSyntheticGesture(std::unique_ptr<SyntheticGesture> gesture) {
133 controller_->QueueSyntheticGesture(
134 std::move(gesture),
135 base::Bind(
136 &SyntheticPointerActionControllerTest::OnSyntheticGestureCompleted,
137 base::Unretained(this)));
138 }
139
140 void FlushInputUntilComplete() {
141 while (target_->flush_requested()) {
142 while (target_->flush_requested()) {
143 target_->ClearFlushRequest();
144 time_ += base::TimeDelta::FromMilliseconds(kFlushInputRateInMs);
145 controller_->Flush(time_);
146 }
147 controller_->OnDidFlushInput();
148 }
149 }
150
151 void OnSyntheticGestureCompleted(SyntheticGesture::Result result) {
152 if (result == SyntheticGesture::GESTURE_FINISHED)
153 num_success_++;
154 else
155 num_failure_++;
156 }
157
158 MockSyntheticGestureTarget* target_;
159 std::unique_ptr<SyntheticGestureController> controller_;
160 base::TimeTicks time_;
161 int num_success_;
162 int num_failure_;
163 std::unique_ptr<std::vector<SyntheticPointerActionParams>> action_param_list_;
164 SyntheticPointerAction::IndexMap index_map_;
165 };
166
167 TEST_F(SyntheticPointerActionControllerTest, PointerTouchAction) {
168 CreateControllerAndTarget<MockSyntheticPointerTouchActionTarget>();
tdresser 2016/06/06 18:36:08 I was thinking we'd have some low level tests whic
lanwei 2016/06/08 14:48:37 The test cases are very similar to SyntheticGestur
lanwei 2016/06/09 14:37:03 Done.
169
170 // Send a touch press for one finger.
171 std::unique_ptr<SyntheticPointer> synthetic_pointer =
172 SyntheticPointer::Create(SyntheticGestureParams::TOUCH_INPUT);
173 SyntheticPointerActionParams params0 = SyntheticPointerActionParams(
174 SyntheticPointerActionParams::PointerActionType::PRESS);
175 params0.gesture_source_type = SyntheticGestureParams::TOUCH_INPUT;
176 params0.set_index(0);
177 params0.set_position(gfx::PointF(54, 89));
178 action_param_list_->push_back(params0);
179
180 std::unique_ptr<SyntheticGesture> gesture(new SyntheticPointerAction(
181 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
182 QueueSyntheticGesture(std::move(gesture));
183 FlushInputUntilComplete();
184
185 MockSyntheticPointerTouchActionTarget* pointer_touch_target =
186 static_cast<MockSyntheticPointerTouchActionTarget*>(target_);
187 EXPECT_EQ(1, num_success_);
188 EXPECT_EQ(0, num_failure_);
189 EXPECT_EQ(pointer_touch_target->type(), WebInputEvent::TouchStart);
190 EXPECT_EQ(pointer_touch_target->indexes(0), index_map_[0]);
191 EXPECT_EQ(pointer_touch_target->positions(0), params0.position());
192 EXPECT_EQ(pointer_touch_target->states(0), WebTouchPoint::StatePressed);
193 ASSERT_EQ(pointer_touch_target->touch_length(), 1U);
194
195 // Send a touch move for the first finger and a touch press for the second
196 // finger.
197 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
198 params0.set_pointer_action_type(
199 SyntheticPointerActionParams::PointerActionType::MOVE);
200 params0.set_position(gfx::PointF(133, 156));
201 SyntheticPointerActionParams params1 = SyntheticPointerActionParams(
202 SyntheticPointerActionParams::PointerActionType::PRESS);
203 params1.gesture_source_type = SyntheticGestureParams::TOUCH_INPUT;
204 params1.set_index(1);
205 params1.set_position(gfx::PointF(79, 132));
206 action_param_list_->push_back(params0);
207 action_param_list_->push_back(params1);
208 gesture.reset(new SyntheticPointerAction(
209 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
210 QueueSyntheticGesture(std::move(gesture));
211 FlushInputUntilComplete();
212
213 pointer_touch_target =
214 static_cast<MockSyntheticPointerTouchActionTarget*>(target_);
215 EXPECT_EQ(2, num_success_);
216 EXPECT_EQ(0, num_failure_);
217 // The type of the SyntheticWebTouchEvent is the action of the last finger.
218 EXPECT_EQ(pointer_touch_target->type(), WebInputEvent::TouchStart);
219 EXPECT_EQ(pointer_touch_target->indexes(0), index_map_[0]);
220 EXPECT_EQ(pointer_touch_target->positions(0), params0.position());
221 EXPECT_EQ(pointer_touch_target->states(0), WebTouchPoint::StateMoved);
222 EXPECT_EQ(pointer_touch_target->indexes(1), index_map_[1]);
223 EXPECT_EQ(pointer_touch_target->positions(1), params1.position());
224 EXPECT_EQ(pointer_touch_target->states(1), WebTouchPoint::StatePressed);
225 ASSERT_EQ(pointer_touch_target->touch_length(), 2U);
226
227 // Send a touch move for the second finger.
228 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
229 params1.set_pointer_action_type(
230 SyntheticPointerActionParams::PointerActionType::MOVE);
231 params1.set_position(gfx::PointF(87, 253));
232 action_param_list_->push_back(params1);
233 gesture.reset(new SyntheticPointerAction(
234 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
235 QueueSyntheticGesture(std::move(gesture));
236 FlushInputUntilComplete();
237
238 pointer_touch_target =
239 static_cast<MockSyntheticPointerTouchActionTarget*>(target_);
240 EXPECT_EQ(3, num_success_);
241 EXPECT_EQ(0, num_failure_);
242 EXPECT_EQ(pointer_touch_target->type(), WebInputEvent::TouchMove);
243 EXPECT_EQ(pointer_touch_target->indexes(1), index_map_[1]);
244 EXPECT_EQ(pointer_touch_target->positions(1), params1.position());
245 EXPECT_EQ(pointer_touch_target->states(1), WebTouchPoint::StateMoved);
246 ASSERT_EQ(pointer_touch_target->touch_length(), 2U);
247
248 // Send touch releases for both fingers.
249 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
250 params0.set_pointer_action_type(
251 SyntheticPointerActionParams::PointerActionType::RELEASE);
252 params1.set_pointer_action_type(
253 SyntheticPointerActionParams::PointerActionType::RELEASE);
254 action_param_list_->push_back(params0);
255 action_param_list_->push_back(params1);
256 int index0 = index_map_[0];
257 int index1 = index_map_[1];
258 gesture.reset(new SyntheticPointerAction(
259 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
260 QueueSyntheticGesture(std::move(gesture));
261 FlushInputUntilComplete();
262
263 EXPECT_EQ(4, num_success_);
264 EXPECT_EQ(0, num_failure_);
265 EXPECT_EQ(pointer_touch_target->type(), WebInputEvent::TouchEnd);
266 EXPECT_EQ(pointer_touch_target->indexes(0), index0);
267 EXPECT_EQ(-1, index_map_[0]);
268 EXPECT_EQ(pointer_touch_target->states(0), WebTouchPoint::StateReleased);
269 EXPECT_EQ(pointer_touch_target->indexes(1), index1);
270 EXPECT_EQ(-1, index_map_[1]);
271 EXPECT_EQ(pointer_touch_target->states(1), WebTouchPoint::StateReleased);
272 ASSERT_EQ(pointer_touch_target->touch_length(), 2U);
273 }
274
275 TEST_F(SyntheticPointerActionControllerTest, PointerTouchActionInvalid) {
276 CreateControllerAndTarget<MockSyntheticPointerTouchActionTarget>();
277
278 // Users sent a wrong index for the touch action.
279 std::unique_ptr<SyntheticPointer> synthetic_pointer =
280 SyntheticPointer::Create(SyntheticGestureParams::TOUCH_INPUT);
281 SyntheticPointerActionParams params0 = SyntheticPointerActionParams(
282 SyntheticPointerActionParams::PointerActionType::PRESS);
283 params0.gesture_source_type = SyntheticGestureParams::TOUCH_INPUT;
284 params0.set_index(-1);
285 params0.set_position(gfx::PointF(54, 89));
286 action_param_list_->push_back(params0);
287
288 std::unique_ptr<SyntheticGesture> gesture(new SyntheticPointerAction(
289 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
290 QueueSyntheticGesture(std::move(gesture));
291 FlushInputUntilComplete();
292
293 EXPECT_EQ(0, num_success_);
294 EXPECT_EQ(1, num_failure_);
295
296 // Users' gesture source type does not match with the touch action.
297 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
298 params0.gesture_source_type = SyntheticGestureParams::MOUSE_INPUT;
299 params0.set_index(0);
300 action_param_list_->push_back(params0);
301
302 gesture.reset(new SyntheticPointerAction(
303 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
304 QueueSyntheticGesture(std::move(gesture));
305 FlushInputUntilComplete();
306
307 EXPECT_EQ(0, num_success_);
308 EXPECT_EQ(2, num_failure_);
309
310 // Cannot send a touch move or touch release without sending a touch press
311 // first.
312 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
313 params0.gesture_source_type = SyntheticGestureParams::TOUCH_INPUT;
314 params0.set_index(0);
315 params0.set_pointer_action_type(
316 SyntheticPointerActionParams::PointerActionType::MOVE);
317 action_param_list_->push_back(params0);
318
319 gesture.reset(new SyntheticPointerAction(
320 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
321 QueueSyntheticGesture(std::move(gesture));
322 FlushInputUntilComplete();
323
324 EXPECT_EQ(0, num_success_);
325 EXPECT_EQ(3, num_failure_);
326
327 // Send a touch press for one finger.
328 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
329 params0.gesture_source_type = SyntheticGestureParams::TOUCH_INPUT;
330 params0.set_index(0);
331 params0.set_pointer_action_type(
332 SyntheticPointerActionParams::PointerActionType::PRESS);
333 action_param_list_->push_back(params0);
334
335 gesture.reset(new SyntheticPointerAction(
336 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
337 QueueSyntheticGesture(std::move(gesture));
338 FlushInputUntilComplete();
339
340 EXPECT_EQ(1, num_success_);
341 EXPECT_EQ(3, num_failure_);
342
343 // Cannot send a touch press again without releasing the finger.
344 action_param_list_.reset(new std::vector<SyntheticPointerActionParams>());
345 params0.gesture_source_type = SyntheticGestureParams::TOUCH_INPUT;
346 params0.set_index(0);
347 params0.set_pointer_action_type(
348 SyntheticPointerActionParams::PointerActionType::PRESS);
349 action_param_list_->push_back(params0);
350
351 gesture.reset(new SyntheticPointerAction(
352 std::move(action_param_list_), synthetic_pointer.get(), &index_map_));
353 QueueSyntheticGesture(std::move(gesture));
354 FlushInputUntilComplete();
355
356 EXPECT_EQ(1, num_success_);
357 EXPECT_EQ(4, num_failure_);
358 }
359
360 } // namespace
361
362 } // namespace content
OLDNEW

Powered by Google App Engine
This is Rietveld 408576698