| // Copyright 2017 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "third_party/blink/renderer/core/dom/document.h" |
| #include "third_party/blink/renderer/core/dom/node_computed_style.h" |
| #include "third_party/blink/renderer/core/frame/local_dom_window.h" |
| #include "third_party/blink/renderer/core/input/event_handler.h" |
| #include "third_party/blink/renderer/core/input/scroll_manager.h" |
| #include "third_party/blink/renderer/core/layout/layout_box.h" |
| #include "third_party/blink/renderer/core/style/computed_style.h" |
| #include "third_party/blink/renderer/core/testing/sim/sim_compositor.h" |
| #include "third_party/blink/renderer/core/testing/sim/sim_request.h" |
| #include "third_party/blink/renderer/core/testing/sim/sim_test.h" |
| |
| namespace blink { |
| |
| class ScrollSnapTest : public SimTest { |
| protected: |
| void SetUpForDiv(); |
| // The following x, y, hint_x, hint_y, delta_x, delta_y are represents |
| // the pointer/finger's location on touch screen. |
| void GestureScroll(double x, double y, double delta_x, double delta_y); |
| void ScrollBegin(double x, double y, double hint_x, double hint_y); |
| void ScrollUpdate(double x, |
| double y, |
| double delta_x, |
| double delta_y, |
| bool is_in_inertial_phase = false); |
| void ScrollEnd(double x, double y, bool is_in_inertial_phase = false); |
| void SetInitialScrollOffset(double x, double y); |
| }; |
| |
| void ScrollSnapTest::SetUpForDiv() { |
| v8::HandleScope HandleScope(v8::Isolate::GetCurrent()); |
| WebView().MainFrameWidget()->Resize(WebSize(400, 400)); |
| SimRequest request("https://example.com/test.html", "text/html"); |
| LoadURL("https://example.com/test.html"); |
| request.Complete(R"HTML( |
| <!DOCTYPE html> |
| <style> |
| #scroller { |
| width: 140px; |
| height: 160px; |
| overflow: scroll; |
| scroll-snap-type: both mandatory; |
| padding: 0px; |
| } |
| #container { |
| margin: 0px; |
| padding: 0px; |
| width: 500px; |
| height: 500px; |
| } |
| #area { |
| position: relative; |
| left: 200px; |
| top: 200px; |
| width: 100px; |
| height: 100px; |
| scroll-snap-align: start; |
| } |
| </style> |
| <div id='scroller'> |
| <div id='container'> |
| <div id='area'></div> |
| </div> |
| </div> |
| )HTML"); |
| |
| Compositor().BeginFrame(); |
| } |
| |
| void ScrollSnapTest::GestureScroll(double x, |
| double y, |
| double delta_x, |
| double delta_y) { |
| ScrollBegin(x, y, delta_x, delta_y); |
| ScrollUpdate(x, y, delta_x, delta_y); |
| ScrollEnd(x + delta_x, y + delta_y); |
| |
| // Wait for animation to finish. |
| Compositor().BeginFrame(); // update run_state_. |
| Compositor().BeginFrame(); // Set start_time = now. |
| Compositor().BeginFrame(0.3); |
| } |
| |
| void ScrollSnapTest::ScrollBegin(double x, |
| double y, |
| double hint_x, |
| double hint_y) { |
| WebGestureEvent event(WebInputEvent::kGestureScrollBegin, |
| WebInputEvent::kNoModifiers, CurrentTimeTicks(), |
| WebGestureDevice::kWebGestureDeviceTouchscreen); |
| event.SetPositionInWidget(WebFloatPoint(x, y)); |
| event.SetPositionInScreen(WebFloatPoint(x, y)); |
| event.data.scroll_begin.delta_x_hint = hint_x; |
| event.data.scroll_begin.delta_y_hint = hint_y; |
| event.data.scroll_begin.pointer_count = 1; |
| event.SetFrameScale(1); |
| GetDocument().GetFrame()->GetEventHandler().HandleGestureScrollEvent(event); |
| } |
| |
| void ScrollSnapTest::ScrollUpdate(double x, |
| double y, |
| double delta_x, |
| double delta_y, |
| bool is_in_inertial_phase) { |
| WebGestureEvent event(WebInputEvent::kGestureScrollUpdate, |
| WebInputEvent::kNoModifiers, CurrentTimeTicks(), |
| WebGestureDevice::kWebGestureDeviceTouchscreen); |
| event.SetPositionInWidget(WebFloatPoint(x, y)); |
| event.SetPositionInScreen(WebFloatPoint(x, y)); |
| event.data.scroll_update.delta_x = delta_x; |
| event.data.scroll_update.delta_y = delta_y; |
| if (is_in_inertial_phase) { |
| event.data.scroll_update.inertial_phase = WebGestureEvent::kMomentumPhase; |
| event.SetTimeStamp(Compositor().LastFrameTime()); |
| } |
| event.SetFrameScale(1); |
| GetDocument().GetFrame()->GetEventHandler().HandleGestureScrollEvent(event); |
| } |
| |
| void ScrollSnapTest::ScrollEnd(double x, double y, bool is_in_inertial_phase) { |
| WebGestureEvent event(WebInputEvent::kGestureScrollEnd, |
| WebInputEvent::kNoModifiers, CurrentTimeTicks(), |
| WebGestureDevice::kWebGestureDeviceTouchscreen); |
| event.SetPositionInWidget(WebFloatPoint(x, y)); |
| event.SetPositionInScreen(WebFloatPoint(x, y)); |
| event.data.scroll_end.inertial_phase = |
| is_in_inertial_phase ? WebGestureEvent::kMomentumPhase |
| : WebGestureEvent::kNonMomentumPhase; |
| GetDocument().GetFrame()->GetEventHandler().HandleGestureScrollEvent(event); |
| } |
| |
| void ScrollSnapTest::SetInitialScrollOffset(double x, double y) { |
| Element* scroller = GetDocument().getElementById("scroller"); |
| scroller->GetLayoutBox()->SetScrollLeft(LayoutUnit::FromFloatRound(x)); |
| scroller->GetLayoutBox()->SetScrollTop(LayoutUnit::FromFloatRound(y)); |
| ASSERT_EQ(scroller->scrollLeft(), x); |
| ASSERT_EQ(scroller->scrollTop(), y); |
| } |
| |
| TEST_F(ScrollSnapTest, ScrollSnapOnX) { |
| SetUpForDiv(); |
| SetInitialScrollOffset(50, 150); |
| GestureScroll(100, 100, -50, 0); |
| |
| Element* scroller = GetDocument().getElementById("scroller"); |
| // Snaps to align the area at start. |
| ASSERT_EQ(scroller->scrollLeft(), 200); |
| // An x-locked scroll ignores snap points on y. |
| ASSERT_EQ(scroller->scrollTop(), 150); |
| } |
| |
| TEST_F(ScrollSnapTest, ScrollSnapOnY) { |
| SetUpForDiv(); |
| SetInitialScrollOffset(150, 50); |
| GestureScroll(100, 100, 0, -50); |
| |
| Element* scroller = GetDocument().getElementById("scroller"); |
| // A y-locked scroll ignores snap points on x. |
| ASSERT_EQ(scroller->scrollLeft(), 150); |
| // Snaps to align the area at start. |
| ASSERT_EQ(scroller->scrollTop(), 200); |
| } |
| |
| TEST_F(ScrollSnapTest, ScrollSnapOnBoth) { |
| SetUpForDiv(); |
| SetInitialScrollOffset(50, 50); |
| GestureScroll(100, 100, -50, -50); |
| |
| Element* scroller = GetDocument().getElementById("scroller"); |
| // A scroll gesture that has move in both x and y would snap on both axes. |
| ASSERT_EQ(scroller->scrollLeft(), 200); |
| ASSERT_EQ(scroller->scrollTop(), 200); |
| } |
| |
| TEST_F(ScrollSnapTest, AnimateFlingToArriveAtSnapPoint) { |
| SetUpForDiv(); |
| // Vertically align with the area. |
| SetInitialScrollOffset(0, 200); |
| Element* scroller = GetDocument().getElementById("scroller"); |
| ASSERT_EQ(scroller->scrollLeft(), 0); |
| ASSERT_EQ(scroller->scrollTop(), 200); |
| |
| ScrollBegin(100, 100, -5, 0); |
| // Starts with a non-inertial GSU. |
| ScrollUpdate(100, 100, -5, 0); |
| // Fling with an inertial GSU. |
| ScrollUpdate(95, 100, -5, 0, true); |
| ScrollEnd(90, 100); |
| // Animate halfway through the fling. |
| Compositor().BeginFrame(0.25); |
| ASSERT_GT(scroller->scrollLeft(), 150); |
| ASSERT_LT(scroller->scrollLeft(), 180); |
| ASSERT_EQ(scroller->scrollTop(), 200); |
| // Finish the animation. |
| Compositor().BeginFrame(0.6); |
| |
| ASSERT_EQ(scroller->scrollLeft(), 200); |
| ASSERT_EQ(scroller->scrollTop(), 200); |
| } |
| |
| TEST_F(ScrollSnapTest, SnapWhenBodyViewportDefining) { |
| v8::HandleScope HandleScope(v8::Isolate::GetCurrent()); |
| WebView().MainFrameWidget()->Resize(WebSize(300, 300)); |
| SimRequest request("https://example.com/test.html", "text/html"); |
| LoadURL("https://example.com/test.html"); |
| request.Complete(R"HTML( |
| <!DOCTYPE html> |
| <style> |
| body { |
| overflow: scroll; |
| scroll-snap-type: both mandatory; |
| height: 300px; |
| width: 300px; |
| margin: 0px; |
| } |
| #container { |
| margin: 0px; |
| padding: 0px; |
| width: 500px; |
| height: 500px; |
| } |
| #area { |
| position: relative; |
| left: 200px; |
| top: 200px; |
| width: 100px; |
| height: 100px; |
| scroll-snap-align: start; |
| } |
| </style> |
| <div id='container'> |
| <div id='area'></div> |
| </div> |
| )HTML"); |
| Compositor().BeginFrame(); |
| |
| GestureScroll(100, 100, -50, -50); |
| |
| // Sanity check that body is the viewport defining element |
| ASSERT_EQ(GetDocument().body(), GetDocument().ViewportDefiningElement()); |
| |
| // When body is viewport defining and overflows then any snap points on the |
| // body element will be captured by layout view as the snap container. |
| ASSERT_EQ(Window().scrollX(), 200); |
| ASSERT_EQ(Window().scrollY(), 200); |
| } |
| |
| TEST_F(ScrollSnapTest, SnapWhenHtmlViewportDefining) { |
| v8::HandleScope HandleScope(v8::Isolate::GetCurrent()); |
| WebView().MainFrameWidget()->Resize(WebSize(300, 300)); |
| SimRequest request("https://example.com/test.html", "text/html"); |
| LoadURL("https://example.com/test.html"); |
| request.Complete(R"HTML( |
| <!DOCTYPE html> |
| <style> |
| :root { |
| overflow: scroll; |
| scroll-snap-type: both mandatory; |
| height: 300px; |
| width: 300px; |
| } |
| body { |
| margin: 0px; |
| } |
| #container { |
| margin: 0px; |
| padding: 0px; |
| width: 500px; |
| height: 500px; |
| } |
| #area { |
| position: relative; |
| left: 200px; |
| top: 200px; |
| width: 100px; |
| height: 100px; |
| scroll-snap-align: start; |
| } |
| </style> |
| <div id='container'> |
| <div id='area'></div> |
| </div> |
| )HTML"); |
| Compositor().BeginFrame(); |
| |
| GestureScroll(100, 100, -50, -50); |
| |
| // Sanity check that document element is the viewport defining element |
| ASSERT_EQ(GetDocument().documentElement(), |
| GetDocument().ViewportDefiningElement()); |
| |
| // When document is viewport defining and overflows then any snap ponts on the |
| // document element will be captured by layout view as snap container. |
| ASSERT_EQ(Window().scrollX(), 200); |
| ASSERT_EQ(Window().scrollY(), 200); |
| } |
| |
| TEST_F(ScrollSnapTest, SnapWhenBodyOverflowHtmlViewportDefining) { |
| v8::HandleScope HandleScope(v8::Isolate::GetCurrent()); |
| WebView().MainFrameWidget()->Resize(WebSize(300, 300)); |
| SimRequest request("https://example.com/test.html", "text/html"); |
| LoadURL("https://example.com/test.html"); |
| request.Complete(R"HTML( |
| <!DOCTYPE html> |
| <style> |
| :root { |
| overflow: scroll; |
| height: 300px; |
| width: 300px; |
| } |
| body { |
| overflow: scroll; |
| scroll-snap-type: both mandatory; |
| height: 400px; |
| width: 400px; |
| } |
| #container { |
| margin: 0px; |
| padding: 0px; |
| width: 500px; |
| height: 500px; |
| } |
| #area { |
| position: relative; |
| left: 200px; |
| top: 200px; |
| width: 100px; |
| height: 100px; |
| scroll-snap-align: start; |
| } |
| </style> |
| <div id='container'> |
| <div id='area'></div> |
| </div> |
| )HTML"); |
| Compositor().BeginFrame(); |
| |
| GestureScroll(100, 100, -50, -50); |
| |
| // Sanity check that document element is the viewport defining element |
| ASSERT_EQ(GetDocument().documentElement(), |
| GetDocument().ViewportDefiningElement()); |
| |
| // When body and document elements are both scrollable then body element |
| // should capture snap points defined on it as opposed to layout view. |
| Element* body = GetDocument().body(); |
| ASSERT_EQ(body->scrollLeft(), 100); |
| ASSERT_EQ(body->scrollTop(), 100); |
| } |
| |
| } // namespace blink |