QWidget: Call appropriate QWindow method from setGeometry_sys()
When calling resize() from showEvent(), we'd set the full geometry on the widget's QWindow. This resulted in the top-level window being moved to the top-left corner, even though no other call to move() or setGeometry() had happened before. The solution consists on calling the proper QWindow methods depending on whether setGeometry_sys() is called for a move, a resize or both. Furthermore, this needs QWindow::resize() to set its position policy to frame-exclusive. The documentation states that is already the case and we're setting the full geometry on the platform window, so we need to convey that bit of information. This also solves the age-old conundrum: "### why do we have isMove as a parameter?" Change-Id: I2e00fd632929ade14b35ae5e6495ed1ab176d32f Task-number: QTBUG-56277 Reviewed-by: Gabriel de Dietrich <gabriel.dedietrich@qt.io> Reviewed-by: Friedemann Kleint <Friedemann.Kleint@qt.io>
This commit is contained in:
parent
1bbad92e83
commit
b65e30c861
@ -1759,6 +1759,7 @@ void QWindow::resize(int w, int h)
|
|||||||
void QWindow::resize(const QSize &newSize)
|
void QWindow::resize(const QSize &newSize)
|
||||||
{
|
{
|
||||||
Q_D(QWindow);
|
Q_D(QWindow);
|
||||||
|
d->positionPolicy = QWindowPrivate::WindowFrameExclusive;
|
||||||
if (d->platformWindow) {
|
if (d->platformWindow) {
|
||||||
d->platformWindow->setGeometry(QHighDpi::toNativePixels(QRect(position(), newSize), this));
|
d->platformWindow->setGeometry(QHighDpi::toNativePixels(QRect(position(), newSize), this));
|
||||||
} else {
|
} else {
|
||||||
|
@ -7255,7 +7255,8 @@ void QWidgetPrivate::setGeometry_sys(int x, int y, int w, int h, bool isMove)
|
|||||||
QRect r(x, y, w, h);
|
QRect r(x, y, w, h);
|
||||||
|
|
||||||
bool isResize = olds != r.size();
|
bool isResize = olds != r.size();
|
||||||
isMove = oldp != r.topLeft(); //### why do we have isMove as a parameter?
|
if (!isMove)
|
||||||
|
isMove = oldp != r.topLeft();
|
||||||
|
|
||||||
|
|
||||||
// We only care about stuff that changes the geometry, or may
|
// We only care about stuff that changes the geometry, or may
|
||||||
@ -7289,12 +7290,17 @@ void QWidgetPrivate::setGeometry_sys(int x, int y, int w, int h, bool isMove)
|
|||||||
|
|
||||||
if (q->isVisible()) {
|
if (q->isVisible()) {
|
||||||
if (!q->testAttribute(Qt::WA_DontShowOnScreen) && !q->testAttribute(Qt::WA_OutsideWSRange)) {
|
if (!q->testAttribute(Qt::WA_DontShowOnScreen) && !q->testAttribute(Qt::WA_OutsideWSRange)) {
|
||||||
if (q->windowHandle()) {
|
if (QWindow *win = q->windowHandle()) {
|
||||||
if (q->isWindow()) {
|
if (q->isWindow()) {
|
||||||
q->windowHandle()->setGeometry(q->geometry());
|
if (isResize && !isMove)
|
||||||
|
win->resize(w, h);
|
||||||
|
else if (isMove && !isResize)
|
||||||
|
win->setPosition(x, y);
|
||||||
|
else
|
||||||
|
win->setGeometry(q->geometry());
|
||||||
} else {
|
} else {
|
||||||
QPoint posInNativeParent = q->mapTo(q->nativeParentWidget(),QPoint());
|
QPoint posInNativeParent = q->mapTo(q->nativeParentWidget(),QPoint());
|
||||||
q->windowHandle()->setGeometry(QRect(posInNativeParent,r.size()));
|
win->setGeometry(QRect(posInNativeParent,r.size()));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (needsShow)
|
if (needsShow)
|
||||||
|
@ -99,6 +99,8 @@ private slots:
|
|||||||
void tst_eventfilter_on_toplevel();
|
void tst_eventfilter_on_toplevel();
|
||||||
|
|
||||||
void QTBUG_50561_QCocoaBackingStore_paintDevice_crash();
|
void QTBUG_50561_QCocoaBackingStore_paintDevice_crash();
|
||||||
|
|
||||||
|
void QTBUG_56277_resize_on_showEvent();
|
||||||
};
|
};
|
||||||
|
|
||||||
void tst_QWidget_window::initTestCase()
|
void tst_QWidget_window::initTestCase()
|
||||||
@ -861,5 +863,34 @@ void tst_QWidget_window::QTBUG_50561_QCocoaBackingStore_paintDevice_crash()
|
|||||||
w.close();
|
w.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class ResizedOnShowEventWidget : public QWidget
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void showEvent(QShowEvent *) override
|
||||||
|
{
|
||||||
|
const auto *primaryScreen = QApplication::primaryScreen();
|
||||||
|
auto newSize = primaryScreen->availableGeometry().size() / 4;
|
||||||
|
if (newSize == geometry().size())
|
||||||
|
newSize -= QSize(10, 10);
|
||||||
|
resize(newSize);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
void tst_QWidget_window::QTBUG_56277_resize_on_showEvent()
|
||||||
|
{
|
||||||
|
const auto platformName = QGuiApplication::platformName().toLower();
|
||||||
|
if (platformName != "cocoa" && platformName != "windows")
|
||||||
|
QSKIP("This can only be consistently tested on desktop platforms with well-known behavior.");
|
||||||
|
|
||||||
|
ResizedOnShowEventWidget w;
|
||||||
|
w.show();
|
||||||
|
QVERIFY(QTest::qWaitForWindowExposed(&w));
|
||||||
|
const auto *screen = w.windowHandle()->screen();
|
||||||
|
const auto geometry = w.geometry();
|
||||||
|
const int frameHeight = geometry.top() - w.frameGeometry().top();
|
||||||
|
const int topmostY = screen->availableGeometry().top() + frameHeight;
|
||||||
|
QVERIFY(geometry.top() > topmostY || geometry.left() > screen->availableGeometry().left());
|
||||||
|
}
|
||||||
|
|
||||||
QTEST_MAIN(tst_QWidget_window)
|
QTEST_MAIN(tst_QWidget_window)
|
||||||
#include "tst_qwidget_window.moc"
|
#include "tst_qwidget_window.moc"
|
||||||
|
Loading…
Reference in New Issue
Block a user