/* * Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies) * Copyright (C) 2009 Girish Ramakrishnan <girish@forwardbias.in> * Copyright (C) 2006 George Staikos <staikos@kde.org> * Copyright (C) 2006 Dirk Mueller <mueller@kde.org> * Copyright (C) 2006 Zack Rusin <zack@kde.org> * Copyright (C) 2006 Simon Hausmann <hausmann@kde.org> * * 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 COMPUTER, INC. ``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 COMPUTER, INC. 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 "webview.h" #include <QtGui> #include <QGraphicsScene> WebViewGraphicsBased::WebViewGraphicsBased(QWidget* parent) : QGraphicsView(parent) , m_item(new GraphicsWebView) , m_numPaintsTotal(0) , m_numPaintsSinceLastMeasure(0) , m_measureFps(false) , m_resizesToContents(false) , m_machine(0) { setScene(new QGraphicsScene(this)); scene()->addItem(m_item); scene()->setFocusItem(m_item); setFrameShape(QFrame::NoFrame); setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); m_updateTimer = new QTimer(this); m_updateTimer->setInterval(1000); connect(m_updateTimer, SIGNAL(timeout()), this, SLOT(updateFrameRate())); } void WebViewGraphicsBased::setPage(QWebPage* page) { connect(page->mainFrame(), SIGNAL(contentsSizeChanged(const QSize&)), SLOT(contentsSizeChanged(const QSize&))); connect(page, SIGNAL(scrollRequested(int, int, const QRect&)), SLOT(scrollRequested(int, int))); graphicsWebView()->setPage(page); } void WebViewGraphicsBased::scrollRequested(int x, int y) { if (!m_resizesToContents) return; // Turn off interactive mode while scrolling, or QGraphicsView will replay the // last mouse event which may cause WebKit to initiate a drag operation. bool interactive = isInteractive(); setInteractive(false); verticalScrollBar()->setValue(-y); horizontalScrollBar()->setValue(-x); setInteractive(interactive); } void WebViewGraphicsBased::contentsSizeChanged(const QSize& size) { if (m_resizesToContents) scene()->setSceneRect(0, 0, size.width(), size.height()); } void WebViewGraphicsBased::setResizesToContents(bool b) { if (b == m_resizesToContents) return; m_resizesToContents = b; graphicsWebView()->setResizesToContents(m_resizesToContents); // When setting resizesToContents ON, our web view widget will always size as big as the // web content being displayed, and so will the QWebPage's viewport. It implies that internally // WebCore will work as if there was no content rendered offscreen, and then no scrollbars need // drawing. In order to keep scrolling working, we: // // 1) Set QGraphicsView's scrollbars policy back to 'auto'. // 2) Set scene's boundaries rect to an invalid size, which automatically makes it to be as big // as it needs to enclose all items onto it. We do that because QGraphicsView also calculates // the size of its scrollable area according to the amount of content in scene that is rendered // offscreen. // 3) Set QWebPage's preferredContentsSize according to the size of QGraphicsView's viewport, // so WebCore properly lays pages out. // // On the other hand, when toggling resizesToContents OFF, we set back the default values, as // opposite as described above. if (m_resizesToContents) { setHorizontalScrollBarPolicy(Qt::ScrollBarAsNeeded); setVerticalScrollBarPolicy(Qt::ScrollBarAsNeeded); graphicsWebView()->page()->setPreferredContentsSize(size()); QRectF itemRect(graphicsWebView()->geometry().topLeft(), graphicsWebView()->page()->mainFrame()->contentsSize()); graphicsWebView()->setGeometry(itemRect); scene()->setSceneRect(itemRect); } else { setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff); graphicsWebView()->page()->setPreferredContentsSize(QSize()); QRect viewportRect(QPoint(0, 0), size()); graphicsWebView()->setGeometry(viewportRect); scene()->setSceneRect(viewportRect); } } void WebViewGraphicsBased::resizeEvent(QResizeEvent* event) { QGraphicsView::resizeEvent(event); QSize size(event->size()); if (m_resizesToContents) { graphicsWebView()->page()->setPreferredContentsSize(size); return; } QRectF rect(QPoint(0, 0), size); graphicsWebView()->setGeometry(rect); scene()->setSceneRect(rect); } void WebViewGraphicsBased::setFrameRateMeasurementEnabled(bool enabled) { m_measureFps = enabled; if (m_measureFps) { m_lastConsultTime = m_startTime = QTime::currentTime(); m_fpsTimer.start(); m_updateTimer->start(); } else { m_fpsTimer.stop(); m_updateTimer->stop(); } } void WebViewGraphicsBased::updateFrameRate() { const QTime now = QTime::currentTime(); int interval = m_lastConsultTime.msecsTo(now); int frames = m_fpsTimer.numFrames(interval); int current = interval ? frames * 1000 / interval : 0; emit currentFPSUpdated(current); m_lastConsultTime = now; } void WebViewGraphicsBased::animatedFlip() { #ifndef QT_NO_ANIMATION QSizeF center = graphicsWebView()->boundingRect().size() / 2; QPointF centerPoint = QPointF(center.width(), center.height()); graphicsWebView()->setTransformOriginPoint(centerPoint); QPropertyAnimation* animation = new QPropertyAnimation(graphicsWebView(), "rotation", this); animation->setDuration(1000); int rotation = int(graphicsWebView()->rotation()); animation->setStartValue(rotation); animation->setEndValue(rotation + 180 - (rotation % 180)); animation->start(QAbstractAnimation::DeleteWhenStopped); #endif } void WebViewGraphicsBased::animatedYFlip() { #ifndef QT_NO_ANIMATION if (!m_machine) { m_machine = new QStateMachine(this); QState* s0 = new QState(m_machine); s0->assignProperty(this, "yRotation", 0); QState* s1 = new QState(m_machine); s1->assignProperty(this, "yRotation", 90); QAbstractTransition* t1 = s0->addTransition(s1); QPropertyAnimation* yRotationAnim = new QPropertyAnimation(this, "yRotation", this); t1->addAnimation(yRotationAnim); QState* s2 = new QState(m_machine); s2->assignProperty(this, "yRotation", -90); s1->addTransition(s1, SIGNAL(propertiesAssigned()), s2); QState* s3 = new QState(m_machine); s3->assignProperty(this, "yRotation", 0); QAbstractTransition* t2 = s2->addTransition(s3); t2->addAnimation(yRotationAnim); QFinalState* final = new QFinalState(m_machine); s3->addTransition(s3, SIGNAL(propertiesAssigned()), final); m_machine->setInitialState(s0); yRotationAnim->setDuration(1000); } m_machine->start(); #endif } void WebViewGraphicsBased::paintEvent(QPaintEvent* event) { QGraphicsView::paintEvent(event); if (!m_measureFps) return; } static QMenu* createContextMenu(QWebPage* page, QPoint position) { QMenu* menu = page->createStandardContextMenu(); QWebHitTestResult r = page->mainFrame()->hitTestContent(position); if (!r.linkUrl().isEmpty()) { WebPage* webPage = qobject_cast<WebPage*>(page); QAction* newTabAction = menu->addAction("Open in Default &Browser", webPage, SLOT(openUrlInDefaultBrowser())); newTabAction->setData(r.linkUrl()); menu->insertAction(menu->actions().at(2), newTabAction); } return menu; } void GraphicsWebView::mousePressEvent(QGraphicsSceneMouseEvent* event) { setProperty("mouseButtons", QVariant::fromValue(int(event->buttons()))); setProperty("keyboardModifiers", QVariant::fromValue(int(event->modifiers()))); QGraphicsWebView::mousePressEvent(event); } void WebViewTraditional::mousePressEvent(QMouseEvent* event) { setProperty("mouseButtons", QVariant::fromValue(int(event->buttons()))); setProperty("keyboardModifiers", QVariant::fromValue(int(event->modifiers()))); QWebView::mousePressEvent(event); } void GraphicsWebView::contextMenuEvent(QGraphicsSceneContextMenuEvent* event) { QMenu* menu = createContextMenu(page(), event->pos().toPoint()); menu->exec(event->screenPos()); delete menu; } void WebViewTraditional::contextMenuEvent(QContextMenuEvent* event) { QMenu* menu = createContextMenu(page(), event->pos()); menu->exec(event->globalPos()); delete menu; }