[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 1 | // Copyright (c) 2012 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 "content/browser/renderer_host/overscroll_controller.h" |
| 6 | |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 7 | #include <algorithm> |
| 8 | |
[email protected] | 3aad43b | 2013-11-05 20:20:18 | [diff] [blame] | 9 | #include "base/command_line.h" |
[email protected] | fc4b06d0 | 2013-10-23 20:47:11 | [diff] [blame] | 10 | #include "base/logging.h" |
Peter Kasting | b5b3987 | 2019-09-10 08:44:04 | [diff] [blame^] | 11 | #include "base/numerics/ranges.h" |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 12 | #include "content/browser/renderer_host/overscroll_controller_delegate.h" |
[email protected] | 1cd76c2d8 | 2012-11-29 07:36:47 | [diff] [blame] | 13 | #include "content/public/browser/overscroll_configuration.h" |
Sahel Sharify | a29ee211 | 2017-12-15 23:22:54 | [diff] [blame] | 14 | #include "content/public/common/content_features.h" |
[email protected] | 3aad43b | 2013-11-05 20:20:18 | [diff] [blame] | 15 | #include "content/public/common/content_switches.h" |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 16 | |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 17 | namespace content { |
| 18 | |
[email protected] | 3aad43b | 2013-11-05 20:20:18 | [diff] [blame] | 19 | namespace { |
| 20 | |
Daniel Cheng | 224569ee | 2018-04-25 05:45:06 | [diff] [blame] | 21 | // Minimum amount of time after an actual scroll after which a pull-to-refresh |
| 22 | // can start. |
| 23 | constexpr base::TimeDelta kPullToRefreshCoolOffDelay = |
| 24 | base::TimeDelta::FromMilliseconds(600); |
Mohsen Izadi | 5a658351 | 2018-04-23 18:31:19 | [diff] [blame] | 25 | |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 26 | bool IsGestureEventFromTouchpad(const blink::WebInputEvent& event) { |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 27 | DCHECK(blink::WebInputEvent::IsGestureEventType(event.GetType())); |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 28 | const blink::WebGestureEvent& gesture = |
| 29 | static_cast<const blink::WebGestureEvent&>(event); |
Daniel Cheng | 7f9ec90 | 2019-04-18 05:07:00 | [diff] [blame] | 30 | return gesture.SourceDevice() == blink::WebGestureDevice::kTouchpad; |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 31 | } |
| 32 | |
Sahel Sharify | bdaa29e | 2018-09-17 17:42:25 | [diff] [blame] | 33 | bool IsGestureEventFromAutoscroll(const blink::WebGestureEvent event) { |
Daniel Cheng | 7f9ec90 | 2019-04-18 05:07:00 | [diff] [blame] | 34 | return event.SourceDevice() == blink::WebGestureDevice::kSyntheticAutoscroll; |
Sahel Sharify | bdaa29e | 2018-09-17 17:42:25 | [diff] [blame] | 35 | } |
| 36 | |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 37 | bool IsGestureScrollUpdateInertialEvent(const blink::WebInputEvent& event) { |
| 38 | if (event.GetType() != blink::WebInputEvent::kGestureScrollUpdate) |
| 39 | return false; |
| 40 | |
| 41 | const blink::WebGestureEvent& gesture = |
| 42 | static_cast<const blink::WebGestureEvent&>(event); |
| 43 | return gesture.data.scroll_update.inertial_phase == |
Daniel Libby | 6ce6b95a | 2019-05-10 17:06:26 | [diff] [blame] | 44 | blink::WebGestureEvent::InertialPhaseState::kMomentum; |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 45 | } |
| 46 | |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 47 | float ClampAbsoluteValue(float value, float max_abs) { |
| 48 | DCHECK_LT(0.f, max_abs); |
Peter Kasting | b5b3987 | 2019-09-10 08:44:04 | [diff] [blame^] | 49 | return base::ClampToRange(value, -max_abs, max_abs); |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 50 | } |
[email protected] | 3aad43b | 2013-11-05 20:20:18 | [diff] [blame] | 51 | |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 52 | } // namespace |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 53 | |
Sahel Sharify | e6d81f47 | 2018-07-11 20:40:26 | [diff] [blame] | 54 | OverscrollController::OverscrollController() {} |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 55 | |
Mohsen Izadi | 38795d5d | 2017-06-30 17:10:48 | [diff] [blame] | 56 | OverscrollController::~OverscrollController() {} |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 57 | |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 58 | bool OverscrollController::ShouldProcessEvent( |
| 59 | const blink::WebInputEvent& event) { |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 60 | switch (event.GetType()) { |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 61 | case blink::WebInputEvent::kGestureScrollBegin: |
| 62 | case blink::WebInputEvent::kGestureScrollUpdate: |
| 63 | case blink::WebInputEvent::kGestureScrollEnd: { |
dtapuska | 8c4dae1 | 2017-01-13 00:23:06 | [diff] [blame] | 64 | const blink::WebGestureEvent& gesture = |
| 65 | static_cast<const blink::WebGestureEvent&>(event); |
mcnee | 19fd249 | 2017-06-01 14:42:43 | [diff] [blame] | 66 | |
| 67 | // GestureScrollBegin and GestureScrollEnd events are created to wrap |
| 68 | // individual resent GestureScrollUpdates from a plugin. Hence these |
| 69 | // should not be used to indicate the beginning/end of the overscroll. |
| 70 | // TODO(mcnee): When we remove BrowserPlugin, delete this code. |
| 71 | // See crbug.com/533069 |
| 72 | if (gesture.resending_plugin_id != -1 && |
| 73 | event.GetType() != blink::WebInputEvent::kGestureScrollUpdate) |
| 74 | return false; |
| 75 | |
Sahel Sharify | bdaa29e | 2018-09-17 17:42:25 | [diff] [blame] | 76 | // Gesture events with Autoscroll source don't cause overscrolling. |
| 77 | if (IsGestureEventFromAutoscroll(gesture)) |
| 78 | return false; |
| 79 | |
Daniel Libby | e15b80b | 2019-05-24 17:18:24 | [diff] [blame] | 80 | ui::input_types::ScrollGranularity granularity; |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 81 | switch (event.GetType()) { |
| 82 | case blink::WebInputEvent::kGestureScrollBegin: |
Daniel Libby | a37d447a | 2019-05-02 20:06:48 | [diff] [blame] | 83 | granularity = gesture.data.scroll_begin.delta_hint_units; |
dtapuska | 8c4dae1 | 2017-01-13 00:23:06 | [diff] [blame] | 84 | break; |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 85 | case blink::WebInputEvent::kGestureScrollUpdate: |
Daniel Libby | a37d447a | 2019-05-02 20:06:48 | [diff] [blame] | 86 | granularity = gesture.data.scroll_update.delta_units; |
dtapuska | 8c4dae1 | 2017-01-13 00:23:06 | [diff] [blame] | 87 | break; |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 88 | case blink::WebInputEvent::kGestureScrollEnd: |
Daniel Libby | a37d447a | 2019-05-02 20:06:48 | [diff] [blame] | 89 | granularity = gesture.data.scroll_end.delta_units; |
dtapuska | 8c4dae1 | 2017-01-13 00:23:06 | [diff] [blame] | 90 | break; |
| 91 | default: |
Daniel Libby | e15b80b | 2019-05-24 17:18:24 | [diff] [blame] | 92 | granularity = ui::input_types::ScrollGranularity::kScrollByPixel; |
dtapuska | 8c4dae1 | 2017-01-13 00:23:06 | [diff] [blame] | 93 | break; |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 94 | } |
dtapuska | 8c4dae1 | 2017-01-13 00:23:06 | [diff] [blame] | 95 | |
Daniel Libby | e15b80b | 2019-05-24 17:18:24 | [diff] [blame] | 96 | return granularity == |
| 97 | ui::input_types::ScrollGranularity::kScrollByPrecisePixel; |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 98 | } |
dtapuska | 8c4dae1 | 2017-01-13 00:23:06 | [diff] [blame] | 99 | default: |
| 100 | break; |
| 101 | } |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 102 | return true; |
| 103 | } |
| 104 | |
chaopeng | a7585d79 | 2018-04-10 18:02:38 | [diff] [blame] | 105 | bool OverscrollController::ShouldIgnoreInertialEvent( |
| 106 | const blink::WebInputEvent& event) const { |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 107 | return ignore_following_inertial_events_ && |
| 108 | IsGestureScrollUpdateInertialEvent(event); |
chaopeng | a7585d79 | 2018-04-10 18:02:38 | [diff] [blame] | 109 | } |
| 110 | |
[email protected] | 277857a | 2014-06-03 10:38:01 | [diff] [blame] | 111 | bool OverscrollController::WillHandleEvent(const blink::WebInputEvent& event) { |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 112 | if (!ShouldProcessEvent(event)) |
| 113 | return false; |
| 114 | |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 115 | // TODO(mohsen): Consider filtering mouse-wheel events during overscroll. See |
| 116 | // https://crbug.com/772106. |
| 117 | if (event.GetType() == blink::WebInputEvent::kMouseWheel) |
| 118 | return false; |
| 119 | |
Mohsen Izadi | 5a658351 | 2018-04-23 18:31:19 | [diff] [blame] | 120 | if (event.GetType() == blink::WebInputEvent::kGestureScrollBegin) { |
| 121 | ignore_following_inertial_events_ = false; |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 122 | first_inertial_event_time_.reset(); |
Mohsen Izadi | 5a658351 | 2018-04-23 18:31:19 | [diff] [blame] | 123 | time_since_last_ignored_scroll_ = |
Daniel Cheng | 224569ee | 2018-04-25 05:45:06 | [diff] [blame] | 124 | event.TimeStamp() - last_ignored_scroll_time_; |
Mohsen Izadi | 5a658351 | 2018-04-23 18:31:19 | [diff] [blame] | 125 | // Will handle events when processing ACKs to ensure the correct order. |
| 126 | return false; |
| 127 | } |
| 128 | |
| 129 | if (event.GetType() == blink::WebInputEvent::kGestureScrollEnd) { |
| 130 | if (scroll_state_ == ScrollState::CONTENT_CONSUMING || |
| 131 | overscroll_ignored_) { |
Daniel Cheng | 224569ee | 2018-04-25 05:45:06 | [diff] [blame] | 132 | last_ignored_scroll_time_ = event.TimeStamp(); |
Mohsen Izadi | 5a658351 | 2018-04-23 18:31:19 | [diff] [blame] | 133 | } |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 134 | // Will handle events when processing ACKs to ensure the correct order. |
| 135 | return false; |
| 136 | } |
| 137 | |
chaopeng | 1d469b0 | 2018-04-05 15:41:07 | [diff] [blame] | 138 | // Consume the scroll-update events if they are from a inertial scroll (fling) |
| 139 | // event that completed an overscroll gesture. |
chaopeng | a7585d79 | 2018-04-10 18:02:38 | [diff] [blame] | 140 | if (ShouldIgnoreInertialEvent(event)) |
| 141 | return true; |
chaopeng | 1d469b0 | 2018-04-05 15:41:07 | [diff] [blame] | 142 | |
mfomitchev | 8a785bef | 2015-04-13 18:35:05 | [diff] [blame] | 143 | bool reset_scroll_state = false; |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 144 | if (scroll_state_ != ScrollState::NONE || overscroll_delta_x_ || |
| 145 | overscroll_delta_y_) { |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 146 | switch (event.GetType()) { |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 147 | case blink::WebInputEvent::kGestureFlingStart: |
mfomitchev | 8a785bef | 2015-04-13 18:35:05 | [diff] [blame] | 148 | reset_scroll_state = true; |
[email protected] | b31beeb | 2013-06-17 15:54:27 | [diff] [blame] | 149 | break; |
| 150 | |
[email protected] | b31beeb | 2013-06-17 15:54:27 | [diff] [blame] | 151 | default: |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 152 | if (blink::WebInputEvent::IsMouseEventType(event.GetType()) || |
| 153 | blink::WebInputEvent::IsKeyboardEventType(event.GetType())) { |
mfomitchev | 8a785bef | 2015-04-13 18:35:05 | [diff] [blame] | 154 | reset_scroll_state = true; |
[email protected] | b31beeb | 2013-06-17 15:54:27 | [diff] [blame] | 155 | } |
| 156 | break; |
[email protected] | a414164 | 2013-05-23 04:10:58 | [diff] [blame] | 157 | } |
| 158 | } |
| 159 | |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 160 | if (reset_scroll_state) |
| 161 | ResetScrollState(); |
mfomitchev | 8a785bef | 2015-04-13 18:35:05 | [diff] [blame] | 162 | |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 163 | if (DispatchEventCompletesAction(event)) { |
| 164 | CompleteAction(); |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 165 | |
[email protected] | 277857a | 2014-06-03 10:38:01 | [diff] [blame] | 166 | // Let the event be dispatched to the renderer. |
| 167 | return false; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 168 | } |
| 169 | |
| 170 | if (overscroll_mode_ != OVERSCROLL_NONE && DispatchEventResetsState(event)) { |
mfomitchev | 09f0d64a | 2017-03-02 19:40:07 | [diff] [blame] | 171 | SetOverscrollMode(OVERSCROLL_NONE, OverscrollSource::NONE); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 172 | |
| 173 | // Let the event be dispatched to the renderer. |
[email protected] | 277857a | 2014-06-03 10:38:01 | [diff] [blame] | 174 | return false; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 175 | } |
| 176 | |
| 177 | if (overscroll_mode_ != OVERSCROLL_NONE) { |
[email protected] | 2422e5c | 2013-09-13 19:05:02 | [diff] [blame] | 178 | // Consume the event only if it updates the overscroll state. |
| 179 | if (ProcessEventForOverscroll(event)) |
[email protected] | 277857a | 2014-06-03 10:38:01 | [diff] [blame] | 180 | return true; |
mfomitchev | 8a785bef | 2015-04-13 18:35:05 | [diff] [blame] | 181 | } else if (reset_scroll_state) { |
| 182 | overscroll_delta_x_ = overscroll_delta_y_ = 0.f; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 183 | } |
| 184 | |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 185 | // In overscrolling state, consume scroll-update and fling-start events when |
| 186 | // they do not contribute to overscroll in order to prevent content scroll. |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 187 | return scroll_state_ == ScrollState::OVERSCROLLING && |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 188 | (event.GetType() == blink::WebInputEvent::kGestureScrollUpdate || |
| 189 | event.GetType() == blink::WebInputEvent::kGestureFlingStart); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 190 | } |
| 191 | |
Sandra Sun | cbd6da4 | 2018-01-05 15:29:36 | [diff] [blame] | 192 | void OverscrollController::OnDidOverscroll( |
| 193 | const ui::DidOverscrollParams& params) { |
| 194 | // TODO(sunyunjia): We should also decide whether to trigger overscroll, |
| 195 | // update scroll_state_ here. See https://crbug.com/799467. |
Mohsen Izadi | 8018941 | 2018-03-22 17:10:18 | [diff] [blame] | 196 | behavior_ = params.overscroll_behavior; |
Sandra Sun | cbd6da4 | 2018-01-05 15:29:36 | [diff] [blame] | 197 | } |
| 198 | |
[email protected] | 180ef24 | 2013-11-07 06:50:46 | [diff] [blame] | 199 | void OverscrollController::ReceivedEventACK(const blink::WebInputEvent& event, |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 200 | bool processed) { |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 201 | if (!ShouldProcessEvent(event)) |
| 202 | return; |
| 203 | |
chaopeng | a7585d79 | 2018-04-10 18:02:38 | [diff] [blame] | 204 | // An inertial scroll (fling) event from a completed overscroll gesture |
| 205 | // should not modify states below. |
| 206 | if (ShouldIgnoreInertialEvent(event)) |
| 207 | return; |
| 208 | |
[email protected] | a414164 | 2013-05-23 04:10:58 | [diff] [blame] | 209 | if (processed) { |
| 210 | // If a scroll event is consumed by the page, i.e. some content on the page |
| 211 | // has been scrolled, then there is not going to be an overscroll gesture, |
| 212 | // until the current scroll ends, and a new scroll gesture starts. |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 213 | // Similarly, if a mouse-wheel event is consumed, probably the page has |
| 214 | // implemented its own scroll-like behavior and no overscroll should happen. |
| 215 | if (scroll_state_ == ScrollState::NONE && |
| 216 | (event.GetType() == blink::WebInputEvent::kGestureScrollUpdate || |
| 217 | event.GetType() == blink::WebInputEvent::kMouseWheel)) { |
| 218 | scroll_state_ = ScrollState::CONTENT_CONSUMING; |
[email protected] | a414164 | 2013-05-23 04:10:58 | [diff] [blame] | 219 | } |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 220 | // In overscrolling state, only return if we are in an overscroll mode; |
| 221 | // otherwise, we would want to ProcessEventForOverscroll to let it start a |
| 222 | // new overscroll mode. |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 223 | if (scroll_state_ != ScrollState::OVERSCROLLING || |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 224 | overscroll_mode_ != OVERSCROLL_NONE) { |
| 225 | return; |
| 226 | } |
[email protected] | a414164 | 2013-05-23 04:10:58 | [diff] [blame] | 227 | } |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 228 | |
| 229 | if (event.GetType() == blink::WebInputEvent::kMouseWheel) |
| 230 | return; |
| 231 | |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 232 | ProcessEventForOverscroll(event); |
| 233 | } |
| 234 | |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 235 | void OverscrollController::Reset() { |
| 236 | overscroll_mode_ = OVERSCROLL_NONE; |
Mohsen Izadi | 38795d5d | 2017-06-30 17:10:48 | [diff] [blame] | 237 | overscroll_source_ = OverscrollSource::NONE; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 238 | overscroll_delta_x_ = overscroll_delta_y_ = 0.f; |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 239 | ResetScrollState(); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 240 | } |
| 241 | |
[email protected] | 63ad7383 | 2013-06-17 15:41:04 | [diff] [blame] | 242 | void OverscrollController::Cancel() { |
mfomitchev | 09f0d64a | 2017-03-02 19:40:07 | [diff] [blame] | 243 | SetOverscrollMode(OVERSCROLL_NONE, OverscrollSource::NONE); |
[email protected] | 63ad7383 | 2013-06-17 15:41:04 | [diff] [blame] | 244 | overscroll_delta_x_ = overscroll_delta_y_ = 0.f; |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 245 | ResetScrollState(); |
[email protected] | 63ad7383 | 2013-06-17 15:41:04 | [diff] [blame] | 246 | } |
| 247 | |
chaopeng | 1d469b0 | 2018-04-05 15:41:07 | [diff] [blame] | 248 | bool OverscrollController::DispatchEventCompletesAction( |
[email protected] | 180ef24 | 2013-11-07 06:50:46 | [diff] [blame] | 249 | const blink::WebInputEvent& event) const { |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 250 | if (overscroll_mode_ == OVERSCROLL_NONE) |
| 251 | return false; |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 252 | DCHECK_NE(OverscrollSource::NONE, overscroll_source_); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 253 | |
| 254 | // Complete the overscroll gesture if there was a mouse move or a scroll-end |
| 255 | // after the threshold. |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 256 | if (event.GetType() != blink::WebInputEvent::kMouseMove && |
| 257 | event.GetType() != blink::WebInputEvent::kGestureScrollEnd && |
chaopeng | 1d469b0 | 2018-04-05 15:41:07 | [diff] [blame] | 258 | event.GetType() != blink::WebInputEvent::kGestureFlingStart && |
| 259 | event.GetType() != blink::WebInputEvent::kGestureScrollUpdate) |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 260 | return false; |
| 261 | |
chaopeng | 1d469b0 | 2018-04-05 15:41:07 | [diff] [blame] | 262 | // Complete the overscroll gesture for inertial scroll (fling) event from |
| 263 | // touchpad. |
| 264 | if (event.GetType() == blink::WebInputEvent::kGestureScrollUpdate) { |
| 265 | if (overscroll_source_ != OverscrollSource::TOUCHPAD) |
| 266 | return false; |
| 267 | DCHECK(IsGestureEventFromTouchpad(event)); |
| 268 | const blink::WebGestureEvent gesture_event = |
| 269 | static_cast<const blink::WebGestureEvent&>(event); |
| 270 | if (gesture_event.data.scroll_update.inertial_phase != |
Daniel Libby | 6ce6b95a | 2019-05-10 17:06:26 | [diff] [blame] | 271 | blink::WebGestureEvent::InertialPhaseState::kMomentum) |
chaopeng | 1d469b0 | 2018-04-05 15:41:07 | [diff] [blame] | 272 | return false; |
| 273 | } |
| 274 | |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 275 | if (event.GetType() == blink::WebInputEvent::kGestureScrollEnd && |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 276 | overscroll_source_ == OverscrollSource::TOUCHPAD) { |
| 277 | DCHECK(IsGestureEventFromTouchpad(event)); |
Sahel Sharify | a29ee211 | 2017-12-15 23:22:54 | [diff] [blame] | 278 | // Complete the action for a GSE with touchpad source only when it is in |
| 279 | // momentumPhase. |
| 280 | const blink::WebGestureEvent gesture_event = |
| 281 | static_cast<const blink::WebGestureEvent&>(event); |
| 282 | if (gesture_event.data.scroll_end.inertial_phase != |
Daniel Libby | 6ce6b95a | 2019-05-10 17:06:26 | [diff] [blame] | 283 | blink::WebGestureEvent::InertialPhaseState::kMomentum) |
Sahel Sharify | a29ee211 | 2017-12-15 23:22:54 | [diff] [blame] | 284 | return false; |
Mohsen Izadi | 38795d5d | 2017-06-30 17:10:48 | [diff] [blame] | 285 | } |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 286 | |
[email protected] | 0da15c2d | 2013-10-30 16:47:20 | [diff] [blame] | 287 | if (!delegate_) |
| 288 | return false; |
| 289 | |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 290 | if (event.GetType() == blink::WebInputEvent::kGestureFlingStart) { |
[email protected] | 6658c6a | 2013-01-23 23:42:22 | [diff] [blame] | 291 | // Check to see if the fling is in the same direction of the overscroll. |
[email protected] | 180ef24 | 2013-11-07 06:50:46 | [diff] [blame] | 292 | const blink::WebGestureEvent gesture = |
| 293 | static_cast<const blink::WebGestureEvent&>(event); |
[email protected] | 6658c6a | 2013-01-23 23:42:22 | [diff] [blame] | 294 | switch (overscroll_mode_) { |
| 295 | case OVERSCROLL_EAST: |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 296 | if (gesture.data.fling_start.velocity_x < 0) |
[email protected] | 6658c6a | 2013-01-23 23:42:22 | [diff] [blame] | 297 | return false; |
| 298 | break; |
| 299 | case OVERSCROLL_WEST: |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 300 | if (gesture.data.fling_start.velocity_x > 0) |
[email protected] | 6658c6a | 2013-01-23 23:42:22 | [diff] [blame] | 301 | return false; |
| 302 | break; |
| 303 | case OVERSCROLL_NORTH: |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 304 | if (gesture.data.fling_start.velocity_y > 0) |
[email protected] | 6658c6a | 2013-01-23 23:42:22 | [diff] [blame] | 305 | return false; |
| 306 | break; |
| 307 | case OVERSCROLL_SOUTH: |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 308 | if (gesture.data.fling_start.velocity_y < 0) |
[email protected] | 6658c6a | 2013-01-23 23:42:22 | [diff] [blame] | 309 | return false; |
| 310 | break; |
| 311 | case OVERSCROLL_NONE: |
[email protected] | 6658c6a | 2013-01-23 23:42:22 | [diff] [blame] | 312 | NOTREACHED(); |
| 313 | } |
| 314 | } |
| 315 | |
Mohsen Izadi | 4bd624cc | 2017-07-21 19:23:51 | [diff] [blame] | 316 | const gfx::Size size = delegate_->GetDisplaySize(); |
Mohsen Izadi | 38795d5d | 2017-06-30 17:10:48 | [diff] [blame] | 317 | if (size.IsEmpty()) |
| 318 | return false; |
| 319 | |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 320 | const float delta = |
| 321 | overscroll_mode_ == OVERSCROLL_WEST || overscroll_mode_ == OVERSCROLL_EAST |
| 322 | ? overscroll_delta_x_ |
| 323 | : overscroll_delta_y_; |
| 324 | const float ratio = fabs(delta) / std::max(size.width(), size.height()); |
Elly Fong-Jones | 4796097 | 2019-05-16 18:40:59 | [diff] [blame] | 325 | const float threshold = |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 326 | overscroll_source_ == OverscrollSource::TOUCHPAD |
Elly Fong-Jones | 4796097 | 2019-05-16 18:40:59 | [diff] [blame] | 327 | ? OverscrollConfig::kCompleteTouchpadThresholdPercent |
| 328 | : OverscrollConfig::kCompleteTouchscreenThresholdPercent; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 329 | return ratio >= threshold; |
| 330 | } |
| 331 | |
| 332 | bool OverscrollController::DispatchEventResetsState( |
[email protected] | 180ef24 | 2013-11-07 06:50:46 | [diff] [blame] | 333 | const blink::WebInputEvent& event) const { |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 334 | switch (event.GetType()) { |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 335 | // GestureScrollBegin/End ACK will reset overscroll state when necessary. |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 336 | case blink::WebInputEvent::kGestureScrollBegin: |
| 337 | case blink::WebInputEvent::kGestureScrollEnd: |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 338 | case blink::WebInputEvent::kGestureScrollUpdate: |
| 339 | case blink::WebInputEvent::kGestureFlingCancel: |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 340 | return false; |
| 341 | |
| 342 | default: |
[email protected] | c4ddb5c4 | 2012-11-17 01:09:57 | [diff] [blame] | 343 | // Touch events can arrive during an overscroll gesture initiated by |
| 344 | // touch-scrolling. These events should not reset the overscroll state. |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 345 | return !blink::WebInputEvent::IsTouchEventType(event.GetType()); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 346 | } |
| 347 | } |
| 348 | |
[email protected] | 2422e5c | 2013-09-13 19:05:02 | [diff] [blame] | 349 | bool OverscrollController::ProcessEventForOverscroll( |
[email protected] | 180ef24 | 2013-11-07 06:50:46 | [diff] [blame] | 350 | const blink::WebInputEvent& event) { |
[email protected] | 2422e5c | 2013-09-13 19:05:02 | [diff] [blame] | 351 | bool event_processed = false; |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 352 | switch (event.GetType()) { |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 353 | case blink::WebInputEvent::kGestureScrollBegin: { |
Sahel Sharify | e6d81f47 | 2018-07-11 20:40:26 | [diff] [blame] | 354 | if (overscroll_mode_ != OVERSCROLL_NONE) |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 355 | SetOverscrollMode(OVERSCROLL_NONE, OverscrollSource::NONE); |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 356 | break; |
| 357 | } |
| 358 | case blink::WebInputEvent::kGestureScrollEnd: { |
Sahel Sharify | a29ee211 | 2017-12-15 23:22:54 | [diff] [blame] | 359 | // Only reset the state on GestureScrollEnd generated from the touchpad |
| 360 | // when the scrolling is in inertial state. |
| 361 | const blink::WebGestureEvent gesture_event = |
| 362 | static_cast<const blink::WebGestureEvent&>(event); |
Daniel Libby | 6ce6b95a | 2019-05-10 17:06:26 | [diff] [blame] | 363 | bool reset_scroll_state = |
| 364 | !IsGestureEventFromTouchpad(event) || |
| 365 | (gesture_event.data.scroll_end.inertial_phase == |
| 366 | blink::WebGestureEvent::InertialPhaseState::kMomentum); |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 367 | |
| 368 | if (reset_scroll_state) |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 369 | ResetScrollState(); |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 370 | |
| 371 | if (DispatchEventCompletesAction(event)) { |
| 372 | CompleteAction(); |
[email protected] | 2422e5c | 2013-09-13 19:05:02 | [diff] [blame] | 373 | break; |
Chong Zhang | 064c205 | 2017-08-29 16:05:31 | [diff] [blame] | 374 | } |
| 375 | |
| 376 | if (!reset_scroll_state) |
| 377 | break; |
| 378 | |
| 379 | if (overscroll_mode_ != OVERSCROLL_NONE) { |
| 380 | SetOverscrollMode(OVERSCROLL_NONE, OverscrollSource::NONE); |
| 381 | } else { |
| 382 | overscroll_delta_x_ = overscroll_delta_y_ = 0.f; |
| 383 | } |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 384 | break; |
| 385 | } |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 386 | case blink::WebInputEvent::kGestureScrollUpdate: { |
[email protected] | 180ef24 | 2013-11-07 06:50:46 | [diff] [blame] | 387 | const blink::WebGestureEvent& gesture = |
| 388 | static_cast<const blink::WebGestureEvent&>(event); |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 389 | bool is_gesture_scroll_update_inertial_event = |
| 390 | IsGestureScrollUpdateInertialEvent(event); |
dtapuska | 19ebfa51 | 2016-02-19 22:27:40 | [diff] [blame] | 391 | event_processed = ProcessOverscroll( |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 392 | gesture.data.scroll_update.delta_x, |
| 393 | gesture.data.scroll_update.delta_y, |
Daniel Cheng | 7f9ec90 | 2019-04-18 05:07:00 | [diff] [blame] | 394 | gesture.SourceDevice() == blink::WebGestureDevice::kTouchpad, |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 395 | is_gesture_scroll_update_inertial_event); |
| 396 | if (is_gesture_scroll_update_inertial_event) { |
| 397 | // Record the timestamp of first inertial event. |
| 398 | if (!first_inertial_event_time_) { |
| 399 | first_inertial_event_time_ = event.TimeStamp(); |
| 400 | break; |
| 401 | } |
| 402 | base::TimeDelta inertial_event_interval = |
| 403 | event.TimeStamp() - first_inertial_event_time_.value(); |
| 404 | if (inertial_event_interval >= |
| 405 | OverscrollConfig::MaxInertialEventsBeforeOverscrollCancellation()) { |
| 406 | ignore_following_inertial_events_ = true; |
| 407 | // Reset overscroll state if fling didn't complete the overscroll |
| 408 | // gesture within the first 20 inertial events. |
| 409 | Cancel(); |
| 410 | } |
| 411 | } |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 412 | break; |
| 413 | } |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 414 | case blink::WebInputEvent::kGestureFlingStart: { |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 415 | const float kFlingVelocityThreshold = 1100.f; |
[email protected] | 180ef24 | 2013-11-07 06:50:46 | [diff] [blame] | 416 | const blink::WebGestureEvent& gesture = |
| 417 | static_cast<const blink::WebGestureEvent&>(event); |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 418 | float velocity_x = gesture.data.fling_start.velocity_x; |
| 419 | float velocity_y = gesture.data.fling_start.velocity_y; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 420 | if (fabs(velocity_x) > kFlingVelocityThreshold) { |
| 421 | if ((overscroll_mode_ == OVERSCROLL_WEST && velocity_x < 0) || |
| 422 | (overscroll_mode_ == OVERSCROLL_EAST && velocity_x > 0)) { |
| 423 | CompleteAction(); |
[email protected] | 2422e5c | 2013-09-13 19:05:02 | [diff] [blame] | 424 | event_processed = true; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 425 | break; |
| 426 | } |
| 427 | } else if (fabs(velocity_y) > kFlingVelocityThreshold) { |
| 428 | if ((overscroll_mode_ == OVERSCROLL_NORTH && velocity_y < 0) || |
| 429 | (overscroll_mode_ == OVERSCROLL_SOUTH && velocity_y > 0)) { |
| 430 | CompleteAction(); |
[email protected] | 2422e5c | 2013-09-13 19:05:02 | [diff] [blame] | 431 | event_processed = true; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 432 | break; |
| 433 | } |
| 434 | } |
| 435 | |
| 436 | // Reset overscroll state if fling didn't complete the overscroll gesture. |
mfomitchev | 09f0d64a | 2017-03-02 19:40:07 | [diff] [blame] | 437 | SetOverscrollMode(OVERSCROLL_NONE, OverscrollSource::NONE); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 438 | break; |
| 439 | } |
| 440 | |
| 441 | default: |
Blink Reformat | 1c4d759e | 2017-04-09 16:34:54 | [diff] [blame] | 442 | DCHECK(blink::WebInputEvent::IsGestureEventType(event.GetType()) || |
| 443 | blink::WebInputEvent::IsTouchEventType(event.GetType())) |
| 444 | << "Received unexpected event: " << event.GetType(); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 445 | } |
[email protected] | 2422e5c | 2013-09-13 19:05:02 | [diff] [blame] | 446 | return event_processed; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 447 | } |
| 448 | |
[email protected] | bfab3d0 | 2014-08-20 03:16:55 | [diff] [blame] | 449 | bool OverscrollController::ProcessOverscroll(float delta_x, |
[email protected] | a7f99f0 | 2013-08-29 14:15:19 | [diff] [blame] | 450 | float delta_y, |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 451 | bool is_touchpad, |
| 452 | bool is_inertial) { |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 453 | if (scroll_state_ == ScrollState::CONTENT_CONSUMING) |
| 454 | return false; |
| 455 | |
chaopeng | 9736c52 | 2018-05-08 18:32:27 | [diff] [blame] | 456 | // Do not start overscroll for inertial events. |
| 457 | if (overscroll_mode_ == OVERSCROLL_NONE && is_inertial) |
| 458 | return false; |
| 459 | |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 460 | overscroll_delta_x_ += delta_x; |
| 461 | overscroll_delta_y_ += delta_y; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 462 | |
Elly Fong-Jones | 4796097 | 2019-05-16 18:40:59 | [diff] [blame] | 463 | const float start_threshold = |
| 464 | is_touchpad ? OverscrollConfig::kStartTouchpadThresholdDips |
| 465 | : OverscrollConfig::kStartTouchscreenThresholdDips; |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 466 | if (fabs(overscroll_delta_x_) <= start_threshold && |
| 467 | fabs(overscroll_delta_y_) <= start_threshold) { |
mfomitchev | 09f0d64a | 2017-03-02 19:40:07 | [diff] [blame] | 468 | SetOverscrollMode(OVERSCROLL_NONE, OverscrollSource::NONE); |
[email protected] | bfab3d0 | 2014-08-20 03:16:55 | [diff] [blame] | 469 | return true; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 470 | } |
| 471 | |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 472 | if (delegate_) { |
| 473 | base::Optional<float> cap = delegate_->GetMaxOverscrollDelta(); |
| 474 | if (cap) { |
Mohsen Izadi | a652de4c | 2017-07-22 02:30:32 | [diff] [blame] | 475 | DCHECK_LE(0.f, cap.value()); |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 476 | switch (overscroll_mode_) { |
| 477 | case OVERSCROLL_WEST: |
| 478 | case OVERSCROLL_EAST: |
| 479 | overscroll_delta_x_ = ClampAbsoluteValue( |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 480 | overscroll_delta_x_, cap.value() + start_threshold); |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 481 | break; |
| 482 | case OVERSCROLL_NORTH: |
| 483 | case OVERSCROLL_SOUTH: |
| 484 | overscroll_delta_y_ = ClampAbsoluteValue( |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 485 | overscroll_delta_y_, cap.value() + start_threshold); |
Mohsen Izadi | a009153 | 2017-07-06 04:21:05 | [diff] [blame] | 486 | break; |
| 487 | case OVERSCROLL_NONE: |
| 488 | break; |
| 489 | } |
| 490 | } |
| 491 | } |
| 492 | |
[email protected] | a23d04af | 2013-02-19 17:41:10 | [diff] [blame] | 493 | // Compute the current overscroll direction. If the direction is different |
| 494 | // from the current direction, then always switch to no-overscroll mode first |
| 495 | // to make sure that subsequent scroll events go through to the page first. |
| 496 | OverscrollMode new_mode = OVERSCROLL_NONE; |
[email protected] | 14749ddb | 2013-02-23 06:56:10 | [diff] [blame] | 497 | const float kMinRatio = 2.5; |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 498 | if (fabs(overscroll_delta_x_) > start_threshold && |
[email protected] | fa6db25 | 2013-07-26 03:38:20 | [diff] [blame] | 499 | fabs(overscroll_delta_x_) > fabs(overscroll_delta_y_) * kMinRatio) |
[email protected] | a23d04af | 2013-02-19 17:41:10 | [diff] [blame] | 500 | new_mode = overscroll_delta_x_ > 0.f ? OVERSCROLL_EAST : OVERSCROLL_WEST; |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 501 | else if (fabs(overscroll_delta_y_) > start_threshold && |
[email protected] | fa6db25 | 2013-07-26 03:38:20 | [diff] [blame] | 502 | fabs(overscroll_delta_y_) > fabs(overscroll_delta_x_) * kMinRatio) |
[email protected] | a23d04af | 2013-02-19 17:41:10 | [diff] [blame] | 503 | new_mode = overscroll_delta_y_ > 0.f ? OVERSCROLL_SOUTH : OVERSCROLL_NORTH; |
| 504 | |
chaopeng | cd44c762 | 2018-04-13 04:08:21 | [diff] [blame] | 505 | // The horizontal overscroll is used for history navigation. Enable it for |
| 506 | // touchpad only if TouchpadOverscrollHistoryNavigation is enabled. |
| 507 | if ((new_mode == OVERSCROLL_EAST || new_mode == OVERSCROLL_WEST) && |
| 508 | is_touchpad && |
| 509 | !OverscrollConfig::TouchpadOverscrollHistoryNavigationEnabled()) { |
| 510 | new_mode = OVERSCROLL_NONE; |
| 511 | } |
| 512 | |
Mohsen Izadi | 93faac1 | 2017-07-29 04:45:15 | [diff] [blame] | 513 | // The vertical overscroll is used for pull-to-refresh. Enable it only if |
| 514 | // pull-to-refresh is enabled. |
Mohsen Izadi | 8c59ba5 | 2018-04-12 18:52:01 | [diff] [blame] | 515 | if (new_mode == OVERSCROLL_SOUTH || new_mode == OVERSCROLL_NORTH) { |
| 516 | auto ptr_mode = OverscrollConfig::GetPullToRefreshMode(); |
| 517 | if (ptr_mode == OverscrollConfig::PullToRefreshMode::kDisabled || |
| 518 | (ptr_mode == |
| 519 | OverscrollConfig::PullToRefreshMode::kEnabledTouchschreen && |
Mohsen Izadi | 5a658351 | 2018-04-23 18:31:19 | [diff] [blame] | 520 | is_touchpad) || |
Daniel Cheng | 224569ee | 2018-04-25 05:45:06 | [diff] [blame] | 521 | time_since_last_ignored_scroll_ < kPullToRefreshCoolOffDelay) { |
Mohsen Izadi | 5a658351 | 2018-04-23 18:31:19 | [diff] [blame] | 522 | overscroll_ignored_ = true; |
Mohsen Izadi | 8c59ba5 | 2018-04-12 18:52:01 | [diff] [blame] | 523 | new_mode = OVERSCROLL_NONE; |
| 524 | } |
| 525 | } |
[email protected] | 3aad43b | 2013-11-05 20:20:18 | [diff] [blame] | 526 | |
mfomitchev | 09f0d64a | 2017-03-02 19:40:07 | [diff] [blame] | 527 | if (overscroll_mode_ == OVERSCROLL_NONE) { |
| 528 | SetOverscrollMode(new_mode, is_touchpad ? OverscrollSource::TOUCHPAD |
| 529 | : OverscrollSource::TOUCHSCREEN); |
| 530 | } else if (new_mode != overscroll_mode_) { |
| 531 | SetOverscrollMode(OVERSCROLL_NONE, OverscrollSource::NONE); |
| 532 | } |
[email protected] | fa6db25 | 2013-07-26 03:38:20 | [diff] [blame] | 533 | |
| 534 | if (overscroll_mode_ == OVERSCROLL_NONE) |
[email protected] | bfab3d0 | 2014-08-20 03:16:55 | [diff] [blame] | 535 | return false; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 536 | |
Mohsen Izadi | 8f43c2c5 | 2018-10-09 17:17:11 | [diff] [blame] | 537 | overscroll_ignored_ = false; |
| 538 | |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 539 | // Tell the delegate about the overscroll update so that it can update |
| 540 | // the display accordingly (e.g. show history preview etc.). |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 541 | if (delegate_) { |
| 542 | // Do not include the threshold amount when sending the deltas to the |
| 543 | // delegate. |
| 544 | float delegate_delta_x = overscroll_delta_x_; |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 545 | if (fabs(delegate_delta_x) > start_threshold) { |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 546 | if (delegate_delta_x < 0) |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 547 | delegate_delta_x += start_threshold; |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 548 | else |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 549 | delegate_delta_x -= start_threshold; |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 550 | } else { |
| 551 | delegate_delta_x = 0.f; |
| 552 | } |
| 553 | |
| 554 | float delegate_delta_y = overscroll_delta_y_; |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 555 | if (fabs(delegate_delta_y) > start_threshold) { |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 556 | if (delegate_delta_y < 0) |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 557 | delegate_delta_y += start_threshold; |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 558 | else |
Mohsen Izadi | 4ef8db7 | 2017-10-31 00:08:42 | [diff] [blame] | 559 | delegate_delta_y -= start_threshold; |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 560 | } else { |
| 561 | delegate_delta_y = 0.f; |
| 562 | } |
[email protected] | bfab3d0 | 2014-08-20 03:16:55 | [diff] [blame] | 563 | return delegate_->OnOverscrollUpdate(delegate_delta_x, delegate_delta_y); |
[email protected] | 9f4f47e | 2012-11-20 02:21:43 | [diff] [blame] | 564 | } |
[email protected] | bfab3d0 | 2014-08-20 03:16:55 | [diff] [blame] | 565 | return false; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 566 | } |
| 567 | |
| 568 | void OverscrollController::CompleteAction() { |
chaopeng | 1d469b0 | 2018-04-05 15:41:07 | [diff] [blame] | 569 | ignore_following_inertial_events_ = true; |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 570 | if (delegate_) |
| 571 | delegate_->OnOverscrollComplete(overscroll_mode_); |
Mohsen Izadi | f589f4b | 2017-08-24 02:55:48 | [diff] [blame] | 572 | Reset(); |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 573 | } |
| 574 | |
mfomitchev | 09f0d64a | 2017-03-02 19:40:07 | [diff] [blame] | 575 | void OverscrollController::SetOverscrollMode(OverscrollMode mode, |
| 576 | OverscrollSource source) { |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 577 | if (overscroll_mode_ == mode) |
| 578 | return; |
mfomitchev | 09f0d64a | 2017-03-02 19:40:07 | [diff] [blame] | 579 | |
| 580 | // If the mode changes to NONE, source is also NONE. |
| 581 | DCHECK(mode != OVERSCROLL_NONE || source == OverscrollSource::NONE); |
| 582 | |
Mohsen Izadi | f589f4b | 2017-08-24 02:55:48 | [diff] [blame] | 583 | // When setting to a non-NONE mode and there is a locked mode, don't set the |
| 584 | // mode if the new mode is not the same as the locked mode. |
| 585 | if (mode != OVERSCROLL_NONE && locked_mode_ != OVERSCROLL_NONE && |
| 586 | mode != locked_mode_) { |
| 587 | return; |
| 588 | } |
| 589 | |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 590 | OverscrollMode old_mode = overscroll_mode_; |
| 591 | overscroll_mode_ = mode; |
Mohsen Izadi | 38795d5d | 2017-06-30 17:10:48 | [diff] [blame] | 592 | overscroll_source_ = source; |
Mohsen Izadi | f589f4b | 2017-08-24 02:55:48 | [diff] [blame] | 593 | if (overscroll_mode_ == OVERSCROLL_NONE) { |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 594 | overscroll_delta_x_ = overscroll_delta_y_ = 0.f; |
Mohsen Izadi | f589f4b | 2017-08-24 02:55:48 | [diff] [blame] | 595 | } else { |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 596 | scroll_state_ = ScrollState::OVERSCROLLING; |
Mohsen Izadi | f589f4b | 2017-08-24 02:55:48 | [diff] [blame] | 597 | locked_mode_ = overscroll_mode_; |
| 598 | } |
Mohsen Izadi | 8018941 | 2018-03-22 17:10:18 | [diff] [blame] | 599 | if (delegate_) { |
| 600 | delegate_->OnOverscrollModeChange(old_mode, overscroll_mode_, source, |
| 601 | behavior_); |
| 602 | } |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 603 | } |
| 604 | |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 605 | void OverscrollController::ResetScrollState() { |
Mohsen Izadi | 9830436c | 2017-12-21 18:02:46 | [diff] [blame] | 606 | scroll_state_ = ScrollState::NONE; |
Mohsen Izadi | 50180f39 | 2017-09-08 19:29:12 | [diff] [blame] | 607 | locked_mode_ = OVERSCROLL_NONE; |
| 608 | } |
| 609 | |
[email protected] | b4df9df | 2012-11-16 01:58:58 | [diff] [blame] | 610 | } // namespace content |