1 // Copyright 2013 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.
5 #include "content/browser/renderer_host/input/input_router_impl.h"
9 #include "base/auto_reset.h"
10 #include "base/command_line.h"
11 #include "base/metrics/histogram.h"
12 #include "base/strings/string_number_conversions.h"
13 #include "content/browser/renderer_host/input/gesture_event_queue.h"
14 #include "content/browser/renderer_host/input/input_ack_handler.h"
15 #include "content/browser/renderer_host/input/input_router_client.h"
16 #include "content/browser/renderer_host/input/touch_event_queue.h"
17 #include "content/browser/renderer_host/input/touchpad_tap_suppression_controller.h"
18 #include "content/common/content_constants_internal.h"
19 #include "content/common/edit_command.h"
20 #include "content/common/input/input_event_ack_state.h"
21 #include "content/common/input/touch_action.h"
22 #include "content/common/input/web_touch_event_traits.h"
23 #include "content/common/input_messages.h"
24 #include "content/common/view_messages.h"
25 #include "content/public/browser/notification_service.h"
26 #include "content/public/browser/notification_types.h"
27 #include "content/public/browser/user_metrics.h"
28 #include "content/public/common/content_switches.h"
29 #include "ipc/ipc_sender.h"
30 #include "ui/events/event.h"
31 #include "ui/events/keycodes/keyboard_codes.h"
34 using base::TimeDelta
;
35 using base::TimeTicks
;
36 using blink::WebGestureEvent
;
37 using blink::WebInputEvent
;
38 using blink::WebKeyboardEvent
;
39 using blink::WebMouseEvent
;
40 using blink::WebMouseWheelEvent
;
41 using blink::WebTouchEvent
;
46 const char* GetEventAckName(InputEventAckState ack_result
) {
48 case INPUT_EVENT_ACK_STATE_UNKNOWN
: return "UNKNOWN";
49 case INPUT_EVENT_ACK_STATE_CONSUMED
: return "CONSUMED";
50 case INPUT_EVENT_ACK_STATE_NOT_CONSUMED
: return "NOT_CONSUMED";
51 case INPUT_EVENT_ACK_STATE_NO_CONSUMER_EXISTS
: return "NO_CONSUMER_EXISTS";
52 case INPUT_EVENT_ACK_STATE_IGNORED
: return "IGNORED";
54 DLOG(WARNING
) << "Unhandled InputEventAckState in GetEventAckName.";
60 InputRouterImpl::Config::Config() {
63 InputRouterImpl::InputRouterImpl(IPC::Sender
* sender
,
64 InputRouterClient
* client
,
65 InputAckHandler
* ack_handler
,
70 ack_handler_(ack_handler
),
71 routing_id_(routing_id
),
72 select_message_pending_(false),
73 move_caret_pending_(false),
74 mouse_move_pending_(false),
75 mouse_wheel_pending_(false),
76 current_ack_source_(ACK_SOURCE_NONE
),
77 flush_requested_(false),
78 active_renderer_fling_count_(0),
79 touch_event_queue_(this, config
.touch_config
),
80 gesture_event_queue_(this, this, config
.gesture_config
) {
84 UpdateTouchAckTimeoutEnabled();
87 InputRouterImpl::~InputRouterImpl() {
88 STLDeleteElements(&pending_select_messages_
);
91 bool InputRouterImpl::SendInput(scoped_ptr
<IPC::Message
> message
) {
92 DCHECK(IPC_MESSAGE_ID_CLASS(message
->type()) == InputMsgStart
);
93 switch (message
->type()) {
94 // Check for types that require an ACK.
95 case InputMsg_SelectRange::ID
:
96 case InputMsg_MoveRangeSelectionExtent::ID
:
97 return SendSelectMessage(message
.Pass());
98 case InputMsg_MoveCaret::ID
:
99 return SendMoveCaret(message
.Pass());
100 case InputMsg_HandleInputEvent::ID
:
101 NOTREACHED() << "WebInputEvents should never be sent via SendInput.";
104 return Send(message
.release());
108 void InputRouterImpl::SendMouseEvent(
109 const MouseEventWithLatencyInfo
& mouse_event
) {
110 if (mouse_event
.event
.type
== WebInputEvent::MouseDown
&&
111 gesture_event_queue_
.GetTouchpadTapSuppressionController()->
112 ShouldDeferMouseDown(mouse_event
))
114 if (mouse_event
.event
.type
== WebInputEvent::MouseUp
&&
115 gesture_event_queue_
.GetTouchpadTapSuppressionController()->
116 ShouldSuppressMouseUp())
119 SendMouseEventImmediately(mouse_event
);
122 void InputRouterImpl::SendWheelEvent(
123 const MouseWheelEventWithLatencyInfo
& wheel_event
) {
124 if (mouse_wheel_pending_
) {
125 // If there's already a mouse wheel event waiting to be sent to the
126 // renderer, add the new deltas to that event. Not doing so (e.g., by
127 // dropping the old event, as for mouse moves) results in very slow
128 // scrolling on the Mac.
129 if (wheel_event
.event
.hasPreciseScrollingDeltas
)
130 DCHECK(wheel_event
.event
.canScroll
);
131 DCHECK(!(wheel_event
.event
.hasPreciseScrollingDeltas
&&
132 !wheel_event
.event
.canScroll
));
133 if (coalesced_mouse_wheel_events_
.empty() ||
134 (!coalesced_mouse_wheel_events_
.empty() &&
135 !coalesced_mouse_wheel_events_
.back().CanCoalesceWith(wheel_event
))) {
136 coalesced_mouse_wheel_events_
.push_back(wheel_event
);
138 coalesced_mouse_wheel_events_
.back().CoalesceWith(wheel_event
);
139 TRACE_EVENT_INSTANT2("input", "InputRouterImpl::CoalescedWheelEvent",
140 TRACE_EVENT_SCOPE_THREAD
,
142 coalesced_mouse_wheel_events_
.back().event
.deltaX
,
144 coalesced_mouse_wheel_events_
.back().event
.deltaY
);
149 mouse_wheel_pending_
= true;
150 current_wheel_event_
= wheel_event
;
152 LOCAL_HISTOGRAM_COUNTS_100("Renderer.WheelQueueSize",
153 coalesced_mouse_wheel_events_
.size());
155 FilterAndSendWebInputEvent(wheel_event
.event
, wheel_event
.latency
, false);
158 void InputRouterImpl::SendKeyboardEvent(
159 const NativeWebKeyboardEventWithLatencyInfo
& key_event
,
160 bool is_keyboard_shortcut
) {
161 // Put all WebKeyboardEvent objects in a queue since we can't trust the
162 // renderer and we need to give something to the HandleKeyboardEvent
164 key_queue_
.push_back(key_event
);
165 LOCAL_HISTOGRAM_COUNTS_100("Renderer.KeyboardQueueSize", key_queue_
.size());
167 gesture_event_queue_
.FlingHasBeenHalted();
169 // Only forward the non-native portions of our event.
170 FilterAndSendWebInputEvent(key_event
.event
,
172 is_keyboard_shortcut
);
175 void InputRouterImpl::SendGestureEvent(
176 const GestureEventWithLatencyInfo
& original_gesture_event
) {
177 input_stream_validator_
.Validate(original_gesture_event
.event
);
179 GestureEventWithLatencyInfo
gesture_event(original_gesture_event
);
181 if (touch_action_filter_
.FilterGestureEvent(&gesture_event
.event
))
184 if (gesture_event
.event
.sourceDevice
== blink::WebGestureDeviceTouchscreen
)
185 touch_event_queue_
.OnGestureScrollEvent(gesture_event
);
187 gesture_event_queue_
.QueueEvent(gesture_event
);
190 void InputRouterImpl::SendTouchEvent(
191 const TouchEventWithLatencyInfo
& touch_event
) {
192 input_stream_validator_
.Validate(touch_event
.event
);
193 touch_event_queue_
.QueueEvent(touch_event
);
196 // Forwards MouseEvent without passing it through
197 // TouchpadTapSuppressionController.
198 void InputRouterImpl::SendMouseEventImmediately(
199 const MouseEventWithLatencyInfo
& mouse_event
) {
200 // Avoid spamming the renderer with mouse move events. It is important
201 // to note that WM_MOUSEMOVE events are anyways synthetic, but since our
202 // thread is able to rapidly consume WM_MOUSEMOVE events, we may get way
203 // more WM_MOUSEMOVE events than we wish to send to the renderer.
204 if (mouse_event
.event
.type
== WebInputEvent::MouseMove
) {
205 if (mouse_move_pending_
) {
206 if (!next_mouse_move_
)
207 next_mouse_move_
.reset(new MouseEventWithLatencyInfo(mouse_event
));
209 next_mouse_move_
->CoalesceWith(mouse_event
);
212 mouse_move_pending_
= true;
213 current_mouse_move_
= mouse_event
;
216 FilterAndSendWebInputEvent(mouse_event
.event
, mouse_event
.latency
, false);
219 void InputRouterImpl::SendTouchEventImmediately(
220 const TouchEventWithLatencyInfo
& touch_event
) {
221 if (WebTouchEventTraits::IsTouchSequenceStart(touch_event
.event
)) {
222 touch_action_filter_
.ResetTouchAction();
223 // Note that if the previous touch-action was TOUCH_ACTION_NONE, enabling
224 // the timeout here will not take effect until the *following* touch
225 // sequence. This is a desirable side-effect, giving the renderer a chance
226 // to send a touch-action response without racing against the ack timeout.
227 UpdateTouchAckTimeoutEnabled();
230 FilterAndSendWebInputEvent(touch_event
.event
, touch_event
.latency
, false);
233 void InputRouterImpl::SendGestureEventImmediately(
234 const GestureEventWithLatencyInfo
& gesture_event
) {
235 FilterAndSendWebInputEvent(gesture_event
.event
, gesture_event
.latency
, false);
238 const NativeWebKeyboardEvent
* InputRouterImpl::GetLastKeyboardEvent() const {
239 if (key_queue_
.empty())
241 return &key_queue_
.front().event
;
244 void InputRouterImpl::NotifySiteIsMobileOptimized(bool is_mobile_optimized
) {
245 touch_event_queue_
.SetIsMobileOptimizedSite(is_mobile_optimized
);
248 void InputRouterImpl::RequestNotificationWhenFlushed() {
249 flush_requested_
= true;
250 SignalFlushedIfNecessary();
253 bool InputRouterImpl::HasPendingEvents() const {
254 return !touch_event_queue_
.empty() ||
255 !gesture_event_queue_
.empty() ||
256 !key_queue_
.empty() ||
257 mouse_move_pending_
||
258 mouse_wheel_pending_
||
259 select_message_pending_
||
260 move_caret_pending_
||
261 active_renderer_fling_count_
> 0;
264 bool InputRouterImpl::OnMessageReceived(const IPC::Message
& message
) {
266 IPC_BEGIN_MESSAGE_MAP(InputRouterImpl
, message
)
267 IPC_MESSAGE_HANDLER(InputHostMsg_HandleInputEvent_ACK
, OnInputEventAck
)
268 IPC_MESSAGE_HANDLER(InputHostMsg_DidOverscroll
, OnDidOverscroll
)
269 IPC_MESSAGE_HANDLER(InputHostMsg_MoveCaret_ACK
, OnMsgMoveCaretAck
)
270 IPC_MESSAGE_HANDLER(InputHostMsg_SelectRange_ACK
, OnSelectMessageAck
)
271 IPC_MESSAGE_HANDLER(InputHostMsg_MoveRangeSelectionExtent_ACK
,
273 IPC_MESSAGE_HANDLER(ViewHostMsg_HasTouchEventHandlers
,
274 OnHasTouchEventHandlers
)
275 IPC_MESSAGE_HANDLER(InputHostMsg_SetTouchAction
,
277 IPC_MESSAGE_HANDLER(InputHostMsg_DidStopFlinging
, OnDidStopFlinging
)
278 IPC_MESSAGE_UNHANDLED(handled
= false)
279 IPC_END_MESSAGE_MAP()
284 void InputRouterImpl::OnTouchEventAck(const TouchEventWithLatencyInfo
& event
,
285 InputEventAckState ack_result
) {
286 // Touchstart events sent to the renderer indicate a new touch sequence, but
287 // in some cases we may filter out sending the touchstart - catch those here.
288 if (WebTouchEventTraits::IsTouchSequenceStart(event
.event
) &&
289 ack_result
== INPUT_EVENT_ACK_STATE_NO_CONSUMER_EXISTS
) {
290 touch_action_filter_
.ResetTouchAction();
291 UpdateTouchAckTimeoutEnabled();
293 ack_handler_
->OnTouchEventAck(event
, ack_result
);
296 void InputRouterImpl::OnGestureEventAck(
297 const GestureEventWithLatencyInfo
& event
,
298 InputEventAckState ack_result
) {
299 touch_event_queue_
.OnGestureEventAck(event
, ack_result
);
300 ack_handler_
->OnGestureEventAck(event
, ack_result
);
303 bool InputRouterImpl::SendSelectMessage(
304 scoped_ptr
<IPC::Message
> message
) {
305 DCHECK(message
->type() == InputMsg_SelectRange::ID
||
306 message
->type() == InputMsg_MoveRangeSelectionExtent::ID
);
308 // TODO(jdduke): Factor out common logic between selection and caret-related
310 if (select_message_pending_
) {
311 if (!pending_select_messages_
.empty() &&
312 pending_select_messages_
.back()->type() == message
->type()) {
313 delete pending_select_messages_
.back();
314 pending_select_messages_
.pop_back();
317 pending_select_messages_
.push_back(message
.release());
321 select_message_pending_
= true;
322 return Send(message
.release());
325 bool InputRouterImpl::SendMoveCaret(scoped_ptr
<IPC::Message
> message
) {
326 DCHECK(message
->type() == InputMsg_MoveCaret::ID
);
327 if (move_caret_pending_
) {
328 next_move_caret_
= message
.Pass();
332 move_caret_pending_
= true;
333 return Send(message
.release());
336 bool InputRouterImpl::Send(IPC::Message
* message
) {
337 return sender_
->Send(message
);
340 void InputRouterImpl::FilterAndSendWebInputEvent(
341 const WebInputEvent
& input_event
,
342 const ui::LatencyInfo
& latency_info
,
343 bool is_keyboard_shortcut
) {
344 TRACE_EVENT1("input",
345 "InputRouterImpl::FilterAndSendWebInputEvent",
347 WebInputEventTraits::GetName(input_event
.type
));
349 // Any input event cancels a pending mouse move event.
350 next_mouse_move_
.reset();
352 OfferToHandlers(input_event
, latency_info
, is_keyboard_shortcut
);
355 void InputRouterImpl::OfferToHandlers(const WebInputEvent
& input_event
,
356 const ui::LatencyInfo
& latency_info
,
357 bool is_keyboard_shortcut
) {
358 output_stream_validator_
.Validate(input_event
);
360 if (OfferToClient(input_event
, latency_info
))
363 OfferToRenderer(input_event
, latency_info
, is_keyboard_shortcut
);
365 // Touch events should always indicate in the event whether they are
366 // cancelable (respect ACK disposition) or not except touchmove.
367 bool needs_synthetic_ack
=
368 !WebInputEventTraits::WillReceiveAckFromRenderer(input_event
);
370 if (WebInputEvent::isTouchEventType(input_event
.type
) &&
371 input_event
.type
!= WebInputEvent::TouchMove
) {
372 const WebTouchEvent
& touch
= static_cast<const WebTouchEvent
&>(input_event
);
373 DCHECK_EQ(needs_synthetic_ack
, !touch
.cancelable
);
376 // The synthetic acks are sent immediately.
377 if (needs_synthetic_ack
) {
378 ProcessInputEventAck(
379 input_event
.type
, INPUT_EVENT_ACK_STATE_IGNORED
, latency_info
,
380 WebInputEventTraits::GetUniqueTouchEventId(input_event
),
381 IGNORING_DISPOSITION
);
385 bool InputRouterImpl::OfferToClient(const WebInputEvent
& input_event
,
386 const ui::LatencyInfo
& latency_info
) {
387 bool consumed
= false;
389 InputEventAckState filter_ack
=
390 client_
->FilterInputEvent(input_event
, latency_info
);
391 switch (filter_ack
) {
392 case INPUT_EVENT_ACK_STATE_CONSUMED
:
393 case INPUT_EVENT_ACK_STATE_NO_CONSUMER_EXISTS
:
394 // Send the ACK and early exit.
395 next_mouse_move_
.reset();
396 ProcessInputEventAck(
397 input_event
.type
, filter_ack
, latency_info
,
398 WebInputEventTraits::GetUniqueTouchEventId(input_event
), CLIENT
);
399 // WARNING: |this| may be deleted at this point.
402 case INPUT_EVENT_ACK_STATE_UNKNOWN
:
403 // Simply drop the event.
413 bool InputRouterImpl::OfferToRenderer(const WebInputEvent
& input_event
,
414 const ui::LatencyInfo
& latency_info
,
415 bool is_keyboard_shortcut
) {
416 if (Send(new InputMsg_HandleInputEvent(
417 routing_id(), &input_event
, latency_info
, is_keyboard_shortcut
))) {
418 // Ack messages for ignored ack event types should never be sent by the
419 // renderer. Consequently, such event types should not affect event time
420 // or in-flight event count metrics.
421 if (WebInputEventTraits::WillReceiveAckFromRenderer(input_event
)) {
422 input_event_start_time_
= TimeTicks::Now();
423 client_
->IncrementInFlightEventCount();
430 void InputRouterImpl::OnInputEventAck(const InputEventAck
& ack
) {
431 client_
->DecrementInFlightEventCount();
432 // Log the time delta for processing an input event.
433 TimeDelta delta
= TimeTicks::Now() - input_event_start_time_
;
434 UMA_HISTOGRAM_TIMES("MPArch.IIR_InputEventDelta", delta
);
436 if (ack
.overscroll
) {
437 DCHECK(ack
.type
== WebInputEvent::MouseWheel
||
438 ack
.type
== WebInputEvent::GestureScrollUpdate
);
439 OnDidOverscroll(*ack
.overscroll
);
442 ProcessInputEventAck(ack
.type
, ack
.state
, ack
.latency
,
443 ack
.unique_touch_event_id
, RENDERER
);
446 void InputRouterImpl::OnDidOverscroll(const DidOverscrollParams
& params
) {
447 client_
->DidOverscroll(params
);
450 void InputRouterImpl::OnMsgMoveCaretAck() {
451 move_caret_pending_
= false;
452 if (next_move_caret_
)
453 SendMoveCaret(next_move_caret_
.Pass());
456 void InputRouterImpl::OnSelectMessageAck() {
457 select_message_pending_
= false;
458 if (!pending_select_messages_
.empty()) {
459 scoped_ptr
<IPC::Message
> next_message
=
460 make_scoped_ptr(pending_select_messages_
.front());
461 pending_select_messages_
.pop_front();
463 SendSelectMessage(next_message
.Pass());
467 void InputRouterImpl::OnHasTouchEventHandlers(bool has_handlers
) {
468 TRACE_EVENT1("input", "InputRouterImpl::OnHasTouchEventHandlers",
469 "has_handlers", has_handlers
);
471 // Lack of a touch handler indicates that the page either has no touch-action
472 // modifiers or that all its touch-action modifiers are auto. Resetting the
473 // touch-action here allows forwarding of subsequent gestures even if the
474 // underlying touches never reach the router.
475 // TODO(jdduke): Reset touch-action only at the end of a touch sequence to
476 // prevent potentially strange mid-sequence behavior, crbug.com/375940.
478 touch_action_filter_
.ResetTouchAction();
480 touch_event_queue_
.OnHasTouchEventHandlers(has_handlers
);
481 client_
->OnHasTouchEventHandlers(has_handlers
);
484 void InputRouterImpl::OnSetTouchAction(TouchAction touch_action
) {
485 // Synthetic touchstart events should get filtered out in RenderWidget.
486 DCHECK(touch_event_queue_
.IsPendingAckTouchStart());
487 TRACE_EVENT1("input", "InputRouterImpl::OnSetTouchAction",
488 "action", touch_action
);
490 touch_action_filter_
.OnSetTouchAction(touch_action
);
492 // TOUCH_ACTION_NONE should disable the touch ack timeout.
493 UpdateTouchAckTimeoutEnabled();
496 void InputRouterImpl::OnDidStopFlinging() {
497 DCHECK_GT(active_renderer_fling_count_
, 0);
498 // Note that we're only guaranteed to get a fling end notification from the
499 // renderer, not from any other consumers. Consequently, the GestureEventQueue
500 // cannot use this bookkeeping for logic like tap suppression.
501 --active_renderer_fling_count_
;
502 SignalFlushedIfNecessary();
504 client_
->DidStopFlinging();
507 void InputRouterImpl::ProcessInputEventAck(WebInputEvent::Type event_type
,
508 InputEventAckState ack_result
,
509 const ui::LatencyInfo
& latency_info
,
510 uint32 unique_touch_event_id
,
511 AckSource ack_source
) {
512 TRACE_EVENT2("input", "InputRouterImpl::ProcessInputEventAck",
513 "type", WebInputEventTraits::GetName(event_type
),
514 "ack", GetEventAckName(ack_result
));
516 // Note: The keyboard ack must be treated carefully, as it may result in
517 // synchronous destruction of |this|. Handling immediately guards against
518 // future references to |this|, as with |auto_reset_current_ack_source| below.
519 if (WebInputEvent::isKeyboardEventType(event_type
)) {
520 ProcessKeyboardAck(event_type
, ack_result
, latency_info
);
521 // WARNING: |this| may be deleted at this point.
525 base::AutoReset
<AckSource
> auto_reset_current_ack_source(
526 ¤t_ack_source_
, ack_source
);
528 if (WebInputEvent::isMouseEventType(event_type
)) {
529 ProcessMouseAck(event_type
, ack_result
, latency_info
);
530 } else if (event_type
== WebInputEvent::MouseWheel
) {
531 ProcessWheelAck(ack_result
, latency_info
);
532 } else if (WebInputEvent::isTouchEventType(event_type
)) {
533 ProcessTouchAck(ack_result
, latency_info
, unique_touch_event_id
);
534 } else if (WebInputEvent::isGestureEventType(event_type
)) {
535 ProcessGestureAck(event_type
, ack_result
, latency_info
);
536 } else if (event_type
!= WebInputEvent::Undefined
) {
537 ack_handler_
->OnUnexpectedEventAck(InputAckHandler::BAD_ACK_MESSAGE
);
540 SignalFlushedIfNecessary();
543 void InputRouterImpl::ProcessKeyboardAck(blink::WebInputEvent::Type type
,
544 InputEventAckState ack_result
,
545 const ui::LatencyInfo
& latency
) {
546 if (key_queue_
.empty()) {
547 ack_handler_
->OnUnexpectedEventAck(InputAckHandler::UNEXPECTED_ACK
);
548 } else if (key_queue_
.front().event
.type
!= type
) {
549 // Something must be wrong. Clear the |key_queue_| and char event
550 // suppression so that we can resume from the error.
552 ack_handler_
->OnUnexpectedEventAck(InputAckHandler::UNEXPECTED_EVENT_TYPE
);
554 NativeWebKeyboardEventWithLatencyInfo front_item
= key_queue_
.front();
555 front_item
.latency
.AddNewLatencyFrom(latency
);
556 key_queue_
.pop_front();
558 ack_handler_
->OnKeyboardEventAck(front_item
, ack_result
);
559 // WARNING: This InputRouterImpl can be deallocated at this point
560 // (i.e. in the case of Ctrl+W, where the call to
561 // HandleKeyboardEvent destroys this InputRouterImpl).
562 // TODO(jdduke): crbug.com/274029 - Make ack-triggered shutdown async.
566 void InputRouterImpl::ProcessMouseAck(blink::WebInputEvent::Type type
,
567 InputEventAckState ack_result
,
568 const ui::LatencyInfo
& latency
) {
569 if (type
!= WebInputEvent::MouseMove
)
572 current_mouse_move_
.latency
.AddNewLatencyFrom(latency
);
573 ack_handler_
->OnMouseEventAck(current_mouse_move_
, ack_result
);
575 DCHECK(mouse_move_pending_
);
576 mouse_move_pending_
= false;
578 if (next_mouse_move_
) {
579 DCHECK(next_mouse_move_
->event
.type
== WebInputEvent::MouseMove
);
580 scoped_ptr
<MouseEventWithLatencyInfo
> next_mouse_move
581 = next_mouse_move_
.Pass();
582 SendMouseEvent(*next_mouse_move
);
586 void InputRouterImpl::ProcessWheelAck(InputEventAckState ack_result
,
587 const ui::LatencyInfo
& latency
) {
588 // TODO(miletus): Add renderer side latency to each uncoalesced mouse
589 // wheel event and add terminal component to each of them.
590 current_wheel_event_
.latency
.AddNewLatencyFrom(latency
);
592 // Process the unhandled wheel event here before calling SendWheelEvent()
593 // since it will mutate current_wheel_event_.
594 ack_handler_
->OnWheelEventAck(current_wheel_event_
, ack_result
);
596 // Mark the wheel event complete only after the ACKs have been handled above.
597 // For example, ACKing the GesturePinchUpdate could cause another
598 // GesturePinchUpdate to be sent, which should queue a wheel event rather than
599 // send it immediately.
600 mouse_wheel_pending_
= false;
602 // Send the next (coalesced or synthetic) mouse wheel event.
603 if (!coalesced_mouse_wheel_events_
.empty()) {
604 MouseWheelEventWithLatencyInfo next_wheel_event
=
605 coalesced_mouse_wheel_events_
.front();
606 coalesced_mouse_wheel_events_
.pop_front();
607 SendWheelEvent(next_wheel_event
);
611 void InputRouterImpl::ProcessGestureAck(WebInputEvent::Type type
,
612 InputEventAckState ack_result
,
613 const ui::LatencyInfo
& latency
) {
614 if (type
== blink::WebInputEvent::GestureFlingStart
&&
615 ack_result
== INPUT_EVENT_ACK_STATE_CONSUMED
) {
616 ++active_renderer_fling_count_
;
619 // |gesture_event_queue_| will forward to OnGestureEventAck when appropriate.
620 gesture_event_queue_
.ProcessGestureAck(ack_result
, type
, latency
);
623 void InputRouterImpl::ProcessTouchAck(InputEventAckState ack_result
,
624 const ui::LatencyInfo
& latency
,
625 uint32 unique_touch_event_id
) {
626 // |touch_event_queue_| will forward to OnTouchEventAck when appropriate.
627 touch_event_queue_
.ProcessTouchAck(ack_result
, latency
,
628 unique_touch_event_id
);
631 void InputRouterImpl::UpdateTouchAckTimeoutEnabled() {
632 // TOUCH_ACTION_NONE will prevent scrolling, in which case the timeout serves
633 // little purpose. It's also a strong signal that touch handling is critical
634 // to page functionality, so the timeout could do more harm than good.
635 const bool touch_ack_timeout_enabled
=
636 touch_action_filter_
.allowed_touch_action() != TOUCH_ACTION_NONE
;
637 touch_event_queue_
.SetAckTimeoutEnabled(touch_ack_timeout_enabled
);
640 void InputRouterImpl::SignalFlushedIfNecessary() {
641 if (!flush_requested_
)
644 if (HasPendingEvents())
647 flush_requested_
= false;
651 } // namespace content