| /* |
| * 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; |
| } |
| |