All we need are the APIENTRY and WINGDIAPI macros, as those are used in the gl.h header. Define those locally for the time we need them. Use a QT_APIENTRY macro instead of hijacking APIENTRY for when we declare OpenGL functions with the stdcall calling convention. A few build fixes needed in tests that used Windows types without explicitly including windows.h first, or that (incorrectly) included one of the sub-headers of windows.h (like winuser.h). [ChangeLog][Potentially Source-Incompatible Changes][OpenGL] On Windows, the public qopengl.h header no longer includes windows.h. Fixes: QTBUG-120687 Change-Id: I3770ac8eaeee5bcf4e7234e5a2539935a8aa5a7d Reviewed-by: Oliver Wolff <oliver.wolff@qt.io> (cherry picked from commit b13c610f50c714873987d8c4f30f50953ade9aba) Reviewed-by: Qt Cherry-pick Bot <cherrypick_bot@qt-project.org>
305 lines
7.4 KiB
Objective-C
305 lines
7.4 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 <QtCore/qt_windows.h>
|
|
#elif QT_CONFIG(xcb)
|
|
# include <xcb/xcb.h>
|
|
#elif defined(ANDROID)
|
|
# include <QtCore/qjniobject.h>
|
|
# include <QtCore/qjnitypes.h>
|
|
# include <QtCore/qnativeinterface.h>
|
|
Q_DECLARE_JNI_CLASS(View, "android/view/View")
|
|
Q_DECLARE_JNI_CLASS(ViewParent, "android/view/ViewParent")
|
|
#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;
|
|
#elif defined(ANDROID)
|
|
using Handle = QtJniTypes::View;;
|
|
#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);
|
|
}
|
|
|
|
#elif defined (ANDROID)
|
|
NativeWindow::NativeWindow()
|
|
{
|
|
m_handle = QJniObject::construct<QtJniTypes::View, QtJniTypes::Context>(
|
|
QNativeInterface::QAndroidApplication::context());
|
|
m_handle.callMethod<void>("setBackgroundColor", 0xffffaaff);
|
|
}
|
|
|
|
NativeWindow::~NativeWindow()
|
|
{
|
|
}
|
|
|
|
NativeWindow::operator WId() const
|
|
{
|
|
return reinterpret_cast<WId>(m_handle.object());
|
|
}
|
|
|
|
void NativeWindow::setGeometry(const QRect &rect)
|
|
{
|
|
// No-op, the view geometry is handled by the QWindow constructed from it
|
|
}
|
|
|
|
QRect NativeWindow::geometry() const
|
|
{
|
|
int x = m_handle.callMethod<jint>("getX");
|
|
int y = m_handle.callMethod<jint>("getY");
|
|
int w = m_handle.callMethod<jint>("getWidth");
|
|
int h = m_handle.callMethod<jint>("getHeight");
|
|
return QRect(x, y, w, h);
|
|
}
|
|
|
|
WId NativeWindow::parentWinId() const
|
|
{
|
|
// TODO note, the returned object is a ViewParent, not necessarily
|
|
// a View - what is this used for?
|
|
using namespace QtJniTypes;
|
|
ViewParent parentView = m_handle.callMethod<ViewParent>("getParent");
|
|
if (parentView.isValid())
|
|
return reinterpret_cast<WId>(parentView.object());
|
|
return 0L;
|
|
}
|
|
|
|
#endif
|
|
|
|
#endif // NATIVEWINDOW_H
|