| /* |
| * Copyright (C) 2013 Intel Corporation. All rights reserved. |
| * Copyright (C) 2013 Samsung Electronics. 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 THE COPYRIGHT HOLDERS ``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 THE COPYRIGHT HOLDERS OR |
| * 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 "WebView.h" |
| |
| #if USE(COORDINATED_GRAPHICS) |
| #include "APIPageConfiguration.h" |
| #include "AcceleratedDrawingAreaProxy.h" |
| #include "ActivityState.h" |
| #include "CoordinatedGraphicsScene.h" |
| #include "CoordinatedLayerTreeHostProxy.h" |
| #include "DownloadManagerEfl.h" |
| #include "DownloadProxy.h" |
| #include "EwkView.h" |
| #include "InputMethodContextEfl.h" |
| #include "NativeWebMouseEvent.h" |
| #include "NotImplemented.h" |
| #include "WebBackForwardList.h" |
| #include "WebBackForwardListItem.h" |
| #include "WebContextMenuProxyEfl.h" |
| #include "WebPageGroup.h" |
| #include "WebPopupMenuProxyEfl.h" |
| #include "ewk_context_private.h" |
| #include <WebCore/PlatformContextCairo.h> |
| |
| #if ENABLE(FULLSCREEN_API) |
| #include "WebFullScreenManagerProxy.h" |
| #endif |
| |
| #if ENABLE(TOUCH_EVENTS) |
| #include "EwkTouchEvent.h" |
| #endif |
| |
| #if ENABLE(INPUT_TYPE_COLOR) |
| #include "WebColorPickerEfl.h" |
| #endif |
| |
| using namespace EwkViewCallbacks; |
| using namespace WebCore; |
| |
| namespace WebKit { |
| |
| Ref<WebView> WebView::create(API::PageConfiguration& pageConfiuration) |
| { |
| return adoptRef(*new WebView(pageConfiuration)); |
| } |
| |
| WebView::WebView(API::PageConfiguration& pageConfiguration) |
| : m_ewkView(0) |
| , m_focused(false) |
| , m_visible(false) |
| , m_hasRequestedFullScreen(false) |
| , m_opacity(1.0) |
| { |
| // Need to call createWebPage after other data members, specifically m_visible, are initialized. |
| m_page = pageConfiguration.processPool()->createWebPage(*this, pageConfiguration.copy()); |
| m_page->initializeWebPage(); |
| |
| m_page->pageGroup().preferences().setAcceleratedCompositingEnabled(true); |
| m_page->pageGroup().preferences().setForceCompositingMode(true); |
| |
| char* debugVisualsEnvironment = getenv("WEBKIT_SHOW_COMPOSITING_DEBUG_VISUALS"); |
| bool showDebugVisuals = debugVisualsEnvironment && !strcmp(debugVisualsEnvironment, "1"); |
| m_page->pageGroup().preferences().setCompositingBordersVisible(showDebugVisuals); |
| m_page->pageGroup().preferences().setCompositingRepaintCountersVisible(showDebugVisuals); |
| } |
| |
| WebView::~WebView() |
| { |
| if (m_page->isClosed()) |
| return; |
| |
| m_page->close(); |
| } |
| |
| void WebView::setEwkView(EwkView* ewkView) |
| { |
| m_ewkView = ewkView; |
| } |
| |
| void WebView::paintToCairoSurface(cairo_surface_t* surface) |
| { |
| CoordinatedGraphicsScene* scene = coordinatedGraphicsScene(); |
| if (!scene) |
| return; |
| |
| PlatformContextCairo context(cairo_create(surface)); |
| |
| const FloatPoint& position = contentPosition(); |
| double effectiveScale = m_page->deviceScaleFactor() * contentScaleFactor(); |
| |
| cairo_matrix_t transform = { effectiveScale, 0, 0, effectiveScale, - position.x() * m_page->deviceScaleFactor(), - position.y() * m_page->deviceScaleFactor() }; |
| cairo_set_matrix(context.cr(), &transform); |
| scene->paintToGraphicsContext(&context, m_page->pageExtendedBackgroundColor(), m_page->drawsBackground()); |
| } |
| |
| void WebView::setThemePath(const String& theme) |
| { |
| m_page->setThemePath(theme); |
| } |
| |
| #if ENABLE(TOUCH_EVENTS) |
| void WebView::sendTouchEvent(EwkTouchEvent* touchEvent) |
| { |
| ASSERT(touchEvent); |
| m_page->handleTouchEvent(NativeWebTouchEvent(touchEvent, transformFromScene())); |
| } |
| #endif |
| |
| void WebView::sendMouseEvent(const Evas_Event_Mouse_Down* event) |
| { |
| ASSERT(event); |
| m_page->handleMouseEvent(NativeWebMouseEvent(event, transformFromScene(), m_userViewportTransform.toAffineTransform())); |
| } |
| |
| void WebView::sendMouseEvent(const Evas_Event_Mouse_Up* event) |
| { |
| ASSERT(event); |
| m_page->handleMouseEvent(NativeWebMouseEvent(event, transformFromScene(), m_userViewportTransform.toAffineTransform())); |
| } |
| |
| void WebView::sendMouseEvent(const Evas_Event_Mouse_Move* event) |
| { |
| ASSERT(event); |
| m_page->handleMouseEvent(NativeWebMouseEvent(event, transformFromScene(), m_userViewportTransform.toAffineTransform())); |
| } |
| |
| void WebView::setViewBackgroundColor(const WebCore::Color& color) |
| { |
| CoordinatedGraphicsScene* scene = coordinatedGraphicsScene(); |
| if (!scene) |
| return; |
| |
| scene->setViewBackgroundColor(color); |
| } |
| |
| WebCore::Color WebView::viewBackgroundColor() |
| { |
| CoordinatedGraphicsScene* scene = coordinatedGraphicsScene(); |
| if (!scene) |
| return Color(); |
| |
| return scene->viewBackgroundColor(); |
| } |
| |
| void WebView::setContentScaleFactor(float scaleFactor) |
| { |
| m_page->scalePage(scaleFactor, roundedIntPoint(contentPosition())); |
| updateViewportSize(); |
| } |
| |
| void WebView::setActive(bool active) |
| { |
| CoordinatedGraphicsScene* scene = coordinatedGraphicsScene(); |
| if (!scene || scene->isActive() == active) |
| return; |
| |
| scene->setActive(active); |
| m_page->activityStateDidChange(ActivityState::WindowIsActive); |
| } |
| |
| void WebView::setSize(const WebCore::IntSize& size) |
| { |
| if (m_size == size) |
| return; |
| |
| m_size = size; |
| |
| updateViewportSize(); |
| } |
| |
| void WebView::setFocused(bool focused) |
| { |
| if (m_focused == focused) |
| return; |
| |
| m_focused = focused; |
| m_page->activityStateDidChange(ActivityState::IsFocused | ActivityState::WindowIsActive); |
| } |
| |
| void WebView::setVisible(bool visible) |
| { |
| if (m_visible == visible) |
| return; |
| |
| m_visible = visible; |
| m_page->activityStateDidChange(ActivityState::IsVisible); |
| |
| if (auto* drawingArea = static_cast<AcceleratedDrawingAreaProxy*>(page()->drawingArea())) |
| drawingArea->visibilityDidChange(); |
| } |
| |
| void WebView::setUserViewportTranslation(double tx, double ty) |
| { |
| m_userViewportTransform = TransformationMatrix().translate(tx, ty); |
| } |
| |
| IntPoint WebView::userViewportToContents(const IntPoint& point) const |
| { |
| return transformFromScene().mapPoint(point); |
| } |
| |
| IntPoint WebView::userViewportToScene(const WebCore::IntPoint& point) const |
| { |
| return m_userViewportTransform.mapPoint(point); |
| } |
| |
| IntPoint WebView::contentsToUserViewport(const IntPoint& point) const |
| { |
| return transformToScene().mapPoint(point); |
| } |
| |
| void WebView::paintToCurrentGLContext() |
| { |
| CoordinatedGraphicsScene* scene = coordinatedGraphicsScene(); |
| if (!scene) |
| return; |
| |
| const FloatRect& viewport = m_userViewportTransform.mapRect(IntRect(IntPoint(), m_size)); |
| |
| scene->paintToCurrentGLContext(transformToScene().toTransformationMatrix(), m_opacity, viewport, m_page->pageExtendedBackgroundColor(), m_page->drawsBackground(), m_contentPosition); |
| } |
| |
| void WebView::setDrawsBackground(bool drawsBackground) |
| { |
| m_page->setDrawsBackground(drawsBackground); |
| } |
| |
| bool WebView::drawsBackground() const |
| { |
| return m_page->drawsBackground(); |
| } |
| |
| void WebView::setDrawsTransparentBackground(bool transparentBackground) |
| { |
| m_page->setDrawsBackground(!transparentBackground); |
| } |
| |
| bool WebView::drawsTransparentBackground() const |
| { |
| return !m_page->drawsBackground(); |
| } |
| |
| void WebView::suspendActiveDOMObjectsAndAnimations() |
| { |
| m_page->suspendActiveDOMObjectsAndAnimations(); |
| } |
| |
| void WebView::resumeActiveDOMObjectsAndAnimations() |
| { |
| m_page->resumeActiveDOMObjectsAndAnimations(); |
| } |
| |
| #if ENABLE(FULLSCREEN_API) |
| WebFullScreenManagerProxyClient& WebView::fullScreenManagerProxyClient() |
| { |
| return *this; |
| } |
| #endif |
| |
| #if ENABLE(FULLSCREEN_API) |
| // WebFullScreenManagerProxyClient |
| bool WebView::isFullScreen() |
| { |
| return m_hasRequestedFullScreen; |
| } |
| |
| void WebView::enterFullScreen() |
| { |
| if (!m_ewkView || m_hasRequestedFullScreen) |
| return; |
| |
| m_hasRequestedFullScreen = true; |
| |
| WebFullScreenManagerProxy* manager = m_page->fullScreenManager(); |
| manager->willEnterFullScreen(); |
| m_ewkView->enterFullScreen(); |
| manager->didEnterFullScreen(); |
| } |
| |
| void WebView::exitFullScreen() |
| { |
| if (!m_ewkView || !m_hasRequestedFullScreen) |
| return; |
| |
| m_hasRequestedFullScreen = false; |
| |
| WebFullScreenManagerProxy* manager = m_page->fullScreenManager(); |
| manager->willExitFullScreen(); |
| m_ewkView->exitFullScreen(); |
| manager->didExitFullScreen(); |
| } |
| |
| bool WebView::requestExitFullScreen() |
| { |
| if (!isFullScreen()) |
| return false; |
| |
| m_page->fullScreenManager()->requestExitFullScreen(); |
| return true; |
| } |
| #endif |
| |
| void WebView::initializeClient(const WKViewClientBase* client) |
| { |
| m_client.initialize(client); |
| } |
| |
| void WebView::didChangeContentSize(const WebCore::IntSize& size) |
| { |
| if (m_contentsSize == size) |
| return; |
| |
| m_contentsSize = size; |
| m_client.didChangeContentsSize(this, size); |
| |
| updateViewportSize(); |
| } |
| |
| void WebView::didFindZoomableArea(const WebCore::IntPoint& target, const WebCore::IntRect& area) |
| { |
| m_client.didFindZoomableArea(this, target, area); |
| } |
| |
| AffineTransform WebView::transformFromScene() const |
| { |
| return transformToScene().inverse().value_or(AffineTransform()); |
| } |
| |
| AffineTransform WebView::transformToScene() const |
| { |
| FloatPoint position = -m_contentPosition; |
| float effectiveScale = m_page->deviceScaleFactor(); |
| if (m_page->useFixedLayout()) |
| effectiveScale *= contentScaleFactor(); |
| position.scale(effectiveScale, effectiveScale); |
| |
| TransformationMatrix transform = m_userViewportTransform; |
| transform.translate(position.x(), position.y()); |
| transform.scale(effectiveScale); |
| |
| return transform.toAffineTransform(); |
| } |
| |
| CoordinatedGraphicsScene* WebView::coordinatedGraphicsScene() |
| { |
| if (auto* drawingArea = static_cast<AcceleratedDrawingAreaProxy*>(page()->drawingArea())) |
| return drawingArea->coordinatedLayerTreeHostProxy().coordinatedGraphicsScene(); |
| |
| return nullptr; |
| } |
| |
| void WebView::updateViewportSize() |
| { |
| if (auto* drawingArea = static_cast<AcceleratedDrawingAreaProxy*>(page()->drawingArea())) { |
| // Web Process expects sizes in UI units, and not raw device units. |
| drawingArea->setSize(roundedIntSize(dipSize()), IntSize(), IntSize()); |
| FloatRect visibleContentsRect(contentPosition(), visibleContentsSize()); |
| visibleContentsRect.intersect(FloatRect(FloatPoint(), contentsSize())); |
| drawingArea->coordinatedLayerTreeHostProxy().setVisibleContentsRect(visibleContentsRect, FloatPoint()); |
| } |
| } |
| |
| inline WebCore::FloatSize WebView::dipSize() const |
| { |
| FloatSize dipSize(size()); |
| dipSize.scale(1 / m_page->deviceScaleFactor()); |
| |
| return dipSize; |
| } |
| |
| WebCore::FloatSize WebView::visibleContentsSize() const |
| { |
| FloatSize visibleContentsSize(dipSize()); |
| if (m_page->useFixedLayout()) |
| visibleContentsSize.scale(1 / contentScaleFactor()); |
| |
| return visibleContentsSize; |
| } |
| |
| // Page Client |
| |
| std::unique_ptr<DrawingAreaProxy> WebView::createDrawingAreaProxy() |
| { |
| return std::make_unique<AcceleratedDrawingAreaProxy>(*m_page); |
| } |
| |
| void WebView::setViewNeedsDisplay(const WebCore::Region& region) |
| { |
| for (const auto& rect : region.rects()) |
| m_client.viewNeedsDisplay(this, rect); |
| } |
| |
| void WebView::requestScroll(const WebCore::FloatPoint&, const WebCore::IntPoint&, bool) |
| { |
| notImplemented(); |
| } |
| |
| WebCore::IntSize WebView::viewSize() |
| { |
| return roundedIntSize(dipSize()); |
| } |
| |
| bool WebView::isActive() const |
| { |
| const CoordinatedGraphicsScene* scene = const_cast<WebView*>(this)->coordinatedGraphicsScene(); |
| if (!scene) |
| return false; |
| |
| return scene->isActive(); |
| } |
| |
| bool WebView::isViewWindowActive() |
| { |
| notImplemented(); |
| return true; |
| } |
| |
| bool WebView::isViewFocused() |
| { |
| return isFocused(); |
| } |
| |
| bool WebView::isViewVisible() |
| { |
| return isVisible(); |
| } |
| |
| bool WebView::isViewInWindow() |
| { |
| notImplemented(); |
| return true; |
| } |
| |
| void WebView::processDidExit() |
| { |
| m_client.webProcessCrashed(this, m_page->backForwardList().currentItem()->url()); |
| } |
| |
| void WebView::didRelaunchProcess() |
| { |
| m_client.webProcessDidRelaunch(this); |
| } |
| |
| void WebView::pageClosed() |
| { |
| notImplemented(); |
| } |
| |
| void WebView::preferencesDidChange() |
| { |
| notImplemented(); |
| } |
| |
| void WebView::toolTipChanged(const String&, const String& newToolTip) |
| { |
| m_client.didChangeTooltip(this, newToolTip); |
| } |
| |
| void WebView::didCommitLoadForMainFrame(const String&, bool) |
| { |
| setContentPosition(WebCore::FloatPoint()); |
| m_contentsSize = IntSize(); |
| } |
| |
| void WebView::setCursor(const WebCore::Cursor& cursor) |
| { |
| m_ewkView->setCursor(cursor); |
| } |
| |
| void WebView::setCursorHiddenUntilMouseMoves(bool) |
| { |
| notImplemented(); |
| } |
| |
| void WebView::registerEditCommand(PassRefPtr<WebEditCommandProxy> command, WebPageProxy::UndoOrRedo undoOrRedo) |
| { |
| m_undoController.registerEditCommand(command, undoOrRedo); |
| } |
| |
| void WebView::clearAllEditCommands() |
| { |
| m_undoController.clearAllEditCommands(); |
| } |
| |
| bool WebView::canUndoRedo(WebPageProxy::UndoOrRedo undoOrRedo) |
| { |
| return m_undoController.canUndoRedo(undoOrRedo); |
| } |
| |
| void WebView::executeUndoRedo(WebPageProxy::UndoOrRedo undoOrRedo) |
| { |
| m_undoController.executeUndoRedo(undoOrRedo); |
| } |
| |
| IntPoint WebView::screenToRootView(const IntPoint& point) |
| { |
| notImplemented(); |
| return point; |
| } |
| |
| IntRect WebView::rootViewToScreen(const IntRect&) |
| { |
| notImplemented(); |
| return IntRect(); |
| } |
| |
| void WebView::doneWithKeyEvent(const NativeWebKeyboardEvent&, bool) |
| { |
| notImplemented(); |
| } |
| |
| #if ENABLE(TOUCH_EVENTS) |
| void WebView::doneWithTouchEvent(const NativeWebTouchEvent& event, bool wasEventHandled) |
| { |
| m_client.doneWithTouchEvent(this, event, wasEventHandled); |
| } |
| #endif |
| |
| RefPtr<WebPopupMenuProxy> WebView::createPopupMenuProxy(WebPageProxy& page) |
| { |
| return WebPopupMenuProxyEfl::create(*m_ewkView, page); |
| } |
| |
| #if ENABLE(CONTEXT_MENUS) |
| std::unique_ptr<WebContextMenuProxy> WebView::createContextMenuProxy(WebPageProxy& page, const ContextMenuContextData& context, const UserData& userData) |
| { |
| return std::make_unique<WebContextMenuProxyEfl>(m_ewkView, page, context, userData); |
| } |
| #endif |
| |
| #if ENABLE(INPUT_TYPE_COLOR) |
| void WebView::initializeColorPickerClient(const WKColorPickerClientBase* client) |
| { |
| m_colorPickerClient.initialize(client); |
| } |
| |
| RefPtr<WebColorPicker> WebView::createColorPicker(WebPageProxy* page, const WebCore::Color& color, const WebCore::IntRect&) |
| { |
| return WebColorPickerEfl::create(this, page, color); |
| } |
| #endif |
| |
| void WebView::enterAcceleratedCompositingMode(const LayerTreeContext&) |
| { |
| setActive(true); |
| } |
| |
| void WebView::exitAcceleratedCompositingMode() |
| { |
| setActive(false); |
| } |
| |
| void WebView::updateAcceleratedCompositingMode(const LayerTreeContext&) |
| { |
| notImplemented(); |
| } |
| |
| void WebView::updateTextInputState() |
| { |
| if (InputMethodContextEfl* inputMethodContext = m_ewkView->inputMethodContext()) |
| inputMethodContext->updateTextInputState(); |
| } |
| |
| void WebView::handleDownloadRequest(DownloadProxy* download) |
| { |
| EwkContext* context = m_ewkView->ewkContext(); |
| context->downloadManager()->registerDownloadJob(toAPI(download)); |
| } |
| |
| FloatRect WebView::convertToDeviceSpace(const FloatRect& userRect) |
| { |
| if (m_page->useFixedLayout()) { |
| FloatRect result = userRect; |
| result.scale(m_page->deviceScaleFactor()); |
| return result; |
| } |
| // Legacy mode. |
| notImplemented(); |
| return userRect; |
| } |
| |
| FloatRect WebView::convertToUserSpace(const FloatRect& deviceRect) |
| { |
| if (m_page->useFixedLayout()) { |
| FloatRect result = deviceRect; |
| result.scale(1 / m_page->deviceScaleFactor()); |
| return result; |
| } |
| // Legacy mode. |
| notImplemented(); |
| return deviceRect; |
| } |
| |
| void WebView::didChangeViewportProperties(const WebCore::ViewportAttributes& attr) |
| { |
| m_client.didChangeViewportAttributes(this, attr); |
| } |
| |
| #if USE(COORDINATED_GRAPHICS_MULTIPROCESS) |
| void WebView::pageDidRequestScroll(const IntPoint& position) |
| { |
| FloatPoint uiPosition(position); |
| setContentPosition(uiPosition); |
| |
| m_client.didChangeContentsPosition(this, position); |
| } |
| |
| void WebView::didRenderFrame(const WebCore::IntSize& contentsSize, const WebCore::IntRect& coveredRect) |
| { |
| m_client.didRenderFrame(this, contentsSize, coveredRect); |
| } |
| |
| void WebView::pageTransitionViewportReady() |
| { |
| m_client.didCompletePageTransition(this); |
| } |
| |
| void WebView::findZoomableAreaForPoint(const IntPoint& point, const IntSize& size) |
| { |
| m_page->findZoomableAreaForPoint(transformFromScene().mapPoint(point), transformFromScene().mapSize(size)); |
| } |
| #endif |
| |
| } // namespace WebKit |
| |
| #endif // USE(COORDINATED_GRAPHICS) |
| |