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