mirror of
https://github.com/crystalidea/qt6windows7.git
synced 2025-01-24 04:44:31 +08:00
331 lines
12 KiB
C++
331 lines
12 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
|
|
|
|
#include <qpainter.h>
|
|
#include <qrasterwindow.h>
|
|
#include <qscreen.h>
|
|
#include <qpa/qwindowsysteminterface.h>
|
|
#include <qpa/qplatformintegration.h>
|
|
#include <private/qguiapplication_p.h>
|
|
#include <private/qhighdpiscaling_p.h>
|
|
|
|
#include <QTest>
|
|
#include <QSignalSpy>
|
|
|
|
class tst_QScreen: public QObject
|
|
{
|
|
Q_OBJECT
|
|
|
|
private slots:
|
|
void angleBetween_data();
|
|
void angleBetween();
|
|
void transformBetween_data();
|
|
void transformBetween();
|
|
void orientationChange();
|
|
|
|
void grabWindow_data();
|
|
void grabWindow();
|
|
};
|
|
|
|
void tst_QScreen::angleBetween_data()
|
|
{
|
|
QTest::addColumn<uint>("oa");
|
|
QTest::addColumn<uint>("ob");
|
|
QTest::addColumn<int>("expected");
|
|
|
|
QTest::newRow("Portrait Portrait")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< 0;
|
|
|
|
QTest::newRow("Portrait Landscape")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::LandscapeOrientation)
|
|
<< 270;
|
|
|
|
QTest::newRow("Portrait InvertedPortrait")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::InvertedPortraitOrientation)
|
|
<< 180;
|
|
|
|
QTest::newRow("Portrait InvertedLandscape")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::InvertedLandscapeOrientation)
|
|
<< 90;
|
|
|
|
QTest::newRow("InvertedLandscape InvertedPortrait")
|
|
<< uint(Qt::InvertedLandscapeOrientation)
|
|
<< uint(Qt::InvertedPortraitOrientation)
|
|
<< 90;
|
|
|
|
QTest::newRow("InvertedLandscape Landscape")
|
|
<< uint(Qt::InvertedLandscapeOrientation)
|
|
<< uint(Qt::LandscapeOrientation)
|
|
<< 180;
|
|
|
|
QTest::newRow("Landscape Primary")
|
|
<< uint(Qt::LandscapeOrientation)
|
|
<< uint(Qt::PrimaryOrientation)
|
|
<< QGuiApplication::primaryScreen()->angleBetween(Qt::LandscapeOrientation, QGuiApplication::primaryScreen()->primaryOrientation());
|
|
}
|
|
|
|
void tst_QScreen::angleBetween()
|
|
{
|
|
QFETCH( uint, oa );
|
|
QFETCH( uint, ob );
|
|
QFETCH( int, expected );
|
|
|
|
Qt::ScreenOrientation a = Qt::ScreenOrientation(oa);
|
|
Qt::ScreenOrientation b = Qt::ScreenOrientation(ob);
|
|
|
|
QCOMPARE(QGuiApplication::primaryScreen()->angleBetween(a, b), expected);
|
|
QCOMPARE(QGuiApplication::primaryScreen()->angleBetween(b, a), (360 - expected) % 360);
|
|
}
|
|
|
|
void tst_QScreen::transformBetween_data()
|
|
{
|
|
QTest::addColumn<uint>("oa");
|
|
QTest::addColumn<uint>("ob");
|
|
QTest::addColumn<QRect>("rect");
|
|
QTest::addColumn<QTransform>("expected");
|
|
|
|
QRect rect(0, 0, 480, 640);
|
|
|
|
QTest::newRow("Portrait Portrait")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< rect
|
|
<< QTransform();
|
|
|
|
QTest::newRow("Portrait Landscape")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::LandscapeOrientation)
|
|
<< rect
|
|
<< QTransform(0, -1, 1, 0, 0, rect.height());
|
|
|
|
QTest::newRow("Portrait InvertedPortrait")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::InvertedPortraitOrientation)
|
|
<< rect
|
|
<< QTransform(-1, 0, 0, -1, rect.width(), rect.height());
|
|
|
|
QTest::newRow("Portrait InvertedLandscape")
|
|
<< uint(Qt::PortraitOrientation)
|
|
<< uint(Qt::InvertedLandscapeOrientation)
|
|
<< rect
|
|
<< QTransform(0, 1, -1, 0, rect.width(), 0);
|
|
|
|
QTest::newRow("InvertedLandscape InvertedPortrait")
|
|
<< uint(Qt::InvertedLandscapeOrientation)
|
|
<< uint(Qt::InvertedPortraitOrientation)
|
|
<< rect
|
|
<< QTransform(0, 1, -1, 0, rect.width(), 0);
|
|
|
|
QTest::newRow("InvertedLandscape Landscape")
|
|
<< uint(Qt::InvertedLandscapeOrientation)
|
|
<< uint(Qt::LandscapeOrientation)
|
|
<< rect
|
|
<< QTransform(-1, 0, 0, -1, rect.width(), rect.height());
|
|
|
|
QTest::newRow("Landscape Primary")
|
|
<< uint(Qt::LandscapeOrientation)
|
|
<< uint(Qt::PrimaryOrientation)
|
|
<< rect
|
|
<< QGuiApplication::primaryScreen()->transformBetween(Qt::LandscapeOrientation, QGuiApplication::primaryScreen()->primaryOrientation(), rect);
|
|
}
|
|
|
|
void tst_QScreen::transformBetween()
|
|
{
|
|
QFETCH( uint, oa );
|
|
QFETCH( uint, ob );
|
|
QFETCH( QRect, rect );
|
|
QFETCH( QTransform, expected );
|
|
|
|
Qt::ScreenOrientation a = Qt::ScreenOrientation(oa);
|
|
Qt::ScreenOrientation b = Qt::ScreenOrientation(ob);
|
|
|
|
QCOMPARE(QGuiApplication::primaryScreen()->transformBetween(a, b, rect), expected);
|
|
}
|
|
|
|
void tst_QScreen::orientationChange()
|
|
{
|
|
qRegisterMetaType<Qt::ScreenOrientation>("Qt::ScreenOrientation");
|
|
|
|
QScreen *screen = QGuiApplication::primaryScreen();
|
|
QSignalSpy spy(screen, SIGNAL(orientationChanged(Qt::ScreenOrientation)));
|
|
int expectedSignalCount = 0;
|
|
|
|
QWindowSystemInterface::handleScreenOrientationChange(screen, Qt::LandscapeOrientation);
|
|
QWindowSystemInterface::flushWindowSystemEvents();
|
|
QTRY_COMPARE(screen->orientation(), Qt::LandscapeOrientation);
|
|
QCOMPARE(spy.size(), ++expectedSignalCount);
|
|
|
|
QWindowSystemInterface::handleScreenOrientationChange(screen, Qt::PortraitOrientation);
|
|
QWindowSystemInterface::flushWindowSystemEvents();
|
|
QTRY_COMPARE(screen->orientation(), Qt::PortraitOrientation);
|
|
QCOMPARE(spy.size(), ++expectedSignalCount);
|
|
|
|
QWindowSystemInterface::handleScreenOrientationChange(screen, Qt::InvertedLandscapeOrientation);
|
|
QWindowSystemInterface::flushWindowSystemEvents();
|
|
QTRY_COMPARE(screen->orientation(), Qt::InvertedLandscapeOrientation);
|
|
QCOMPARE(spy.size(), ++expectedSignalCount);
|
|
|
|
QWindowSystemInterface::handleScreenOrientationChange(screen, Qt::InvertedPortraitOrientation);
|
|
QWindowSystemInterface::flushWindowSystemEvents();
|
|
QTRY_COMPARE(screen->orientation(), Qt::InvertedPortraitOrientation);
|
|
QCOMPARE(spy.size(), ++expectedSignalCount);
|
|
|
|
QWindowSystemInterface::handleScreenOrientationChange(screen, Qt::LandscapeOrientation);
|
|
QWindowSystemInterface::flushWindowSystemEvents();
|
|
QTRY_COMPARE(screen->orientation(), Qt::LandscapeOrientation);
|
|
QCOMPARE(spy.size(), ++expectedSignalCount);
|
|
}
|
|
|
|
void tst_QScreen::grabWindow_data()
|
|
{
|
|
if (!QGuiApplicationPrivate::platformIntegration()->hasCapability(QPlatformIntegration::ScreenWindowGrabbing)
|
|
|| (QGuiApplication::platformName().toLower() == QStringLiteral("xcb") && !qEnvironmentVariableIsEmpty("WAYLAND_DISPLAY")))
|
|
QSKIP("This platform does not support grabbing windows on screen.");
|
|
QTest::addColumn<int>("screenIndex");
|
|
QTest::addColumn<QByteArray>("screenName");
|
|
QTest::addColumn<bool>("grabWindow");
|
|
QTest::addColumn<QRect>("windowRect");
|
|
QTest::addColumn<QRect>("grabRect");
|
|
|
|
int screenIndex = 0;
|
|
for (const auto screen : QGuiApplication::screens()) {
|
|
const QByteArray screenName = screen->name().toUtf8();
|
|
const QRect availableGeometry = screen->availableGeometry();
|
|
const QPoint topLeft = availableGeometry.topLeft() + QPoint(20, 20);
|
|
QTest::addRow("%s - Window", screenName.data())
|
|
<< screenIndex << screenName << true << QRect(topLeft, QSize(200, 200)) << QRect(0, 0, -1, -1);
|
|
QTest::addRow("%s - Window Section", screenName.data())
|
|
<< screenIndex << screenName << true << QRect(topLeft, QSize(200, 200)) << QRect(50, 50, 100, 100);
|
|
QTest::addRow("%s - Screen", screenName.data())
|
|
<< screenIndex << screenName << false << QRect(topLeft, QSize(200, 200)) << QRect(0, 0, -1, -1);
|
|
QTest::addRow("%s - Screen Section", screenName.data())
|
|
<< screenIndex << screenName << false << QRect(topLeft, QSize(200, 200)) << QRect(topLeft, QSize(200, 200));
|
|
|
|
++screenIndex;
|
|
}
|
|
}
|
|
|
|
void tst_QScreen::grabWindow()
|
|
{
|
|
QFETCH(int, screenIndex);
|
|
QFETCH(QByteArray, screenName);
|
|
QFETCH(bool, grabWindow);
|
|
QFETCH(QRect, windowRect);
|
|
QFETCH(QRect, grabRect);
|
|
|
|
class Window : public QRasterWindow
|
|
{
|
|
public:
|
|
Window(QScreen *scr)
|
|
: image(scr->size(), QImage::Format_ARGB32_Premultiplied)
|
|
{
|
|
setFlags(Qt::CustomizeWindowHint|Qt::FramelessWindowHint|Qt::WindowStaysOnTopHint);
|
|
setScreen(scr);
|
|
image.setDevicePixelRatio(scr->devicePixelRatio());
|
|
}
|
|
QImage image;
|
|
|
|
protected:
|
|
void resizeEvent(QResizeEvent *e) override
|
|
{
|
|
const QSize sz = e->size();
|
|
image = image.scaled(sz * image.devicePixelRatio());
|
|
QPainter painter(&image);
|
|
painter.fillRect(0, 0, sz.width(), sz.height(), Qt::black);
|
|
painter.setPen(QPen(Qt::red, 2));
|
|
painter.drawLine(0, 0, sz.width(), sz.height());
|
|
painter.drawLine(0, sz.height(), sz.width(), 0);
|
|
painter.drawRect(0, 0, sz.width(), sz.height());
|
|
}
|
|
void paintEvent(QPaintEvent *) override
|
|
{
|
|
QPainter painter(this);
|
|
painter.drawImage(0, 0, image);
|
|
}
|
|
};
|
|
const auto screens = QGuiApplication::screens();
|
|
double highestDpr = 0;
|
|
for (auto screen : screens)
|
|
highestDpr = qMax(highestDpr, screen->devicePixelRatio());
|
|
|
|
QScreen *screen = screens.at(screenIndex);
|
|
QCOMPARE(screen->name().toUtf8(), screenName);
|
|
const double screenDpr = screen->devicePixelRatio();
|
|
|
|
if (QHighDpiScaling::isActive()) {
|
|
const float rawFactor = QHighDpiScaling::factor(screen);
|
|
const float roundedFactor = qRound(rawFactor);
|
|
if (!qFuzzyCompare(roundedFactor, rawFactor))
|
|
QSKIP("HighDPI enabled with non-integer factor. Skip due to possible rounding errors.");
|
|
}
|
|
|
|
Window window(screen);
|
|
window.setGeometry(windowRect);
|
|
#ifndef Q_OS_ANDROID
|
|
window.show();
|
|
#else
|
|
window.showNormal();
|
|
#endif
|
|
|
|
if (!QTest::qWaitForWindowExposed(&window))
|
|
QSKIP("Failed to expose window - aborting");
|
|
|
|
// this is necessary because of scrolling effects combined with potential slowness of VMs
|
|
QTest::qWait(1500);
|
|
|
|
QSize expectedGrabSize = grabRect.isValid()
|
|
? grabRect.size()
|
|
: (grabWindow ? windowRect.size() : screen->size());
|
|
// we ask for pixel coordinates, but will get a pixmap with device-specific DPR
|
|
expectedGrabSize *= screen->devicePixelRatio();
|
|
|
|
// the painted image will always be in the screen's DPR
|
|
QImage paintedImage = window.image;
|
|
QCOMPARE(paintedImage.devicePixelRatio(), screenDpr);
|
|
|
|
const QPixmap pixmap = screen->grabWindow(grabWindow
|
|
? window.winId()
|
|
: 0, grabRect.x(), grabRect.y(), grabRect.width(), grabRect.height());
|
|
|
|
QImage grabbedImage = pixmap.toImage();
|
|
const QSize grabbedSize = grabbedImage.size();
|
|
QCOMPARE(grabbedSize, expectedGrabSize);
|
|
|
|
QPoint pixelOffset = QPoint(0, 0);
|
|
if (!grabRect.isValid()) {
|
|
if (grabWindow) {
|
|
// if we grab the entire window, then the grabbed image should be as large as the window
|
|
QCOMPARE(grabbedImage.size(), paintedImage.size());
|
|
} else {
|
|
// if we grab the entire screen, then the grabbed image should be as large as the screen
|
|
QCOMPARE(grabbedImage.size(), screen->size() * screenDpr);
|
|
pixelOffset = window.geometry().topLeft() - screen->geometry().topLeft();
|
|
grabbedImage = grabbedImage.copy(QRect(pixelOffset * screenDpr, window.geometry().size() * screenDpr));
|
|
}
|
|
} else if (grabWindow) {
|
|
// if we grab the section, compare with the corresponding section from the painted image
|
|
const QRect sectionRect = QRect(grabRect.topLeft() * screenDpr,
|
|
grabRect.size() * screenDpr);
|
|
paintedImage = paintedImage.copy(sectionRect);
|
|
}
|
|
QCOMPARE(grabbedImage.size(), paintedImage.size());
|
|
|
|
// the two images might differ in format, or DPR, so instead of comparing them, sample a few pixels
|
|
for (auto point : {
|
|
QPoint(0, 0),
|
|
QPoint(5, 15),
|
|
QPoint(paintedImage.width() - 1, paintedImage.height() - 1),
|
|
QPoint(paintedImage.width() - 5, paintedImage.height() - 10)
|
|
}) {
|
|
QCOMPARE(grabbedImage.pixelColor(point), paintedImage.pixelColor(point));
|
|
}
|
|
}
|
|
|
|
#include <tst_qscreen.moc>
|
|
QTEST_MAIN(tst_QScreen);
|