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