summaryrefslogtreecommitdiffstats
path: root/tests/auto/widgets/graphicsview/qgraphicsproxywidget/tst_qgraphicsproxywidget.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'tests/auto/widgets/graphicsview/qgraphicsproxywidget/tst_qgraphicsproxywidget.cpp')
-rw-r--r--tests/auto/widgets/graphicsview/qgraphicsproxywidget/tst_qgraphicsproxywidget.cpp3649
1 files changed, 3649 insertions, 0 deletions
diff --git a/tests/auto/widgets/graphicsview/qgraphicsproxywidget/tst_qgraphicsproxywidget.cpp b/tests/auto/widgets/graphicsview/qgraphicsproxywidget/tst_qgraphicsproxywidget.cpp
new file mode 100644
index 0000000000..78c545e25a
--- /dev/null
+++ b/tests/auto/widgets/graphicsview/qgraphicsproxywidget/tst_qgraphicsproxywidget.cpp
@@ -0,0 +1,3649 @@
+/****************************************************************************
+**
+** Copyright (C) 2011 Nokia Corporation and/or its subsidiary(-ies).
+** All rights reserved.
+** Contact: Nokia Corporation (qt-info@nokia.com)
+**
+** This file is part of the test suite of the Qt Toolkit.
+**
+** $QT_BEGIN_LICENSE:LGPL$
+** GNU Lesser General Public License Usage
+** This file may be used under the terms of the GNU Lesser General Public
+** License version 2.1 as published by the Free Software Foundation and
+** appearing in the file LICENSE.LGPL included in the packaging of this
+** file. Please review the following information to ensure the GNU Lesser
+** General Public License version 2.1 requirements will be met:
+** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
+**
+** In addition, as a special exception, Nokia gives you certain additional
+** rights. These rights are described in the Nokia Qt LGPL Exception
+** version 1.1, included in the file LGPL_EXCEPTION.txt in this package.
+**
+** GNU General Public License Usage
+** Alternatively, this file may be used under the terms of the GNU General
+** Public License version 3.0 as published by the Free Software Foundation
+** and appearing in the file LICENSE.GPL included in the packaging of this
+** file. Please review the following information to ensure the GNU General
+** Public License version 3.0 requirements will be met:
+** http://www.gnu.org/copyleft/gpl.html.
+**
+** Other Usage
+** Alternatively, this file may be used in accordance with the terms and
+** conditions contained in a signed written agreement between you and Nokia.
+**
+**
+**
+**
+**
+** $QT_END_LICENSE$
+**
+****************************************************************************/
+
+
+#include <QtTest/QtTest>
+#include <QtGui>
+#include <QtWidgets>
+#include <private/qgraphicsproxywidget_p.h>
+#include <private/qlayoutengine_p.h> // qSmartMin functions...
+#if defined(Q_WS_MAC) && !defined(QT_NO_STYLE_MAC)
+#include <QMacStyle>
+#endif
+#ifdef Q_WS_X11
+#include <private/qt_x11_p.h>
+#endif
+
+static void sendMouseMove(QWidget *widget, const QPoint &point, Qt::MouseButton button = Qt::NoButton)
+{
+ QMouseEvent event(QEvent::MouseMove, point, widget->mapToGlobal(point), button, button, 0);
+ QApplication::sendEvent(widget, &event);
+}
+
+/*
+ Notes:
+
+ 1) The proxy and the widget geometries are linked.
+ proxy resize => widget resize => stop (no livelock)
+ widget resize => proxy resize => stop (no livelock)
+
+ 2) As far as possible, the properties are linked.
+ proxy enable => widget enable => stop
+ widget disabled => proxy disabled => stop
+
+ 3) Windowed state is linked
+ Windowed proxy state => windowed widget state => stop
+ Windowed widget state => windowed proxy state => stop
+*/
+
+class EventSpy : public QObject
+{
+public:
+ EventSpy(QObject *receiver)
+ {
+ receiver->installEventFilter(this);
+ }
+
+ QMap<QEvent::Type, int> counts;
+
+protected:
+ bool eventFilter(QObject *, QEvent *event)
+ {
+ ++counts[event->type()];
+ return false;
+ }
+};
+
+class tst_QGraphicsProxyWidget : public QObject
+{
+ Q_OBJECT
+
+public slots:
+ void initTestCase();
+ void cleanupTestCase();
+ void init();
+ void cleanup();
+
+private slots:
+ void qgraphicsproxywidget_data();
+ void qgraphicsproxywidget();
+ void paint();
+ void paint_2();
+ void setWidget_data();
+ void setWidget();
+ void eventFilter_data();
+ void eventFilter();
+ void focusInEvent_data();
+ void focusInEvent();
+ void focusInEventNoWidget();
+ void focusNextPrevChild_data();
+ void focusNextPrevChild();
+ void focusOutEvent_data();
+ void focusOutEvent();
+#if !defined(Q_OS_WINCE) || (defined(GWES_ICONCURS) && !defined(QT_NO_CURSOR))
+ void hoverEnterLeaveEvent_data();
+ void hoverEnterLeaveEvent();
+#endif
+ void hoverMoveEvent_data();
+ void hoverMoveEvent();
+ void keyPressEvent_data();
+ void keyPressEvent();
+ void keyReleaseEvent_data();
+ void keyReleaseEvent();
+ void mouseDoubleClickEvent_data();
+ void mouseDoubleClickEvent();
+ void mousePressReleaseEvent_data();
+ void mousePressReleaseEvent();
+ void resizeEvent_data();
+ void resizeEvent();
+ void paintEvent();
+ void wheelEvent();
+ void sizeHint_data();
+ void sizeHint();
+ void sizePolicy();
+ void minimumSize();
+ void maximumSize();
+ void scrollUpdate();
+ void setWidget_simple();
+ void setWidget_ownership();
+ void resize_simple_data();
+ void resize_simple();
+ void symmetricMove();
+ void symmetricResize();
+ void symmetricEnabled();
+ void symmetricVisible();
+ void tabFocus_simpleWidget();
+ void tabFocus_simpleTwoWidgets();
+ void tabFocus_complexWidget();
+ void tabFocus_complexTwoWidgets();
+ void setFocus_simpleWidget();
+ void setFocus_simpleTwoWidgets();
+ void setFocus_complexTwoWidgets();
+ void popup_basic();
+ void popup_subwidget();
+#if !defined(QT_NO_CURSOR) && (!defined(Q_OS_WINCE) || defined(GWES_ICONCURS))
+ void changingCursor_basic();
+#endif
+ void tooltip_basic();
+ void childPos_data();
+ void childPos();
+ void autoShow();
+ void windowOpacity();
+ void stylePropagation();
+ void palettePropagation();
+ void fontPropagation();
+ void dontCrashWhenDie();
+ void createProxyForChildWidget();
+ void actionsContextMenu();
+ void actionsContextMenu_data();
+ void deleteProxyForChildWidget();
+ void bypassGraphicsProxyWidget_data();
+ void bypassGraphicsProxyWidget();
+ void dragDrop();
+ void windowFlags_data();
+ void windowFlags();
+ void comboboxWindowFlags();
+ void updateAndDelete();
+ void inputMethod();
+ void clickFocus();
+ void windowFrameMargins();
+ void QTBUG_6986_sendMouseEventToAlienWidget();
+};
+
+// Subclass that exposes the protected functions.
+class SubQGraphicsProxyWidget : public QGraphicsProxyWidget
+{
+
+public:
+ SubQGraphicsProxyWidget(QGraphicsItem *parent = 0) : QGraphicsProxyWidget(parent),
+ paintCount(0), keyPress(0), focusOut(0)
+ {}
+
+ bool call_eventFilter(QObject* object, QEvent* event)
+ { return SubQGraphicsProxyWidget::eventFilter(object, event); }
+
+ void call_focusInEvent(QFocusEvent* event)
+ { return SubQGraphicsProxyWidget::focusInEvent(event); }
+
+ bool call_focusNextPrevChild(bool next)
+ { return SubQGraphicsProxyWidget::focusNextPrevChild(next); }
+
+ void call_focusOutEvent(QFocusEvent* event)
+ { return SubQGraphicsProxyWidget::focusOutEvent(event); }
+
+ void call_hideEvent(QHideEvent* event)
+ { return SubQGraphicsProxyWidget::hideEvent(event); }
+
+ void call_hoverEnterEvent(QGraphicsSceneHoverEvent* event)
+ { return SubQGraphicsProxyWidget::hoverEnterEvent(event); }
+
+ void call_hoverLeaveEvent(QGraphicsSceneHoverEvent* event)
+ { return SubQGraphicsProxyWidget::hoverLeaveEvent(event); }
+
+ void call_hoverMoveEvent(QGraphicsSceneHoverEvent* event)
+ { return SubQGraphicsProxyWidget::hoverMoveEvent(event); }
+
+ void call_keyPressEvent(QKeyEvent* event)
+ { return SubQGraphicsProxyWidget::keyPressEvent(event); }
+
+ void call_keyReleaseEvent(QKeyEvent* event)
+ { return SubQGraphicsProxyWidget::keyReleaseEvent(event); }
+
+ void call_mouseDoubleClickEvent(QGraphicsSceneMouseEvent* event)
+ { return SubQGraphicsProxyWidget::mouseDoubleClickEvent(event); }
+
+ void call_mouseMoveEvent(QGraphicsSceneMouseEvent* event)
+ { return SubQGraphicsProxyWidget::mouseMoveEvent(event); }
+
+ void call_mousePressEvent(QGraphicsSceneMouseEvent* event)
+ { return SubQGraphicsProxyWidget::mousePressEvent(event); }
+
+ void call_mouseReleaseEvent(QGraphicsSceneMouseEvent* event)
+ { return SubQGraphicsProxyWidget::mouseReleaseEvent(event); }
+
+ void call_resizeEvent(QGraphicsSceneResizeEvent* event)
+ { return SubQGraphicsProxyWidget::resizeEvent(event); }
+
+ QSizeF call_sizeHint(Qt::SizeHint which, QSizeF const& constraint = QSizeF()) const
+ { return SubQGraphicsProxyWidget::sizeHint(which, constraint); }
+
+ void call_showEvent(QShowEvent* event)
+ { return SubQGraphicsProxyWidget::showEvent(event); }
+
+ void paint (QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget = 0) {
+ paintCount++;
+ QGraphicsProxyWidget::paint(painter, option, widget);
+ }
+
+ void focusOutEvent(QFocusEvent *event)
+ {
+ focusOut++;
+ QGraphicsProxyWidget::focusOutEvent(event);
+ }
+
+ bool eventFilter(QObject *object, QEvent *event) {
+ if (event->type() == QEvent::KeyPress && object == widget())
+ keyPress++;
+ return QGraphicsProxyWidget::eventFilter(object, event);
+ }
+ int paintCount;
+ int keyPress;
+ int focusOut;
+};
+
+class WheelWidget : public QWidget
+{
+public:
+ WheelWidget() : wheelEventCalled(false) { setFocusPolicy(Qt::WheelFocus); }
+
+ virtual void wheelEvent(QWheelEvent *event) { event->accept(); wheelEventCalled = true; }
+
+ bool wheelEventCalled;
+};
+
+// This will be called before the first test function is executed.
+// It is only called once.
+void tst_QGraphicsProxyWidget::initTestCase()
+{
+#ifdef Q_OS_WINCE //disable magic for WindowsCE
+ qApp->setAutoMaximizeThreshold(-1);
+#endif
+}
+
+// This will be called after the last test function is executed.
+// It is only called once.
+void tst_QGraphicsProxyWidget::cleanupTestCase()
+{
+}
+
+// This will be called before each test function is executed.
+void tst_QGraphicsProxyWidget::init()
+{
+}
+
+// This will be called after every test function.
+void tst_QGraphicsProxyWidget::cleanup()
+{
+}
+
+void tst_QGraphicsProxyWidget::qgraphicsproxywidget_data()
+{
+}
+
+void tst_QGraphicsProxyWidget::qgraphicsproxywidget()
+{
+ SubQGraphicsProxyWidget proxy;
+ proxy.paint(0, 0, 0);
+ proxy.setWidget(0);
+ QVERIFY(proxy.type() == QGraphicsProxyWidget::Type);
+ QVERIFY(!proxy.widget());
+ QEvent event(QEvent::None);
+ proxy.call_eventFilter(0, &event);
+ QFocusEvent focusEvent(QEvent::FocusIn);
+ focusEvent.ignore();
+ proxy.call_focusInEvent(&focusEvent);
+ QCOMPARE(focusEvent.isAccepted(), false);
+ QCOMPARE(proxy.call_focusNextPrevChild(false), false);
+ QCOMPARE(proxy.call_focusNextPrevChild(true), false);
+ proxy.call_focusOutEvent(&focusEvent);
+ QHideEvent hideEvent;
+ proxy.call_hideEvent(&hideEvent);
+ QGraphicsSceneHoverEvent hoverEvent;
+ proxy.call_hoverEnterEvent(&hoverEvent);
+ proxy.call_hoverLeaveEvent(&hoverEvent);
+ proxy.call_hoverMoveEvent(&hoverEvent);
+ QKeyEvent keyEvent(QEvent::KeyPress, 0, Qt::NoModifier);
+ proxy.call_keyPressEvent(&keyEvent);
+ proxy.call_keyReleaseEvent(&keyEvent);
+ QGraphicsSceneMouseEvent mouseEvent;
+ proxy.call_mouseDoubleClickEvent(&mouseEvent);
+ proxy.call_mouseMoveEvent(&mouseEvent);
+ proxy.call_mousePressEvent(&mouseEvent);
+ proxy.call_mouseReleaseEvent(&mouseEvent);
+ QGraphicsSceneResizeEvent resizeEvent;
+ proxy.call_resizeEvent(&resizeEvent);
+ QShowEvent showEvent;
+ proxy.call_showEvent(&showEvent);
+ proxy.call_sizeHint(Qt::PreferredSize, QSizeF());
+}
+
+// public void paint(QPainter* painter, QStyleOptionGraphicsItem const* option, QWidget* widget)
+void tst_QGraphicsProxyWidget::paint()
+{
+ SubQGraphicsProxyWidget proxy;
+ proxy.paint(0, 0, 0);
+}
+
+class MyProxyWidget : public QGraphicsProxyWidget
+{
+public:
+ void paint(QPainter *painter, const QStyleOptionGraphicsItem *option, QWidget *widget)
+ {
+ // Make sure QGraphicsProxyWidget::paint does not modify the render hints set on the painter.
+ painter->setRenderHints(QPainter::Antialiasing | QPainter::SmoothPixmapTransform
+ | QPainter::NonCosmeticDefaultPen | QPainter::TextAntialiasing);
+ const QPainter::RenderHints oldRenderHints = painter->renderHints();
+ QGraphicsProxyWidget::paint(painter, option, widget);
+ QCOMPARE(painter->renderHints(), oldRenderHints);
+ }
+};
+
+void tst_QGraphicsProxyWidget::paint_2()
+{
+ MyProxyWidget *proxyWidget = new MyProxyWidget;
+ proxyWidget->setWidget(new QLineEdit);
+
+ QGraphicsScene scene;
+ scene.addItem(proxyWidget);
+ scene.setSceneRect(scene.itemsBoundingRect());
+
+ // Trigger repaint.
+ QPixmap pixmap(scene.sceneRect().toRect().size());
+ QPainter painter(&pixmap);
+ scene.render(&painter);
+}
+
+void tst_QGraphicsProxyWidget::setWidget_data()
+{
+ QTest::addColumn<bool>("widgetExists");
+ QTest::addColumn<bool>("insertWidget");
+ QTest::addColumn<bool>("hasParent");
+ QTest::addColumn<bool>("proxyHasParent");
+
+ QTest::newRow("setWidget(0)") << false << false << false << false;
+ QTest::newRow("setWidget(widget)") << false << true << false << false;
+ QTest::newRow("setWidget(widgetWParent)") << false << true << true << false;
+ QTest::newRow("setWidget(1), setWidget(0)") << true << false << false << false;
+ QTest::newRow("setWidget(1), setWidget(widget)") << true << true << false << false;
+ QTest::newRow("setWidget(1), setWidget(widgetWParent)") << true << true << true << false;
+ QTest::newRow("p setWidget(0)") << false << false << false << true;
+ QTest::newRow("p setWidget(widget)") << false << true << false << true;
+ QTest::newRow("p setWidget(widgetWParent)") << false << true << true << true;
+ QTest::newRow("p setWidget(1), setWidget(0)") << true << false << false << true;
+ QTest::newRow("p setWidget(1), setWidget(widget)") << true << true << false << true;
+ QTest::newRow("p setWidget(1), setWidget(widgetWParent)") << true << true << true << true;
+}
+
+// public void setWidget(QWidget* widget)
+void tst_QGraphicsProxyWidget::setWidget()
+{
+ QFETCH(bool, widgetExists);
+ QFETCH(bool, insertWidget);
+ QFETCH(bool, hasParent);
+ QFETCH(bool, proxyHasParent);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+#ifdef Q_WS_X11
+ qt_x11_wait_for_window_manager(&view);
+#endif
+ QPointer<SubQGraphicsProxyWidget> proxy = new SubQGraphicsProxyWidget;
+ SubQGraphicsProxyWidget parentProxy;
+ scene.addItem(proxy);
+ scene.addItem(&parentProxy);
+ if (proxyHasParent)
+ proxy->setParent(&parentProxy);
+ QPointer<QWidget> existingSubWidget = new QWidget;
+ proxy->setVisible(false);
+ proxy->setEnabled(false);
+
+ if (widgetExists) {
+ existingSubWidget->setAttribute(Qt::WA_QuitOnClose, true);
+ proxy->setWidget(existingSubWidget);
+ }
+
+ QWidget *widget = new QWidget;
+#ifndef QT_NO_CURSOR
+ widget->setCursor(Qt::IBeamCursor);
+#endif
+ widget->setPalette(QPalette(Qt::magenta));
+ widget->setLayoutDirection(Qt::RightToLeft);
+ QCleanlooksStyle cleanlooksStyle;
+ widget->setStyle(&cleanlooksStyle);
+ widget->setFont(QFont("Times"));
+ widget->setVisible(true);
+ QApplication::setActiveWindow(widget);
+ widget->activateWindow();
+ widget->setEnabled(true);
+ widget->resize(325, 241);
+ widget->setMinimumSize(100, 200);
+ widget->setMaximumSize(1000, 2000);
+ widget->setFocusPolicy(Qt::TabFocus);
+ widget->setContentsMargins(10, 29, 19, 81);
+ widget->setFocus(Qt::TabFocusReason);
+ widget->setAcceptDrops(true);
+ QTRY_VERIFY(widget->hasFocus());
+ QVERIFY(widget->isActiveWindow());
+
+ QWidget parentWidget;
+ if (hasParent)
+ widget->setParent(&parentWidget);
+
+ QWidget *subWidget = insertWidget ? widget : 0;
+ bool shouldBeInsertable = !hasParent && subWidget;
+ if (shouldBeInsertable)
+ subWidget->setAttribute(Qt::WA_QuitOnClose, true);
+
+ proxy->setWidget(subWidget);
+
+ if (shouldBeInsertable) {
+ QCOMPARE(proxy->widget(), subWidget);
+ QVERIFY(subWidget->testAttribute(Qt::WA_DontShowOnScreen));
+ QVERIFY(!subWidget->testAttribute(Qt::WA_QuitOnClose));
+ QCOMPARE(proxy->acceptHoverEvents(), true);
+#ifndef QT_NO_CURSOR
+ QVERIFY(proxy->hasCursor());
+
+ // These should match
+ QCOMPARE(proxy->cursor().shape(), widget->cursor().shape());
+#endif
+ //###QCOMPARE(proxy->palette(), widget->palette());
+ QCOMPARE(proxy->layoutDirection(), widget->layoutDirection());
+ QCOMPARE(proxy->style(), widget->style());
+ QCOMPARE(proxy->isVisible(), widget->isVisible());
+ QCOMPARE(proxy->isEnabled(), widget->isEnabled());
+ QVERIFY(proxy->flags() & QGraphicsItem::ItemIsFocusable);
+ QCOMPARE(proxy->effectiveSizeHint(Qt::MinimumSize).toSize(),
+ qSmartMinSize(widget));
+ QCOMPARE(proxy->minimumSize().toSize(),
+ qSmartMinSize(widget) );
+ QCOMPARE(proxy->maximumSize().toSize(), QSize(1000, 2000));
+ QCOMPARE(proxy->effectiveSizeHint(Qt::PreferredSize).toSize(),
+ qSmartMinSize(widget));
+ QCOMPARE(proxy->size().toSize(), widget->size());
+ QCOMPARE(proxy->rect().toRect(), widget->rect());
+ QCOMPARE(proxy->focusPolicy(), Qt::WheelFocus);
+ QVERIFY(proxy->acceptDrops());
+ QCOMPARE(proxy->acceptsHoverEvents(), true); // to get widget enter events
+ int left, top, right, bottom;
+ widget->getContentsMargins(&left, &top, &right, &bottom);
+ qreal rleft, rtop, rright, rbottom;
+ proxy->getContentsMargins(&rleft, &rtop, &rright, &rbottom);
+ QCOMPARE((qreal)left, rleft);
+ QCOMPARE((qreal)top, rtop);
+ QCOMPARE((qreal)right, rright);
+ QCOMPARE((qreal)bottom, rbottom);
+ } else {
+ // proxy shouldn't mess with the widget if it can't insert it.
+ QCOMPARE(proxy->widget(), (QWidget*)0);
+ QCOMPARE(proxy->acceptHoverEvents(), false);
+ if (subWidget) {
+ QVERIFY(!subWidget->testAttribute(Qt::WA_DontShowOnScreen));
+ QVERIFY(subWidget->testAttribute(Qt::WA_QuitOnClose));
+ // reset
+ subWidget->setAttribute(Qt::WA_QuitOnClose, false);
+ }
+ }
+
+ if (widgetExists) {
+ QCOMPARE(existingSubWidget->parent(), static_cast<QObject*>(0));
+ QVERIFY(!existingSubWidget->testAttribute(Qt::WA_DontShowOnScreen));
+ QVERIFY(!existingSubWidget->testAttribute(Qt::WA_QuitOnClose));
+ }
+
+ if (hasParent)
+ widget->setParent(0);
+
+ delete widget;
+ if (shouldBeInsertable)
+ QVERIFY(!proxy);
+ delete existingSubWidget;
+ if (!shouldBeInsertable) {
+ QVERIFY(proxy);
+ delete proxy;
+ }
+ QVERIFY(!proxy);
+}
+
+Q_DECLARE_METATYPE(QEvent::Type)
+void tst_QGraphicsProxyWidget::eventFilter_data()
+{
+ QTest::addColumn<QEvent::Type>("eventType");
+ QTest::addColumn<bool>("fromObject"); // big grin evil
+ QTest::newRow("none") << QEvent::None << false;
+ for (int i = 0; i < 2; ++i) {
+ bool fromObject = (i == 0);
+ QTest::newRow(QString("resize %1").arg(fromObject).toLatin1()) << QEvent::Resize << fromObject;
+ QTest::newRow(QString("move %1").arg(fromObject).toLatin1()) << QEvent::Move << fromObject;
+ QTest::newRow(QString("hide %1").arg(fromObject).toLatin1()) << QEvent::Hide << fromObject;
+ QTest::newRow(QString("show %1").arg(fromObject).toLatin1()) << QEvent::Show << fromObject;
+ QTest::newRow(QString("enabled %1").arg(fromObject).toLatin1()) << QEvent::EnabledChange << fromObject;
+ QTest::newRow(QString("focusIn %1").arg(fromObject).toLatin1()) << QEvent::FocusIn << fromObject;
+ QTest::newRow(QString("focusOut %1").arg(fromObject).toLatin1()) << QEvent::FocusOut << fromObject;
+ QTest::newRow(QString("keyPress %1").arg(fromObject).toLatin1()) << QEvent::KeyPress << fromObject;
+ }
+}
+
+// protected bool eventFilter(QObject* object, QEvent* event)
+void tst_QGraphicsProxyWidget::eventFilter()
+{
+ QFETCH(QEvent::Type, eventType);
+ QFETCH(bool, fromObject);
+
+ QGraphicsScene scene;
+ QEvent windowActivate(QEvent::WindowActivate);
+ qApp->sendEvent(&scene, &windowActivate);
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ scene.addItem(proxy);
+
+ QWidget *widget = new QWidget(0, Qt::FramelessWindowHint);
+ widget->setFocusPolicy(Qt::TabFocus);
+ widget->resize(10, 10);
+ widget->show();
+ proxy->setWidget(widget);
+ proxy->show();
+
+ // mirror whatever is happening to the widget
+ // don't get in a loop
+ switch (eventType) {
+ case QEvent::None: {
+ QEvent event(QEvent::None);
+ proxy->call_eventFilter(widget, &event);
+ break;
+ }
+ case QEvent::Resize: {
+ QSize oldSize = widget->size();
+ QSize newSize = QSize(100, 100);
+ if (fromObject) {
+ widget->resize(newSize);
+ } else {
+ proxy->resize(newSize);
+ }
+ QCOMPARE(proxy->size().toSize(), newSize);
+ QCOMPARE(widget->size(), newSize);
+ break;
+ }
+ case QEvent::Move: {
+ QPoint oldPoint = widget->pos();
+ QPoint newPoint = QPoint(100, 100);
+ if (fromObject) {
+ widget->move(newPoint);
+ } else {
+ proxy->setPos(newPoint);
+ }
+ QCOMPARE(proxy->pos().toPoint(), newPoint);
+ QCOMPARE(widget->pos(), newPoint);
+ break;
+ }
+ case QEvent::Hide: {
+ // A hide event can only come from a widget
+ if (fromObject) {
+ widget->setFocus(Qt::TabFocusReason);
+ widget->hide();
+ } else {
+ QHideEvent event;
+ proxy->call_eventFilter(widget, &event);
+ }
+ QCOMPARE(proxy->isVisible(), false);
+ break;
+ }
+ case QEvent::Show: {
+ // A show event can either come from a widget or somewhere else
+ widget->hide();
+ if (fromObject) {
+ widget->show();
+ } else {
+ QShowEvent event;
+ proxy->call_eventFilter(widget, &event);
+ }
+ QCOMPARE(proxy->isVisible(), true);
+ break;
+ }
+ case QEvent::EnabledChange: {
+ widget->setEnabled(false);
+ proxy->setEnabled(false);
+ if (fromObject) {
+ widget->setEnabled(true);
+ QCOMPARE(proxy->isEnabled(), true);
+ widget->setEnabled(false);
+ QCOMPARE(proxy->isEnabled(), false);
+ } else {
+ QEvent event(QEvent::EnabledChange);
+ proxy->call_eventFilter(widget, &event);
+ // match the widget not the event
+ QCOMPARE(proxy->isEnabled(), false);
+ }
+ break;
+ }
+ case QEvent::FocusIn: {
+ if (fromObject) {
+ widget->setFocus(Qt::TabFocusReason);
+ QVERIFY(proxy->hasFocus());
+ }
+ break;
+ }
+ case QEvent::FocusOut: {
+ widget->setFocus(Qt::TabFocusReason);
+ QVERIFY(proxy->hasFocus());
+ QVERIFY(widget->hasFocus());
+ if (fromObject) {
+ widget->clearFocus();
+ QVERIFY(!proxy->hasFocus());
+ }
+ break;
+ }
+ case QEvent::KeyPress: {
+ if (fromObject) {
+ QTest::keyPress(widget, Qt::Key_A, Qt::NoModifier);
+ } else {
+ QKeyEvent event(QEvent::KeyPress, Qt::Key_A, Qt::NoModifier);
+ proxy->call_eventFilter(widget, &event);
+ }
+ QCOMPARE(proxy->keyPress, 1);
+ break;
+ }
+ default:
+ break;
+ }
+}
+
+void tst_QGraphicsProxyWidget::focusInEvent_data()
+{
+ QTest::addColumn<bool>("widgetHasFocus");
+ QTest::addColumn<bool>("widgetCanHaveFocus");
+ QTest::newRow("no focus, can't get") << false << false;
+ QTest::newRow("no focus, can get") << false << true;
+ QTest::newRow("has focus, can't get") << true << false;
+ QTest::newRow("has focus, can get") << true << true;
+ // ### add test for widget having a focusNextPrevChild
+}
+
+// protected void focusInEvent(QFocusEvent* event)
+void tst_QGraphicsProxyWidget::focusInEvent()
+{
+ // ### This test is just plain old broken
+ QFETCH(bool, widgetHasFocus);
+ QFETCH(bool, widgetCanHaveFocus);
+
+ QGraphicsScene scene;
+ QEvent windowActivate(QEvent::WindowActivate);
+ qApp->sendEvent(&scene, &windowActivate);
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ proxy->setEnabled(true);
+ scene.addItem(proxy);
+ proxy->setVisible(true);
+
+ QWidget *widget = new QWidget;
+ widget->resize(100, 100);
+ if (widgetCanHaveFocus)
+ widget->setFocusPolicy(Qt::WheelFocus);
+ widget->show();
+
+ if (widgetHasFocus)
+ widget->setFocus(Qt::TabFocusReason);
+
+ proxy->setWidget(widget);
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, true); // <- shouldn't need to do this
+
+ // ### This test is just plain old broken - sending a focus in event
+ // does not cause items to gain input focus. The widget has focus
+ // because the proxy has focus, not because it got this event.
+
+ QFocusEvent event(QEvent::FocusIn, Qt::TabFocusReason);
+ event.ignore();
+ proxy->call_focusInEvent(&event);
+ QTRY_COMPARE(widget->hasFocus(), widgetCanHaveFocus);
+}
+
+void tst_QGraphicsProxyWidget::focusInEventNoWidget()
+{
+ QGraphicsView view;
+ QGraphicsScene scene(&view);
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ proxy->setEnabled(true);
+ scene.addItem(proxy);
+ proxy->setVisible(true);
+ view.show();
+
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, true); // <- shouldn't need to do this
+ QFocusEvent event(QEvent::FocusIn, Qt::TabFocusReason);
+ event.ignore();
+ //should not crash
+ proxy->call_focusInEvent(&event);
+}
+
+void tst_QGraphicsProxyWidget::focusNextPrevChild_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::addColumn<bool>("hasScene");
+ QTest::addColumn<bool>("next");
+ QTest::addColumn<bool>("focusNextPrevChild");
+
+ for (int i = 0; i < 2; ++i) {
+ for (int j = 0; j < 2; ++j) {
+ for (int k = 0; k < 2; ++k) {
+ bool next = (i == 0);
+ bool hasWidget = (j == 0);
+ bool hasScene = (k == 0);
+ bool result = hasScene && hasWidget;
+ QString name = QString("Forward: %1, hasWidget: %2, hasScene: %3, result: %4").arg(next).arg(hasWidget).arg(hasScene).arg(result);
+ QTest::newRow(name.toLatin1()) << hasWidget << hasScene << next << result;
+ }
+ }
+ }
+}
+
+// protected bool focusNextPrevChild(bool next)
+void tst_QGraphicsProxyWidget::focusNextPrevChild()
+{
+ QFETCH(bool, next);
+ QFETCH(bool, focusNextPrevChild);
+ QFETCH(bool, hasWidget);
+ QFETCH(bool, hasScene);
+
+ // If a widget has its own focusNextPrevChild we need to respect it
+ // otherwise respect the scene
+ // Respect the widget over the scene!
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+
+ QLabel *widget = new QLabel;
+ // I can't believe designer adds this much junk!
+ widget->setText("<html><head><meta name=\"qrichtext\" content=\"1\" /><style type=\"text/css\"> p, li { white-space: pre-wrap; } </style></head><body style=\" font-family:'Sans Serif'; font-size:9pt; font-weight:400; font-style:normal;\"> <p style=\" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;\"><a href=\"http://www.slashdot.org\"><span style=\" text-decoration: underline; color:#0000ff;\">old</span></a> foo <a href=\"http://www.reddit.org\"><span style=\" text-decoration: underline; color:#0000ff;\">new</span></a></p></body></html>");
+ widget->setTextInteractionFlags(Qt::TextBrowserInteraction);
+
+ if (hasWidget)
+ proxy->setWidget(widget);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+ QApplication::setActiveWindow(&view);
+ QTest::qWaitForWindowShown(&view);
+ QTRY_COMPARE(QApplication::activeWindow(), (QWidget*)&view);
+ if (hasScene) {
+ scene.addItem(proxy);
+ proxy->show();
+
+ // widget should take precedence over scene so make scene.focusNextPrevChild return false
+ // so we know that a true can only come from the widget
+ if (!(hasWidget && hasScene)) {
+ QGraphicsTextItem *item = new QGraphicsTextItem("Foo");
+ item->setTextInteractionFlags(Qt::TextBrowserInteraction);
+ scene.addItem(item);
+ item->setPos(50, 40);
+ }
+ scene.setFocusItem(proxy);
+ QVERIFY(proxy->hasFocus());
+ }
+
+ QCOMPARE(proxy->call_focusNextPrevChild(next), focusNextPrevChild);
+
+ if (!hasScene)
+ delete proxy;
+}
+
+void tst_QGraphicsProxyWidget::focusOutEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::addColumn<bool>("call");
+ QTest::newRow("no widget, focus to other widget") << false << false;
+ QTest::newRow("no widget, focusOutCalled") << false << true;
+ QTest::newRow("widget, focus to other widget") << true << false;
+ QTest::newRow("widget, focusOutCalled") << true << true;
+}
+
+// protected void focusOutEvent(QFocusEvent* event)
+void tst_QGraphicsProxyWidget::focusOutEvent()
+{
+ QFETCH(bool, hasWidget);
+ QFETCH(bool, call);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ scene.addItem(proxy);
+ view.show();
+ QApplication::setActiveWindow(&view);
+ view.activateWindow();
+ view.setFocus();
+ QTest::qWaitForWindowShown(&view);
+ QTRY_VERIFY(view.isVisible());
+ QTRY_COMPARE(QApplication::activeWindow(), (QWidget*)&view);
+
+ QWidget *widget = new QWidget;
+ widget->setFocusPolicy(Qt::WheelFocus);
+ if (hasWidget)
+ proxy->setWidget(widget);
+ proxy->show();
+ proxy->setFocus();
+ QVERIFY(proxy->hasFocus());
+ QEXPECT_FAIL("widget, focus to other widget", "Widget should have focus but doesn't", Continue);
+ QEXPECT_FAIL("widget, focusOutCalled", "Widget should have focus but doesn't", Continue);
+ QCOMPARE(widget->hasFocus(), hasWidget);
+
+ if (!call) {
+ QWidget *other = new QLineEdit(&view);
+ other->show();
+ QApplication::processEvents();
+ QTRY_VERIFY(other->isVisible());
+ other->setFocus();
+ QTRY_VERIFY(other->hasFocus());
+ qApp->processEvents();
+ QTRY_COMPARE(proxy->hasFocus(), false);
+ QVERIFY(proxy->focusOut);
+ QCOMPARE(widget->hasFocus(), false);
+ } else {
+ {
+ /*
+ ### Test doesn't make sense
+ QFocusEvent focusEvent(QEvent::FocusOut);
+ proxy->call_focusOutEvent(&focusEvent);
+ QCOMPARE(focusEvent.isAccepted(), hasWidget);
+ qApp->processEvents();
+ QCOMPARE(proxy->paintCount, hasWidget ? 1 : 0);
+ */
+ }
+ {
+ /*
+ ### Test doesn't make sense
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, false);
+ QFocusEvent focusEvent(QEvent::FocusOut);
+ proxy->call_focusOutEvent(&focusEvent);
+ QCOMPARE(focusEvent.isAccepted(), hasWidget);
+ qApp->processEvents();
+ QCOMPARE(proxy->paintCount, 0);
+ */
+ }
+ }
+}
+
+class EventLogger : public QWidget
+{
+public:
+ EventLogger() : QWidget(), enterCount(0), leaveCount(0), moveCount(0),
+ hoverEnter(0), hoverLeave(0), hoverMove(0)
+ {
+ installEventFilter(this);
+ }
+
+ void enterEvent(QEvent *event)
+ {
+ enterCount++;
+ QWidget::enterEvent(event);
+ }
+
+ void leaveEvent(QEvent *event )
+ {
+ leaveCount++;
+ QWidget::leaveEvent(event);
+ }
+
+ void mouseMoveEvent(QMouseEvent *event)
+ {
+ event->setAccepted(true);
+ moveCount++;
+ QWidget::mouseMoveEvent(event);
+ }
+
+ int enterCount;
+ int leaveCount;
+ int moveCount;
+
+ int hoverEnter;
+ int hoverLeave;
+ int hoverMove;
+protected:
+ bool eventFilter(QObject *object, QEvent *event)
+ {
+ switch (event->type()) {
+ case QEvent::HoverEnter:
+ hoverEnter++;
+ break;
+ case QEvent::HoverLeave:
+ hoverLeave++;
+ break;
+ case QEvent::HoverMove:
+ hoverMove++;
+ break;
+ default:
+ break;
+ }
+ return QWidget::eventFilter(object, event);
+ }
+};
+
+// protected void hoverEnterEvent(QGraphicsSceneHoverEvent* event)
+#if !defined(Q_OS_WINCE) || (defined(GWES_ICONCURS) && !defined(QT_NO_CURSOR))
+void tst_QGraphicsProxyWidget::hoverEnterLeaveEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::addColumn<bool>("hoverEnabled");
+ QTest::newRow("widget, no hover") << true << false;
+ QTest::newRow("no widget, no hover") << false << false;
+ QTest::newRow("widget, hover") << true << true;
+ QTest::newRow("no widget, hover") << false << true;
+}
+
+void tst_QGraphicsProxyWidget::hoverEnterLeaveEvent()
+{
+ QFETCH(bool, hasWidget);
+ QFETCH(bool, hoverEnabled);
+
+ // proxy should translate this into events that the widget would expect
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ //do not let the window manager move the window while we are moving the mouse on it
+ view.setWindowFlags(Qt::X11BypassWindowManagerHint);
+ view.show();
+#ifdef Q_WS_X11
+ qt_x11_wait_for_window_manager(&view);
+#endif
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ EventLogger *widget = new EventLogger;
+ widget->resize(50, 50);
+ widget->setAttribute(Qt::WA_Hover, hoverEnabled);
+ widget->setMouseTracking(true);
+ view.resize(100, 100);
+ if (hasWidget)
+ proxy->setWidget(widget);
+ proxy->setPos(50, 0);
+ scene.addItem(proxy);
+ QTest::qWait(30);
+ QTest::mouseMove(&view, QPoint(10, 10));
+ QTest::qWait(30);
+
+ // in
+ QTest::mouseMove(&view, QPoint(50, 50));
+ QTRY_COMPARE(widget->testAttribute(Qt::WA_UnderMouse), hasWidget ? true : false);
+ // ### this attribute isn't supported
+ QCOMPARE(widget->enterCount, hasWidget ? 1 : 0);
+ QCOMPARE(widget->hoverEnter, (hasWidget && hoverEnabled) ? 1 : 0);
+ // does not work on all platforms
+ //QCOMPARE(widget->moveCount, 0);
+
+ // out
+ QTest::mouseMove(&view, QPoint(10, 10));
+ // QTRY_COMPARE(widget->testAttribute(Qt::WA_UnderMouse), false);
+ // ### this attribute isn't supported
+ QTRY_COMPARE(widget->leaveCount, hasWidget ? 1 : 0);
+ QTRY_COMPARE(widget->hoverLeave, (hasWidget && hoverEnabled) ? 1 : 0);
+ // does not work on all platforms
+ //QCOMPARE(widget->moveCount, 0);
+
+ if (!hasWidget)
+ delete widget;
+}
+#endif
+
+void tst_QGraphicsProxyWidget::hoverMoveEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::addColumn<bool>("hoverEnabled");
+ QTest::addColumn<bool>("mouseTracking");
+ QTest::addColumn<bool>("mouseDown");
+ for (int i = 0; i < 2; ++i) {
+ for (int j = 0; j < 2; ++j) {
+ for (int k = 0; k < 2; ++k) {
+ for (int l = 0; l < 2; ++l) {
+ bool hasWidget = (i == 0);
+ bool hoverEnabled = (j == 0);
+ bool mouseTracking = (k == 0);
+ bool mouseDown = (l == 0);
+ QString name = QString("hasWidget:%1, hover:%2, mouseTracking:%3, mouseDown: %4").arg(hasWidget).arg(hoverEnabled).arg(mouseTracking).arg(mouseDown);
+ QTest::newRow(name.toLatin1()) << hasWidget << hoverEnabled << mouseTracking << mouseDown;
+ }
+ }
+ }
+ }
+}
+
+// protected void hoverMoveEvent(QGraphicsSceneHoverEvent* event)
+void tst_QGraphicsProxyWidget::hoverMoveEvent()
+{
+ QFETCH(bool, hasWidget);
+ QFETCH(bool, hoverEnabled);
+ QFETCH(bool, mouseTracking);
+ QFETCH(bool, mouseDown);
+
+ QSKIP("Ambiguous test...", SkipAll);
+
+ // proxy should translate the move events to what the widget would expect
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, true); // ### remove me!!!
+ EventLogger *widget = new EventLogger;
+ widget->resize(50, 50);
+ widget->setAttribute(Qt::WA_Hover, hoverEnabled);
+ if (mouseTracking)
+ widget->setMouseTracking(true);
+ view.resize(100, 100);
+ if (hasWidget)
+ proxy->setWidget(widget);
+ proxy->setPos(50, 0);
+ scene.addItem(proxy);
+
+ // in
+ QTest::mouseMove(&view, QPoint(50, 50));
+ QTest::qWait(12);
+
+ if (mouseDown)
+ QTest::mousePress(view.viewport(), Qt::LeftButton);
+
+ // move a little bit
+ QTest::mouseMove(&view, QPoint(60, 60));
+ QTRY_COMPARE(widget->hoverEnter, (hasWidget && hoverEnabled) ? 1 : 0);
+ QCOMPARE(widget->moveCount, (hasWidget && mouseTracking) || (hasWidget && mouseDown) ? 1 : 0);
+
+ if (!hasWidget)
+ delete widget;
+}
+
+void tst_QGraphicsProxyWidget::keyPressEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::newRow("widget") << true;
+ QTest::newRow("no widget") << false;
+}
+
+// protected void keyPressEvent(QKeyEvent* event)
+void tst_QGraphicsProxyWidget::keyPressEvent()
+{
+ QFETCH(bool, hasWidget);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+ view.viewport()->setFocus();
+ QApplication::setActiveWindow(&view);
+ QTest::qWaitForWindowShown(&view);
+ QTRY_COMPARE(QApplication::activeWindow(), (QWidget*)&view);
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, true); // ### remove me!!!
+
+ QLineEdit *widget = new QLineEdit;
+ widget->resize(50, 50);
+ view.resize(100, 100);
+ if (hasWidget) {
+ proxy->setWidget(widget);
+ proxy->show();
+ }
+ proxy->setPos(50, 0);
+ scene.addItem(proxy);
+ proxy->setFocus();
+
+ QTest::keyPress(view.viewport(), 'x');
+
+ QTRY_COMPARE(widget->text(), hasWidget ? QString("x") : QString());
+
+ if (!hasWidget)
+ delete widget;
+}
+
+void tst_QGraphicsProxyWidget::keyReleaseEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::newRow("widget") << true;
+ QTest::newRow("no widget") << false;
+}
+
+// protected void keyReleaseEvent(QKeyEvent* event)
+void tst_QGraphicsProxyWidget::keyReleaseEvent()
+{
+ QFETCH(bool, hasWidget);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+ QApplication::setActiveWindow(&view);
+ QTest::qWaitForWindowShown(&view);
+ QTRY_COMPARE(QApplication::activeWindow(), (QWidget*)&view);
+
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, true); // ### remove me!!!
+ QPushButton *widget = new QPushButton;
+ QSignalSpy spy(widget, SIGNAL(clicked()));
+ widget->resize(50, 50);
+ view.resize(100, 100);
+ if (hasWidget) {
+ proxy->setWidget(widget);
+ proxy->show();
+ }
+ proxy->setPos(50, 0);
+ scene.addItem(proxy);
+ proxy->setFocus();
+
+ QTest::keyPress(view.viewport(), Qt::Key_Space);
+ QTRY_COMPARE(spy.count(), 0);
+ QTest::keyRelease(view.viewport(), Qt::Key_Space);
+ QTRY_COMPARE(spy.count(), (hasWidget) ? 1 : 0);
+
+ if (!hasWidget)
+ delete widget;
+}
+
+void tst_QGraphicsProxyWidget::mouseDoubleClickEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::newRow("widget") << true;
+ QTest::newRow("no widget") << false;
+}
+
+// protected void mouseDoubleClickEvent(QGraphicsSceneMouseEvent* event)
+void tst_QGraphicsProxyWidget::mouseDoubleClickEvent()
+{
+ QFETCH(bool, hasWidget);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+
+ QApplication::setActiveWindow(&view);
+ QTest::qWaitForWindowShown(&view);
+ QTRY_COMPARE(QApplication::activeWindow(), (QWidget*)&view);
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, true); // ### remove me!!!
+ QLineEdit *widget = new QLineEdit;
+ widget->setText("foo");
+ widget->resize(50, 50);
+ view.resize(100, 100);
+ if (hasWidget) {
+ proxy->setWidget(widget);
+ proxy->show();
+ }
+ proxy->setPos(50, 0);
+ scene.addItem(proxy);
+ proxy->setFocus();
+
+ QTest::mouseMove(view.viewport(), view.mapFromScene(proxy->mapToScene(15, proxy->boundingRect().center().y())));
+ QTest::mousePress(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(proxy->mapToScene(15, proxy->boundingRect().center().y())));
+ QTest::mouseRelease(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(proxy->mapToScene(15, proxy->boundingRect().center().y())));
+ QTest::mouseDClick(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(proxy->mapToScene(15, proxy->boundingRect().center().y())));
+ QTest::mouseRelease(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(proxy->mapToScene(15, proxy->boundingRect().center().y())));
+
+ QTRY_COMPARE(widget->selectedText(), hasWidget ? QString("foo") : QString());
+
+ if (!hasWidget)
+ delete widget;
+}
+
+void tst_QGraphicsProxyWidget::mousePressReleaseEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::newRow("widget") << true;
+ QTest::newRow("no widget") << false;
+}
+
+// protected void mousePressEvent(QGraphicsSceneMouseEvent* event)
+void tst_QGraphicsProxyWidget::mousePressReleaseEvent()
+{
+ QFETCH(bool, hasWidget);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+ QTest::qWaitForWindowShown(&view);
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ proxy->setFlag(QGraphicsItem::ItemIsFocusable, true); // ### remove me!!!
+ QPushButton *widget = new QPushButton;
+ QSignalSpy spy(widget, SIGNAL(clicked()));
+ widget->resize(50, 50);
+ view.resize(100, 100);
+ if (hasWidget) {
+ proxy->setWidget(widget);
+ proxy->show();
+ }
+ proxy->setPos(50, 0);
+ scene.addItem(proxy);
+ proxy->setFocus();
+
+ QTest::mousePress(view.viewport(), Qt::LeftButton, 0,
+ view.mapFromScene(proxy->mapToScene(proxy->boundingRect().center())));
+ QTRY_COMPARE(spy.count(), 0);
+ QTest::mouseRelease(view.viewport(), Qt::LeftButton, 0,
+ view.mapFromScene(proxy->mapToScene(proxy->boundingRect().center())));
+ QTRY_COMPARE(spy.count(), (hasWidget) ? 1 : 0);
+
+ if (!hasWidget)
+ delete widget;
+}
+
+void tst_QGraphicsProxyWidget::resizeEvent_data()
+{
+ QTest::addColumn<bool>("hasWidget");
+ QTest::newRow("widget") << true;
+ QTest::newRow("no widget") << false;
+}
+
+// protected void resizeEvent(QGraphicsSceneResizeEvent* event)
+void tst_QGraphicsProxyWidget::resizeEvent()
+{
+ QFETCH(bool, hasWidget);
+
+ SubQGraphicsProxyWidget proxy;
+ QWidget *widget = new QWidget;
+ if (hasWidget)
+ proxy.setWidget(widget);
+
+ QSize newSize(100, 100);
+ QGraphicsSceneResizeEvent event;
+ event.setOldSize(QSize(10, 10));
+ event.setNewSize(newSize);
+ proxy.call_resizeEvent(&event);
+ if (hasWidget)
+ QCOMPARE(widget->size(), newSize);
+ if (!hasWidget)
+ delete widget;
+}
+
+void tst_QGraphicsProxyWidget::paintEvent()
+{
+ //we test that calling update on a widget inside a QGraphicsView is triggering a repaint
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+ QApplication::setActiveWindow(&view);
+ QTest::qWaitForWindowShown(&view);
+ QTRY_VERIFY(view.isActiveWindow());
+
+ SubQGraphicsProxyWidget proxy;
+
+ QWidget *w = new QWidget;
+ //showing the widget here seems to create a bug in Graphics View
+ //this bug prevents the widget from being updated
+
+ w->show();
+ QTest::qWaitForWindowShown(w);
+ QApplication::processEvents();
+ QTest::qWait(30);
+ proxy.setWidget(w);
+ scene.addItem(&proxy);
+
+ //make sure we flush all the paint events
+ QTRY_VERIFY(proxy.paintCount > 1);
+ QTest::qWait(30);
+ proxy.paintCount = 0;
+
+ w->update();
+ QTRY_COMPARE(proxy.paintCount, 1); //the widget should have been painted now
+}
+
+
+void tst_QGraphicsProxyWidget::wheelEvent()
+{
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+#ifdef Q_WS_X11
+ qt_x11_wait_for_window_manager(&view);
+#endif
+
+ WheelWidget *wheelWidget = new WheelWidget();
+ wheelWidget->setFixedSize(400, 400);
+
+ QGraphicsProxyWidget *proxy = scene.addWidget(wheelWidget);
+ proxy->setVisible(true);
+
+ QGraphicsSceneWheelEvent event(QEvent::GraphicsSceneWheel);
+ event.setScenePos(QPoint(50, 50));
+ event.setAccepted(false);
+ wheelWidget->wheelEventCalled = false;
+
+ QApplication::sendEvent(&scene, &event);
+
+ QVERIFY(event.isAccepted());
+ QVERIFY(wheelWidget->wheelEventCalled);
+}
+
+Q_DECLARE_METATYPE(Qt::SizeHint)
+void tst_QGraphicsProxyWidget::sizeHint_data()
+{
+ QTest::addColumn<Qt::SizeHint>("which");
+ QTest::addColumn<QSizeF>("constraint");
+ QTest::addColumn<QSizeF>("sizeHint");
+ QTest::addColumn<bool>("hasWidget");
+
+ for (int i = 0; i < 2; ++i) {
+ bool hasWidget = (i == 0);
+ // ### What should these do?
+ QTest::newRow("min") << Qt::MinimumSize << QSizeF() << QSizeF() << hasWidget;
+ QTest::newRow("pre") << Qt::PreferredSize << QSizeF() << QSizeF() << hasWidget;
+ QTest::newRow("max") << Qt::MaximumSize << QSizeF() << QSizeF() << hasWidget;
+ QTest::newRow("mindes") << Qt::MinimumDescent << QSizeF() << QSizeF() << hasWidget;
+ QTest::newRow("nsize") << Qt::NSizeHints << QSizeF() << QSizeF() << hasWidget;
+ }
+}
+
+// protected QSizeF sizeHint(Qt::SizeHint which, QSizeF const& constraint = QSizeF()) const
+void tst_QGraphicsProxyWidget::sizeHint()
+{
+ QFETCH(Qt::SizeHint, which);
+ QFETCH(QSizeF, constraint);
+ QFETCH(QSizeF, sizeHint);
+ QFETCH(bool, hasWidget);
+ QSKIP("Broken test", SkipAll);
+ SubQGraphicsProxyWidget proxy;
+ QWidget *widget = new QWidget;
+ if (hasWidget)
+ proxy.setWidget(widget);
+ QCOMPARE(proxy.call_sizeHint(which, constraint), sizeHint);
+ if (!hasWidget)
+ delete widget;
+}
+
+void tst_QGraphicsProxyWidget::sizePolicy()
+{
+ for (int p = 0; p < 2; ++p) {
+ bool hasWidget = (p == 0 ? true : false);
+ SubQGraphicsProxyWidget proxy;
+ QWidget *widget = new QWidget;
+ QSizePolicy proxyPol(QSizePolicy::Maximum, QSizePolicy::Expanding);
+ proxy.setSizePolicy(proxyPol);
+ QSizePolicy widgetPol(QSizePolicy::Fixed, QSizePolicy::Minimum);
+ widget->setSizePolicy(widgetPol);
+
+ QCOMPARE(proxy.sizePolicy(), proxyPol);
+ QCOMPARE(widget->sizePolicy(), widgetPol);
+ if (hasWidget) {
+ proxy.setWidget(widget);
+ QCOMPARE(proxy.sizePolicy(), widgetPol);
+ } else {
+ QCOMPARE(proxy.sizePolicy(), proxyPol);
+ }
+ QCOMPARE(widget->sizePolicy(), widgetPol);
+
+ proxy.setSizePolicy(widgetPol);
+ widget->setSizePolicy(proxyPol);
+ if (hasWidget)
+ QCOMPARE(proxy.sizePolicy(), proxyPol);
+ else
+ QCOMPARE(proxy.sizePolicy(), widgetPol);
+ }
+}
+
+void tst_QGraphicsProxyWidget::minimumSize()
+{
+ SubQGraphicsProxyWidget proxy;
+ QWidget *widget = new QWidget;
+ QSize minSize(50, 50);
+ widget->setMinimumSize(minSize);
+ proxy.resize(30, 30);
+ widget->resize(30,30);
+ QCOMPARE(proxy.size(), QSizeF(30, 30));
+ proxy.setWidget(widget);
+ QCOMPARE(proxy.size().toSize(), minSize);
+ QCOMPARE(proxy.minimumSize().toSize(), minSize);
+ widget->setMinimumSize(70, 70);
+ QCOMPARE(proxy.minimumSize(), QSizeF(70, 70));
+ QCOMPARE(proxy.size(), QSizeF(70, 70));
+}
+
+void tst_QGraphicsProxyWidget::maximumSize()
+{
+ SubQGraphicsProxyWidget proxy;
+ QWidget *widget = new QWidget;
+ QSize maxSize(150, 150);
+ widget->setMaximumSize(maxSize);
+ proxy.resize(200, 200);
+ widget->resize(200,200);
+ QCOMPARE(proxy.size(), QSizeF(200, 200));
+ proxy.setWidget(widget);
+ QCOMPARE(proxy.size().toSize(), maxSize);
+ QCOMPARE(proxy.maximumSize().toSize(), maxSize);
+ widget->setMaximumSize(70, 70);
+ QCOMPARE(proxy.maximumSize(), QSizeF(70, 70));
+ QCOMPARE(proxy.size(), QSizeF(70, 70));
+}
+
+class View : public QGraphicsView
+{
+public:
+ View(QGraphicsScene *scene, QWidget *parent = 0)
+ : QGraphicsView(scene, parent), npaints(0)
+ { }
+ QRegion paintEventRegion;
+ int npaints;
+protected:
+ void paintEvent(QPaintEvent *event)
+ {
+ ++npaints;
+ paintEventRegion += event->region();
+ QGraphicsView::paintEvent(event);
+ }
+};
+
+class ScrollWidget : public QWidget
+{
+ Q_OBJECT
+public:
+ ScrollWidget() : npaints(0)
+ {
+ resize(200, 200);
+ }
+ QRegion paintEventRegion;
+ int npaints;
+
+public slots:
+ void updateScroll()
+ {
+ update(0, 0, 200, 10);
+ scroll(0, 10, QRect(0, 0, 100, 20));
+ }
+
+protected:
+ void paintEvent(QPaintEvent *event)
+ {
+ ++npaints;
+ paintEventRegion += event->region();
+ QPainter painter(this);
+ painter.fillRect(event->rect(), Qt::blue);
+ }
+};
+
+void tst_QGraphicsProxyWidget::scrollUpdate()
+{
+ ScrollWidget *widget = new ScrollWidget;
+
+ QGraphicsScene scene;
+ scene.addWidget(widget);
+
+ View view(&scene);
+ view.show();
+ QTest::qWaitForWindowShown(&view);
+ QTRY_VERIFY(view.npaints >= 1);
+ QTest::qWait(20);
+ widget->paintEventRegion = QRegion();
+ widget->npaints = 0;
+ view.paintEventRegion = QRegion();
+ view.npaints = 0;
+ QTimer::singleShot(0, widget, SLOT(updateScroll()));
+ QTRY_COMPARE(view.npaints, 2);
+ // QRect(0, 0, 200, 12) is the first update, expanded (-2, -2, 2, 2)
+ // QRect(0, 12, 102, 10) is the scroll update, expanded (-2, -2, 2, 2),
+ // intersected with the above update.
+ QCOMPARE(view.paintEventRegion.rects(),
+ QVector<QRect>() << QRect(0, 0, 200, 12) << QRect(0, 12, 102, 10));
+ QCOMPARE(widget->npaints, 2);
+ QCOMPARE(widget->paintEventRegion.rects(),
+ QVector<QRect>() << QRect(0, 0, 200, 12) << QRect(0, 12, 102, 10));
+}
+
+void tst_QGraphicsProxyWidget::setWidget_simple()
+{
+ QGraphicsProxyWidget proxy;
+ QLineEdit *lineEdit = new QLineEdit;
+ proxy.setWidget(lineEdit);
+
+ QVERIFY(lineEdit->testAttribute(Qt::WA_DontShowOnScreen));
+ // Size hints
+ // ### size hints are tested in a different test
+ // QCOMPARE(proxy.effectiveSizeHint(Qt::MinimumSize).toSize(), lineEdit->minimumSizeHint());
+ // QCOMPARE(proxy.effectiveSizeHint(Qt::MaximumSize).toSize(), lineEdit->maximumSize());
+ // QCOMPARE(proxy.effectiveSizeHint(Qt::PreferredSize).toSize(), lineEdit->sizeHint());
+ QCOMPARE(proxy.size().toSize(), lineEdit->minimumSizeHint().expandedTo(lineEdit->size()));
+ QRect rect = lineEdit->rect();
+ rect.setSize(rect.size().expandedTo(lineEdit->minimumSizeHint()));
+ QCOMPARE(proxy.rect().toRect(), rect);
+
+ // Properties
+ // QCOMPARE(proxy.focusPolicy(), lineEdit->focusPolicy());
+ // QCOMPARE(proxy.palette(), lineEdit->palette());
+#ifndef QT_NO_CURSOR
+ QCOMPARE(proxy.cursor().shape(), lineEdit->cursor().shape());
+#endif
+ QCOMPARE(proxy.layoutDirection(), lineEdit->layoutDirection());
+ QCOMPARE(proxy.style(), lineEdit->style());
+ QCOMPARE(proxy.font(), lineEdit->font());
+ QCOMPARE(proxy.isEnabled(), lineEdit->isEnabled());
+ QCOMPARE(proxy.isVisible(), lineEdit->isVisible());
+}
+
+void tst_QGraphicsProxyWidget::setWidget_ownership()
+{
+ QPointer<QLineEdit> lineEdit = new QLineEdit;
+ QPointer<QLineEdit> lineEdit2 = new QLineEdit;
+ QVERIFY(lineEdit);
+ {
+ // Create a proxy and transfer ownership to it
+ QGraphicsProxyWidget proxy;
+ proxy.setWidget(lineEdit);
+ QCOMPARE(proxy.widget(), (QWidget *)lineEdit);
+
+ // Remove the widget without destroying it.
+ proxy.setWidget(0);
+ QVERIFY(!proxy.widget());
+ QVERIFY(lineEdit);
+
+ // Assign the widget again and switch to another widget.
+ proxy.setWidget(lineEdit);
+ proxy.setWidget(lineEdit2);
+ QCOMPARE(proxy.widget(), (QWidget *)lineEdit2);
+
+ // Assign the first widget, and destroy the proxy.
+ proxy.setWidget(lineEdit);
+ }
+ QVERIFY(!lineEdit);
+ QVERIFY(lineEdit2);
+
+ QGraphicsScene scene;
+ QPointer<QGraphicsProxyWidget> proxy = scene.addWidget(lineEdit2);
+
+ delete lineEdit2;
+ QVERIFY(!proxy);
+}
+
+void tst_QGraphicsProxyWidget::resize_simple_data()
+{
+ QTest::addColumn<QSizeF>("size");
+
+ QTest::newRow("200, 200") << QSizeF(200, 200);
+#if !defined(QT_ARCH_ARM) && !defined(Q_OS_WINCE)
+ QTest::newRow("1000, 1000") << QSizeF(1000, 1000);
+ // Since 4.5, 10000x10000 runs out of memory.
+ // QTest::newRow("10000, 10000") << QSizeF(10000, 10000);
+#endif
+}
+
+void tst_QGraphicsProxyWidget::resize_simple()
+{
+ QFETCH(QSizeF, size);
+
+ QGraphicsProxyWidget proxy;
+ QWidget *widget = new QWidget;
+ widget->setGeometry(0, 0, (int)size.width(), (int)size.height());
+ proxy.setWidget(widget);
+ widget->show();
+ QCOMPARE(widget->pos(), QPoint());
+
+ // The proxy resizes itself, the line edit follows
+ proxy.resize(size);
+ QCOMPARE(proxy.size(), size);
+ QCOMPARE(proxy.size().toSize(), widget->size());
+
+ // The line edit resizes itself, the proxy follows (no loopback/live lock)
+ QSize doubleSize = size.toSize() * 2;
+ widget->resize(doubleSize);
+ QCOMPARE(widget->size(), doubleSize);
+ QCOMPARE(widget->size(), proxy.size().toSize());
+}
+
+void tst_QGraphicsProxyWidget::symmetricMove()
+{
+ QGraphicsProxyWidget proxy;
+ QLineEdit *lineEdit = new QLineEdit;
+ proxy.setWidget(lineEdit);
+ lineEdit->show();
+
+ proxy.setPos(10, 10);
+ QCOMPARE(proxy.pos(), QPointF(10, 10));
+ QCOMPARE(lineEdit->pos(), QPoint(10, 10));
+
+ lineEdit->move(5, 5);
+ QCOMPARE(proxy.pos(), QPointF(5, 5));
+ QCOMPARE(lineEdit->pos(), QPoint(5, 5));
+}
+
+void tst_QGraphicsProxyWidget::symmetricResize()
+{
+ QGraphicsProxyWidget proxy;
+ QLineEdit *lineEdit = new QLineEdit;
+ proxy.setWidget(lineEdit);
+ lineEdit->show();
+
+ proxy.resize(256, 256);
+ QCOMPARE(proxy.size(), QSizeF(256, 256));
+ QCOMPARE(lineEdit->size(), QSize(256, 256));
+
+ lineEdit->resize(512, 512);
+ QCOMPARE(proxy.size(), QSizeF(512, 512));
+ QCOMPARE(lineEdit->size(), QSize(512, 512));
+}
+
+void tst_QGraphicsProxyWidget::symmetricVisible()
+{
+ QGraphicsProxyWidget proxy;
+ QLineEdit *lineEdit = new QLineEdit;
+ proxy.setWidget(lineEdit);
+ lineEdit->show();
+
+ QCOMPARE(proxy.isVisible(), lineEdit->isVisible());
+
+ proxy.hide();
+ QCOMPARE(proxy.isVisible(), lineEdit->isVisible());
+ proxy.show();
+ QCOMPARE(proxy.isVisible(), lineEdit->isVisible());
+ lineEdit->hide();
+ QCOMPARE(proxy.isVisible(), lineEdit->isVisible());
+ lineEdit->show();
+ QCOMPARE(proxy.isVisible(), lineEdit->isVisible());
+ }
+
+void tst_QGraphicsProxyWidget::symmetricEnabled()
+{
+ QGraphicsProxyWidget proxy;
+ QLineEdit *lineEdit = new QLineEdit;
+ proxy.setWidget(lineEdit);
+ lineEdit->show();
+
+ QCOMPARE(proxy.isEnabled(), lineEdit->isEnabled());
+ proxy.setEnabled(false);
+ QCOMPARE(proxy.isEnabled(), lineEdit->isEnabled());
+ proxy.setEnabled(true);
+ QCOMPARE(proxy.isEnabled(), lineEdit->isEnabled());
+ lineEdit->setEnabled(false);
+ QCOMPARE(proxy.isEnabled(), lineEdit->isEnabled());
+ lineEdit->setEnabled(true);
+ QCOMPARE(proxy.isEnabled(), lineEdit->isEnabled());
+}
+
+void tst_QGraphicsProxyWidget::tabFocus_simpleWidget()
+{
+ QGraphicsScene scene;
+ QLineEdit *edit = new QLineEdit;
+ QGraphicsProxyWidget *editProxy = scene.addWidget(edit);
+ editProxy->show();
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+ QGraphicsView *view = new QGraphicsView(&scene);
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(view);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ window.show();
+ QApplication::setActiveWindow(&window);
+ window.activateWindow();
+ QTest::qWaitForWindowShown(&window);
+
+ leftDial->setFocus();
+ QApplication::processEvents();
+ QTRY_VERIFY(leftDial->hasFocus());
+
+ EventSpy eventSpy(edit);
+
+ // Tab into line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QTRY_VERIFY(!leftDial->hasFocus());
+ QTRY_VERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QVERIFY(scene.hasFocus());
+ QVERIFY(editProxy->hasFocus());
+ QVERIFY(edit->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 0);
+
+ // Tab into right dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QTRY_VERIFY(!view->hasFocus());
+ QVERIFY(!view->viewport()->hasFocus());
+ QVERIFY(!scene.hasFocus());
+ QVERIFY(!editProxy->hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QTRY_VERIFY(rightDial->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+
+ // Backtab into line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QTRY_VERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QTRY_VERIFY(scene.hasFocus());
+ QVERIFY(editProxy->hasFocus());
+ QVERIFY(edit->hasFocus());
+ QVERIFY(!rightDial->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+
+ // Backtab into left dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QTRY_VERIFY(!view->hasFocus());
+ QVERIFY(!view->viewport()->hasFocus());
+ QVERIFY(!scene.hasFocus());
+ QVERIFY(!editProxy->hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QTRY_VERIFY(leftDial->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 2);
+
+ delete view;
+}
+
+void tst_QGraphicsProxyWidget::tabFocus_simpleTwoWidgets()
+{
+ QGraphicsScene scene;
+ QLineEdit *edit = new QLineEdit;
+ QLineEdit *edit2 = new QLineEdit;
+ QGraphicsProxyWidget *editProxy = scene.addWidget(edit);
+ editProxy->show();
+ QGraphicsProxyWidget *editProxy2 = scene.addWidget(edit2);
+ editProxy2->show();
+ editProxy2->setPos(0, editProxy->rect().height() * 1.1);
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+ QGraphicsView *view = new QGraphicsView(&scene);
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(view);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ window.show();
+ QApplication::setActiveWindow(&window);
+ window.activateWindow();
+ QTest::qWaitForWindowShown(&window);
+
+ leftDial->setFocus();
+ QApplication::processEvents();
+ QTRY_VERIFY(leftDial->hasFocus());
+
+ EventSpy eventSpy(edit);
+ EventSpy eventSpy2(edit2);
+
+ // Tab into line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!leftDial->hasFocus());
+ QVERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QVERIFY(scene.hasFocus());
+ QVERIFY(editProxy->hasFocus());
+ QVERIFY(edit->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 0);
+
+ // Tab into second line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QVERIFY(scene.hasFocus());
+ QVERIFY(!editProxy->hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QVERIFY(editProxy2->hasFocus());
+ QVERIFY(edit2->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 0);
+
+ // Tab into right dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!view->hasFocus());
+ QVERIFY(!view->viewport()->hasFocus());
+ QVERIFY(!scene.hasFocus());
+ QVERIFY(!editProxy->hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QVERIFY(!editProxy2->hasFocus());
+ QVERIFY(!edit2->hasFocus());
+ QVERIFY(rightDial->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 1);
+
+ // Backtab into line edit 2
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QVERIFY(scene.hasFocus());
+ QVERIFY(!editProxy->hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QVERIFY(editProxy2->hasFocus());
+ QVERIFY(edit2->hasFocus());
+ QVERIFY(!rightDial->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 1);
+
+ // Backtab into line edit 1
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QVERIFY(scene.hasFocus());
+ QVERIFY(editProxy->hasFocus());
+ QVERIFY(edit->hasFocus());
+ QVERIFY(!editProxy2->hasFocus());
+ QVERIFY(!edit2->hasFocus());
+ QVERIFY(!rightDial->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 2);
+
+ // Backtab into left dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!view->hasFocus());
+ QVERIFY(!view->viewport()->hasFocus());
+ QVERIFY(!scene.hasFocus());
+ QVERIFY(!editProxy->hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QVERIFY(leftDial->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 2);
+
+ delete view;
+}
+
+void tst_QGraphicsProxyWidget::tabFocus_complexWidget()
+{
+ QGraphicsScene scene;
+
+ QLineEdit *edit1 = new QLineEdit;
+ edit1->setText("QLineEdit 1");
+ QLineEdit *edit2 = new QLineEdit;
+ edit2->setText("QLineEdit 2");
+ QVBoxLayout *vlayout = new QVBoxLayout;
+ vlayout->addWidget(edit1);
+ vlayout->addWidget(edit2);
+
+ QGroupBox *box = new QGroupBox("QGroupBox");
+ box->setCheckable(true);
+ box->setChecked(true);
+ box->setLayout(vlayout);
+
+ QGraphicsProxyWidget *proxy = scene.addWidget(box);
+ proxy->show();
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+ QGraphicsView *view = new QGraphicsView(&scene);
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(view);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ window.show();
+ QApplication::setActiveWindow(&window);
+ window.activateWindow();
+ QTest::qWaitForWindowShown(&window);
+
+ leftDial->setFocus();
+ QApplication::processEvents();
+ QTRY_VERIFY(leftDial->hasFocus());
+
+ EventSpy eventSpy(edit1);
+ EventSpy eventSpy2(edit2);
+ EventSpy eventSpyBox(box);
+
+ // Tab into group box
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!leftDial->hasFocus());
+ QVERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QVERIFY(scene.hasFocus());
+ QVERIFY(proxy->hasFocus());
+ QVERIFY(box->hasFocus());
+
+ // Tab into line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ edit1->hasFocus();
+ QVERIFY(!box->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 0);
+
+ // Tab into line edit 2
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ edit2->hasFocus();
+ QVERIFY(!edit1->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 0);
+
+ // Tab into right dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!edit2->hasFocus());
+ rightDial->hasFocus();
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 1);
+
+ // Backtab into line edit 2
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!rightDial->hasFocus());
+ edit2->hasFocus();
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 1);
+
+ // Backtab into line edit 1
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!edit2->hasFocus());
+ edit1->hasFocus();
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 2);
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 2);
+
+ // Backtab into line box
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!edit1->hasFocus());
+ box->hasFocus();
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 2);
+
+ // Backtab into left dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!box->hasFocus());
+ leftDial->hasFocus();
+
+ delete view;
+}
+
+void tst_QGraphicsProxyWidget::tabFocus_complexTwoWidgets()
+{
+ // ### add event spies to this test.
+ QGraphicsScene scene;
+
+ QLineEdit *edit1 = new QLineEdit;
+ edit1->setText("QLineEdit 1");
+ QLineEdit *edit2 = new QLineEdit;
+ edit2->setText("QLineEdit 2");
+ QFontComboBox *fontComboBox = new QFontComboBox;
+ QVBoxLayout *vlayout = new QVBoxLayout;
+ vlayout->addWidget(edit1);
+ vlayout->addWidget(fontComboBox);
+ vlayout->addWidget(edit2);
+
+ QGroupBox *box = new QGroupBox("QGroupBox");
+ box->setCheckable(true);
+ box->setChecked(true);
+ box->setLayout(vlayout);
+
+ QLineEdit *edit1_2 = new QLineEdit;
+ edit1_2->setText("QLineEdit 1_2");
+ QLineEdit *edit2_2 = new QLineEdit;
+ edit2_2->setText("QLineEdit 2_2");
+ QFontComboBox *fontComboBox2 = new QFontComboBox;
+ vlayout = new QVBoxLayout;
+ vlayout->addWidget(edit1_2);
+ vlayout->addWidget(fontComboBox2);
+ vlayout->addWidget(edit2_2);
+
+ QGroupBox *box_2 = new QGroupBox("QGroupBox 2");
+ box_2->setCheckable(true);
+ box_2->setChecked(true);
+ box_2->setLayout(vlayout);
+
+ QGraphicsProxyWidget *proxy = scene.addWidget(box);
+ proxy->show();
+
+ QGraphicsProxyWidget *proxy_2 = scene.addWidget(box_2);
+ proxy_2->setPos(proxy->boundingRect().width() * 1.2, 0);
+ proxy_2->show();
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+ QGraphicsView *view = new QGraphicsView(&scene);
+ view->setRenderHint(QPainter::Antialiasing);
+ view->rotate(45);
+ view->scale(0.5, 0.5);
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(view);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ window.show();
+ QApplication::setActiveWindow(&window);
+ window.activateWindow();
+ QTest::qWaitForWindowShown(&window);
+ QTRY_COMPARE(QApplication::activeWindow(), &window);
+
+ leftDial->setFocus();
+ QApplication::processEvents();
+ QTRY_VERIFY(leftDial->hasFocus());
+
+ EventSpy eventSpy(edit1);
+ EventSpy eventSpy2(edit2);
+ EventSpy eventSpy3(fontComboBox);
+ EventSpy eventSpy1_2(edit1_2);
+ EventSpy eventSpy2_2(edit2_2);
+ EventSpy eventSpy2_3(fontComboBox2);
+ EventSpy eventSpyBox(box);
+
+ // Tab into group box
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!leftDial->hasFocus());
+ QVERIFY(view->hasFocus());
+ QVERIFY(view->viewport()->hasFocus());
+ QVERIFY(scene.hasFocus());
+ QVERIFY(proxy->hasFocus());
+ QVERIFY(box->hasFocus());
+
+ // Tab into line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ edit1->hasFocus();
+ QVERIFY(!box->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 0);
+
+ // Tab to the font combobox
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ fontComboBox->hasFocus();
+ QVERIFY(!edit2->hasFocus());
+ QCOMPARE(eventSpy3.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy3.counts[QEvent::FocusOut], 0);
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+
+ // Tab into line edit 2
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ edit2->hasFocus();
+ QVERIFY(!edit1->hasFocus());
+ QCOMPARE(eventSpy2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2.counts[QEvent::FocusOut], 0);
+ QCOMPARE(eventSpy3.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+
+ // Tab into right box
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!edit2->hasFocus());
+ box_2->hasFocus();
+
+ // Tab into right top line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!box_2->hasFocus());
+ edit1_2->hasFocus();
+ QCOMPARE(eventSpy1_2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy1_2.counts[QEvent::FocusOut], 0);
+
+ // Tab into right font combobox
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!edit1_2->hasFocus());
+ fontComboBox2->hasFocus();
+ QCOMPARE(eventSpy1_2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy1_2.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2_3.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2_3.counts[QEvent::FocusOut], 0);
+
+ // Tab into right bottom line edit
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!edit1_2->hasFocus());
+ edit2_2->hasFocus();
+ QCOMPARE(eventSpy1_2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy1_2.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2_3.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2_3.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusOut], 0);
+
+ // Tab into right dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Tab);
+ QApplication::processEvents();
+ QVERIFY(!edit2->hasFocus());
+ rightDial->hasFocus();
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusOut], 1);
+
+ // Backtab into line edit 2
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!rightDial->hasFocus());
+ edit2_2->hasFocus();
+
+ // Backtab into the right font combobox
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!edit2_2->hasFocus());
+ fontComboBox2->hasFocus();
+
+ // Backtab into line edit 1
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!edit2_2->hasFocus());
+ edit1_2->hasFocus();
+
+ // Backtab into line box
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!edit1_2->hasFocus());
+ box_2->hasFocus();
+
+ // Backtab into line edit 2
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!rightDial->hasFocus());
+ edit2->hasFocus();
+
+ // Backtab into the font combobox
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!edit2->hasFocus());
+ fontComboBox->hasFocus();
+
+ // Backtab into line edit 1
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!fontComboBox->hasFocus());
+ edit1->hasFocus();
+
+ // Backtab into line box
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!edit1->hasFocus());
+ box->hasFocus();
+
+ // Backtab into left dial
+ QTest::keyPress(QApplication::focusWidget(), Qt::Key_Backtab);
+ QApplication::processEvents();
+ QVERIFY(!box->hasFocus());
+ leftDial->hasFocus();
+
+ delete view;
+}
+
+void tst_QGraphicsProxyWidget::setFocus_simpleWidget()
+{
+ QGraphicsScene scene;
+ QLineEdit *edit = new QLineEdit;
+ QGraphicsProxyWidget *editProxy = scene.addWidget(edit);
+ editProxy->show();
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+ QGraphicsView *view = new QGraphicsView(&scene);
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(view);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ window.show();
+ QApplication::setActiveWindow(&window);
+ window.activateWindow();
+ QTest::qWaitForWindowShown(&window);
+ QTRY_COMPARE(QApplication::activeWindow(), &window);
+
+ leftDial->setFocus();
+ QApplication::processEvents();
+ QTRY_VERIFY(leftDial->hasFocus());
+
+ EventSpy eventSpy(edit);
+
+ // Calling setFocus for the line edit when the view doesn't have input
+ // focus does not steal focus from the dial. The edit, proxy and scene
+ // have focus but only in their own little space.
+ edit->setFocus();
+ QVERIFY(scene.hasFocus());
+ QVERIFY(edit->hasFocus());
+ QVERIFY(!view->hasFocus());
+ QVERIFY(!view->viewport()->hasFocus());
+ QVERIFY(leftDial->hasFocus());
+
+ // Clearing focus is a local operation.
+ edit->clearFocus();
+ QVERIFY(scene.hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QVERIFY(!view->hasFocus());
+ QVERIFY(leftDial->hasFocus());
+
+ view->setFocus();
+ QVERIFY(scene.hasFocus());
+ QVERIFY(view->hasFocus());
+ QVERIFY(!leftDial->hasFocus());
+ QVERIFY(!edit->hasFocus());
+
+ scene.clearFocus();
+ QVERIFY(!scene.hasFocus());
+
+ edit->setFocus();
+ QVERIFY(scene.hasFocus());
+ QVERIFY(edit->hasFocus());
+ QVERIFY(editProxy->hasFocus());
+
+ // Symmetry
+ editProxy->clearFocus();
+ QVERIFY(!edit->hasFocus());
+
+ delete view;
+}
+
+void tst_QGraphicsProxyWidget::setFocus_simpleTwoWidgets()
+{
+ QGraphicsScene scene;
+ QLineEdit *edit = new QLineEdit;
+ QGraphicsProxyWidget *editProxy = scene.addWidget(edit);
+ editProxy->show();
+ QLineEdit *edit2 = new QLineEdit;
+ QGraphicsProxyWidget *edit2Proxy = scene.addWidget(edit2);
+ edit2Proxy->show();
+ edit2Proxy->setPos(editProxy->boundingRect().width() * 1.01, 0);
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+ QGraphicsView *view = new QGraphicsView(&scene);
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(view);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ window.show();
+ QApplication::setActiveWindow(&window);
+ window.activateWindow();
+ QTest::qWaitForWindowShown(&window);
+ QTRY_COMPARE(QApplication::activeWindow(), &window);
+
+ leftDial->setFocus();
+ QApplication::processEvents();
+ QTRY_VERIFY(leftDial->hasFocus());
+
+ EventSpy eventSpy(edit);
+
+ view->setFocus();
+ QVERIFY(!edit->hasFocus());
+
+ edit->setFocus();
+ QVERIFY(scene.hasFocus());
+ QVERIFY(edit->hasFocus());
+ QVERIFY(editProxy->hasFocus());
+
+ edit2->setFocus();
+ QVERIFY(scene.hasFocus());
+ QVERIFY(!edit->hasFocus());
+ QVERIFY(!editProxy->hasFocus());
+ QVERIFY(edit2->hasFocus());
+ QVERIFY(edit2Proxy->hasFocus());
+
+ delete view;
+}
+
+void tst_QGraphicsProxyWidget::setFocus_complexTwoWidgets()
+{
+ // ### add event spies to this test.
+ QGraphicsScene scene;
+
+ QLineEdit *edit1 = new QLineEdit;
+ edit1->setText("QLineEdit 1");
+ QLineEdit *edit2 = new QLineEdit;
+ edit2->setText("QLineEdit 2");
+ QVBoxLayout *vlayout = new QVBoxLayout;
+ vlayout->addWidget(edit1);
+ vlayout->addWidget(edit2);
+
+ QGroupBox *box = new QGroupBox("QGroupBox");
+ box->setCheckable(true);
+ box->setChecked(true);
+ box->setLayout(vlayout);
+
+ QLineEdit *edit1_2 = new QLineEdit;
+ edit1_2->setText("QLineEdit 1_2");
+ QLineEdit *edit2_2 = new QLineEdit;
+ edit2_2->setText("QLineEdit 2_2");
+ vlayout = new QVBoxLayout;
+ vlayout->addWidget(edit1_2);
+ vlayout->addWidget(edit2_2);
+
+ QGroupBox *box_2 = new QGroupBox("QGroupBox 2");
+ box_2->setCheckable(true);
+ box_2->setChecked(true);
+ box_2->setLayout(vlayout);
+
+ QGraphicsProxyWidget *proxy = scene.addWidget(box);
+ proxy->show();
+
+ QGraphicsProxyWidget *proxy_2 = scene.addWidget(box_2);
+ proxy_2->setPos(proxy->boundingRect().width() * 1.2, 0);
+ proxy_2->show();
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+ QGraphicsView *view = new QGraphicsView(&scene);
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(view);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ window.show();
+ QApplication::setActiveWindow(&window);
+ window.activateWindow();
+ QTest::qWaitForWindowShown(&window);
+ QTRY_COMPARE(QApplication::activeWindow(), &window);
+
+ leftDial->setFocus();
+ QApplication::processEvents();
+ QTRY_VERIFY(leftDial->hasFocus());
+
+ EventSpy eventSpy(edit1);
+ EventSpy eventSpy2(edit2);
+ EventSpy eventSpyBox(box);
+ EventSpy eventSpy_2(edit1_2);
+ EventSpy eventSpy2_2(edit2_2);
+ EventSpy eventSpyBox_2(box_2);
+
+ view->setFocus();
+
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 0);
+
+ edit1->setFocus();
+ QApplication::processEvents();
+ QVERIFY(scene.hasFocus());
+ QVERIFY(edit1->hasFocus());
+ QVERIFY(!box->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpyBox.counts[QEvent::FocusIn], 0);
+
+ edit2_2->setFocus();
+ QApplication::processEvents();
+ QVERIFY(!edit1->hasFocus());
+ QVERIFY(!box_2->hasFocus());
+ QVERIFY(edit2_2->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpyBox.counts[QEvent::FocusIn], 0);
+ QCOMPARE(eventSpyBox_2.counts[QEvent::FocusIn], 0);
+
+ box->setFocus();
+ QApplication::processEvents();
+ QVERIFY(!edit2_2->hasFocus());
+ QVERIFY(!edit1->hasFocus());
+ QVERIFY(box->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpyBox.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpyBox.counts[QEvent::FocusOut], 0);
+ QCOMPARE(eventSpyBox_2.counts[QEvent::FocusIn], 0);
+ QCOMPARE(eventSpyBox_2.counts[QEvent::FocusOut], 0);
+
+ edit2_2->setFocus();
+ QApplication::processEvents();
+ QVERIFY(edit2_2->hasFocus());
+ QVERIFY(!edit1->hasFocus());
+ QVERIFY(!box->hasFocus());
+ QVERIFY(!box_2->hasFocus());
+ QCOMPARE(eventSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusIn], 2);
+ QCOMPARE(eventSpy2_2.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpyBox.counts[QEvent::FocusIn], 1);
+ QCOMPARE(eventSpyBox.counts[QEvent::FocusOut], 1);
+ QCOMPARE(eventSpyBox_2.counts[QEvent::FocusIn], 0);
+ QCOMPARE(eventSpyBox_2.counts[QEvent::FocusOut], 0);
+
+ delete view;
+}
+
+void tst_QGraphicsProxyWidget::popup_basic()
+{
+ // ProxyWidget should automatically create proxy's when the widget creates a child
+ QGraphicsScene *scene = new QGraphicsScene;
+ QGraphicsView view(scene);
+ view.setAlignment(Qt::AlignLeft | Qt::AlignTop);
+ view.setGeometry(0, 100, 480, 500);
+ view.show();
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ QComboBox *box = new QComboBox;
+ box->setGeometry(0, 0, 320, 40);
+ box->addItems(QStringList() << "monday" << "tuesday" << "wednesday"
+ << "thursday" << "saturday" << "sunday");
+ QCOMPARE(proxy->childItems().count(), 0);
+ proxy->setWidget(box);
+ proxy->show();
+ scene->addItem(proxy);
+
+ QCOMPARE(box->pos(), QPoint());
+ QCOMPARE(proxy->pos(), QPointF());
+
+ QTest::qWaitForWindowShown(&view);
+ QTest::qWait(125);
+ QApplication::processEvents();
+
+ QTest::mousePress(view.viewport(), Qt::LeftButton, 0,
+ view.mapFromScene(proxy->mapToScene(proxy->boundingRect().center())));
+
+ QTRY_COMPARE(box->pos(), QPoint());
+
+ QCOMPARE(proxy->childItems().count(), 1);
+ QGraphicsProxyWidget *child = (QGraphicsProxyWidget*)(proxy->childItems())[0];
+ QVERIFY(child->isWidget());
+ QVERIFY(child->widget());
+ QStyleOptionComboBox opt;
+ opt.initFrom(box);
+ opt.editable = box->isEditable();
+ if (box->style()->styleHint(QStyle::SH_ComboBox_Popup, &opt))
+ QSKIP("Does not work due to SH_Combobox_Popup", SkipAll);
+ QCOMPARE(child->widget()->parent(), static_cast<QObject*>(box));
+
+ QTRY_COMPARE(proxy->pos(), QPointF(box->pos()));
+ QCOMPARE(child->x(), qreal(box->x()));
+ QCOMPARE(child->y(), qreal(box->rect().bottom()));
+#ifndef Q_OS_WIN
+ // The popup's coordinates on Windows are in global space,
+ // regardless.
+ QCOMPARE(child->widget()->x(), box->x());
+ QCOMPARE(child->widget()->y(), box->rect().bottom());
+ QCOMPARE(child->geometry().toRect(), child->widget()->geometry());
+#endif
+ QTest::qWait(12);
+}
+
+void tst_QGraphicsProxyWidget::popup_subwidget()
+{
+ QGroupBox *groupBox = new QGroupBox;
+ groupBox->setTitle("GroupBox");
+ groupBox->setCheckable(true);
+
+ QComboBox *box = new QComboBox;
+ box->addItems(QStringList() << "monday" << "tuesday" << "wednesday"
+ << "thursday" << "saturday" << "sunday");
+
+ QVBoxLayout *layout = new QVBoxLayout;
+ layout->addWidget(new QLineEdit("QLineEdit"));
+ layout->addWidget(box);
+ layout->addWidget(new QLineEdit("QLineEdit"));
+ groupBox->setLayout(layout);
+
+ QGraphicsScene scene;
+ QGraphicsProxyWidget *groupBoxProxy = scene.addWidget(groupBox);
+ groupBox->show();
+ groupBox->move(10000, 10000);
+ QCOMPARE(groupBox->pos(), QPoint(10000, 10000));
+ QCOMPARE(groupBoxProxy->pos(), QPointF(10000, 10000));
+
+ QGraphicsView view(&scene);
+ view.show();
+#ifdef Q_WS_X11
+ qt_x11_wait_for_window_manager(&view);
+#endif
+
+ box->showPopup();
+
+ QVERIFY(!groupBoxProxy->childItems().isEmpty());
+
+ QStyleOptionComboBox opt;
+ opt.initFrom(box);
+ opt.editable = box->isEditable();
+ if (box->style()->styleHint(QStyle::SH_ComboBox_Popup, &opt))
+ QSKIP("Does not work due to SH_Combobox_Popup", SkipAll);
+ QGraphicsProxyWidget *child = (QGraphicsProxyWidget*)(groupBoxProxy->childItems())[0];
+ QWidget *popup = child->widget();
+ QCOMPARE(popup->parent(), static_cast<QObject*>(box));
+ QCOMPARE(popup->x(), box->mapToGlobal(QPoint()).x());
+ QCOMPARE(popup->y(), QRect(box->mapToGlobal(QPoint()), box->size()).bottom());
+ QCOMPARE(popup->size(), child->size().toSize());
+}
+
+#if !defined(QT_NO_CURSOR) && (!defined(Q_OS_WINCE) || defined(GWES_ICONCURS))
+void tst_QGraphicsProxyWidget::changingCursor_basic()
+{
+ // Confirm that mouse events are working properly by checking that
+ // when moving the mouse over a line edit it will change the cursor into the I
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+
+ SubQGraphicsProxyWidget *proxy = new SubQGraphicsProxyWidget;
+ QLineEdit *widget = new QLineEdit;
+ proxy->setWidget(widget);
+ proxy->show();
+ scene.addItem(proxy);
+ QApplication::setActiveWindow(&view);
+ QTest::qWaitForWindowShown(&view);
+ QApplication::processEvents();
+ QTRY_VERIFY(view.isActiveWindow());
+
+ // in
+ QTest::mouseMove(view.viewport(), view.mapFromScene(proxy->mapToScene(proxy->boundingRect().center())));
+ sendMouseMove(view.viewport(), view.mapFromScene(proxy->mapToScene(proxy->boundingRect().center())));
+ QTRY_COMPARE(view.viewport()->cursor().shape(), Qt::IBeamCursor);
+
+ // out
+ QTest::mouseMove(view.viewport(), QPoint(1, 1));
+ sendMouseMove(view.viewport(), QPoint(1, 1));
+ QTRY_COMPARE(view.viewport()->cursor().shape(), Qt::ArrowCursor);
+}
+#endif
+
+void tst_QGraphicsProxyWidget::tooltip_basic()
+{
+ QString toolTip = "Qt rocks!";
+ QString toolTip2 = "Qt rocks even more!";
+
+ QPushButton *button = new QPushButton("button");
+ QGraphicsProxyWidget *proxy = new QGraphicsProxyWidget;
+ QGraphicsProxyWidgetPrivate *proxyd = static_cast<QGraphicsProxyWidgetPrivate *>(QGraphicsItemPrivate::get(proxy));
+ proxy->setWidget(button);
+ proxyd->lastWidgetUnderMouse = button; // force widget under mouse
+
+ QVERIFY(button->toolTip().isEmpty());
+ QVERIFY(proxy->toolTip().isEmpty());
+ // Check that setting the tooltip on the proxy also set it on the widget
+ proxy->setToolTip(toolTip);
+ QCOMPARE(proxy->toolTip(), toolTip);
+ QCOMPARE(button->toolTip(), toolTip);
+ // Check that setting the tooltip on the widget also set it on the proxy
+ button->setToolTip(toolTip2);
+ QCOMPARE(proxy->toolTip(), toolTip2);
+ QCOMPARE(button->toolTip(), toolTip2);
+
+ QGraphicsScene scene;
+ scene.addItem(proxy);
+
+ QGraphicsView view(&scene);
+ view.setFixedSize(200, 200);
+ view.show();
+ QTest::qWaitForWindowShown(&view);
+ {
+ QHelpEvent helpEvent(QEvent::ToolTip, view.viewport()->rect().topLeft(),
+ view.viewport()->mapToGlobal(view.viewport()->rect().topLeft()));
+ QApplication::sendEvent(view.viewport(), &helpEvent);
+ QTest::qWait(350);
+
+ bool foundView = false;
+ bool foundTipLabel = false;
+ foreach (QWidget *widget, QApplication::topLevelWidgets()) {
+ if (widget == &view)
+ foundView = true;
+ if (widget->inherits("QTipLabel"))
+ foundTipLabel = true;
+ }
+ QVERIFY(foundView);
+ QVERIFY(!foundTipLabel);
+ }
+
+ {
+ QHelpEvent helpEvent(QEvent::ToolTip, view.mapFromScene(proxy->boundingRect().center()),
+ view.viewport()->mapToGlobal(view.mapFromScene(proxy->boundingRect().center())));
+ QApplication::sendEvent(view.viewport(), &helpEvent);
+ QTest::qWait(350);
+
+ bool foundView = false;
+ bool foundTipLabel = false;
+ foreach (QWidget *widget, QApplication::topLevelWidgets()) {
+ if (widget == &view)
+ foundView = true;
+ if (widget->inherits("QTipLabel"))
+ foundTipLabel = true;
+ }
+ QVERIFY(foundView);
+ QVERIFY(foundTipLabel);
+ }
+}
+
+void tst_QGraphicsProxyWidget::childPos_data()
+{
+ QTest::addColumn<bool>("moveCombo");
+ QTest::addColumn<QPoint>("comboPos");
+ QTest::addColumn<QPointF>("proxyPos");
+ QTest::addColumn<QPointF>("menuPos");
+
+ QTest::newRow("0") << true << QPoint() << QPointF() << QPointF();
+ QTest::newRow("1") << true << QPoint(10, 0) << QPointF(10, 0) << QPointF();
+ QTest::newRow("2") << true << QPoint(100, 0) << QPointF(100, 0) << QPointF();
+ QTest::newRow("3") << true << QPoint(1000, 0) << QPointF(1000, 0) << QPointF();
+ QTest::newRow("4") << true << QPoint(10000, 0) << QPointF(10000, 0) << QPointF();
+ QTest::newRow("5") << true << QPoint(-10000, 0) << QPointF(-10000, 0) << QPointF();
+ QTest::newRow("6") << true << QPoint(-1000, 0) << QPointF(-1000, 0) << QPointF();
+ QTest::newRow("7") << true << QPoint(-100, 0) << QPointF(-100, 0) << QPointF();
+ QTest::newRow("8") << true << QPoint(-10, 0) << QPointF(-10, 0) << QPointF();
+ QTest::newRow("0-") << false << QPoint() << QPointF() << QPointF();
+ QTest::newRow("1-") << false << QPoint(10, 0) << QPointF(10, 0) << QPointF();
+ QTest::newRow("2-") << false << QPoint(100, 0) << QPointF(100, 0) << QPointF();
+ QTest::newRow("3-") << false << QPoint(1000, 0) << QPointF(1000, 0) << QPointF();
+ QTest::newRow("4-") << false << QPoint(10000, 0) << QPointF(10000, 0) << QPointF();
+ QTest::newRow("5-") << false << QPoint(-10000, 0) << QPointF(-10000, 0) << QPointF();
+ QTest::newRow("6-") << false << QPoint(-1000, 0) << QPointF(-1000, 0) << QPointF();
+ QTest::newRow("7-") << false << QPoint(-100, 0) << QPointF(-100, 0) << QPointF();
+ QTest::newRow("8-") << false << QPoint(-10, 0) << QPointF(-10, 0) << QPointF();
+}
+
+void tst_QGraphicsProxyWidget::childPos()
+{
+#ifdef Q_OS_IRIX
+ QSKIP("This test is not reliable on IRIX.", SkipAll);
+#endif
+ QFETCH(bool, moveCombo);
+ QFETCH(QPoint, comboPos);
+ QFETCH(QPointF, proxyPos);
+ QFETCH(QPointF, menuPos);
+
+ QGraphicsScene scene;
+
+ QComboBox *box = new QComboBox;
+ box->addItem("Item 1");
+ box->addItem("Item 2");
+ box->addItem("Item 3");
+ box->addItem("Item 4");
+
+ if (moveCombo)
+ box->move(comboPos);
+
+ QGraphicsProxyWidget *proxy = scene.addWidget(box);
+ proxy->show();
+ QVERIFY(proxy->isVisible());
+ QVERIFY(box->isVisible());
+
+ if (!moveCombo)
+ proxy->setPos(proxyPos);
+
+ QCOMPARE(proxy->pos(), proxyPos);
+ QCOMPARE(box->pos(), comboPos);
+
+ for (int i = 0; i < 2; ++i) {
+ box->showPopup();
+ QApplication::processEvents();
+ QApplication::processEvents();
+
+ QWidget *menu = 0;
+ foreach (QObject *child, box->children()) {
+ if ((menu = qobject_cast<QWidget *>(child)))
+ break;
+ }
+ QVERIFY(menu);
+ QVERIFY(menu->isVisible());
+ QVERIFY(menu->testAttribute(Qt::WA_DontShowOnScreen));
+
+ QCOMPARE(proxy->childItems().size(), 1);
+ QGraphicsProxyWidget *proxyChild = 0;
+ foreach (QGraphicsItem *child, proxy->childItems()) {
+ if (child->isWidget() && (proxyChild = qobject_cast<QGraphicsProxyWidget *>(static_cast<QGraphicsWidget *>(child))))
+ break;
+ }
+ QVERIFY(proxyChild);
+ QVERIFY(proxyChild->isVisible());
+ qreal expectedXPosition = 0.0;
+#if defined(Q_WS_MAC) && !defined(QT_NO_STYLE_MAC)
+ // The Mac style wants the popup to show up at QPoint(4 - 11, 1).
+ // See QMacStyle::subControlRect SC_ComboBoxListBoxPopup.
+ if (qobject_cast<QMacStyle *>(QApplication::style()))
+ expectedXPosition = qreal(4 - 11);
+#endif
+ QCOMPARE(proxyChild->pos().x(), expectedXPosition);
+ menu->hide();
+ }
+}
+
+void tst_QGraphicsProxyWidget::autoShow()
+{
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+
+ QGraphicsProxyWidget *proxy1 = scene.addWidget(new QPushButton("Button1"));
+
+ QPushButton *button2 = new QPushButton("Button2");
+ button2->hide();
+ QGraphicsProxyWidget *proxy2 = new QGraphicsProxyWidget();
+ proxy2->setWidget(button2);
+ scene.addItem(proxy2);
+
+ view.show();
+ QApplication::processEvents();
+
+ QCOMPARE(proxy1->isVisible(), true);
+ QCOMPARE(proxy2->isVisible(), false);
+
+}
+
+Q_DECLARE_METATYPE(QList<QRectF>)
+void tst_QGraphicsProxyWidget::windowOpacity()
+{
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+
+ QWidget *widget = new QWidget;
+ widget->resize(100, 100);
+ QGraphicsProxyWidget *proxy = scene.addWidget(widget);
+ proxy->setCacheMode(QGraphicsItem::ItemCoordinateCache);
+
+ QApplication::setActiveWindow(&view);
+ view.show();
+ QTest::qWaitForWindowShown(&view);
+ QApplication::sendPostedEvents();
+ QTRY_VERIFY(view.isActiveWindow());
+
+ qRegisterMetaType<QList<QRectF> >("QList<QRectF>");
+ QSignalSpy signalSpy(&scene, SIGNAL(changed(const QList<QRectF> &)));
+
+ EventSpy eventSpy(widget);
+ QVERIFY(widget->isVisible());
+
+ widget->setWindowOpacity(0.5);
+ QApplication::processEvents();
+
+ // Make sure setWindowOpacity triggers an update on the scene,
+ // and not on the widget or the proxy itself. The entire proxy needs an update
+ // in case it has a window decoration. Update: QGraphicsItem::CacheMode is
+ // disabled on platforms without alpha channel support in QPixmap (e.g.,
+ // X11 without XRender).
+ int paints = 0;
+#ifdef Q_WS_X11
+ paints = !X11->use_xrender;
+#endif
+ QTRY_COMPARE(eventSpy.counts[QEvent::UpdateRequest], 0);
+ QTRY_COMPARE(eventSpy.counts[QEvent::Paint], paints);
+
+ QCOMPARE(signalSpy.count(), 1);
+ const QList<QVariant> arguments = signalSpy.takeFirst();
+ const QList<QRectF> updateRects = qvariant_cast<QList<QRectF> >(arguments.at(0));
+ QCOMPARE(updateRects.size(), 1);
+ QCOMPARE(updateRects.at(0), proxy->sceneBoundingRect());
+}
+
+void tst_QGraphicsProxyWidget::stylePropagation()
+{
+ QPointer<QWindowsStyle> windowsStyle = new QWindowsStyle;
+
+ QLineEdit *edit = new QLineEdit;
+ QGraphicsProxyWidget proxy;
+ proxy.setWidget(edit);
+
+ EventSpy editSpy(edit);
+ EventSpy proxySpy(&proxy);
+
+ // Widget to proxy
+ QCOMPARE(proxy.style(), QApplication::style());
+ edit->setStyle(windowsStyle);
+ QCOMPARE(editSpy.counts[QEvent::StyleChange], 1);
+ QCOMPARE(proxySpy.counts[QEvent::StyleChange], 1);
+ QCOMPARE(proxy.style(), (QStyle *)windowsStyle);
+ edit->setStyle(0);
+ QCOMPARE(editSpy.counts[QEvent::StyleChange], 2);
+ QCOMPARE(proxySpy.counts[QEvent::StyleChange], 2);
+ QCOMPARE(proxy.style(), QApplication::style());
+ QCOMPARE(edit->style(), QApplication::style());
+ QVERIFY(windowsStyle); // not deleted
+
+ // Proxy to widget
+ proxy.setStyle(windowsStyle);
+ QCOMPARE(editSpy.counts[QEvent::StyleChange], 3);
+ QCOMPARE(proxySpy.counts[QEvent::StyleChange], 3);
+ QCOMPARE(edit->style(), (QStyle *)windowsStyle);
+ proxy.setStyle(0);
+ QCOMPARE(editSpy.counts[QEvent::StyleChange], 4);
+ QCOMPARE(proxySpy.counts[QEvent::StyleChange], 4);
+ QCOMPARE(proxy.style(), QApplication::style());
+ QCOMPARE(edit->style(), QApplication::style());
+ QVERIFY(windowsStyle); // not deleted
+
+ delete windowsStyle;
+}
+
+void tst_QGraphicsProxyWidget::palettePropagation()
+{
+ // Construct a palette with an unlikely setup
+ QPalette lineEditPalette = QApplication::palette("QLineEdit");
+ QPalette palette = lineEditPalette;
+ palette.setBrush(QPalette::Text, Qt::red);
+
+ QLineEdit *edit = new QLineEdit;
+ QGraphicsProxyWidget proxy;
+ proxy.setWidget(edit);
+
+ EventSpy editSpy(edit);
+ EventSpy proxySpy(&proxy);
+
+ // Widget to proxy (no change)
+ QVERIFY(!edit->testAttribute(Qt::WA_SetPalette));
+ edit->setPalette(palette);
+ QCOMPARE(editSpy.counts[QEvent::PaletteChange], 1);
+ QCOMPARE(proxySpy.counts[QEvent::PaletteChange], 0);
+ QVERIFY(edit->testAttribute(Qt::WA_SetPalette));
+ QVERIFY(!proxy.testAttribute(Qt::WA_SetPalette));
+ QCOMPARE(proxy.palette(), QPalette());
+ edit->setPalette(QPalette());
+ QCOMPARE(editSpy.counts[QEvent::PaletteChange], 2);
+ QCOMPARE(proxySpy.counts[QEvent::PaletteChange], 0);
+ QVERIFY(!edit->testAttribute(Qt::WA_SetPalette));
+ QVERIFY(!proxy.testAttribute(Qt::WA_SetPalette));
+ QCOMPARE(proxy.palette(), QPalette());
+
+ // Proxy to widget
+ proxy.setPalette(palette);
+ QVERIFY(proxy.testAttribute(Qt::WA_SetPalette));
+ QCOMPARE(editSpy.counts[QEvent::PaletteChange], 3);
+ QCOMPARE(proxySpy.counts[QEvent::PaletteChange], 1);
+ QVERIFY(!edit->testAttribute(Qt::WA_SetPalette));
+ QCOMPARE(edit->palette(), palette);
+ QCOMPARE(edit->palette(), proxy.palette());
+ QCOMPARE(edit->palette().color(QPalette::Text), QColor(Qt::red));
+ proxy.setPalette(QPalette());
+ QVERIFY(!proxy.testAttribute(Qt::WA_SetPalette));
+ QVERIFY(!edit->testAttribute(Qt::WA_SetPalette));
+ QCOMPARE(editSpy.counts[QEvent::PaletteChange], 4);
+ QCOMPARE(proxySpy.counts[QEvent::PaletteChange], 2);
+ QCOMPARE(edit->palette(), lineEditPalette);
+}
+
+void tst_QGraphicsProxyWidget::fontPropagation()
+{
+ // Construct a font with an unlikely setup
+ QGraphicsScene scene;
+ QFont lineEditFont = QApplication::font("QLineEdit");
+ QFont font = lineEditFont;
+ font.setPointSize(43);
+
+ QLineEdit *edit = new QLineEdit;
+ QGraphicsProxyWidget proxy;
+ proxy.setWidget(edit);
+
+ scene.addItem(&proxy);
+ EventSpy editSpy(edit);
+ EventSpy proxySpy(&proxy);
+
+ // Widget to proxy (no change)
+ QVERIFY(!edit->testAttribute(Qt::WA_SetFont));
+ edit->setFont(font);
+ QCOMPARE(editSpy.counts[QEvent::FontChange], 1);
+ QCOMPARE(proxySpy.counts[QEvent::FontChange], 0);
+ QVERIFY(edit->testAttribute(Qt::WA_SetFont));
+ QVERIFY(!proxy.testAttribute(Qt::WA_SetFont));
+ QCOMPARE(proxy.font(), lineEditFont);
+ edit->setFont(QFont());
+ QCOMPARE(editSpy.counts[QEvent::FontChange], 2);
+ QCOMPARE(proxySpy.counts[QEvent::FontChange], 0);
+ QVERIFY(!edit->testAttribute(Qt::WA_SetFont));
+ QVERIFY(!proxy.testAttribute(Qt::WA_SetFont));
+ QCOMPARE(proxy.font(), lineEditFont);
+
+ // Proxy to widget
+ proxy.setFont(font);
+ QApplication::processEvents(); // wait for QEvent::Polish
+ QVERIFY(proxy.testAttribute(Qt::WA_SetFont));
+ QCOMPARE(editSpy.counts[QEvent::FontChange], 3);
+ QCOMPARE(proxySpy.counts[QEvent::FontChange], 1);
+ QVERIFY(!edit->testAttribute(Qt::WA_SetFont));
+ QCOMPARE(edit->font(), font);
+ QCOMPARE(edit->font(), proxy.font());
+ QCOMPARE(edit->font().pointSize(), 43);
+ proxy.setFont(QFont());
+ QVERIFY(!proxy.testAttribute(Qt::WA_SetFont));
+ QVERIFY(!edit->testAttribute(Qt::WA_SetFont));
+ QCOMPARE(editSpy.counts[QEvent::FontChange], 4);
+ QCOMPARE(proxySpy.counts[QEvent::FontChange], 2);
+ QCOMPARE(edit->font(), lineEditFont);
+
+ proxy.setFont(font);
+ QLineEdit *edit2 = new QLineEdit;
+ proxy.setWidget(edit2);
+ delete edit;
+ QCOMPARE(edit2->font().pointSize(), 43);
+}
+
+class MainWidget : public QMainWindow
+{
+Q_OBJECT
+public:
+ MainWidget() : QMainWindow() {
+ view = new QGraphicsView(this);
+ scene = new QGraphicsScene(this);
+ item = new QGraphicsWidget();
+ widget = new QGraphicsProxyWidget(item);
+ layout = new QGraphicsLinearLayout(item);
+ layout->addItem(widget);
+ item->setLayout(layout);
+ button = new QPushButton("Push me");
+ widget->setWidget(button);
+ setCentralWidget(view);
+ scene->addItem(item);
+ view->setScene(scene);
+ scene->setFocusItem(item);
+ layout->activate();
+ }
+ QGraphicsView* view;
+ QGraphicsScene* scene;
+ QGraphicsWidget * item;
+ QGraphicsLinearLayout * layout;
+ QGraphicsProxyWidget * widget;
+ QPushButton * button;
+};
+
+void tst_QGraphicsProxyWidget::dontCrashWhenDie()
+{
+ MainWidget *w = new MainWidget();
+ w->show();
+ QTest::qWaitForWindowShown(w);
+ QTest::qWait(100);
+ QTest::mouseMove(w->view->viewport(), w->view->mapFromScene(w->widget->mapToScene(w->widget->boundingRect().center())));
+ delete w->item;
+
+ QApplication::processEvents();
+ delete w;
+}
+
+void tst_QGraphicsProxyWidget::createProxyForChildWidget()
+{
+ QGraphicsScene scene;
+
+ QLineEdit *edit1 = new QLineEdit;
+ edit1->setText("QLineEdit 1");
+ QLineEdit *edit2 = new QLineEdit;
+ edit2->setText("QLineEdit 2");
+ QCheckBox *checkbox = new QCheckBox("QCheckBox");
+ QVBoxLayout *vlayout = new QVBoxLayout;
+
+ vlayout->addWidget(edit1);
+ vlayout->addWidget(edit2);
+ vlayout->addWidget(checkbox);
+ vlayout->insertStretch(-1);
+
+ QGroupBox *box = new QGroupBox("QGroupBox");
+ box->setCheckable(true);
+ box->setChecked(true);
+ box->setLayout(vlayout);
+
+ QDial *leftDial = new QDial;
+ QDial *rightDial = new QDial;
+
+ QWidget window;
+ QHBoxLayout *layout = new QHBoxLayout;
+ layout->addWidget(leftDial);
+ layout->addWidget(box);
+ layout->addWidget(rightDial);
+ window.setLayout(layout);
+
+ QVERIFY(window.graphicsProxyWidget() == 0);
+ QVERIFY(checkbox->graphicsProxyWidget() == 0);
+
+ QGraphicsProxyWidget *windowProxy = scene.addWidget(&window);
+ QGraphicsView view(&scene);
+ view.show();
+ view.resize(500,500);
+
+ QVERIFY(window.graphicsProxyWidget() == windowProxy);
+ QVERIFY(box->graphicsProxyWidget() == 0);
+ QVERIFY(checkbox->graphicsProxyWidget() == 0);
+
+ QPointer<QGraphicsProxyWidget> checkboxProxy = windowProxy->createProxyForChildWidget(checkbox);
+
+ QGraphicsProxyWidget *boxProxy = box->graphicsProxyWidget();
+
+ QVERIFY(boxProxy);
+ QVERIFY(checkbox->graphicsProxyWidget() == checkboxProxy);
+ QVERIFY(checkboxProxy->parentItem() == boxProxy);
+ QVERIFY(boxProxy->parentItem() == windowProxy);
+
+ QVERIFY(checkboxProxy->mapToScene(QPointF()) == checkbox->mapTo(&window, QPoint()));
+ QVERIFY(checkboxProxy->size() == checkbox->size());
+ QVERIFY(boxProxy->size() == box->size());
+
+ window.resize(500,500);
+ QVERIFY(windowProxy->size() == QSize(500,500));
+ QVERIFY(checkboxProxy->mapToScene(QPointF()) == checkbox->mapTo(&window, QPoint()));
+ QVERIFY(checkboxProxy->size() == checkbox->size());
+ QVERIFY(boxProxy->size() == box->size());
+
+ QTest::qWait(10);
+
+
+ QSignalSpy spy(checkbox, SIGNAL(clicked()));
+
+ QTest::mousePress(view.viewport(), Qt::LeftButton, 0,
+ view.mapFromScene(checkboxProxy->mapToScene(QPointF(8,8))));
+ QTRY_COMPARE(spy.count(), 0);
+ QTest::mouseRelease(view.viewport(), Qt::LeftButton, 0,
+ view.mapFromScene(checkboxProxy->mapToScene(QPointF(8,8))));
+ QTRY_COMPARE(spy.count(), 1);
+
+
+
+ boxProxy->setWidget(0);
+
+ QVERIFY(checkbox->graphicsProxyWidget() == 0);
+ QVERIFY(box->graphicsProxyWidget() == 0);
+ QVERIFY(checkboxProxy == 0);
+
+ delete boxProxy;
+}
+
+class ContextMenuWidget : public QWidget
+{
+ Q_OBJECT
+public:
+ ContextMenuWidget()
+ : embeddedPopup(false),
+ gotContextMenuEvent(false)
+ { }
+ bool embeddedPopup;
+ bool gotContextMenuEvent;
+protected:
+ bool event(QEvent *event)
+ {
+ if (event->type() == QEvent::ContextMenu)
+ QTimer::singleShot(0, this, SLOT(checkMenu()));
+ return QWidget::event(event);
+ }
+ void contextMenuEvent(QContextMenuEvent *)
+ {
+ gotContextMenuEvent = true;
+ }
+
+private slots:
+ void checkMenu()
+ {
+ if (qFindChild<QMenu *>(this))
+ embeddedPopup = true;
+ hide();
+ }
+};
+
+void tst_QGraphicsProxyWidget::actionsContextMenu_data()
+{
+ QTest::addColumn<bool>("actionsContextMenu");
+ QTest::addColumn<bool>("hasFocus");
+
+ QTest::newRow("without actionsContextMenu and with focus") << false << true;
+ QTest::newRow("without actionsContextMenu and without focus") << false << false;
+ QTest::newRow("with actionsContextMenu and focus") << true << true;
+ QTest::newRow("with actionsContextMenu without focus") << true << false;
+}
+
+void tst_QGraphicsProxyWidget::actionsContextMenu()
+{
+ QFETCH(bool, hasFocus);
+ QFETCH(bool, actionsContextMenu);
+
+ ContextMenuWidget *widget = new ContextMenuWidget;
+ if (actionsContextMenu) {
+ widget->addAction(new QAction("item 1", widget));
+ widget->addAction(new QAction("item 2", widget));
+ widget->addAction(new QAction("item 3", widget));
+ widget->setContextMenuPolicy(Qt::ActionsContextMenu);
+ }
+ QGraphicsScene scene;
+
+ QGraphicsView view(&scene);
+ view.show();
+ QApplication::setActiveWindow(&view);
+ QTest::qWaitForWindowShown(&view);
+ view.setFocus();
+ QTRY_VERIFY(view.hasFocus());
+
+ if (hasFocus)
+ scene.addWidget(widget)->setFocus();
+ else
+ scene.addWidget(widget)->clearFocus();
+
+ QApplication::processEvents();
+
+ QContextMenuEvent contextMenuEvent(QContextMenuEvent::Mouse,
+ view.viewport()->rect().center(),
+ view.viewport()->mapToGlobal(view.viewport()->rect().center()));
+ contextMenuEvent.accept();
+ qApp->sendEvent(view.viewport(), &contextMenuEvent);
+
+ if (hasFocus) {
+ if (actionsContextMenu) {
+ //actionsContextMenu embedded popup but no contextMenuEvent (widget has focus)
+ QVERIFY(widget->embeddedPopup);
+ QVERIFY(!widget->gotContextMenuEvent);
+ } else {
+ //no embedded popup but contextMenuEvent (widget has focus)
+ QVERIFY(!widget->embeddedPopup);
+ QVERIFY(widget->gotContextMenuEvent);
+ }
+ } else {
+ //qgraphicsproxywidget doesn't have the focus, the widget must not receive any contextMenuEvent and must not create any QMenu
+ QVERIFY(!widget->embeddedPopup);
+ QVERIFY(!widget->gotContextMenuEvent);
+ }
+
+}
+
+
+void tst_QGraphicsProxyWidget::deleteProxyForChildWidget()
+{
+ QDialog dialog;
+ dialog.resize(320, 120);
+ dialog.move(80, 40);
+
+ QGraphicsScene scene;
+ scene.setSceneRect(QRectF(0, 0, 640, 480));
+ QGraphicsView view(&scene);
+ QComboBox *combo = new QComboBox;
+ combo->addItems(QStringList() << "red" << "green" << "blue" << "white" << "black" << "yellow" << "cyan" << "magenta");
+ dialog.setLayout(new QVBoxLayout);
+ dialog.layout()->addWidget(combo);
+
+ QGraphicsProxyWidget *proxy = scene.addWidget(&dialog);
+ view.show();
+
+ proxy->setWidget(0);
+ //just don't crash
+ QApplication::processEvents();
+ delete combo;
+}
+
+void tst_QGraphicsProxyWidget::bypassGraphicsProxyWidget_data()
+{
+ QTest::addColumn<bool>("bypass");
+
+ QTest::newRow("autoembed") << false;
+ QTest::newRow("bypass") << true;
+}
+
+void tst_QGraphicsProxyWidget::bypassGraphicsProxyWidget()
+{
+ QFETCH(bool, bypass);
+
+ QWidget *widget = new QWidget;
+ widget->resize(100, 100);
+
+ QGraphicsScene scene;
+ QGraphicsView view(&scene);
+ view.show();
+
+ QGraphicsProxyWidget *proxy = scene.addWidget(widget);
+
+ QCOMPARE(proxy->widget(), widget);
+ QVERIFY(proxy->childItems().isEmpty());
+
+ Qt::WindowFlags flags;
+ flags |= Qt::Dialog;
+ if (bypass)
+ flags |= Qt::BypassGraphicsProxyWidget;
+ QFileDialog *dialog = new QFileDialog(widget, flags);
+ dialog->setOption(QFileDialog::DontUseNativeDialog, true);
+ dialog->show();
+
+ QCOMPARE(proxy->childItems().size(), bypass ? 0 : 1);
+ if (!bypass)
+ QCOMPARE(((QGraphicsProxyWidget *)proxy->childItems().first())->widget(), (QWidget *)dialog);
+
+ dialog->hide();
+ QApplication::processEvents();
+ delete dialog;
+ delete widget;
+}
+
+static void makeDndEvent(QGraphicsSceneDragDropEvent *event, QGraphicsView *view, const QPointF &pos)
+{
+ event->setScenePos(pos);
+ event->setScreenPos(view->mapToGlobal(view->mapFromScene(pos)));
+ event->setButtons(Qt::LeftButton);
+ event->setModifiers(0);
+ event->setPossibleActions(Qt::CopyAction);
+ event->setProposedAction(Qt::CopyAction);
+ event->setDropAction(Qt::CopyAction);
+ event->setWidget(view->viewport());
+ event->setSource(view->viewport());
+ event->ignore();
+}
+
+void tst_QGraphicsProxyWidget::dragDrop()
+{
+ QPushButton *button = new QPushButton; // acceptDrops(false)
+ QLineEdit *edit = new QLineEdit; // acceptDrops(true)
+
+ QGraphicsScene scene;
+ QGraphicsProxyWidget *buttonProxy = scene.addWidget(button);
+ QGraphicsProxyWidget *editProxy = scene.addWidget(edit);
+ QVERIFY(buttonProxy->acceptDrops());
+ QVERIFY(editProxy->acceptDrops());
+
+ button->setGeometry(0, 0, 100, 50);
+ edit->setGeometry(0, 60, 100, 50);
+
+ QGraphicsView view(&scene);
+ view.show();
+#ifdef Q_WS_X11
+ qt_x11_wait_for_window_manager(&view);
+#endif
+
+ QMimeData data;
+ data.setText("hei");
+ {
+ QGraphicsSceneDragDropEvent event(QEvent::GraphicsSceneDragEnter);
+ makeDndEvent(&event, &view, QPointF(50, 25));
+ event.setMimeData(&data);
+ qApp->sendEvent(&scene, &event);
+ QVERIFY(event.isAccepted());
+ }
+ {
+ QGraphicsSceneDragDropEvent event(QEvent::GraphicsSceneDragMove);
+ makeDndEvent(&event, &view, QPointF(50, 25));
+ event.setMimeData(&data);
+ qApp->sendEvent(&scene, &event);
+ QVERIFY(!event.isAccepted());
+ }
+ {
+ QGraphicsSceneDragDropEvent event(QEvent::GraphicsSceneDragMove);
+ makeDndEvent(&event, &view, QPointF(50, 75));
+ event.setMimeData(&data);
+ qApp->sendEvent(&scene, &event);
+ QVERIFY(event.isAccepted());
+ }
+ {
+ QGraphicsSceneDragDropEvent event(QEvent::GraphicsSceneDrop);
+ makeDndEvent(&event, &view, QPointF(50, 75));
+ event.setMimeData(&data);
+ qApp->sendEvent(&scene, &event);
+ QVERIFY(event.isAccepted());
+ }
+ QCOMPARE(edit->text(), QString("hei"));
+}
+
+void tst_QGraphicsProxyWidget::windowFlags_data()
+{
+ QTest::addColumn<int>("proxyFlags");
+ QTest::addColumn<int>("widgetFlags");
+ QTest::addColumn<int>("resultingProxyFlags");
+ QTest::addColumn<int>("resultingWidgetFlags");
+
+ QTest::newRow("proxy(0) widget(0)") << 0 << 0 << 0 << int(Qt::Window);
+ QTest::newRow("proxy(window)") << int(Qt::Window) << 0 << int(Qt::Window) << int(Qt::Window);
+ QTest::newRow("proxy(window) widget(window)") << int(Qt::Window) << int(Qt::Window) << int(Qt::Window) << int(Qt::Window);
+ QTest::newRow("proxy(0) widget(window)") << int(0) << int(Qt::Window) << int(0) << int(Qt::Window);
+}
+
+void tst_QGraphicsProxyWidget::windowFlags()
+{
+ QFETCH(int, proxyFlags);
+ QFETCH(int, widgetFlags);
+ QFETCH(int, resultingProxyFlags);
+ QFETCH(int, resultingWidgetFlags);
+ Qt::WindowFlags proxyWFlags = Qt::WindowFlags(proxyFlags);
+ Qt::WindowFlags widgetWFlags = Qt::WindowFlags(widgetFlags);
+ Qt::WindowFlags resultingProxyWFlags = Qt::WindowFlags(resultingProxyFlags);
+ Qt::WindowFlags resultingWidgetWFlags = Qt::WindowFlags(resultingWidgetFlags);
+
+ QGraphicsProxyWidget proxy(0, proxyWFlags);
+ QVERIFY((proxy.windowFlags() & proxyWFlags) == proxyWFlags);
+
+ QWidget *widget = new QWidget(0, widgetWFlags);
+ QVERIFY((widget->windowFlags() & widgetWFlags) == widgetWFlags);
+
+ proxy.setWidget(widget);
+
+ if (resultingProxyFlags == 0)
+ QVERIFY(!proxy.windowFlags());
+ else
+ QVERIFY((proxy.windowFlags() & resultingProxyWFlags) == resultingProxyWFlags);
+ QVERIFY((widget->windowFlags() & resultingWidgetWFlags) == resultingWidgetWFlags);
+}
+
+void tst_QGraphicsProxyWidget::comboboxWindowFlags()
+{
+ QComboBox *comboBox = new QComboBox;
+ comboBox->addItem("Item 1");
+ comboBox->addItem("Item 2");
+ comboBox->addItem("Item 3");
+ QWidget *embedWidget = comboBox;
+
+ QGraphicsScene scene;
+ QGraphicsProxyWidget *proxy = scene.addWidget(embedWidget);
+ proxy->setWindowFlags(Qt::Window);
+ QVERIFY(embedWidget->isWindow());
+ QVERIFY(proxy->isWindow());
+
+ comboBox->showPopup();
+
+ QCOMPARE(proxy->childItems().size(), 1);
+ QGraphicsItem *popupProxy = proxy->childItems().first();
+ QVERIFY(popupProxy->isWindow());
+ QVERIFY((static_cast<QGraphicsWidget *>(popupProxy)->windowFlags() & Qt::Popup) == Qt::Popup);
+}
+
+void tst_QGraphicsProxyWidget::updateAndDelete()
+{
+ QGraphicsScene scene;
+ QGraphicsProxyWidget *proxy = scene.addWidget(new QPushButton("Hello World"));
+ View view(&scene);
+ view.show();
+#ifdef Q_WS_X11
+ qt_x11_wait_for_window_manager(&view);
+#endif
+ QTRY_VERIFY(view.npaints > 0);
+
+ const QRect itemDeviceBoundingRect = proxy->deviceTransform(view.viewportTransform())
+ .mapRect(proxy->boundingRect()).toRect();
+ const QRegion expectedRegion = itemDeviceBoundingRect.adjusted(-2, -2, 2, 2);
+
+ view.npaints = 0;
+ view.paintEventRegion = QRegion();
+
+ // Update and hide.
+ proxy->update();
+ proxy->hide();
+ QTRY_COMPARE(view.npaints, 1);
+ QCOMPARE(view.paintEventRegion, expectedRegion);
+
+ proxy->show();
+ QTest::qWait(50);
+ view.npaints = 0;
+ view.paintEventRegion = QRegion();
+
+ // Update and delete.
+ proxy->update();
+ delete proxy;
+ QTRY_COMPARE(view.npaints, 1);
+ QCOMPARE(view.paintEventRegion, expectedRegion);
+}
+
+class InputMethod_LineEdit : public QLineEdit
+{
+ bool event(QEvent *e)
+ {
+ if (e->type() == QEvent::InputMethod)
+ ++inputMethodEvents;
+ return QLineEdit::event(e);
+ }
+public:
+ int inputMethodEvents;
+};
+
+void tst_QGraphicsProxyWidget::inputMethod()
+{
+ QGraphicsScene scene;
+
+ // check that the proxy is initialized with the correct input method sensitivity
+ for (int i = 0; i < 2; ++i)
+ {
+ QLineEdit *lineEdit = new QLineEdit;
+ lineEdit->setAttribute(Qt::WA_InputMethodEnabled, !!i);
+ QGraphicsProxyWidget *proxy = scene.addWidget(lineEdit);
+ QCOMPARE(!!(proxy->flags() & QGraphicsItem::ItemAcceptsInputMethod), !!i);
+ }
+
+ // check that input method events are only forwarded to widgets with focus
+ for (int i = 0; i < 2; ++i)
+ {
+ InputMethod_LineEdit *lineEdit = new InputMethod_LineEdit;
+ lineEdit->setAttribute(Qt::WA_InputMethodEnabled, true);
+ QGraphicsProxyWidget *proxy = scene.addWidget(lineEdit);
+
+ if (i)
+ lineEdit->setFocus();
+
+ lineEdit->inputMethodEvents = 0;
+ QInputMethodEvent event;
+ qApp->sendEvent(proxy, &event);
+ QCOMPARE(lineEdit->inputMethodEvents, i);
+ }
+
+ scene.clear();
+ QGraphicsView view(&scene);
+ QWidget *w = new QWidget;
+ w->setLayout(new QVBoxLayout(w));
+ QLineEdit *lineEdit = new QLineEdit;
+ lineEdit->setEchoMode(QLineEdit::Password);
+ w->layout()->addWidget(lineEdit);
+ lineEdit->setAttribute(Qt::WA_InputMethodEnabled, true);
+ QGraphicsProxyWidget *proxy = scene.addWidget(w);
+ view.show();
+ QTest::qWaitForWindowShown(&view);
+ QTRY_VERIFY(!(proxy->flags() & QGraphicsItem::ItemAcceptsInputMethod));
+ lineEdit->setFocus();
+ QVERIFY((proxy->flags() & QGraphicsItem::ItemAcceptsInputMethod));
+}
+
+void tst_QGraphicsProxyWidget::clickFocus()
+{
+ QGraphicsScene scene;
+ scene.setItemIndexMethod(QGraphicsScene::NoIndex);
+ QGraphicsProxyWidget *proxy = scene.addWidget(new QLineEdit);
+
+ QGraphicsView view(&scene);
+
+ {
+ EventSpy proxySpy(proxy);
+ EventSpy widgetSpy(proxy->widget());
+
+ view.setFrameStyle(0);
+ view.resize(300, 300);
+ view.show();
+#ifdef Q_WS_X11
+ qt_x11_wait_for_window_manager(&view);
+#endif
+ QApplication::setActiveWindow(&view);
+ QTRY_COMPARE(QApplication::activeWindow(), (QWidget*)&view);
+
+ QVERIFY(!proxy->hasFocus());
+ QVERIFY(!proxy->widget()->hasFocus());
+
+ QCOMPARE(proxySpy.counts[QEvent::FocusIn], 0);
+ QCOMPARE(proxySpy.counts[QEvent::FocusOut], 0);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusIn], 0);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusOut], 0);
+
+ QPointF lineEditCenter = proxy->mapToScene(proxy->boundingRect().center());
+ // Spontaneous mouse click sets focus on a clickable widget.
+ for (int retry = 0; retry < 50 && !proxy->hasFocus(); retry++)
+ QTest::mouseClick(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(lineEditCenter));
+ QVERIFY(proxy->hasFocus());
+ QVERIFY(proxy->widget()->hasFocus());
+ QCOMPARE(proxySpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusIn], 1);
+
+ scene.setFocusItem(0);
+ QVERIFY(!proxy->hasFocus());
+ QVERIFY(!proxy->widget()->hasFocus());
+ QCOMPARE(proxySpy.counts[QEvent::FocusOut], 1);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusOut], 1);
+
+ // Non-spontaneous mouse click sets focus if the widget has been clicked before
+ {
+ QGraphicsSceneMouseEvent event(QEvent::GraphicsSceneMousePress);
+ event.setScenePos(lineEditCenter);
+ event.setButton(Qt::LeftButton);
+ qApp->sendEvent(&scene, &event);
+ QVERIFY(proxy->hasFocus());
+ QVERIFY(proxy->widget()->hasFocus());
+ QCOMPARE(proxySpy.counts[QEvent::FocusIn], 2);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusIn], 2);
+ }
+ }
+
+ scene.setFocusItem(0);
+ proxy->setWidget(new QLineEdit); // resets focusWidget
+
+ {
+ QPointF lineEditCenter = proxy->mapToScene(proxy->boundingRect().center());
+ EventSpy proxySpy(proxy);
+ EventSpy widgetSpy(proxy->widget());
+ QVERIFY(!proxy->hasFocus());
+ QVERIFY(!proxy->widget()->hasFocus());
+ QCOMPARE(proxySpy.counts[QEvent::FocusOut], 0);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusOut], 0);
+
+ // Non-spontaneous mouse click does not set focus on the embedded widget.
+ {
+ QGraphicsSceneMouseEvent event(QEvent::GraphicsSceneMousePress);
+ event.setScenePos(lineEditCenter);
+ event.setButton(Qt::LeftButton);
+ qApp->sendEvent(&scene, &event);
+ QVERIFY(!proxy->hasFocus());
+ QVERIFY(!proxy->widget()->hasFocus());
+ QCOMPARE(proxySpy.counts[QEvent::FocusIn], 0);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusIn], 0);
+ }
+
+ scene.setFocusItem(0);
+ QVERIFY(!proxy->hasFocus());
+ QVERIFY(!proxy->widget()->hasFocus());
+ QCOMPARE(proxySpy.counts[QEvent::FocusOut], 0);
+ QCOMPARE(widgetSpy.counts[QEvent::FocusOut], 0);
+
+ // Spontaneous click on non-clickable widget does not give focus.
+ proxy->widget()->setFocusPolicy(Qt::NoFocus);
+ QTest::mouseClick(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(lineEditCenter));
+ QVERIFY(!proxy->hasFocus());
+ QVERIFY(!proxy->widget()->hasFocus());
+
+ // Multiple clicks should only result in one FocusIn.
+ proxy->widget()->setFocusPolicy(Qt::StrongFocus);
+ scene.setFocusItem(0);
+ QVERIFY(!proxy->hasFocus());
+ QVERIFY(!proxy->widget()->hasFocus());
+ QTest::mouseClick(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(lineEditCenter));
+ QTest::mouseClick(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(lineEditCenter));
+ QVERIFY(proxy->hasFocus());
+ QVERIFY(proxy->widget()->hasFocus());
+ QCOMPARE(widgetSpy.counts[QEvent::FocusIn], 1);
+ QCOMPARE(proxySpy.counts[QEvent::FocusIn], 1);
+ }
+}
+
+void tst_QGraphicsProxyWidget::windowFrameMargins()
+{
+ // Make sure the top margin is non-zero when passing Qt::Window.
+ QGraphicsProxyWidget *proxy = new QGraphicsProxyWidget(0, Qt::Window);
+
+ qreal left, top, right, bottom;
+ proxy->getWindowFrameMargins(&left, &top, &right, &bottom);
+ QVERIFY(top > 0);
+
+ proxy->setWidget(new QPushButton("testtest"));
+ proxy->getWindowFrameMargins(&left, &top, &right, &bottom);
+ QVERIFY(top > 0);
+
+ QGraphicsScene scene;
+ scene.addItem(proxy);
+ proxy->getWindowFrameMargins(&left, &top, &right, &bottom);
+ QVERIFY(top > 0);
+
+ proxy->unsetWindowFrameMargins();
+ proxy->getWindowFrameMargins(&left, &top, &right, &bottom);
+ QVERIFY(top > 0);
+}
+
+class HoverButton : public QPushButton
+{
+public:
+ HoverButton(QWidget *parent = 0) : QPushButton(parent), hoverLeaveReceived(false)
+ {}
+
+ bool hoverLeaveReceived;
+
+ bool event(QEvent* e)
+ {
+ if(QEvent::HoverLeave == e->type())
+ hoverLeaveReceived = true;
+ return QPushButton::event(e);
+ }
+};
+
+class Scene : public QGraphicsScene
+{
+Q_OBJECT
+public:
+ Scene() {
+ QWidget *background = new QWidget;
+ background->setGeometry(0, 0, 500, 500);
+ hoverButton = new HoverButton;
+ hoverButton->setParent(background);
+ hoverButton->setText("Second button");
+ hoverButton->setGeometry(10, 10, 200, 50);
+ addWidget(background);
+
+ QPushButton *hideButton = new QPushButton("I'm a button with a very very long text");
+ hideButton->setGeometry(10, 10, 400, 50);
+ topButton = addWidget(hideButton);
+ connect(hideButton, SIGNAL(clicked()), this, SLOT(hideButton()));
+ topButton->setFocus();
+ }
+
+ QGraphicsProxyWidget *topButton;
+ HoverButton *hoverButton;
+
+public slots:
+ void hideButton() {
+ QCursor::setPos(600,600);
+ topButton->hide();
+ }
+};
+
+void tst_QGraphicsProxyWidget::QTBUG_6986_sendMouseEventToAlienWidget()
+{
+#if defined(Q_OS_MAC) || defined(Q_OS_WIN) || defined(QT_NO_CURSOR)
+ QSKIP("Test case unstable on this platform", SkipAll);
+#endif
+ QGraphicsView view;
+ Scene scene;
+ view.setScene(&scene);
+ view.resize(600, 600);
+ QApplication::setActiveWindow(&view);
+ view.show();
+ QTest::qWaitForWindowShown(&view);
+ QTRY_COMPARE(QApplication::activeWindow(), &view);
+ QCursor::setPos(view.mapToGlobal(view.mapFromScene(scene.topButton->boundingRect().center())));
+ QTest::mouseClick(view.viewport(), Qt::LeftButton, 0, view.mapFromScene(scene.topButton->scenePos()));
+ QTRY_COMPARE(scene.hoverButton->hoverLeaveReceived, true);
+}
+
+QTEST_MAIN(tst_QGraphicsProxyWidget)
+#include "tst_qgraphicsproxywidget.moc"