/* * Copyright (C) 2012 Google Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS BE LIABLE FOR ANY * DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include "config.h" #include "ScrollingCoordinator.h" #include "CompositorFakeWebGraphicsContext3D.h" #include "FakeWebCompositorOutputSurface.h" #include "FrameTestHelpers.h" #include "RenderLayerCompositor.h" #include "RenderView.h" #include "URLTestHelpers.h" #include "WebCompositorInitializer.h" #include "WebFrameClient.h" #include "WebFrameImpl.h" #include "WebSettings.h" #include "WebViewClient.h" #include "WebViewImpl.h" #include #include #include #include #if ENABLE(ACCELERATED_OVERFLOW_SCROLLING) #include "GraphicsLayerChromium.h" #endif using namespace WebCore; using namespace WebKit; namespace { class MockWebViewClient : public WebViewClient { public: virtual WebCompositorOutputSurface* createOutputSurface() OVERRIDE { return FakeWebCompositorOutputSurface::create(CompositorFakeWebGraphicsContext3D::create(WebGraphicsContext3D::Attributes())).leakPtr(); } }; class MockWebFrameClient : public WebFrameClient { }; class ScrollingCoordinatorChromiumTest : public testing::Test { public: ScrollingCoordinatorChromiumTest() : m_baseURL("http://www.test.com/") , m_webCompositorInitializer(0) { // We cannot reuse FrameTestHelpers::createWebViewAndLoad here because the compositing // settings need to be set before the page is loaded. m_webViewImpl = static_cast(WebView::create(&m_mockWebViewClient)); m_webViewImpl->settings()->setJavaScriptEnabled(true); m_webViewImpl->settings()->setForceCompositingMode(true); m_webViewImpl->settings()->setAcceleratedCompositingEnabled(true); m_webViewImpl->settings()->setAcceleratedCompositingForFixedPositionEnabled(true); m_webViewImpl->settings()->setFixedPositionCreatesStackingContext(true); m_webViewImpl->initializeMainFrame(&m_mockWebFrameClient); m_webViewImpl->resize(IntSize(320, 240)); } virtual ~ScrollingCoordinatorChromiumTest() { webkit_support::UnregisterAllMockedURLs(); m_webViewImpl->close(); } void navigateTo(const std::string& url) { FrameTestHelpers::loadFrame(m_webViewImpl->mainFrame(), url); webkit_support::ServeAsynchronousMockedRequests(); } void registerMockedHttpURLLoad(const std::string& fileName) { URLTestHelpers::registerMockedURLFromBaseURL(WebString::fromUTF8(m_baseURL.c_str()), WebString::fromUTF8(fileName.c_str())); } WebLayer* getRootScrollLayer() { RenderLayerCompositor* compositor = m_webViewImpl->mainFrameImpl()->frame()->contentRenderer()->compositor(); ASSERT(compositor); ASSERT(compositor->scrollLayer()); WebLayer* webScrollLayer = static_cast(compositor->scrollLayer()->platformLayer()); return webScrollLayer; } protected: std::string m_baseURL; MockWebFrameClient m_mockWebFrameClient; MockWebViewClient m_mockWebViewClient; WebKitTests::WebCompositorInitializer m_webCompositorInitializer; WebViewImpl* m_webViewImpl; }; TEST_F(ScrollingCoordinatorChromiumTest, fastScrollingByDefault) { navigateTo("about:blank"); // Make sure the scrolling coordinator is active. FrameView* frameView = m_webViewImpl->mainFrameImpl()->frameView(); Page* page = m_webViewImpl->mainFrameImpl()->frame()->page(); ASSERT_TRUE(page->scrollingCoordinator()); ASSERT_TRUE(page->scrollingCoordinator()->coordinatesScrollingForFrameView(frameView)); // Fast scrolling should be enabled by default. WebLayer* rootScrollLayer = getRootScrollLayer(); ASSERT_TRUE(rootScrollLayer->scrollable()); ASSERT_FALSE(rootScrollLayer->shouldScrollOnMainThread()); ASSERT_FALSE(rootScrollLayer->haveWheelEventHandlers()); } TEST_F(ScrollingCoordinatorChromiumTest, fastScrollingForFixedPosition) { registerMockedHttpURLLoad("fixed-position.html"); navigateTo(m_baseURL + "fixed-position.html"); Page* page = m_webViewImpl->mainFrameImpl()->frame()->page(); ASSERT_TRUE(page->scrollingCoordinator()->supportsFixedPositionLayers()); // Fixed position should not fall back to main thread scrolling. WebLayer* rootScrollLayer = getRootScrollLayer(); ASSERT_FALSE(rootScrollLayer->shouldScrollOnMainThread()); // Verify the properties of the fixed position element starting from the RenderObject all the // way to the WebLayer. Element* fixedElement = m_webViewImpl->mainFrameImpl()->frame()->document()->getElementById("fixed"); ASSERT(fixedElement); RenderObject* renderer = fixedElement->renderer(); ASSERT_TRUE(renderer->isBoxModelObject()); ASSERT_TRUE(renderer->hasLayer()); RenderLayer* layer = toRenderBoxModelObject(renderer)->layer(); ASSERT_TRUE(layer->isComposited()); RenderLayerBacking* layerBacking = layer->backing(); WebLayer* webLayer = static_cast(layerBacking->graphicsLayer()->platformLayer()); ASSERT_TRUE(webLayer->fixedToContainerLayer()); } TEST_F(ScrollingCoordinatorChromiumTest, nonFastScrollableRegion) { registerMockedHttpURLLoad("non-fast-scrollable.html"); navigateTo(m_baseURL + "non-fast-scrollable.html"); WebLayer* rootScrollLayer = getRootScrollLayer(); WebVector nonFastScrollableRegion = rootScrollLayer->nonFastScrollableRegion(); ASSERT_EQ(1u, nonFastScrollableRegion.size()); ASSERT_EQ(WebRect(8, 8, 10, 10), nonFastScrollableRegion[0]); } TEST_F(ScrollingCoordinatorChromiumTest, wheelEventHandler) { registerMockedHttpURLLoad("wheel-event-handler.html"); navigateTo(m_baseURL + "wheel-event-handler.html"); WebLayer* rootScrollLayer = getRootScrollLayer(); ASSERT_TRUE(rootScrollLayer->haveWheelEventHandlers()); } TEST_F(ScrollingCoordinatorChromiumTest, clippedBodyTest) { registerMockedHttpURLLoad("clipped-body.html"); navigateTo(m_baseURL + "clipped-body.html"); WebLayer* rootScrollLayer = getRootScrollLayer(); ASSERT_EQ(0u, rootScrollLayer->nonFastScrollableRegion().size()); } #if ENABLE(ACCELERATED_OVERFLOW_SCROLLING) TEST_F(ScrollingCoordinatorChromiumTest, touchOverflowScrolling) { registerMockedHttpURLLoad("touch-overflow-scrolling.html"); navigateTo(m_baseURL + "touch-overflow-scrolling.html"); // Verify the properties of the accelerated scrolling element starting from the RenderObject // all the way to the WebLayer. Element* scrollableElement = m_webViewImpl->mainFrameImpl()->frame()->document()->getElementById("scrollable"); ASSERT(scrollableElement); RenderObject* renderer = scrollableElement->renderer(); ASSERT_TRUE(renderer->isBoxModelObject()); ASSERT_TRUE(renderer->hasLayer()); RenderLayer* layer = toRenderBoxModelObject(renderer)->layer(); ASSERT_TRUE(layer->usesCompositedScrolling()); ASSERT_TRUE(layer->isComposited()); RenderLayerBacking* layerBacking = layer->backing(); ASSERT_TRUE(layerBacking->hasScrollingLayer()); ASSERT(layerBacking->scrollingContentsLayer()); GraphicsLayerChromium* graphicsLayerChromium = static_cast(layerBacking->scrollingContentsLayer()); ASSERT_EQ(layer, graphicsLayerChromium->scrollableArea()); WebLayer* webScrollLayer = static_cast(layerBacking->scrollingContentsLayer()->platformLayer()); ASSERT_TRUE(webScrollLayer->scrollable()); } #endif // ENABLE(ACCELERATED_OVERFLOW_SCROLLING) } // namespace