Rename WebKitTools to Tools
[WebKit-https.git] / Tools / QtTestBrowser / webview.cpp
1 /*
2  * Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies)
3  * Copyright (C) 2009 Girish Ramakrishnan <girish@forwardbias.in>
4  * Copyright (C) 2006 George Staikos <staikos@kde.org>
5  * Copyright (C) 2006 Dirk Mueller <mueller@kde.org>
6  * Copyright (C) 2006 Zack Rusin <zack@kde.org>
7  * Copyright (C) 2006 Simon Hausmann <hausmann@kde.org>
8  *
9  * All rights reserved.
10  *
11  * Redistribution and use in source and binary forms, with or without
12  * modification, are permitted provided that the following conditions
13  * are met:
14  * 1. Redistributions of source code must retain the above copyright
15  *    notice, this list of conditions and the following disclaimer.
16  * 2. Redistributions in binary form must reproduce the above copyright
17  *    notice, this list of conditions and the following disclaimer in the
18  *    documentation and/or other materials provided with the distribution.
19  *
20  * THIS SOFTWARE IS PROVIDED BY APPLE COMPUTER, INC. ``AS IS'' AND ANY
21  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
22  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
23  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE COMPUTER, INC. OR
24  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
25  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
26  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
27  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
28  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
29  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
30  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
31  */
32
33 #include "webview.h"
34
35 #include <QtGui>
36 #include <QGraphicsScene>
37
38 WebViewGraphicsBased::WebViewGraphicsBased(QWidget* parent)
39     : QGraphicsView(parent)
40     , m_item(new GraphicsWebView)
41     , m_numPaintsTotal(0)
42     , m_numPaintsSinceLastMeasure(0)
43     , m_measureFps(false)
44     , m_resizesToContents(false)
45     , m_machine(0)
46 {
47     setScene(new QGraphicsScene(this));
48     scene()->addItem(m_item);
49     scene()->setFocusItem(m_item);
50
51     setFrameShape(QFrame::NoFrame);
52     setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
53     setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
54
55     m_updateTimer = new QTimer(this);
56     m_updateTimer->setInterval(1000);
57     connect(m_updateTimer, SIGNAL(timeout()), this, SLOT(updateFrameRate()));
58 }
59
60 void WebViewGraphicsBased::setPage(QWebPage* page)
61 {
62     connect(page->mainFrame(), SIGNAL(contentsSizeChanged(const QSize&)), SLOT(contentsSizeChanged(const QSize&)));
63     connect(page, SIGNAL(scrollRequested(int, int, const QRect&)), SLOT(scrollRequested(int, int)));
64     graphicsWebView()->setPage(page);
65 }
66
67 void WebViewGraphicsBased::scrollRequested(int x, int y)
68 {
69     if (!m_resizesToContents)
70         return;
71
72     // Turn off interactive mode while scrolling, or QGraphicsView will replay the
73     // last mouse event which may cause WebKit to initiate a drag operation.
74     bool interactive = isInteractive();
75     setInteractive(false);
76
77     verticalScrollBar()->setValue(-y);
78     horizontalScrollBar()->setValue(-x);
79
80     setInteractive(interactive);
81 }
82
83 void WebViewGraphicsBased::contentsSizeChanged(const QSize& size)
84 {
85     if (m_resizesToContents)
86         scene()->setSceneRect(0, 0, size.width(), size.height());
87 }
88
89 void WebViewGraphicsBased::setResizesToContents(bool b)
90 {
91     if (b == m_resizesToContents)
92         return;
93
94     m_resizesToContents = b;
95     graphicsWebView()->setResizesToContents(m_resizesToContents);
96
97     // When setting resizesToContents ON, our web view widget will always size as big as the
98     // web content being displayed, and so will the QWebPage's viewport. It implies that internally
99     // WebCore will work as if there was no content rendered offscreen, and then no scrollbars need
100     // drawing. In order to keep scrolling working, we:
101     //
102     // 1) Set QGraphicsView's scrollbars policy back to 'auto'.
103     // 2) Set scene's boundaries rect to an invalid size, which automatically makes it to be as big
104     //    as it needs to enclose all items onto it. We do that because QGraphicsView also calculates
105     //    the size of its scrollable area according to the amount of content in scene that is rendered
106     //    offscreen.
107     // 3) Set QWebPage's preferredContentsSize according to the size of QGraphicsView's viewport,
108     //    so WebCore properly lays pages out.
109     //
110     // On the other hand, when toggling resizesToContents OFF, we set back the default values, as
111     // opposite as described above.
112     if (m_resizesToContents) {
113         setHorizontalScrollBarPolicy(Qt::ScrollBarAsNeeded);
114         setVerticalScrollBarPolicy(Qt::ScrollBarAsNeeded);
115         graphicsWebView()->page()->setPreferredContentsSize(size());
116         QRectF itemRect(graphicsWebView()->geometry().topLeft(), graphicsWebView()->page()->mainFrame()->contentsSize());
117         graphicsWebView()->setGeometry(itemRect);
118         scene()->setSceneRect(itemRect);
119     } else {
120         setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
121         setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
122         graphicsWebView()->page()->setPreferredContentsSize(QSize());
123         QRect viewportRect(QPoint(0, 0), size());
124         graphicsWebView()->setGeometry(viewportRect);
125         scene()->setSceneRect(viewportRect);
126     }
127 }
128
129 void WebViewGraphicsBased::resizeEvent(QResizeEvent* event)
130 {
131     QGraphicsView::resizeEvent(event);
132
133     QSize size(event->size());
134
135     if (m_resizesToContents) {
136         graphicsWebView()->page()->setPreferredContentsSize(size);
137         return;
138     }
139
140     QRectF rect(QPoint(0, 0), size);
141     graphicsWebView()->setGeometry(rect);
142     scene()->setSceneRect(rect);
143 }
144
145 void WebViewGraphicsBased::setFrameRateMeasurementEnabled(bool enabled)
146 {
147     m_measureFps = enabled;
148     if (m_measureFps) {
149         m_lastConsultTime = m_startTime = QTime::currentTime();
150         m_fpsTimer.start();
151         m_updateTimer->start();
152     } else {
153         m_fpsTimer.stop();
154         m_updateTimer->stop();
155     }
156 }
157
158 void WebViewGraphicsBased::updateFrameRate()
159 {
160     const QTime now = QTime::currentTime();
161     int interval = m_lastConsultTime.msecsTo(now);
162     int frames = m_fpsTimer.numFrames(interval);
163     int current = interval ? frames * 1000 / interval : 0;
164
165     emit currentFPSUpdated(current);
166
167     m_lastConsultTime = now;
168 }
169
170 void WebViewGraphicsBased::animatedFlip()
171 {
172     QSizeF center = graphicsWebView()->boundingRect().size() / 2;
173     QPointF centerPoint = QPointF(center.width(), center.height());
174     graphicsWebView()->setTransformOriginPoint(centerPoint);
175
176     QPropertyAnimation* animation = new QPropertyAnimation(graphicsWebView(), "rotation", this);
177     animation->setDuration(1000);
178
179     int rotation = int(graphicsWebView()->rotation());
180
181     animation->setStartValue(rotation);
182     animation->setEndValue(rotation + 180 - (rotation % 180));
183
184     animation->start(QAbstractAnimation::DeleteWhenStopped);
185 }
186
187 void WebViewGraphicsBased::animatedYFlip()
188 {
189     if (!m_machine) {
190         m_machine = new QStateMachine(this);
191
192         QState* s0 = new QState(m_machine);
193         s0->assignProperty(this, "yRotation", 0);
194
195         QState* s1 = new QState(m_machine);
196         s1->assignProperty(this, "yRotation", 90);
197
198         QAbstractTransition* t1 = s0->addTransition(s1);
199         QPropertyAnimation* yRotationAnim = new QPropertyAnimation(this, "yRotation", this);
200         t1->addAnimation(yRotationAnim);
201
202         QState* s2 = new QState(m_machine);
203         s2->assignProperty(this, "yRotation", -90);
204         s1->addTransition(s1, SIGNAL(propertiesAssigned()), s2);
205
206         QState* s3 = new QState(m_machine);
207         s3->assignProperty(this, "yRotation", 0);
208
209         QAbstractTransition* t2 = s2->addTransition(s3);
210         t2->addAnimation(yRotationAnim);
211
212         QFinalState* final = new QFinalState(m_machine);
213         s3->addTransition(s3, SIGNAL(propertiesAssigned()), final);
214
215         m_machine->setInitialState(s0);
216         yRotationAnim->setDuration(1000);
217     }
218
219     m_machine->start();
220 }
221
222 void WebViewGraphicsBased::paintEvent(QPaintEvent* event)
223 {
224     QGraphicsView::paintEvent(event);
225     if (!m_measureFps)
226         return;
227 }
228
229 static QMenu* createContextMenu(QWebPage* page, QPoint position)
230 {
231     QMenu* menu = page->createStandardContextMenu();
232
233     QWebHitTestResult r = page->mainFrame()->hitTestContent(position);
234
235     if (!r.linkUrl().isEmpty()) {
236         WebPage* webPage = qobject_cast<WebPage*>(page);
237         QAction* newTabAction = menu->addAction("Open in Default &Browser", webPage, SLOT(openUrlInDefaultBrowser()));
238         newTabAction->setData(r.linkUrl());
239         menu->insertAction(menu->actions().at(2), newTabAction);
240     }
241     return menu;
242 }
243
244 void GraphicsWebView::mousePressEvent(QGraphicsSceneMouseEvent* event)
245 {
246     setProperty("mouseButtons", QVariant::fromValue(int(event->buttons())));
247     setProperty("keyboardModifiers", QVariant::fromValue(int(event->modifiers())));
248
249     QGraphicsWebView::mousePressEvent(event);
250 }
251
252 void WebViewTraditional::mousePressEvent(QMouseEvent* event)
253 {
254     setProperty("mouseButtons", QVariant::fromValue(int(event->buttons())));
255     setProperty("keyboardModifiers", QVariant::fromValue(int(event->modifiers())));
256
257     QWebView::mousePressEvent(event);
258 }
259
260 void GraphicsWebView::contextMenuEvent(QGraphicsSceneContextMenuEvent* event)
261 {
262     QMenu* menu = createContextMenu(page(), event->pos().toPoint());
263     menu->exec(event->screenPos());
264     delete menu;
265 }
266
267 void WebViewTraditional::contextMenuEvent(QContextMenuEvent* event)
268 {
269     QMenu* menu = createContextMenu(page(), event->pos());
270     menu->exec(event->globalPos());
271     delete menu;
272 }
273