From a46e7027ce7191f2e3ffb6a633e077bb78f7ea68 Mon Sep 17 00:00:00 2001
From: Frank Praznik <[EMAIL REDACTED]>
Date: Sat, 29 Jun 2024 17:07:53 -0400
Subject: [PATCH] video: Allow setting the parents of toplevel windows
Allow setting a parent/child relationship on toplevel windows, which allows raising sets of windows together, and allows child windows to always float above their parents.
Modal windows are now set by setting the parent, then toggling modal status, as the previous interface duplicated functionality now handled by SDL_SetWindowParent().
---
include/SDL3/SDL_video.h | 41 +++++++---
src/dynapi/SDL_dynapi.sym | 3 +-
src/dynapi/SDL_dynapi_overrides.h | 3 +-
src/dynapi/SDL_dynapi_procs.h | 3 +-
src/video/SDL_sysvideo.h | 3 +-
src/video/SDL_video.c | 98 +++++++++++++----------
src/video/cocoa/SDL_cocoavideo.m | 3 +-
src/video/cocoa/SDL_cocoawindow.h | 3 +-
src/video/cocoa/SDL_cocoawindow.m | 110 ++++++++++++++++++--------
src/video/haiku/SDL_bvideo.cc | 2 +
src/video/haiku/SDL_bwindow.cc | 25 +++---
src/video/haiku/SDL_bwindow.h | 2 +
src/video/wayland/SDL_waylandvideo.c | 3 +-
src/video/wayland/SDL_waylandwindow.c | 98 ++++++++++++++---------
src/video/wayland/SDL_waylandwindow.h | 5 +-
src/video/windows/SDL_windowsvideo.c | 3 +-
src/video/windows/SDL_windowswindow.c | 59 +++++++-------
src/video/windows/SDL_windowswindow.h | 3 +-
src/video/x11/SDL_x11video.c | 3 +-
src/video/x11/SDL_x11window.c | 38 ++++++---
src/video/x11/SDL_x11window.h | 3 +-
test/testmodal.c | 24 ++++--
22 files changed, 336 insertions(+), 199 deletions(-)
diff --git a/include/SDL3/SDL_video.h b/include/SDL3/SDL_video.h
index 7cce828aa7dcb..fd80ef1edbd1e 100644
--- a/include/SDL3/SDL_video.h
+++ b/include/SDL3/SDL_video.h
@@ -2223,24 +2223,47 @@ extern SDL_DECLSPEC SDL_bool SDLCALL SDL_SetWindowOpacity(SDL_Window *window, fl
*/
extern SDL_DECLSPEC float SDLCALL SDL_GetWindowOpacity(SDL_Window *window);
-/**
- * Set the window as a modal to a parent window.
+ /**
+ * Set the window as a child of a parent window.
+ *
+ * If the window is already the child of an existing window, it will be reparented
+ * to the new owner. Setting the parent window to null unparents the window and
+ * removes child window status.
*
- * If the window is already modal to an existing window, it will be reparented
- * to the new owner. Setting the parent window to null unparents the modal
- * window and removes modal status.
+ * Attempting to set the parent of a window that is currently in the modal state will fail.
+ * Use SDL_SetWindowModalFor() to cancel the modal status before attempting to change
+ * the parent.
*
- * Setting a window as modal to a parent that is a descendent of the modal
+ * Setting a parent window that is currently the sibling or descendent of the child
* window results in undefined behavior.
*
- * \param modal_window the window that should be set modal.
- * \param parent_window the parent window for the modal window.
+ * \param window the window that should become the child of a parent.
+ * \param parent the new parent window for the child window.
+ * \returns SDL_TRUE on success or SDL_FALSE on failure; call
+ * SDL_GetError() for more information.
+ *
+ * \since This function is available since SDL 3.0.0.
+ *
+ * \sa SDL_SetWindowModal
+ */
+ extern SDL_DECLSPEC SDL_bool SDLCALL SDL_SetWindowParent(SDL_Window *window, SDL_Window *parent);
+
+/**
+ * Toggle the state of the window as modal.
+ *
+ * To enable modal status on a window, the window must currently be the child window of a parent,
+ * or toggling modal status on will fail.
+ *
+ * \param window the window on which to set the modal state.
+ * \param modal SDL_TRUE to toggle modal status on, SDL_FALSE to toggle it off.
* \returns SDL_TRUE on success or SDL_FALSE on failure; call SDL_GetError()
* for more information.
*
* \since This function is available since SDL 3.0.0.
+ *
+ * \sa SDL_SetWindowParent
*/
-extern SDL_DECLSPEC SDL_bool SDLCALL SDL_SetWindowModalFor(SDL_Window *modal_window, SDL_Window *parent_window);
+extern SDL_DECLSPEC SDL_bool SDLCALL SDL_SetWindowModal(SDL_Window *window, SDL_bool modal);
/**
* Set whether the window may have input focus.
diff --git a/src/dynapi/SDL_dynapi.sym b/src/dynapi/SDL_dynapi.sym
index 9ce6598ee4bd3..f3b8de3c440a4 100644
--- a/src/dynapi/SDL_dynapi.sym
+++ b/src/dynapi/SDL_dynapi.sym
@@ -854,10 +854,11 @@ SDL3_0.0.0 {
SDL_SetWindowKeyboardGrab;
SDL_SetWindowMaximumSize;
SDL_SetWindowMinimumSize;
- SDL_SetWindowModalFor;
+ SDL_SetWindowModal;
SDL_SetWindowMouseGrab;
SDL_SetWindowMouseRect;
SDL_SetWindowOpacity;
+ SDL_SetWindowParent;
SDL_SetWindowPosition;
SDL_SetWindowRelativeMouseMode;
SDL_SetWindowResizable;
diff --git a/src/dynapi/SDL_dynapi_overrides.h b/src/dynapi/SDL_dynapi_overrides.h
index a4b486adb5358..66e618b7789df 100644
--- a/src/dynapi/SDL_dynapi_overrides.h
+++ b/src/dynapi/SDL_dynapi_overrides.h
@@ -879,10 +879,11 @@
#define SDL_SetWindowKeyboardGrab SDL_SetWindowKeyboardGrab_REAL
#define SDL_SetWindowMaximumSize SDL_SetWindowMaximumSize_REAL
#define SDL_SetWindowMinimumSize SDL_SetWindowMinimumSize_REAL
-#define SDL_SetWindowModalFor SDL_SetWindowModalFor_REAL
+#define SDL_SetWindowModal SDL_SetWindowModal_REAL
#define SDL_SetWindowMouseGrab SDL_SetWindowMouseGrab_REAL
#define SDL_SetWindowMouseRect SDL_SetWindowMouseRect_REAL
#define SDL_SetWindowOpacity SDL_SetWindowOpacity_REAL
+#define SDL_SetWindowParent SDL_SetWindowParent_REAL
#define SDL_SetWindowPosition SDL_SetWindowPosition_REAL
#define SDL_SetWindowRelativeMouseMode SDL_SetWindowRelativeMouseMode_REAL
#define SDL_SetWindowResizable SDL_SetWindowResizable_REAL
diff --git a/src/dynapi/SDL_dynapi_procs.h b/src/dynapi/SDL_dynapi_procs.h
index 7292c9c8d5e47..54f542f37f643 100644
--- a/src/dynapi/SDL_dynapi_procs.h
+++ b/src/dynapi/SDL_dynapi_procs.h
@@ -889,10 +889,11 @@ SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowIcon,(SDL_Window *a, SDL_Surface *b),(a,b)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowKeyboardGrab,(SDL_Window *a, SDL_bool b),(a,b),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowMaximumSize,(SDL_Window *a, int b, int c),(a,b,c),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowMinimumSize,(SDL_Window *a, int b, int c),(a,b,c),return)
-SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowModalFor,(SDL_Window *a, SDL_Window *b),(a,b),return)
+SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowModal,(SDL_Window *a, SDL_bool b),(a,b),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowMouseGrab,(SDL_Window *a, SDL_bool b),(a,b),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowMouseRect,(SDL_Window *a, const SDL_Rect *b),(a,b),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowOpacity,(SDL_Window *a, float b),(a,b),return)
+SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowParent,(SDL_Window *a, SDL_Window *b),(a,b),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowPosition,(SDL_Window *a, int b, int c),(a,b,c),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowRelativeMouseMode,(SDL_Window *a, SDL_bool b),(a,b),return)
SDL_DYNAPI_PROC(SDL_bool,SDL_SetWindowResizable,(SDL_Window *a, SDL_bool b),(a,b),return)
diff --git a/src/video/SDL_sysvideo.h b/src/video/SDL_sysvideo.h
index 862405e13b183..56d0d18ce47ab 100644
--- a/src/video/SDL_sysvideo.h
+++ b/src/video/SDL_sysvideo.h
@@ -264,7 +264,8 @@ struct SDL_VideoDevice
bool (*GetWindowBordersSize)(SDL_VideoDevice *_this, SDL_Window *window, int *top, int *left, int *bottom, int *right);
void (*GetWindowSizeInPixels)(SDL_VideoDevice *_this, SDL_Window *window, int *w, int *h);
bool (*SetWindowOpacity)(SDL_VideoDevice *_this, SDL_Window *window, float opacity);
- bool (*SetWindowModalFor)(SDL_VideoDevice *_this, SDL_Window *modal_window, SDL_Window *parent_window);
+ bool (*SetWindowParent)(SDL_VideoDevice *_this, SDL_Window *window, SDL_Window *parent);
+ bool (*SetWindowModal)(SDL_VideoDevice *_this, SDL_Window *window, bool modal);
void (*ShowWindow)(SDL_VideoDevice *_this, SDL_Window *window);
void (*HideWindow)(SDL_VideoDevice *_this, SDL_Window *window);
void (*RaiseWindow)(SDL_VideoDevice *_this, SDL_Window *window);
diff --git a/src/video/SDL_video.c b/src/video/SDL_video.c
index f28abf80bbd6c..b192e18e5f2c6 100644
--- a/src/video/SDL_video.c
+++ b/src/video/SDL_video.c
@@ -219,7 +219,7 @@ static void SDL_SyncIfRequired(SDL_Window *window)
}
}
-static void SDL_SetWindowParent(SDL_Window *window, SDL_Window *parent)
+static void SDL_UpdateWindowHierarchy(SDL_Window *window, SDL_Window *parent)
{
// Unlink the window from the existing parent.
if (window->parent) {
@@ -2169,6 +2169,10 @@ static void ApplyWindowFlags(SDL_Window *window, SDL_WindowFlags flags)
SDL_MinimizeWindow(window);
}
+ if (flags & SDL_WINDOW_MODAL) {
+ SDL_SetWindowModal(window, true);
+ }
+
if (flags & SDL_WINDOW_MOUSE_GRABBED) {
SDL_SetWindowMouseGrab(window, true);
}
@@ -2432,10 +2436,8 @@ SDL_Window *SDL_CreateWindowWithProperties(SDL_PropertiesID props)
}
_this->windows = window;
- // Set the parent before creation if this is non-modal, otherwise it will be set later.
- if (!(flags & SDL_WINDOW_MODAL)) {
- SDL_SetWindowParent(window, parent);
- }
+ // Set the parent before creation.
+ SDL_UpdateWindowHierarchy(window, parent);
if (_this->CreateSDLWindow && !_this->CreateSDLWindow(_this, window, props)) {
SDL_DestroyWindow(window);
@@ -2462,10 +2464,6 @@ SDL_Window *SDL_CreateWindowWithProperties(SDL_PropertiesID props)
*/
flags = window->flags;
#endif
-
- if (flags & SDL_WINDOW_MODAL) {
- SDL_SetWindowModalFor(window, parent);
- }
if (title) {
SDL_SetWindowTitle(window, title);
}
@@ -2525,7 +2523,6 @@ bool SDL_RecreateWindow(SDL_Window *window, SDL_WindowFlags flags)
bool need_vulkan_unload = false;
bool need_vulkan_load = false;
SDL_WindowFlags graphics_flags;
- SDL_Window *parent = window->parent;
// ensure no more than one of these flags is set
graphics_flags = flags & (SDL_WINDOW_OPENGL | SDL_WINDOW_METAL | SDL_WINDOW_VULKAN);
@@ -2552,7 +2549,7 @@ bool SDL_RecreateWindow(SDL_Window *window, SDL_WindowFlags flags)
// If this is a modal dialog, clear the modal status.
if (window->flags & SDL_WINDOW_MODAL) {
- SDL_SetWindowModalFor(window, NULL);
+ SDL_SetWindowModal(window, false);
}
// Restore video mode, etc.
@@ -2642,10 +2639,6 @@ bool SDL_RecreateWindow(SDL_Window *window, SDL_WindowFlags flags)
window->flags |= SDL_WINDOW_EXTERNAL;
}
- if (flags & SDL_WINDOW_MODAL) {
- SDL_SetWindowModalFor(window, parent);
- }
-
if (_this->SetWindowTitle && window->title) {
_this->SetWindowTitle(_this, window);
}
@@ -3579,40 +3572,59 @@ float SDL_GetWindowOpacity(SDL_Window *window)
return window->opacity;
}
-SDL_bool SDL_SetWindowModalFor(SDL_Window *modal_window, SDL_Window *parent_window)
+SDL_bool SDL_SetWindowParent(SDL_Window *window, SDL_Window *parent)
{
- bool result;
+ CHECK_WINDOW_MAGIC(window, false);
+ CHECK_WINDOW_NOT_POPUP(window, false);
+
+ if (parent) {
+ CHECK_WINDOW_MAGIC(parent, false);
+ CHECK_WINDOW_NOT_POPUP(parent, false);
+ }
- CHECK_WINDOW_MAGIC(modal_window, false);
- CHECK_WINDOW_NOT_POPUP(modal_window, false);
+ if (!_this->SetWindowParent) {
+ return SDL_Unsupported();
+ }
- if (parent_window) {
- CHECK_WINDOW_MAGIC(parent_window, false);
- CHECK_WINDOW_NOT_POPUP(parent_window, false);
+ if (window->flags & SDL_WINDOW_MODAL) {
+ return SDL_SetError("Modal windows cannot change parents; call SDL_SetWindowModal() to clear modal status first.");
}
- if (!_this->SetWindowModalFor) {
+ if (window->parent == parent) {
+ return true;
+ }
+
+ const SDL_bool ret = _this->SetWindowParent(_this, window, parent);
+ SDL_UpdateWindowHierarchy(window, ret ? parent : NULL);
+
+ return ret;
+}
+
+SDL_bool SDL_SetWindowModal(SDL_Window *window, SDL_bool modal)
+{
+ CHECK_WINDOW_MAGIC(window, false);
+ CHECK_WINDOW_NOT_POPUP(window, false);
+
+ if (!_this->SetWindowModal) {
return SDL_Unsupported();
}
- if (parent_window) {
- modal_window->flags |= SDL_WINDOW_MODAL;
- } else if (modal_window->flags & SDL_WINDOW_MODAL) {
- modal_window->flags &= ~SDL_WINDOW_MODAL;
+ if (modal) {
+ if (!window->parent) {
+ return SDL_SetError("Window must have a parent to enable the modal state; use SDL_SetWindowParent() to set the parent first.");
+ }
+ window->flags |= SDL_WINDOW_MODAL;
+ } else if (window->flags & SDL_WINDOW_MODAL) {
+ window->flags &= ~SDL_WINDOW_MODAL;
} else {
- return true; // Not modal; nothing to do.
+ return true; // Already not modal, so nothing to do.
}
- result = _this->SetWindowModalFor(_this, modal_window, parent_window);
-
- /* The existing parent might be needed when changing the modal status,
- * so don't change the hierarchy until after setting the new modal state.
- */
- if (result) {
- SDL_SetWindowParent(modal_window, parent_window);
+ if (window->flags & SDL_WINDOW_HIDDEN) {
+ return true;
}
- return result;
+ return _this->SetWindowModal(_this, window, modal);
}
SDL_bool SDL_SetWindowFocusable(SDL_Window *window, SDL_bool focusable)
@@ -4101,12 +4113,12 @@ void SDL_DestroyWindow(SDL_Window *window)
SDL_DestroyProperties(window->text_input_props);
SDL_DestroyProperties(window->props);
- /* Clear the modal status, but don't unset the parent, as it may be
- * needed later in the destruction process if a backend needs to
- * update the input focus.
+ /* Clear the modal status, but don't unset the parent just yet, as it
+ * may be needed later in the destruction process if a backend needs
+ * to update the input focus.
*/
- if (_this->SetWindowModalFor && (window->flags & SDL_WINDOW_MODAL)) {
- _this->SetWindowModalFor(_this, window, NULL);
+ if (_this->SetWindowModal && (window->flags & SDL_WINDOW_MODAL)) {
+ _this->SetWindowModal(_this, window, false);
}
// Make sure the destroyed window isn't referenced by any display as a fullscreen window.
@@ -4168,9 +4180,9 @@ void SDL_DestroyWindow(SDL_Window *window)
SDL_DestroySurface(window->icon);
// Unlink the window from its siblings.
- SDL_SetWindowParent(window, NULL);
+ SDL_UpdateWindowHierarchy(window, NULL);
- // Unlink the window from the list
+ // Unlink the window from the global window list
if (window->next) {
window->next->prev = window->prev;
}
diff --git a/src/video/cocoa/SDL_cocoavideo.m b/src/video/cocoa/SDL_cocoavideo.m
index 85795e1410d20..93b9153171d05 100644
--- a/src/video/cocoa/SDL_cocoavideo.m
+++ b/src/video/cocoa/SDL_cocoavideo.m
@@ -122,7 +122,8 @@ static void Cocoa_DeleteDevice(SDL_VideoDevice *device)
device->UpdateWindowShape = Cocoa_UpdateWindowShape;
device->FlashWindow = Cocoa_FlashWindow;
device->SetWindowFocusable = Cocoa_SetWindowFocusable;
- device->SetWindowModalFor = Cocoa_SetWindowModalFor;
+ device->SetWindowParent = Cocoa_SetWindowParent;
+ device->SetWindowModal = Cocoa_SetWindowModal;
device->SyncWindow = Cocoa_SyncWindow;
#ifdef SDL_VIDEO_OPENGL_CGL
diff --git a/src/video/cocoa/SDL_cocoawindow.h b/src/video/cocoa/SDL_cocoawindow.h
index 3e0413871be56..dd30c200a8134 100644
--- a/src/video/cocoa/SDL_cocoawindow.h
+++ b/src/video/cocoa/SDL_cocoawindow.h
@@ -183,7 +183,8 @@ extern bool Cocoa_SetWindowHitTest(SDL_Window *window, bool enabled);
extern void Cocoa_AcceptDragAndDrop(SDL_Window *window, bool accept);
extern bool Cocoa_FlashWindow(SDL_VideoDevice *_this, SDL_Window *window, SDL_FlashOperation operation);
extern bool Cocoa_SetWindowFocusable(SDL_VideoDevice *_this, SDL_Window *window, bool focusable);
-extern bool Cocoa_SetWindowModalFor(SDL_VideoDevice *_this, SDL_Window *modal_window, SDL_Window *parent_window);
+extern bool Cocoa_SetWindowModal(SDL_VideoDevice *_this, SDL_Window *window, bool modal);
+extern bool Cocoa_SetWindowParent(SDL_VideoDevice *_this, SDL_Window *window, SDL_Window *parent);
extern bool Cocoa_SyncWindow(SDL_VideoDevice *_this, SDL_Window *window);
#endif // SDL_cocoawindow_h_
diff --git a/src/video/cocoa/SDL_cocoawindow.m b/src/video/cocoa/SDL_cocoawindow.m
index ba12e02f69965..21e437f5c0c06 100644
--- a/src/video/cocoa/SDL_cocoawindow.m
+++ b/src/video/cocoa/SDL_cocoawindow.m
@@ -2075,15 +2075,27 @@ static bool SetupWindowData(SDL_VideoDevice *_this, SDL_Window *window, NSWindow
window->flags &= ~SDL_WINDOW_MINIMIZED;
}
+ if (window->parent) {
+ NSWindow *nsparent = ((__bridge SDL_CocoaWindowData *)window->parent->internal).nswindow;
+ [nsparent addChildWindow:nswindow ordered:NSWindowAbove];
+
+ /* FIXME: Should not need to call addChildWindow then orderOut.
+ Attaching a hidden child window to a hidden parent window will cause the child window
+ to show when the parent does. We therefore shouldn't attach the child window here as we're
+ going to do so when the child window is explicitly shown later but skipping the addChildWindow
+ entirely causes the child window to not get key focus correctly the first time it's shown. Adding
+ then immediately ordering out (removing) the window does work. */
+ if (window->flags & SDL_WINDOW_HIDDEN) {
+ [nswindow orderOut:nil];
+ }
+ }
+
if (!SDL_WINDOW_IS_POPUP(window)) {
if ([nswindow isKeyWindow]) {
window->flags |= SDL_WINDOW_INPUT_FOCUS;
Cocoa_SetKeyboardFocus(data.window);
}
} else {
- NSWindow *nsparent = ((__bridge SDL_CocoaWindowData *)window->parent->internal).nswindow;
- [nsparent addChildWindow:nswindow ordered:NSWindowAbove];
-
if (window->flags & SDL_WINDOW_TOOLTIP) {
[nswindow setIgnoresMouseEvents:YES];
} else if (window->flags & SDL_WINDOW_POPUP_MENU) {
@@ -2091,16 +2103,6 @@ static bool SetupWindowData(SDL_VideoDevice *_this, SDL_Window *window, NSWindow
Cocoa_SetKeyboardFocus(window);
}
}
-
- /* FIXME: Should not need to call addChildWindow then orderOut.
- Attaching a hidden child window to a hidden parent window will cause the child window
- to show when the parent does. We therefore shouldn't attach the child window here as we're
- going to do so when the child window is explicitly shown later but skipping the addChildWindow
- entirely causes the child window to not get key focus correctly the first time it's shown. Adding
- then immediately ordering out (removing) the window does work. */
- if (window->flags & SDL_WINDOW_HIDDEN) {
- [nswindow orderOut:nil];
- }
}
if (nswindow.isOpaque) {
@@ -2465,14 +2467,15 @@ void Cocoa_ShowWindow(SDL_VideoDevice *_this, SDL_Window *window)
if (![nswindow isMiniaturized]) {
[windowData.listener pauseVisibleObservation];
- if (SDL_WINDOW_IS_POPUP(window)) {
+ if (window->parent) {
NSWindow *nsparent = ((__bridge SDL_CocoaWindowData *)window->parent->internal).nswindow;
[nsparent addChildWindow:nswindow ordered:NSWindowAbove];
- } else {
- if ((window->flags & SDL_WINDOW_MODAL) && window->parent) {
- Cocoa_SetWindowModalFor(_this, window, window->parent);
- }
+ if (window->flags & SDL_WINDOW_MODAL) {
+ Cocoa_SetWindowModal(_this, window, true);
+ }
+ }
+ if (!SDL_WINDOW_IS_POPUP(window)) {
if (bActivate) {
[nswindow makeKeyAndOrderFront:nil];
} else {
@@ -2482,9 +2485,9 @@ void Cocoa_ShowWindow(SDL_VideoDevice *_this, SDL_Window *window)
}
}
}
- [nswindow setIsVisible:YES];
- [windowData.listener resumeVisibleObservation];
}
+ [nswindow setIsVisible:YES];
+ [windowData.listener resumeVisibleObservation];
}
}
@@ -2492,6 +2495,7 @@ void Cocoa_HideWindow(SDL_VideoDevice *_this, SDL_Window *window)
{
@autoreleasepool {
NSWindow *nswindow = ((__bridge SDL_CocoaWindowData *)window->internal).nswindow;
+ const BOOL waskey = [nswindow isKeyWindow];
/* orderOut has no effect on miniaturized windows, so close must be used to remove
* the window from the desktop and window list in this case.
@@ -2509,9 +2513,9 @@ void Cocoa_HideWindow(SDL_VideoDevice *_this, SDL_Window *window)
/* If this window is the source of a modal session, end it when
* hidden, or other windows will be prevented from closing.
*/
- Cocoa_SetWindowModalFor(_this, window, NULL);
+ Cocoa_SetWindowModal(_this, window, false);
- // Transfer keyboard focus back to the parent
+ // Transfer keyboard focus back to the parent when closing a popup menu
if (window->flags & SDL_WINDOW_POPUP_MENU) {
if (window == SDL_GetKeyboardFocus()) {
SDL_Window *new_focus = window->parent;
@@ -2523,6 +2527,20 @@ void Cocoa_HideWindow(SDL_VideoDevice *_this, SDL_Window *window)
Cocoa_SetKeyboardFocus(new_focus);
}
+ } else if (window->parent && waskey) {
+ /* Key status is not automatically set on the parent when a child is hidden. Check if the
+ * child window was key, and set the first visible parent to be key if so.
+ */
+ SDL_Window *new_focus = window->parent;
+
+ while (new_focus->parent != NULL && (new_focus->is_hiding || new_focus->is_destroying)) {
+ new_focus = new_focus->parent;
+ }
+
+ if (new_focus) {
+ NSWindow *newkey = ((__bridge SDL_CocoaWindowData *)window->internal).nswindow;
+ [newkey makeKeyAndOrderFront:nil];
+ }
}
}
}
@@ -2539,19 +2557,21 @@ void Cocoa_RaiseWindow(SDL_VideoDevice *_this, SDL_Window *window)
*/
[windowData.listener pauseVisibleObservation];
if (![nswindow isMiniaturized] && [nswindow isVisible]) {
- if (SDL_WINDOW_IS_POPUP(window)) {
+ if (window->parent) {
NSWindow *nsparent = ((__bridge SDL_CocoaWindowData *)window->parent->internal).nswindow;
[nsparent addChildWindow:nswindow ordered:NSWindowAbove];
- if (bActivate) {
- [nswindow makeKeyWindow];
- }
- } else {
+ }
+ if (!SDL_WINDOW_IS_POPUP(window)) {
if (bActivate) {
[NSApp activateIgnoringOtherApps:YES];
[nswindow makeKeyAndOrderFront:nil];
} else {
[nswindow orderFront:nil];
}
+ } else {
+ if (bActivate) {
+ [nswindow makeKeyWindow];
+ }
}
}
[windowData.listener resumeVisibleObservation];
@@ -2943,7 +2963,7 @@ void Cocoa_DestroyWindow(SDL_VideoDevice *_this, SDL_Window *window)
*/
if (topmost_data.keyboard_focus == window) {
SDL_Window *new_focus = window;
- while(new_focus->parent && (new_focus->is_hiding || new_focus->is_destroying)) {
+ while (new_focus->parent && (new_focus->is_hiding || new_focus->is_destroying)) {
new_focus = new_focus->parent;
}
@@ -3054,18 +3074,38 @@ void Cocoa_AcceptDragAndDrop(SDL_Window *window, bool accept)
}
}
-bool Cocoa_SetWindowModalFor(SDL_VideoDevice *_this, SDL_Window *modal_window, SDL_Window *parent_window)
+bool Cocoa_SetWindowParent(SDL_VideoDevice *_this, SDL_Window *window, SDL_Window *parent)
{
@autoreleasepool {
- SDL_CocoaWindowData *modal_data = (__bridge SDL_CocoaWindowData *)modal_window->internal;
+ SDL_CocoaWindowData *child_data = (__bridge SDL_CocoaWindowData *)window->internal;
+
+ // Remove an existing parent.
+ if (child_data.nswindow.parentWindow) {
+ NSWindow *nsparent = ((__bridge SDL_CocoaWindowData *)window->parent->internal).nswindow;
+ [nsparent removeChildWindow:child_data.nswindow];
+ }
+
+ if (parent) {
+ SDL_CocoaWindowData *parent_data = (__bridge SDL_CocoaWindowData *)parent->internal;
+ [parent_data.nswindow addChildWindow:child_data.nswindow ordered:NSWindowAbove];
+ }
+ }
+
+ return true;
+}
+
+bool Cocoa_SetWindowModal(SDL_VideoDevice *_this, SDL_Window *window, bool modal)
+{
+ @autoreleasepool {
+ SDL_CocoaWindowData *data = (__bridge SDL_CocoaWindowData *)window->internal;
- if (modal_data.modal_session) {
- [NSApp endModalSession:modal_data.modal_session];
- modal_data.modal_session = nil;
+ if (data.modal_session) {
+ [NSApp endModalSession:data.modal_session];
+ data.modal_session = nil;
}
- if (parent_window) {
- modal_data.modal_session = [NSApp beginModalSessionForWindow:modal_data.nswindow];
+ if (modal) {
+ data.modal_session = [NSApp beginModalSessionForWindow:data.nswindow];
}
}
diff --git a/src/video/haiku/SDL_bvideo.cc b/src/video/haiku/SDL_bvideo.cc
index 942fc966c3701..23521f73f29b8 100644
--- a/src/video/haiku/SDL_bvideo.cc
+++ b/src/video/haiku/SDL_bvideo.cc
@@ -82,6 +82,8 @@ static SDL_VideoDevice * HAIKU_CreateDevice(void)
device->SetWindowFullscreen = HAIKU_SetWindowFullscreen;
device->SetWindowMouseGrab = HAIKU_SetWindowMouseGrab;
device->SetWindowMinimumSize = HAIKU_SetWindowMinimumSize;
+ device->SetWindowParent = HAIKU_SetWindowParent;
+ device->SetWindowModal = HAIKU_SetWindowModal;
device->DestroyWindow = HAIKU_DestroyWindow;
device->CreateWindowFramebuffer = HAIKU_CreateWindowFramebuffer;
device->UpdateWindowFramebuffer = HAIKU_UpdateWindowFramebuffer;
diff --git a/src/video/haiku/SDL_bwindow.cc b/src/video/haiku/SDL_bwindow.cc
index b6eca9db3d2c8..de9bc2484d696 100644
--- a/src/video/haiku/SDL_bwindow.cc
+++ b/src/video/haiku/SDL_bwindow.cc
@@ -188,21 +188,22 @@ bool HAIKU_SetWindowMouseGrab(SDL_VideoDevice *_this, SDL_Window * window, bool
return SDL_Unsupported();
}
-bool HAIKU_SetWindowModalFor(SDL_VideoDevice *_this, SDL_Window *modal_window, SDL_Window *parent_window)
+bool HAIKU_SetWindowParent(SDL_VideoDevice *_this, SDL_Window * window, SDL_Window *parent)
{
- if (modal_window->parent && modal_window->parent != parent_window) {
- // Remove from the subset of a previous parent.
- _ToBeWin(modal_window)->RemoveFromSubset(_ToBeWin(modal_window->parent));
- }
+ return true;
+}
- if (parent_window) {
- _ToBeWin(modal_window)->SetLook(B_MODAL_WINDOW_LOOK);
- _ToBeWin(modal_window)->SetFeel(B_MODAL_SUBSET_WINDOW_FEEL);
- _ToBeWin(modal_window)->AddToSubset(_ToBeWin(parent_window));
+bool HAIKU_SetWindowModal(SDL_VideoDevice *_this, SDL_Window *window, bool modal)
+{
+ if (modal) {
+ _ToBeWin(window)->SetLook(B_MODAL_WINDOW_LOOK);
+ _ToBeWin(window)->SetFeel(B_MODAL_SUBSET_WINDOW_FEEL);
+ _ToBeWin(window)->AddToSubset(_ToBeWin(window->parent));
} else {
- window_look look = (modal_window->flags & SDL_WINDOW_BORDERLESS) ? B_NO_BORDER_WINDOW_LOOK : B_TITLED_WINDOW_LOOK;
- _ToBeWin(modal_window)->SetLook(look);
- _ToBeWin(modal_window)->SetFeel(B_NORMAL_WINDOW_FEEL);
+ window_look look = (window->flags & SDL_WINDOW_BORDERLESS) ? B_NO_BORDER_WINDOW_LOOK : B_TITLED_WINDOW_LOOK;
+ _ToBeWin(window)->RemoveFromSubset(_ToBeWin(window->parent));
+ _ToBeWin(window)->SetLook(look);
+ _ToBeWin(window)->SetFeel(B_NORMAL_WINDOW_FEEL);
}
return true;
diff --git a/src/video/haiku/SDL_bwindow.h b/src/video/haiku/SDL_bwindow.h
index 98c88d77fd636..1bc711d146e35 100644
--- a/src/video/haiku/SDL_bwindow.h
+++ b/src/video/haiku/SDL_bwindow.h
@@ -39,6 +39,8 @@ extern void HAIKU_SetWindowBordered(SDL_VideoDevice *_this, SDL_Window *window,
extern void HAIKU_SetWindowResizable(SDL_VideoDevice *_this, SDL_Window *window, bool resizable);
extern SDL_FullscreenResult HAIKU_SetWindowFullscreen(SDL_VideoDevice *_this, SDL_Window *window, SDL_VideoDisplay *display, SDL_FullscreenOp fullscreen);
extern bool HAIKU_SetWindowMouseGrab(SDL_VideoDevice *_this, SDL_Window *window, bool grabbed);
+extern bool HAIKU_SetWindowParent(SDL_VideoDevice *_this, SDL_Window *window, SDL_Window *parent);
+extern bool HAIKU_SetWindowModal(SDL_VideoDevice *_this, SDL_Window *window, bool modal);
extern void HAIKU_DestroyWindow(SDL_VideoDevice *_this, SDL_Window *window);
#endif
diff --git a/src/video/wayland/SDL_waylandvideo.c b/src/video/wayland/SDL_waylandvideo.c
index a3898c4da8015..32cc022e7ff55 100644
--- a/src/video/wayland/SDL_waylandvideo.c
+++ b/src/video/wayland/SDL_waylandvideo.c
@@ -547,7 +547,8 @@ static SDL_VideoDevice *Wayland_CreateDevice(bool require_preferred_protocols)
device->SetWindowSize = Wayland_SetWindowSize;
device->SetWindowMinimumSize = Wayland_SetWindowMinimumSize;
device->SetWindowMaximumSize = Wayland_SetWindowMaximumSize;
- device->SetWindowModalFor = Wayland_SetWindowModalFor;
+ device->SetWindowParent = Wayland_SetWindowParent;
+ device->SetWindowModal = Wayland_SetWindowModal;
device->SetWindowOpacity = Wayland_SetWindowOpacity;
device->SetWindowTitle = Wayland_SetWindowTitle;
device->SetWindowIcon = Wayland_SetWindowIcon;
diff --git a/src/video/wayland/SDL_waylandwindow.c b/src/video/wayland/SDL_waylandwindow.c
index 4596a3078ef03..24ee3eefdecb1 100644
--- a/src/video/wayland/SDL_waylandwindow.c
+++ b/src/video/wayland/SDL_waylandwindow.c
@@ -672,8 +672,11 @@ static void surface_frame_done(void *data, struct wl_callback *cb, uint32_t time
for (SDL_Window *w = wind->sdlwindow->first_child; w; w = w->next_sibling) {
if (w->internal->surface_status == WAYLAND_SURFACE_STATUS_SHOW_PENDING) {
Wayland_ShowWindow(SDL_GetVideoDevice(), w);
- } else if ((w->flags & SDL_WINDOW_MODAL) && w->internal->modal_reparenting_required) {
- Wayland_SetWindowModalFor(SDL_GetVideoDevice(), w, w->parent);
+ } else if (w->internal->reparenting_required) {
+ Wayland_SetWindowParent(SDL_GetVideoDevice(), w, w->parent);
+ if (w->flags & SDL_WINDOW_MODAL) {
+ Wayland_SetWindowModal(SDL_GetVideoDevice(), w, true);
+ }
}
}
@@ -1564,58 +1567,75 @@ bool Wayland_SetWindowHitTe
(Patch may be truncated, please check the link at the top of this post.)