Chromium Code Reviews| Index: Source/core/frame/FrameViewTest.cpp |
| diff --git a/Source/core/frame/FrameViewTest.cpp b/Source/core/frame/FrameViewTest.cpp |
| new file mode 100644 |
| index 0000000000000000000000000000000000000000..aeaf53e16d8f74f11b223e920413459b232b2b74 |
| --- /dev/null |
| +++ b/Source/core/frame/FrameViewTest.cpp |
| @@ -0,0 +1,258 @@ |
| +// Copyright 2015 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 "config.h" |
| + |
| +#include "core/frame/FrameView.h" |
| + |
| +#include "core/dom/Document.h" |
| +#include "core/frame/FrameOwner.h" |
| +#include "core/frame/LocalFrame.h" |
| +#include "core/loader/EmptyClients.h" |
| +#include "core/page/Page.h" |
| +#include "core/testing/DummyPageHolder.h" |
| +#include "platform/testing/UnitTestHelpers.h" |
| +#include <gtest/gtest.h> |
| + |
| +namespace blink { |
| + |
| +class RootFrameLoaderClient; |
| + |
| +class MockChromeClient final : public EmptyChromeClient { |
|
esprehn
2015/09/21 21:27:21
Can we just write a Sim test instead?
Sami
2015/09/23 18:31:07
Looks much nicer that way. Done. One minor issue w
|
| +public: |
| + MockChromeClient() |
| + : m_didScheduleAnimation(false) { } |
| + ~MockChromeClient() override { } |
| + |
| + void scheduleAnimation() override |
| + { |
| + m_didScheduleAnimation = true; |
| + } |
| + |
| + bool animationScheduled() const |
| + { |
| + return m_didScheduleAnimation; |
| + } |
| + |
| + void resetAnimationScheduled() |
| + { |
| + m_didScheduleAnimation = false; |
| + } |
| +private: |
| + bool m_didScheduleAnimation; |
| +}; |
| + |
| +class StubFrameLoaderClientWithParent final : public EmptyFrameLoaderClient { |
| +public: |
| + static PassOwnPtrWillBeRawPtr<StubFrameLoaderClientWithParent> create(Frame* parent) |
| + { |
| + return adoptPtrWillBeNoop(new StubFrameLoaderClientWithParent(parent)); |
| + } |
| + |
| + DEFINE_INLINE_VIRTUAL_TRACE() |
| + { |
| + visitor->trace(m_parent); |
| + EmptyFrameLoaderClient::trace(visitor); |
| + } |
| + |
| + Frame* parent() const override { return m_parent.get(); } |
| + |
| +private: |
| + explicit StubFrameLoaderClientWithParent(Frame* parent) |
| + : m_parent(parent) |
| + { |
| + } |
| + |
| + RawPtrWillBeMember<Frame> m_parent; |
| +}; |
| + |
| +class StubFrameOwner : public NoBaseWillBeGarbageCollectedFinalized<StubFrameOwner>, public FrameOwner { |
| + WILL_BE_USING_GARBAGE_COLLECTED_MIXIN(StubFrameOwner); |
| +public: |
| + static PassOwnPtrWillBeRawPtr<StubFrameOwner> create() |
| + { |
| + return adoptPtrWillBeNoop(new StubFrameOwner); |
| + } |
| + |
| + DEFINE_INLINE_VIRTUAL_TRACE() { FrameOwner::trace(visitor); } |
| + |
| + bool isLocal() const override { return false; } |
| + SandboxFlags sandboxFlags() const override { return SandboxNone; } |
| + void dispatchLoad() override { } |
| + void renderFallbackContent() override { } |
| +}; |
| + |
| +class FrameViewTest : public ::testing::Test { |
| +public: |
| + using SecurityOriginStatus = FrameView::SecurityOriginStatus; |
| + using ViewportVisibility = FrameView::ViewportVisibility; |
| + |
| + void SetUp() override; |
| + void TearDown() override; |
| + |
| + LocalFrame* childFrame() const |
| + { |
| + return m_childFrame.get(); |
| + } |
| + |
| + ViewportVisibility viewportVisibility(FrameView* frameView) |
| + { |
| + return frameView->m_viewportVisibility; |
| + } |
| + |
| + void enableThrottling(FrameView* frameView) |
| + { |
| + frameView->m_viewportVisibility = ViewportVisibility::Hidden; |
| + frameView->m_securityOriginStatusForThrottling = SecurityOriginStatus::IsCrossOrigin; |
| + EXPECT_TRUE(frameView->shouldThrottleRenderingPipeline()); |
| + } |
| + |
| +protected: |
| + MockChromeClient m_mockChromeClient; |
| + Page::PageClients m_pageClients; |
| + OwnPtr<DummyPageHolder> m_dummyPageHolder; |
| + RefPtrWillBePersistent<Document> m_document; |
| + OwnPtrWillBePersistent<StubFrameOwner> m_frameOwner; |
| + |
| + OwnPtrWillBePersistent<StubFrameLoaderClientWithParent> m_childClient; |
| + RefPtrWillBePersistent<LocalFrame> m_childFrame; |
| +}; |
| + |
| +class RootFrameLoaderClient final : public EmptyFrameLoaderClient { |
| +public: |
| + static PassOwnPtrWillBeRawPtr<RootFrameLoaderClient> create(FrameViewTest* testSuite) |
| + { |
| + return adoptPtrWillBeNoop(new RootFrameLoaderClient(testSuite)); |
| + } |
| + |
| + Frame* firstChild() const override { return m_testSuite->childFrame(); } |
| + |
| +private: |
| + explicit RootFrameLoaderClient(FrameViewTest* testSuite) |
| + : m_testSuite(testSuite) |
| + { |
| + } |
| + |
| + FrameViewTest* m_testSuite; |
| +}; |
| + |
| +void FrameViewTest::SetUp() |
| +{ |
| + // Create a root frame and one child frame. The FrameLoaderClients will connect the frames in the frame tree. |
| + fillWithEmptyClients(m_pageClients); |
| + m_pageClients.chromeClient = &m_mockChromeClient; |
| + OwnPtrWillBePersistent<RootFrameLoaderClient> frameLoaderClient = RootFrameLoaderClient::create(this); |
| + m_dummyPageHolder = DummyPageHolder::create(IntSize(640, 480), &m_pageClients, frameLoaderClient.release()); |
| + m_document = &m_dummyPageHolder->document(); |
| + m_frameOwner = StubFrameOwner::create(); |
| + |
| + m_childClient = StubFrameLoaderClientWithParent::create(m_document->frame()); |
| + m_childFrame = LocalFrame::create(m_childClient.get(), m_document->frame()->host(), m_frameOwner.get()); |
| + m_childFrame->setView(FrameView::create(m_childFrame.get(), IntSize(320, 240))); |
| + m_childFrame->init(); |
| + |
| + // Also hook up the frames in the FrameView tree. |
| + m_document->view()->addChild(m_childFrame->view()); |
| +} |
| + |
| +void FrameViewTest::TearDown() |
| +{ |
| + m_document->view()->removeChild(m_childFrame->view()); |
| + m_childFrame->detach(FrameDetachType::Remove); |
| + m_childFrame = nullptr; |
| +} |
| + |
| +TEST_F(FrameViewTest, viewportVisibility) |
| +{ |
| + // Initially both frames are visible. |
| + FrameView* frameView = m_document->view(); |
| + FrameView* childFrameView = m_childFrame->view(); |
| + EXPECT_EQ(ViewportVisibility::Visible, viewportVisibility(frameView)); |
| + EXPECT_EQ(ViewportVisibility::Visible, viewportVisibility(childFrameView)); |
| + |
| + // Moving the child fully outside the parent makes it invisible. |
| + childFrameView->move(0, frameView->height()); |
| + frameView->updateAllLifecyclePhases(); |
| + frameView->updateThrottling(); |
| + EXPECT_EQ(ViewportVisibility::Visible, viewportVisibility(frameView)); |
| + EXPECT_EQ(ViewportVisibility::Hidden, viewportVisibility(childFrameView)); |
| + |
| + // A partially visible child is considered visible. |
| + childFrameView->move(-childFrameView->width() / 2, 0); |
| + frameView->updateAllLifecyclePhases(); |
| + frameView->updateThrottling(); |
| + EXPECT_EQ(ViewportVisibility::Visible, viewportVisibility(frameView)); |
| + EXPECT_EQ(ViewportVisibility::Visible, viewportVisibility(childFrameView)); |
| +} |
| + |
| +TEST_F(FrameViewTest, viewportVisibilityFullyClipped) |
| +{ |
| + // A child which is fully clipped away by its ancestor should become invisible. |
| + FrameView* frameView = m_document->view(); |
| + FrameView* childFrameView = m_childFrame->view(); |
| + frameView->resize(0, 0); |
| + frameView->updateAllLifecyclePhases(); |
| + frameView->updateThrottling(); |
| + EXPECT_EQ(ViewportVisibility::Visible, viewportVisibility(frameView)); |
| + EXPECT_EQ(ViewportVisibility::Hidden, viewportVisibility(childFrameView)); |
| +} |
| + |
| +TEST_F(FrameViewTest, hiddenCrossOriginFramesAreThrottled) |
| +{ |
| + FrameView* frameView = m_document->view(); |
| + FrameView* childFrameView = m_childFrame->view(); |
| + EXPECT_FALSE(frameView->shouldThrottleRenderingPipeline()); |
| + EXPECT_FALSE(childFrameView->shouldThrottleRenderingPipeline()); |
| + |
| + // Just being hidden doesn't make a frame throttled. |
| + childFrameView->move(0, frameView->height()); |
| + frameView->updateAllLifecyclePhases(); |
| + frameView->updateThrottling(); |
| + EXPECT_FALSE(frameView->shouldThrottleRenderingPipeline()); |
| + EXPECT_FALSE(childFrameView->shouldThrottleRenderingPipeline()); |
| + |
| + // Also making the frame cross origin enables throttling. |
| + m_childFrame->securityContext()->setSecurityOrigin(SecurityOrigin::createUnique()); |
| + frameView->updateThrottling(); |
| + EXPECT_FALSE(frameView->shouldThrottleRenderingPipeline()); |
| + EXPECT_TRUE(childFrameView->shouldThrottleRenderingPipeline()); |
| +} |
| + |
| +TEST_F(FrameViewTest, throttledLifecycleUpdate) |
| +{ |
| + FrameView* frameView = m_document->view(); |
| + FrameView* childFrameView = m_childFrame->view(); |
| + enableThrottling(childFrameView); |
| + |
| + childFrameView->setNeedsLayout(); |
| + EXPECT_TRUE(childFrameView->needsLayout()); |
| + frameView->updateAllLifecyclePhases(DocumentLifecycle::ThrottlingMode::Allow); |
| + EXPECT_TRUE(childFrameView->needsLayout()); |
| + |
| + frameView->updateAllLifecyclePhases(DocumentLifecycle::ThrottlingMode::Disallow); |
| + EXPECT_FALSE(childFrameView->needsLayout()); |
| +} |
| + |
| +TEST_F(FrameViewTest, unthrottledFrameSchedulesVisualUpdate) |
| +{ |
| + FrameView* frameView = m_document->view(); |
| + FrameView* childFrameView = m_childFrame->view(); |
| + m_childFrame->securityContext()->setSecurityOrigin(SecurityOrigin::createUnique()); |
| + |
| + // First make the child hidden to enable throttling. |
| + childFrameView->move(0, frameView->height()); |
| + frameView->updateAllLifecyclePhases(DocumentLifecycle::ThrottlingMode::Allow); |
| + m_mockChromeClient.resetAnimationScheduled(); |
| + frameView->updateThrottling(); |
| + EXPECT_FALSE(m_mockChromeClient.animationScheduled()); |
| + |
| + // Then bring it back on-screen. This should schedule a visual update. |
| + childFrameView->move(0, 0); |
| + frameView->updateAllLifecyclePhases(DocumentLifecycle::ThrottlingMode::Allow); |
| + m_mockChromeClient.resetAnimationScheduled(); |
| + frameView->updateThrottling(); |
| + EXPECT_TRUE(m_mockChromeClient.animationScheduled()); |
| +} |
| + |
| +} // namespace blink |