xcb: Implement native window helper for embeddedwindows/foreign window test
Change-Id: I73720f8f49a5d7e5df7c95bf4b17ef910180e01c Reviewed-by: Liang Qi <liang.qi@qt.io> (cherry picked from commit 3cb3c1d9a8e5cfe262be8d6e9800f604d4d48c97) Reviewed-by: Qt Cherry-pick Bot <cherrypick_bot@qt-project.org>
This commit is contained in:
parent
d28a9f512e
commit
8efaa9f830
@ -14,7 +14,7 @@ qt_internal_add_test(tst_qwindow
|
|||||||
Qt::GuiPrivate
|
Qt::GuiPrivate
|
||||||
)
|
)
|
||||||
|
|
||||||
if(APPLE OR WIN32)
|
if(APPLE OR WIN32 OR QT_FEATURE_xcb)
|
||||||
qt_internal_add_test(tst_foreignwindow
|
qt_internal_add_test(tst_foreignwindow
|
||||||
LOWDPI
|
LOWDPI
|
||||||
SOURCES
|
SOURCES
|
||||||
@ -29,6 +29,10 @@ if(APPLE OR WIN32)
|
|||||||
set_source_files_properties(tst_foreignwindow.cpp PROPERTIES LANGUAGE OBJCXX)
|
set_source_files_properties(tst_foreignwindow.cpp PROPERTIES LANGUAGE OBJCXX)
|
||||||
set_property(TARGET tst_foreignwindow PROPERTY PROPERTY MACOSX_BUNDLE TRUE)
|
set_property(TARGET tst_foreignwindow PROPERTY PROPERTY MACOSX_BUNDLE TRUE)
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
|
if(QT_FEATURE_xcb)
|
||||||
|
target_link_libraries(tst_foreignwindow PRIVATE XCB::XCB)
|
||||||
|
endif()
|
||||||
endif()
|
endif()
|
||||||
|
|
||||||
## Scopes:
|
## Scopes:
|
||||||
|
@ -60,12 +60,15 @@ void tst_ForeignWindow::initialState()
|
|||||||
|
|
||||||
const QRect initialGeometry(123, 456, 321, 654);
|
const QRect initialGeometry(123, 456, 321, 654);
|
||||||
nativeWindow.setGeometry(initialGeometry);
|
nativeWindow.setGeometry(initialGeometry);
|
||||||
|
QTRY_COMPARE(nativeWindow.geometry(), initialGeometry);
|
||||||
|
|
||||||
std::unique_ptr<QWindow> foreignWindow(QWindow::fromWinId(nativeWindow));
|
std::unique_ptr<QWindow> foreignWindow(QWindow::fromWinId(nativeWindow));
|
||||||
QCOMPARE(nativeWindow.geometry(), initialGeometry);
|
QCOMPARE(nativeWindow.geometry(), initialGeometry);
|
||||||
|
|
||||||
// For extra bonus points, the foreign window should actually
|
// For extra bonus points, the foreign window should actually
|
||||||
// reflect the state of the native window.
|
// reflect the state of the native window.
|
||||||
|
if (!QGuiApplication::platformName().compare(QLatin1String("xcb"), Qt::CaseInsensitive))
|
||||||
|
QEXPECT_FAIL("", "QXcbWindow does not pick up foreign window geometry", Continue);
|
||||||
QCOMPARE(foreignWindow->geometry(), initialGeometry);
|
QCOMPARE(foreignWindow->geometry(), initialGeometry);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -83,7 +86,7 @@ void tst_ForeignWindow::embedForeignWindow()
|
|||||||
// As a prerequisite to that, we must be able to reparent the foreign window
|
// As a prerequisite to that, we must be able to reparent the foreign window
|
||||||
std::unique_ptr<QWindow> foreignWindow(QWindow::fromWinId(nativeWindow));
|
std::unique_ptr<QWindow> foreignWindow(QWindow::fromWinId(nativeWindow));
|
||||||
foreignWindow.release()->setParent(&parentWindow);
|
foreignWindow.release()->setParent(&parentWindow);
|
||||||
QCOMPARE(nativeWindow.parentWinId(), parentWindow.winId());
|
QTRY_COMPARE(nativeWindow.parentWinId(), parentWindow.winId());
|
||||||
}
|
}
|
||||||
|
|
||||||
#include <tst_foreignwindow.moc>
|
#include <tst_foreignwindow.moc>
|
||||||
|
@ -8,6 +8,10 @@ qt_internal_add_manual_test(embeddedwindows
|
|||||||
Qt::Gui
|
Qt::Gui
|
||||||
)
|
)
|
||||||
|
|
||||||
|
if(QT_FEATURE_xcb)
|
||||||
|
target_link_libraries(embeddedwindows PRIVATE XCB::XCB)
|
||||||
|
endif()
|
||||||
|
|
||||||
if(APPLE)
|
if(APPLE)
|
||||||
enable_language(OBJCXX)
|
enable_language(OBJCXX)
|
||||||
set_source_files_properties(main.cpp PROPERTIES LANGUAGE OBJCXX)
|
set_source_files_properties(main.cpp PROPERTIES LANGUAGE OBJCXX)
|
||||||
|
@ -3,7 +3,7 @@
|
|||||||
|
|
||||||
#include <QtGui>
|
#include <QtGui>
|
||||||
|
|
||||||
#if defined(Q_OS_MACOS) || defined(Q_OS_IOS) || defined(Q_OS_WIN)
|
#if defined(Q_OS_MACOS) || defined(Q_OS_IOS) || defined(Q_OS_WIN) || QT_CONFIG(xcb)
|
||||||
#include "../../shared/nativewindow.h"
|
#include "../../shared/nativewindow.h"
|
||||||
#define HAVE_NATIVE_WINDOW
|
#define HAVE_NATIVE_WINDOW
|
||||||
#endif
|
#endif
|
||||||
|
@ -12,6 +12,8 @@
|
|||||||
# define VIEW_BASE UIView
|
# define VIEW_BASE UIView
|
||||||
#elif defined(Q_OS_WIN)
|
#elif defined(Q_OS_WIN)
|
||||||
# include <winuser.h>
|
# include <winuser.h>
|
||||||
|
#elif QT_CONFIG(xcb)
|
||||||
|
# include <xcb/xcb.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
class NativeWindow
|
class NativeWindow
|
||||||
@ -21,7 +23,7 @@ public:
|
|||||||
NativeWindow();
|
NativeWindow();
|
||||||
~NativeWindow();
|
~NativeWindow();
|
||||||
|
|
||||||
operator WId() const { return reinterpret_cast<WId>(m_handle); }
|
operator WId() const;
|
||||||
WId parentWinId() const;
|
WId parentWinId() const;
|
||||||
|
|
||||||
void setGeometry(const QRect &rect);
|
void setGeometry(const QRect &rect);
|
||||||
@ -32,6 +34,8 @@ private:
|
|||||||
VIEW_BASE *m_handle = nullptr;
|
VIEW_BASE *m_handle = nullptr;
|
||||||
#elif defined(Q_OS_WIN)
|
#elif defined(Q_OS_WIN)
|
||||||
HWND m_handle = nullptr;
|
HWND m_handle = nullptr;
|
||||||
|
#elif QT_CONFIG(xcb)
|
||||||
|
xcb_window_t m_handle = 0;
|
||||||
#endif
|
#endif
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -78,6 +82,11 @@ QRect NativeWindow::geometry() const
|
|||||||
return QRectF::fromCGRect(m_handle.frame).toRect();
|
return QRectF::fromCGRect(m_handle.frame).toRect();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NativeWindow::operator WId() const
|
||||||
|
{
|
||||||
|
return reinterpret_cast<WId>(m_handle);
|
||||||
|
}
|
||||||
|
|
||||||
WId NativeWindow::parentWinId() const
|
WId NativeWindow::parentWinId() const
|
||||||
{
|
{
|
||||||
return WId(m_handle.superview);
|
return WId(m_handle.superview);
|
||||||
@ -122,11 +131,78 @@ QRect NativeWindow::geometry() const
|
|||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
NativeWindow::operator WId() const
|
||||||
|
{
|
||||||
|
return reinterpret_cast<WId>(m_handle);
|
||||||
|
}
|
||||||
|
|
||||||
WId NativeWindow::parentWinId() const
|
WId NativeWindow::parentWinId() const
|
||||||
{
|
{
|
||||||
return WId(GetAncestor(m_handle, GA_PARENT));
|
return WId(GetAncestor(m_handle, GA_PARENT));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#elif QT_CONFIG(xcb)
|
||||||
|
|
||||||
|
struct Connection
|
||||||
|
{
|
||||||
|
Connection() : m_connection(xcb_connect(nullptr, nullptr)) {}
|
||||||
|
~Connection() { xcb_disconnect(m_connection); }
|
||||||
|
operator xcb_connection_t*() const { return m_connection; }
|
||||||
|
xcb_connection_t *m_connection = nullptr;
|
||||||
|
};
|
||||||
|
|
||||||
|
static Connection connection;
|
||||||
|
|
||||||
|
NativeWindow::NativeWindow()
|
||||||
|
{
|
||||||
|
m_handle = xcb_generate_id(connection);
|
||||||
|
|
||||||
|
xcb_screen_t *screen = xcb_setup_roots_iterator(xcb_get_setup(connection)).data;
|
||||||
|
|
||||||
|
xcb_create_window(connection, XCB_COPY_FROM_PARENT, m_handle,
|
||||||
|
screen->root, 0, 0, 1, 1, 0, XCB_WINDOW_CLASS_INPUT_OUTPUT,
|
||||||
|
screen->root_visual, XCB_CW_BACK_PIXEL,
|
||||||
|
(const uint32_t []){ 0xffffaaff });
|
||||||
|
|
||||||
|
xcb_flush(connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
NativeWindow::~NativeWindow()
|
||||||
|
{
|
||||||
|
xcb_destroy_window(connection, m_handle);
|
||||||
|
}
|
||||||
|
|
||||||
|
void NativeWindow::setGeometry(const QRect &rect)
|
||||||
|
{
|
||||||
|
const quint32 mask = XCB_CONFIG_WINDOW_X | XCB_CONFIG_WINDOW_Y
|
||||||
|
| XCB_CONFIG_WINDOW_WIDTH | XCB_CONFIG_WINDOW_HEIGHT;
|
||||||
|
const qint32 values[] = { rect.x(), rect.y(), rect.width(), rect.height() };
|
||||||
|
xcb_configure_window(connection, m_handle, mask,
|
||||||
|
reinterpret_cast<const quint32*>(values));
|
||||||
|
xcb_flush(connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
QRect NativeWindow::geometry() const
|
||||||
|
{
|
||||||
|
xcb_get_geometry_reply_t *geometry = xcb_get_geometry_reply(
|
||||||
|
connection, xcb_get_geometry(connection, m_handle), nullptr);
|
||||||
|
const auto cleanup = qScopeGuard([&]{ free(geometry); });
|
||||||
|
return QRect(geometry->x, geometry->y, geometry->width, geometry->height);
|
||||||
|
}
|
||||||
|
|
||||||
|
NativeWindow::operator WId() const
|
||||||
|
{
|
||||||
|
return m_handle;
|
||||||
|
}
|
||||||
|
|
||||||
|
WId NativeWindow::parentWinId() const
|
||||||
|
{
|
||||||
|
xcb_query_tree_reply_t *tree = xcb_query_tree_reply(
|
||||||
|
connection, xcb_query_tree(connection, m_handle), nullptr);
|
||||||
|
const auto cleanup = qScopeGuard([&]{ free(tree); });
|
||||||
|
return tree->parent;
|
||||||
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#endif // NATIVEWINDOW_H
|
#endif // NATIVEWINDOW_H
|
||||||
|
Loading…
x
Reference in New Issue
Block a user