qt5base-lts/tests/shared/nativewindow.h
Tor Arne Vestbø c3a1fe7bd7 Prevent reparenting of foreign window embedding container
A foreign window used to embed a Qt window into it should not end up
with changes to its own parent, as its only job is to give the embedded
Qt window a parent handle.

Pick-to: 6.6
Change-Id: If1bc89658fedf449d266bc0cc750c90b6a841a68
Reviewed-by: Richard Moe Gustavsen <richard.gustavsen@qt.io>
Reviewed-by: Oliver Wolff <oliver.wolff@qt.io>
2023-11-15 18:25:04 +01:00

255 lines
5.9 KiB
Objective-C

// Copyright (C) 2023 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0
#ifndef NATIVEWINDOW_H
#define NATIVEWINDOW_H
#if defined(Q_OS_MACOS)
# include <AppKit/AppKit.h>
# define VIEW_BASE NSView
#elif defined(Q_OS_IOS)
# include <UIKit/UIKit.h>
# define VIEW_BASE UIView
#elif defined(Q_OS_WIN)
# include <winuser.h>
#elif QT_CONFIG(xcb)
# include <xcb/xcb.h>
#endif
class NativeWindow
{
Q_DISABLE_COPY(NativeWindow)
public:
#if defined(Q_OS_MACOS)
using Handle = NSView*;
#elif defined(Q_OS_IOS)
using Handle = UIView*;
#elif defined(Q_OS_WIN)
using Handle = HWND;
#elif QT_CONFIG(xcb)
using Handle = xcb_window_t;
#endif
NativeWindow();
~NativeWindow();
operator WId() const;
WId parentWinId() const;
bool isParentOf(WId childWinId);
void setParent(WId parent);
void setGeometry(const QRect &rect);
QRect geometry() const;
private:
Handle m_handle = {};
};
#if defined(Q_OS_MACOS) || defined(Q_OS_IOS)
@interface View : VIEW_BASE
@end
@implementation View
- (instancetype)init
{
if ((self = [super init])) {
#if defined(Q_OS_MACOS)
self.wantsLayer = YES;
#endif
self.layer.backgroundColor = CGColorCreateGenericRGB(1.0, 0.5, 1.0, 1.0);
}
return self;
}
- (void)dealloc
{
[super dealloc];
}
@end
NativeWindow::NativeWindow()
: m_handle([View new])
{
}
NativeWindow::~NativeWindow()
{
[m_handle release];
}
void NativeWindow::setGeometry(const QRect &rect)
{
m_handle.frame = QRectF(rect).toCGRect();
}
QRect NativeWindow::geometry() const
{
return QRectF::fromCGRect(m_handle.frame).toRect();
}
NativeWindow::operator WId() const
{
return reinterpret_cast<WId>(m_handle);
}
WId NativeWindow::parentWinId() const
{
return WId(m_handle.superview);
}
bool NativeWindow::isParentOf(WId childWinId)
{
auto *subview = reinterpret_cast<Handle>(childWinId);
return subview.superview == m_handle;
}
void NativeWindow::setParent(WId parent)
{
if (auto *superview = reinterpret_cast<Handle>(parent))
[superview addSubview:m_handle];
else
[m_handle removeFromSuperview];
}
#elif defined(Q_OS_WIN)
NativeWindow::NativeWindow()
{
static const LPCWSTR className = []{
WNDCLASS wc = {};
wc.lpfnWndProc = DefWindowProc;
wc.hInstance = GetModuleHandle(nullptr);
wc.lpszClassName = L"Native Window";
wc.hbrBackground = CreateSolidBrush(RGB(255, 128, 255));
RegisterClass(&wc);
return wc.lpszClassName;
}();
m_handle = CreateWindowEx(0, className, nullptr, WS_POPUP,
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
nullptr, nullptr, GetModuleHandle(nullptr), nullptr);
}
NativeWindow::~NativeWindow()
{
DestroyWindow(m_handle);
}
void NativeWindow::setGeometry(const QRect &rect)
{
MoveWindow(m_handle, rect.x(), rect.y(), rect.width(), rect.height(), false);
}
QRect NativeWindow::geometry() const
{
WINDOWPLACEMENT wp;
wp.length = sizeof(WINDOWPLACEMENT);
if (GetWindowPlacement(m_handle, &wp)) {
RECT r = wp.rcNormalPosition;
return QRect(r.left, r.top, r.right - r.left, r.bottom - r.top);
}
return {};
}
NativeWindow::operator WId() const
{
return reinterpret_cast<WId>(m_handle);
}
WId NativeWindow::parentWinId() const
{
return WId(GetAncestor(m_handle, GA_PARENT));
}
bool NativeWindow::isParentOf(WId childWinId)
{
return GetAncestor(Handle(childWinId), GA_PARENT) == m_handle;
}
void NativeWindow::setParent(WId parent)
{
SetParent(m_handle, Handle(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;
}
bool NativeWindow::isParentOf(WId childWinId)
{
xcb_query_tree_reply_t *tree = xcb_query_tree_reply(
connection, xcb_query_tree(connection, Handle(childWinId)), nullptr);
const auto cleanup = qScopeGuard([&]{ free(tree); });
return tree->parent == m_handle;
}
void NativeWindow::setParent(WId parent)
{
xcb_screen_t *screen = xcb_setup_roots_iterator(xcb_get_setup(connection)).data;
xcb_reparent_window(connection, m_handle,
parent ? Handle(parent) : screen->root, 0, 0);
}
#endif
#endif // NATIVEWINDOW_H