Add ICU message format support
[chromium-blink-merge.git] / content / browser / renderer_host / input / input_router_impl.cc
blob77d3283b6cf835aa74d72d274df79c85e64a44a9
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"
7 #include <math.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"
33 using base::Time;
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;
43 namespace content {
44 namespace {
46 const char* GetEventAckName(InputEventAckState ack_result) {
47 switch(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.";
55 return "";
58 } // namespace
60 InputRouterImpl::Config::Config() {
63 InputRouterImpl::InputRouterImpl(IPC::Sender* sender,
64 InputRouterClient* client,
65 InputAckHandler* ack_handler,
66 int routing_id,
67 const Config& config)
68 : sender_(sender),
69 client_(client),
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) {
81 DCHECK(sender);
82 DCHECK(client);
83 DCHECK(ack_handler);
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.";
102 return false;
103 default:
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))
113 return;
114 if (mouse_event.event.type == WebInputEvent::MouseUp &&
115 gesture_event_queue_.GetTouchpadTapSuppressionController()->
116 ShouldSuppressMouseUp())
117 return;
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);
137 } else {
138 coalesced_mouse_wheel_events_.back().CoalesceWith(wheel_event);
139 TRACE_EVENT_INSTANT2("input", "InputRouterImpl::CoalescedWheelEvent",
140 TRACE_EVENT_SCOPE_THREAD,
141 "total_dx",
142 coalesced_mouse_wheel_events_.back().event.deltaX,
143 "total_dy",
144 coalesced_mouse_wheel_events_.back().event.deltaY);
146 return;
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
163 // handler.
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,
171 key_event.latency,
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))
182 return;
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));
208 else
209 next_mouse_move_->CoalesceWith(mouse_event);
210 return;
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())
240 return NULL;
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) {
265 bool handled = true;
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,
272 OnSelectMessageAck)
273 IPC_MESSAGE_HANDLER(ViewHostMsg_HasTouchEventHandlers,
274 OnHasTouchEventHandlers)
275 IPC_MESSAGE_HANDLER(InputHostMsg_SetTouchAction,
276 OnSetTouchAction)
277 IPC_MESSAGE_HANDLER(InputHostMsg_DidStopFlinging, OnDidStopFlinging)
278 IPC_MESSAGE_UNHANDLED(handled = false)
279 IPC_END_MESSAGE_MAP()
281 return handled;
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
309 // IPC messages.
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());
318 return true;
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();
329 return true;
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",
346 "type",
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))
361 return;
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.
400 consumed = true;
401 break;
402 case INPUT_EVENT_ACK_STATE_UNKNOWN:
403 // Simply drop the event.
404 consumed = true;
405 break;
406 default:
407 break;
410 return consumed;
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();
425 return true;
427 return false;
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.
477 if (!has_handlers)
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.
522 return;
525 base::AutoReset<AckSource> auto_reset_current_ack_source(
526 &current_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.
551 key_queue_.clear();
552 ack_handler_->OnUnexpectedEventAck(InputAckHandler::UNEXPECTED_EVENT_TYPE);
553 } else {
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)
570 return;
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_)
642 return;
644 if (HasPendingEvents())
645 return;
647 flush_requested_ = false;
648 client_->DidFlush();
651 } // namespace content