/* * Copyright (C) 2014 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 "PageOverlayController.h" #include "Chrome.h" #include "ChromeClient.h" #include "Frame.h" #include "FrameView.h" #include "GraphicsContext.h" #include "GraphicsLayer.h" #include "MainFrame.h" #include "Page.h" #include "PageOverlay.h" #include "ScrollingCoordinator.h" #include "Settings.h" #include "TiledBacking.h" // FIXME: Someone needs to call didChangeSettings() if we want dynamic updates of layer border/repaint counter settings. namespace WebCore { PageOverlayController::PageOverlayController(MainFrame& mainFrame) : m_initialized(false) , m_mainFrame(mainFrame) { } PageOverlayController::~PageOverlayController() { } void PageOverlayController::createRootLayersIfNeeded() { if (m_initialized) return; m_initialized = true; ASSERT(!m_documentOverlayRootLayer); ASSERT(!m_viewOverlayRootLayer); m_documentOverlayRootLayer = GraphicsLayer::create(m_mainFrame.page()->chrome().client().graphicsLayerFactory(), *this); m_viewOverlayRootLayer = GraphicsLayer::create(m_mainFrame.page()->chrome().client().graphicsLayerFactory(), *this); #ifndef NDEBUG m_documentOverlayRootLayer->setName("Page Overlay container (document-relative)"); m_viewOverlayRootLayer->setName("Page Overlay container (view-relative)"); #endif } GraphicsLayer& PageOverlayController::documentOverlayRootLayer() { createRootLayersIfNeeded(); return *m_documentOverlayRootLayer; } GraphicsLayer& PageOverlayController::viewOverlayRootLayer() { createRootLayersIfNeeded(); return *m_viewOverlayRootLayer; } static void updateOverlayGeometry(PageOverlay& overlay, GraphicsLayer& graphicsLayer) { IntRect overlayFrame = overlay.frame(); if (overlayFrame.location() == graphicsLayer.position() && overlayFrame.size() == graphicsLayer.size()) return; graphicsLayer.setPosition(overlayFrame.location()); graphicsLayer.setSize(overlayFrame.size()); } void PageOverlayController::installPageOverlay(PassRefPtr pageOverlay, PageOverlay::FadeMode fadeMode) { createRootLayersIfNeeded(); RefPtr overlay = pageOverlay; if (m_pageOverlays.contains(overlay)) return; m_pageOverlays.append(overlay); std::unique_ptr layer = GraphicsLayer::create(m_mainFrame.page()->chrome().client().graphicsLayerFactory(), *this); layer->setAnchorPoint(FloatPoint3D()); layer->setBackgroundColor(overlay->backgroundColor()); #ifndef NDEBUG layer->setName("Page Overlay content"); #endif updateSettingsForLayer(*layer); switch (overlay->overlayType()) { case PageOverlay::OverlayType::View: m_viewOverlayRootLayer->addChild(layer.get()); break; case PageOverlay::OverlayType::Document: m_documentOverlayRootLayer->addChild(layer.get()); break; } GraphicsLayer& rawLayer = *layer; m_overlayGraphicsLayers.set(overlay.get(), WTFMove(layer)); updateForceSynchronousScrollLayerPositionUpdates(); overlay->setPage(m_mainFrame.page()); if (FrameView* frameView = m_mainFrame.view()) frameView->enterCompositingMode(); updateOverlayGeometry(*overlay, rawLayer); if (fadeMode == PageOverlay::FadeMode::Fade) overlay->startFadeInAnimation(); } void PageOverlayController::uninstallPageOverlay(PageOverlay* overlay, PageOverlay::FadeMode fadeMode) { if (fadeMode == PageOverlay::FadeMode::Fade) { overlay->startFadeOutAnimation(); return; } overlay->setPage(nullptr); m_overlayGraphicsLayers.take(overlay)->removeFromParent(); bool removed = m_pageOverlays.removeFirst(overlay); ASSERT_UNUSED(removed, removed); updateForceSynchronousScrollLayerPositionUpdates(); } void PageOverlayController::updateForceSynchronousScrollLayerPositionUpdates() { #if ENABLE(ASYNC_SCROLLING) bool forceSynchronousScrollLayerPositionUpdates = false; for (auto& overlay : m_pageOverlays) { if (overlay->needsSynchronousScrolling()) forceSynchronousScrollLayerPositionUpdates = true; } if (ScrollingCoordinator* scrollingCoordinator = m_mainFrame.page()->scrollingCoordinator()) scrollingCoordinator->setForceSynchronousScrollLayerPositionUpdates(forceSynchronousScrollLayerPositionUpdates); #endif } void PageOverlayController::setPageOverlayNeedsDisplay(PageOverlay& overlay, const WebCore::IntRect& dirtyRect) { ASSERT(m_pageOverlays.contains(&overlay)); GraphicsLayer& graphicsLayer = *m_overlayGraphicsLayers.get(&overlay); if (!graphicsLayer.drawsContent()) { graphicsLayer.setDrawsContent(true); updateOverlayGeometry(overlay, graphicsLayer); } graphicsLayer.setNeedsDisplayInRect(dirtyRect); } void PageOverlayController::setPageOverlayOpacity(PageOverlay& overlay, float opacity) { ASSERT(m_pageOverlays.contains(&overlay)); m_overlayGraphicsLayers.get(&overlay)->setOpacity(opacity); } void PageOverlayController::clearPageOverlay(PageOverlay& overlay) { ASSERT(m_pageOverlays.contains(&overlay)); m_overlayGraphicsLayers.get(&overlay)->setDrawsContent(false); } GraphicsLayer& PageOverlayController::layerForOverlay(PageOverlay& overlay) const { ASSERT(m_pageOverlays.contains(&overlay)); return *m_overlayGraphicsLayers.get(&overlay); } void PageOverlayController::willAttachRootLayer() { for (auto& overlayAndLayer : m_overlayGraphicsLayers) updateOverlayGeometry(*overlayAndLayer.key, *overlayAndLayer.value); } void PageOverlayController::didChangeViewSize() { for (auto& overlayAndLayer : m_overlayGraphicsLayers) { if (overlayAndLayer.key->overlayType() == PageOverlay::OverlayType::View) updateOverlayGeometry(*overlayAndLayer.key, *overlayAndLayer.value); } } void PageOverlayController::didChangeDocumentSize() { for (auto& overlayAndLayer : m_overlayGraphicsLayers) { if (overlayAndLayer.key->overlayType() == PageOverlay::OverlayType::Document) updateOverlayGeometry(*overlayAndLayer.key, *overlayAndLayer.value); } } void PageOverlayController::didChangeSettings() { // FIXME: We should apply these settings to all overlay sublayers recursively. for (auto& graphicsLayer : m_overlayGraphicsLayers.values()) updateSettingsForLayer(*graphicsLayer); } void PageOverlayController::didChangeDeviceScaleFactor() { createRootLayersIfNeeded(); m_documentOverlayRootLayer->noteDeviceOrPageScaleFactorChangedIncludingDescendants(); m_viewOverlayRootLayer->noteDeviceOrPageScaleFactorChangedIncludingDescendants(); for (auto& graphicsLayer : m_overlayGraphicsLayers.values()) graphicsLayer->setNeedsDisplay(); } void PageOverlayController::didChangeExposedRect() { m_mainFrame.page()->chrome().client().scheduleCompositingLayerFlush(); } void PageOverlayController::didScrollFrame(Frame& frame) { for (auto& overlayAndLayer : m_overlayGraphicsLayers) { if (overlayAndLayer.key->overlayType() == PageOverlay::OverlayType::View || !frame.isMainFrame()) overlayAndLayer.value->setNeedsDisplay(); overlayAndLayer.key->didScrollFrame(frame); } } void PageOverlayController::updateSettingsForLayer(GraphicsLayer& layer) { Settings& settings = m_mainFrame.settings(); layer.setAcceleratesDrawing(settings.acceleratedDrawingEnabled()); layer.setShowDebugBorder(settings.showDebugBorders()); layer.setShowRepaintCounter(settings.showRepaintCounter()); } bool PageOverlayController::handleMouseEvent(const PlatformMouseEvent& mouseEvent) { if (m_pageOverlays.isEmpty()) return false; for (auto it = m_pageOverlays.rbegin(), end = m_pageOverlays.rend(); it != end; ++it) { if ((*it)->mouseEvent(mouseEvent)) return true; } return false; } bool PageOverlayController::copyAccessibilityAttributeStringValueForPoint(String attribute, FloatPoint parameter, String& value) { if (m_pageOverlays.isEmpty()) return false; for (auto it = m_pageOverlays.rbegin(), end = m_pageOverlays.rend(); it != end; ++it) { if ((*it)->copyAccessibilityAttributeStringValueForPoint(attribute, parameter, value)) return true; } return false; } bool PageOverlayController::copyAccessibilityAttributeBoolValueForPoint(String attribute, FloatPoint parameter, bool& value) { if (m_pageOverlays.isEmpty()) return false; for (auto it = m_pageOverlays.rbegin(), end = m_pageOverlays.rend(); it != end; ++it) { if ((*it)->copyAccessibilityAttributeBoolValueForPoint(attribute, parameter, value)) return true; } return false; } Vector PageOverlayController::copyAccessibilityAttributesNames(bool parameterizedNames) { if (m_pageOverlays.isEmpty()) return { }; for (auto it = m_pageOverlays.rbegin(), end = m_pageOverlays.rend(); it != end; ++it) { Vector names = (*it)->copyAccessibilityAttributeNames(parameterizedNames); if (!names.isEmpty()) return names; } return { }; } void PageOverlayController::paintContents(const WebCore::GraphicsLayer* graphicsLayer, WebCore::GraphicsContext& graphicsContext, WebCore::GraphicsLayerPaintingPhase, const WebCore::FloatRect& clipRect) { for (auto& overlayAndGraphicsLayer : m_overlayGraphicsLayers) { if (overlayAndGraphicsLayer.value.get() != graphicsLayer) continue; GraphicsContextStateSaver stateSaver(graphicsContext); graphicsContext.clip(clipRect); overlayAndGraphicsLayer.key->drawRect(graphicsContext, enclosingIntRect(clipRect)); return; } } float PageOverlayController::deviceScaleFactor() const { if (Page* page = m_mainFrame.page()) return page->deviceScaleFactor(); return 1; } void PageOverlayController::notifyFlushRequired(const WebCore::GraphicsLayer*) { if (Page* page = m_mainFrame.page()) page->chrome().client().scheduleCompositingLayerFlush(); } void PageOverlayController::didChangeOverlayFrame(PageOverlay& overlay) { ASSERT(m_pageOverlays.contains(&overlay)); updateOverlayGeometry(overlay, *m_overlayGraphicsLayers.get(&overlay)); } void PageOverlayController::didChangeOverlayBackgroundColor(PageOverlay& overlay) { ASSERT(m_pageOverlays.contains(&overlay)); m_overlayGraphicsLayers.get(&overlay)->setBackgroundColor(overlay.backgroundColor()); } bool PageOverlayController::shouldSkipLayerInDump(const GraphicsLayer*, LayerTreeAsTextBehavior behavior) const { return !(behavior & LayerTreeAsTextIncludePageOverlayLayers); } } // namespace WebKit