blob: 8edb3a3e9b829c1a794587bc2b8f7708ccef7024 [file] [log] [blame]
/****************************************************************************
**
** Copyright (C) 2017 The Qt Company Ltd.
** Contact: https://www.qt.io/licensing/
**
** This file is part of the test suite of the Qt Toolkit.
**
** $QT_BEGIN_LICENSE:GPL-EXCEPT$
** Commercial License Usage
** Licensees holding valid commercial Qt licenses may use this file in
** accordance with the commercial license agreement provided with the
** Software or, alternatively, in accordance with the terms contained in
** a written agreement between you and The Qt Company. For licensing terms
** and conditions see https://www.qt.io/terms-conditions. For further
** information use the contact form at https://www.qt.io/contact-us.
**
** GNU General Public License Usage
** Alternatively, this file may be used under the terms of the GNU
** General Public License version 3 as published by the Free Software
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
** included in the packaging of this file. Please review the following
** information to ensure the GNU General Public License requirements will
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
**
** $QT_END_LICENSE$
**
****************************************************************************/
#include "itemwindow.h"
#include <QtGui/QGuiApplication>
#include <QtCore/QCommandLineOption>
#include <QtCore/QCommandLineParser>
#include <QtCore/QDebug>
#include <QtCore/QStringList>
#ifdef Q_OS_WIN
# include <qpa/qplatformnativeinterface.h>
# include <QtCore/QMetaObject>
# include <QtCore/qt_windows.h>
#endif
#include <eventfilter.h> // diaglib
#include <qwindowdump.h>
#include <iostream>
QT_USE_NAMESPACE
static const char usage[] =
"\nEmbeds a QWindow into a native foreign window passed on the command line.\n"
"When no window ID is passed, a test window is created (Windows only).";
static QString windowTitle()
{
return QLatin1String(QT_VERSION_STR) + QLatin1Char(' ') + QGuiApplication::platformName();
}
#ifdef Q_OS_WIN
// Helper to create a native test window (Windows)
static QString registerWindowClass(const QString &name)
{
QString result;
void *proc = DefWindowProc;
QPlatformNativeInterface *ni = QGuiApplication::platformNativeInterface();
if (!QMetaObject::invokeMethod(ni, "registerWindowClass", Qt::DirectConnection,
Q_RETURN_ARG(QString, result),
Q_ARG(QString, name),
Q_ARG(void *, proc))) {
qWarning("registerWindowClass failed");
}
return result;
}
static HWND createNativeWindow(const QString &name)
{
const HWND hwnd =
CreateWindowEx(0, reinterpret_cast<const wchar_t *>(name.utf16()),
L"NativeWindow", WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
0, 0, GetModuleHandle(NULL), NULL);
if (!hwnd) {
qErrnoWarning("Cannot create window \"%s\"", qPrintable(name));
return 0;
}
const QString text = windowTitle() + QLatin1String(" 0x") + QString::number(quint64(hwnd), 16);
SetWindowText(hwnd, reinterpret_cast<const wchar_t *>(text.utf16()));
return hwnd;
}
#endif // Q_OS_WIN
// Helper functions for simple management of native windows.
static WId createNativeTestWindow()
{
WId result = 0;
#ifdef Q_OS_WIN
const QString className = registerWindowClass(QLatin1String("TestClass") + windowTitle());
const HWND nativeWin = createNativeWindow(className);
result = WId(nativeWin);
#else // Q_OS_WIN
Q_UNIMPLEMENTED();
#endif
return result;
}
static void showNativeWindow(WId wid)
{
#ifdef Q_OS_WIN
ShowWindow(HWND(wid), SW_SHOW);
#else // Q_OS_WIN
Q_UNUSED(wid)
Q_UNIMPLEMENTED();
#endif
}
static void setFocusToNativeWindow(WId wid)
{
#ifdef Q_OS_WIN
SetFocus(HWND(wid));
#else // Q_OS_WIN
Q_UNUSED(wid)
Q_UNIMPLEMENTED();
#endif
}
static void destroyNativeWindow(WId wid)
{
#ifdef Q_OS_WIN
DestroyWindow(HWND(wid));
#else // Q_OS_WIN
Q_UNUSED(wid)
Q_UNIMPLEMENTED();
#endif
}
// Main test window to be embedded into foreign window with some buttons.
class EmbeddedTestWindow : public ItemWindow {
Q_OBJECT
public:
explicit EmbeddedTestWindow(QWindow *parent = nullptr) : ItemWindow(parent)
{
const int spacing = 10;
const QSize buttonSize(100, 30);
const int width = 3 * buttonSize.width() + 4 * spacing;
QPoint pos(spacing, spacing);
addItem(new TextItem(::windowTitle(), QRect(pos, QSize(width - 2 * spacing, buttonSize.height())),
Qt::white));
pos.ry() += 2 * spacing + buttonSize.height();
ButtonItem *mgi = new ButtonItem("Map to global", QRect(pos, buttonSize),
QColor(Qt::yellow).lighter(), this);
connect(mgi, &ButtonItem::clicked, this, &EmbeddedTestWindow::testMapToGlobal);
addItem(mgi);
pos.rx() += buttonSize.width() + spacing;
ButtonItem *di = new ButtonItem("Dump Wins", QRect(pos, buttonSize),
QColor(Qt::cyan).lighter(), this);
connect(di, &ButtonItem::clicked, this, [] () { QtDiag::dumpAllWindows(); });
addItem(di);
pos.rx() += buttonSize.width() + spacing;
ButtonItem *qi = new ButtonItem("Quit", QRect(pos, buttonSize),
QColor(Qt::red).lighter(), this);
qi->setShortcut(Qt::CTRL + Qt::Key_Q);
connect(qi, &ButtonItem::clicked, qApp, &QCoreApplication::quit);
addItem(qi);
setBackground(Qt::lightGray);
resize(width, pos.y() + buttonSize.height() + spacing);
}
public slots:
void testMapToGlobal();
};
void EmbeddedTestWindow::testMapToGlobal()
{
const QPoint globalPos = mapToGlobal(QPoint(0,0));
qDebug() << "mapToGlobal(QPoint(0,0)" << globalPos
<< "cursor at:" << QCursor::pos();
}
struct EventFilterOption
{
const char *name;
const char *description;
QtDiag::EventFilter::EventCategories categories;
};
EventFilterOption eventFilterOptions[] = {
{"mouse-events", "Dump mouse events.", QtDiag::EventFilter::MouseEvents},
{"keyboard-events", "Dump keyboard events.", QtDiag::EventFilter::KeyEvents},
{"state-events", "Dump state/focus change events.", QtDiag::EventFilter::StateChangeEvents | QtDiag::EventFilter::FocusEvents}
};
static inline bool isOptionSet(int argc, char *argv[], const char *option)
{
return (argv + argc) !=
std::find_if(argv + 1, argv + argc,
[option] (const char *arg) { return !qstrcmp(arg, option); });
}
int main(int argc, char *argv[])
{
// Check for no scaling before QApplication is instantiated.
if (isOptionSet(argc, argv, "-s"))
QCoreApplication::setAttribute(Qt::AA_DisableHighDpiScaling);
QCoreApplication::setApplicationVersion(QLatin1String(QT_VERSION_STR));
QGuiApplication::setApplicationDisplayName("Foreign Window Embedding Tester");
QGuiApplication app(argc, argv);
// Process command line
QCommandLineParser parser;
parser.setSingleDashWordOptionMode(QCommandLineParser::ParseAsLongOptions);
parser.setApplicationDescription(QLatin1String(usage));
parser.addHelpOption();
parser.addVersionOption();
QCommandLineOption noScalingDummy(QStringLiteral("s"),
QStringLiteral("Disable High DPI scaling."));
parser.addOption(noScalingDummy);
const int eventFilterOptionCount = int(sizeof(eventFilterOptions) / sizeof(eventFilterOptions[0]));
for (int i = 0; i < eventFilterOptionCount; ++i) {
parser.addOption(QCommandLineOption(QLatin1String(eventFilterOptions[i].name),
QLatin1String(eventFilterOptions[i].description)));
}
parser.addPositionalArgument(QStringLiteral("[windows]"), QStringLiteral("Window ID."));
parser.process(QCoreApplication::arguments());
QtDiag::EventFilter::EventCategories eventCategories = 0;
for (int i = 0; i < eventFilterOptionCount; ++i) {
if (parser.isSet(QLatin1String(eventFilterOptions[i].name)))
eventCategories |= eventFilterOptions[i].categories;
}
if (eventCategories)
app.installEventFilter(new QtDiag::EventFilter(eventCategories, &app));
// Obtain foreign window to test with.
WId testForeignWinId = 0;
bool createdTestWindow = false;
if (parser.positionalArguments().isEmpty()) {
testForeignWinId = createNativeTestWindow();
if (!testForeignWinId)
parser.showHelp(-1);
showNativeWindow(testForeignWinId);
createdTestWindow = true;
} else {
bool ok;
const QString &winIdArgument = parser.positionalArguments().constFirst();
testForeignWinId = winIdArgument.toULongLong(&ok, 0);
if (!ok) {
std::cerr << "Invalid window id: \"" << qPrintable(winIdArgument) << "\"\n";
return -1;
}
}
if (!testForeignWinId)
parser.showHelp(1);
QWindow *foreignWindow = QWindow::fromWinId(testForeignWinId);
if (!foreignWindow)
return -2;
foreignWindow->setObjectName("ForeignWindow");
EmbeddedTestWindow *embeddedTestWindow = new EmbeddedTestWindow(foreignWindow);
embeddedTestWindow->setObjectName("EmbeddedTestWindow");
embeddedTestWindow->show();
setFocusToNativeWindow(embeddedTestWindow->winId()); // Windows: Set keyboard focus.
const int exitCode = app.exec();
delete embeddedTestWindow;
delete foreignWindow;
if (createdTestWindow)
destroyNativeWindow(testForeignWinId);
return exitCode;
}
#include "main.moc"