/* * Copyright (C) 2010 Apple 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 "PageOverlay.h" #include "Frame.h" #include "FrameView.h" #include "GraphicsContext.h" #include "Page.h" #include "PageOverlayController.h" #include "PlatformMouseEvent.h" #include "ScrollbarTheme.h" namespace WebCore { static const Seconds fadeAnimationDuration { 200_ms }; static const double fadeAnimationFrameRate = 30; static PageOverlay::PageOverlayID generatePageOverlayID() { static PageOverlay::PageOverlayID pageOverlayID; return ++pageOverlayID; } Ref PageOverlay::create(Client& client, OverlayType overlayType) { return adoptRef(*new PageOverlay(client, overlayType)); } PageOverlay::PageOverlay(Client& client, OverlayType overlayType) : m_client(client) , m_fadeAnimationTimer(*this, &PageOverlay::fadeAnimationTimerFired) , m_fadeAnimationDuration(fadeAnimationDuration) , m_needsSynchronousScrolling(overlayType == OverlayType::View) , m_overlayType(overlayType) , m_pageOverlayID(generatePageOverlayID()) { } PageOverlay::~PageOverlay() = default; PageOverlayController* PageOverlay::controller() const { if (!m_page) return nullptr; return &m_page->pageOverlayController(); } IntRect PageOverlay::bounds() const { if (!m_overrideFrame.isEmpty()) return { { }, m_overrideFrame.size() }; FrameView* frameView = m_page->mainFrame().view(); if (!frameView) return IntRect(); switch (m_overlayType) { case OverlayType::View: { int width = frameView->width(); int height = frameView->height(); if (!ScrollbarTheme::theme().usesOverlayScrollbars()) { if (frameView->verticalScrollbar()) width -= frameView->verticalScrollbar()->width(); if (frameView->horizontalScrollbar()) height -= frameView->horizontalScrollbar()->height(); } return IntRect(0, 0, width, height); } case OverlayType::Document: return IntRect(IntPoint(), frameView->contentsSize()); } ASSERT_NOT_REACHED(); return IntRect(IntPoint(), frameView->contentsSize()); } IntRect PageOverlay::frame() const { if (!m_overrideFrame.isEmpty()) return m_overrideFrame; return bounds(); } void PageOverlay::setFrame(IntRect frame) { if (m_overrideFrame == frame) return; m_overrideFrame = frame; if (auto pageOverlayController = controller()) pageOverlayController->didChangeOverlayFrame(*this); } IntSize PageOverlay::viewToOverlayOffset() const { switch (m_overlayType) { case OverlayType::View: return IntSize(); case OverlayType::Document: { FrameView* frameView = m_page->mainFrame().view(); return frameView ? toIntSize(frameView->viewToContents(IntPoint())) : IntSize(); } } return IntSize(); } void PageOverlay::setBackgroundColor(const Color& backgroundColor) { if (m_backgroundColor == backgroundColor) return; m_backgroundColor = backgroundColor; if (auto pageOverlayController = controller()) pageOverlayController->didChangeOverlayBackgroundColor(*this); } void PageOverlay::setPage(Page* page) { m_client.willMoveToPage(*this, page); m_page = page; m_client.didMoveToPage(*this, page); m_fadeAnimationTimer.stop(); } void PageOverlay::setNeedsDisplay(const IntRect& dirtyRect) { if (auto pageOverlayController = controller()) { if (m_fadeAnimationType != FadeAnimationType::NoAnimation) pageOverlayController->setPageOverlayOpacity(*this, m_fractionFadedIn); pageOverlayController->setPageOverlayNeedsDisplay(*this, dirtyRect); } } void PageOverlay::setNeedsDisplay() { setNeedsDisplay(bounds()); } void PageOverlay::drawRect(GraphicsContext& graphicsContext, const IntRect& dirtyRect) { // If the dirty rect is outside the bounds, ignore it. IntRect paintRect = intersection(dirtyRect, bounds()); if (paintRect.isEmpty()) return; GraphicsContextStateSaver stateSaver(graphicsContext); if (m_overlayType == PageOverlay::OverlayType::Document) { if (FrameView* frameView = m_page->mainFrame().view()) { auto offset = frameView->scrollOrigin(); graphicsContext.translate(toFloatSize(offset)); paintRect.moveBy(-offset); } } m_client.drawRect(*this, graphicsContext, paintRect); } bool PageOverlay::mouseEvent(const PlatformMouseEvent& mouseEvent) { IntPoint mousePositionInOverlayCoordinates(mouseEvent.position()); if (m_overlayType == PageOverlay::OverlayType::Document) mousePositionInOverlayCoordinates = m_page->mainFrame().view()->windowToContents(mousePositionInOverlayCoordinates); mousePositionInOverlayCoordinates.moveBy(-frame().location()); // Ignore events outside the bounds. if (m_shouldIgnoreMouseEventsOutsideBounds && !bounds().contains(mousePositionInOverlayCoordinates)) return false; return m_client.mouseEvent(*this, mouseEvent); } void PageOverlay::didScrollFrame(Frame& frame) { m_client.didScrollFrame(*this, frame); } bool PageOverlay::copyAccessibilityAttributeStringValueForPoint(String attribute, FloatPoint parameter, String& value) { return m_client.copyAccessibilityAttributeStringValueForPoint(*this, attribute, parameter, value); } bool PageOverlay::copyAccessibilityAttributeBoolValueForPoint(String attribute, FloatPoint parameter, bool& value) { return m_client.copyAccessibilityAttributeBoolValueForPoint(*this, attribute, parameter, value); } Vector PageOverlay::copyAccessibilityAttributeNames(bool parameterizedNames) { return m_client.copyAccessibilityAttributeNames(*this, parameterizedNames); } void PageOverlay::startFadeInAnimation() { if (m_fadeAnimationType == FadeInAnimation && m_fadeAnimationTimer.isActive()) return; m_fractionFadedIn = 0; m_fadeAnimationType = FadeInAnimation; startFadeAnimation(); } void PageOverlay::startFadeOutAnimation() { if (m_fadeAnimationType == FadeOutAnimation && m_fadeAnimationTimer.isActive()) return; m_fractionFadedIn = 1; m_fadeAnimationType = FadeOutAnimation; startFadeAnimation(); } void PageOverlay::stopFadeOutAnimation() { m_fractionFadedIn = 1.0; m_fadeAnimationTimer.stop(); } void PageOverlay::startFadeAnimation() { m_fadeAnimationStartTime = WallTime::now(); m_fadeAnimationTimer.startRepeating(1_s / fadeAnimationFrameRate); } void PageOverlay::fadeAnimationTimerFired() { float animationProgress = (WallTime::now() - m_fadeAnimationStartTime) / m_fadeAnimationDuration; if (animationProgress >= 1.0) animationProgress = 1.0; double sine = sin(piOverTwoFloat * animationProgress); float fadeAnimationValue = sine * sine; m_fractionFadedIn = (m_fadeAnimationType == FadeInAnimation) ? fadeAnimationValue : 1 - fadeAnimationValue; controller()->setPageOverlayOpacity(*this, m_fractionFadedIn); if (animationProgress == 1.0) { m_fadeAnimationTimer.stop(); bool wasFadingOut = m_fadeAnimationType == FadeOutAnimation; m_fadeAnimationType = NoAnimation; // If this was a fade out, uninstall the page overlay. if (wasFadingOut) controller()->uninstallPageOverlay(*this, PageOverlay::FadeMode::DoNotFade); } } void PageOverlay::clear() { if (auto pageOverlayController = controller()) pageOverlayController->clearPageOverlay(*this); } GraphicsLayer& PageOverlay::layer() { return controller()->layerForOverlay(*this); } } // namespace WebKit