| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580 |
- /*
- Simple DirectMedia Layer
- Copyright (C) 1997-2023 Sam Lantinga <slouken@libsdl.org>
- This software is provided 'as-is', without any express or implied
- warranty. In no event will the authors be held liable for any damages
- arising from the use of this software.
- Permission is granted to anyone to use this software for any purpose,
- including commercial applications, and to alter it and redistribute it
- freely, subject to the following restrictions:
- 1. The origin of this software must not be misrepresented; you must not
- claim that you wrote the original software. If you use this software
- in a product, an acknowledgment in the product documentation would be
- appreciated but is not required.
- 2. Altered source versions must be plainly marked as such, and must not be
- misrepresented as being the original software.
- 3. This notice may not be removed or altered from any source distribution.
- */
- #include "SDL_internal.h"
- #if SDL_VIDEO_DRIVER_WINDOWS
- #include "../../core/windows/SDL_windows.h"
- #include "../SDL_sysvideo.h"
- #include "../SDL_pixels_c.h"
- #include "../../events/SDL_keyboard_c.h"
- #include "../../events/SDL_mouse_c.h"
- #include "../../events/SDL_windowevents_c.h"
- #include "../../SDL_hints_c.h"
- #include "SDL_windowsvideo.h"
- #include "SDL_windowswindow.h"
- #include "SDL_windowsshape.h"
- /* Dropfile support */
- #include <shellapi.h>
- #include <SDL3/SDL_syswm.h>
- /* Dark mode support */
- #ifndef DWMWA_USE_IMMERSIVE_DARK_MODE
- #define DWMWA_USE_IMMERSIVE_DARK_MODE 20
- #endif
- typedef HRESULT (WINAPI *DwmSetWindowAttribute_t)(HWND hwnd, DWORD dwAttribute, LPCVOID pvAttribute, DWORD cbAttribute);
- /* Windows CE compatibility */
- #ifndef SWP_NOCOPYBITS
- #define SWP_NOCOPYBITS 0
- #endif
- /* #define HIGHDPI_DEBUG */
- /* Fake window to help with DirectInput events. */
- HWND SDL_HelperWindow = NULL;
- static const TCHAR *SDL_HelperWindowClassName = TEXT("SDLHelperWindowInputCatcher");
- static const TCHAR *SDL_HelperWindowName = TEXT("SDLHelperWindowInputMsgWindow");
- static ATOM SDL_HelperWindowClass = 0;
- /* For borderless Windows, still want the following flag:
- - WS_MINIMIZEBOX: window will respond to Windows minimize commands sent to all windows, such as windows key + m, shaking title bar, etc.
- Additionally, non-fullscreen windows can add:
- - WS_CAPTION: this seems to enable the Windows minimize animation
- - WS_SYSMENU: enables system context menu on task bar
- This will also cause the task bar to overlap the window and other windowed behaviors, so only use this for windows that shouldn't appear to be fullscreen
- */
- #define STYLE_BASIC (WS_CLIPSIBLINGS | WS_CLIPCHILDREN)
- #define STYLE_FULLSCREEN (WS_POPUP | WS_MINIMIZEBOX)
- #define STYLE_BORDERLESS (WS_POPUP | WS_MINIMIZEBOX)
- #define STYLE_BORDERLESS_WINDOWED (WS_POPUP | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX)
- #define STYLE_NORMAL (WS_OVERLAPPED | WS_CAPTION | WS_SYSMENU | WS_MINIMIZEBOX)
- #define STYLE_RESIZABLE (WS_THICKFRAME | WS_MAXIMIZEBOX)
- #define STYLE_MASK (STYLE_FULLSCREEN | STYLE_BORDERLESS | STYLE_NORMAL | STYLE_RESIZABLE)
- static DWORD GetWindowStyle(SDL_Window *window)
- {
- DWORD style = 0;
- if (SDL_WINDOW_IS_POPUP(window)) {
- style |= WS_POPUP;
- } else if (window->flags & SDL_WINDOW_FULLSCREEN) {
- style |= STYLE_FULLSCREEN;
- } else {
- if (window->flags & SDL_WINDOW_BORDERLESS) {
- /* This behavior more closely matches other platform where the window is borderless
- but still interacts with the window manager (e.g. task bar shows above it, it can
- be resized to fit within usable desktop area, etc.)
- */
- if (SDL_GetHintBoolean("SDL_BORDERLESS_WINDOWED_STYLE", SDL_TRUE)) {
- style |= STYLE_BORDERLESS_WINDOWED;
- } else {
- style |= STYLE_BORDERLESS;
- }
- } else {
- style |= STYLE_NORMAL;
- }
- if (window->flags & SDL_WINDOW_RESIZABLE) {
- /* You can have a borderless resizable window, but Windows doesn't always draw it correctly,
- see https://bugzilla.libsdl.org/show_bug.cgi?id=4466
- */
- if (!(window->flags & SDL_WINDOW_BORDERLESS) ||
- SDL_GetHintBoolean("SDL_BORDERLESS_RESIZABLE_STYLE", SDL_FALSE)) {
- style |= STYLE_RESIZABLE;
- }
- }
- /* Need to set initialize minimize style, or when we call ShowWindow with WS_MINIMIZE it will activate a random window */
- if (window->flags & SDL_WINDOW_MINIMIZED) {
- style |= WS_MINIMIZE;
- }
- }
- return style;
- }
- static DWORD GetWindowStyleEx(SDL_Window *window)
- {
- DWORD style = WS_EX_APPWINDOW;
- if (SDL_WINDOW_IS_POPUP(window)) {
- style = WS_EX_TOOLWINDOW | WS_EX_NOACTIVATE;
- }
- return style;
- }
- /**
- * Returns arguments to pass to SetWindowPos - the window rect, including frame, in Windows coordinates.
- * Can be called before we have a HWND.
- */
- static void WIN_AdjustWindowRectWithStyle(SDL_Window *window, DWORD style, BOOL menu, int *x, int *y, int *width, int *height, SDL_bool use_current)
- {
- SDL_VideoData *videodata = SDL_GetVideoDevice() ? SDL_GetVideoDevice()->driverdata : NULL;
- RECT rect;
- int dpi = 96;
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- UINT frame_dpi;
- #endif
- /* Client rect, in SDL screen coordinates */
- SDL_RelativeToGlobalForWindow(window,
- (use_current ? window->x : window->windowed.x),
- (use_current ? window->y : window->windowed.y),
- x, y);
- *width = (use_current ? window->w : window->windowed.w);
- *height = (use_current ? window->h : window->windowed.h);
- /* Convert client rect from SDL coordinates to pixels (no-op if DPI scaling not enabled) */
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- WIN_ScreenPointFromSDL(x, y, &dpi);
- #endif
- /* Note, use the guessed DPI returned from WIN_ScreenPointFromSDL rather than the cached one in
- data->scaling_dpi.
- - This is called before the window is created, so we can't rely on data->scaling_dpi
- - Bug workaround: when leaving exclusive fullscreen, the cached DPI and window DPI reported
- by GetDpiForWindow will be wrong, and would cause windows shrinking slightly when
- going from exclusive fullscreen to windowed on a HighDPI monitor with scaling if we used them.
- */
- *width = MulDiv(*width, dpi, 96);
- *height = MulDiv(*height, dpi, 96);
- /* Copy the client size in pixels into this rect structure,
- which we'll then adjust with AdjustWindowRectEx */
- rect.left = 0;
- rect.top = 0;
- rect.right = *width;
- rect.bottom = *height;
- /* borderless windows will have WM_NCCALCSIZE return 0 for the non-client area. When this happens, it looks like windows will send a resize message
- expanding the window client area to the previous window + chrome size, so shouldn't need to adjust the window size for the set styles.
- */
- if (!(window->flags & SDL_WINDOW_BORDERLESS)) {
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- AdjustWindowRectEx(&rect, style, menu, 0);
- #else
- if (WIN_IsPerMonitorV2DPIAware(SDL_GetVideoDevice())) {
- /* With per-monitor v2, the window border/titlebar size depend on the DPI, so we need to call AdjustWindowRectExForDpi instead of
- AdjustWindowRectEx. */
- UINT unused;
- RECT screen_rect;
- HMONITOR mon;
- screen_rect.left = *x;
- screen_rect.top = *y;
- screen_rect.right = (LONG)*x + *width;
- screen_rect.bottom = (LONG)*y + *height;
- mon = MonitorFromRect(&screen_rect, MONITOR_DEFAULTTONEAREST);
- if (videodata != NULL) {
- /* GetDpiForMonitor docs promise to return the same hdpi / vdpi */
- if (videodata->GetDpiForMonitor(mon, MDT_EFFECTIVE_DPI, &frame_dpi, &unused) != S_OK) {
- frame_dpi = 96;
- }
- videodata->AdjustWindowRectExForDpi(&rect, style, menu, 0, frame_dpi);
- }
- } else {
- AdjustWindowRectEx(&rect, style, menu, 0);
- }
- #endif
- }
- /* Final rect in Windows screen space, including the frame */
- *x += rect.left;
- *y += rect.top;
- *width = (rect.right - rect.left);
- *height = (rect.bottom - rect.top);
- #ifdef HIGHDPI_DEBUG
- SDL_Log("WIN_AdjustWindowRectWithStyle: in: %d, %d, %dx%d, returning: %d, %d, %dx%d, used dpi %d for frame calculation",
- (use_current ? window->x : window->windowed.x),
- (use_current ? window->y : window->windowed.y),
- (use_current ? window->w : window->windowed.w),
- (use_current ? window->h : window->windowed.h),
- *x, *y, *width, *height, frame_dpi);
- #endif
- }
- static void WIN_AdjustWindowRect(SDL_Window *window, int *x, int *y, int *width, int *height, SDL_bool use_current)
- {
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- DWORD style;
- BOOL menu;
- style = GetWindowLong(hwnd, GWL_STYLE);
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- menu = FALSE;
- #else
- menu = (style & WS_CHILDWINDOW) ? FALSE : (GetMenu(hwnd) != NULL);
- #endif
- WIN_AdjustWindowRectWithStyle(window, style, menu, x, y, width, height, use_current);
- }
- void WIN_SetWindowPositionInternal(SDL_Window *window, UINT flags)
- {
- SDL_Window *child_window;
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- HWND top;
- int x, y;
- int w, h;
- /* Figure out what the window area will be */
- if (SDL_ShouldAllowTopmost() && (window->flags & SDL_WINDOW_ALWAYS_ON_TOP)) {
- top = HWND_TOPMOST;
- } else {
- top = HWND_NOTOPMOST;
- }
- WIN_AdjustWindowRect(window, &x, &y, &w, &h, SDL_TRUE);
- data->expected_resize = SDL_TRUE;
- SetWindowPos(hwnd, top, x, y, w, h, flags);
- data->expected_resize = SDL_FALSE;
- /* Update any child windows */
- for (child_window = window->first_child; child_window != NULL; child_window = child_window->next_sibling) {
- WIN_SetWindowPositionInternal(child_window, flags);
- }
- }
- static void SDLCALL WIN_MouseRelativeModeCenterChanged(void *userdata, const char *name, const char *oldValue, const char *hint)
- {
- SDL_WindowData *data = (SDL_WindowData *)userdata;
- data->mouse_relative_mode_center = SDL_GetStringBoolean(hint, SDL_TRUE);
- }
- static int WIN_GetScalingDPIForHWND(const SDL_VideoData *videodata, HWND hwnd)
- {
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- return 96;
- #else
- /* DPI scaling not requested? */
- if (!videodata->dpi_scaling_enabled) {
- return 96;
- }
- /* Window 10+ */
- if (videodata->GetDpiForWindow) {
- return videodata->GetDpiForWindow(hwnd);
- }
- /* Window 8.1+ */
- if (videodata->GetDpiForMonitor) {
- HMONITOR monitor = MonitorFromWindow(hwnd, MONITOR_DEFAULTTONEAREST);
- if (monitor) {
- UINT dpi_uint, unused;
- if (S_OK == videodata->GetDpiForMonitor(monitor, MDT_EFFECTIVE_DPI, &dpi_uint, &unused)) {
- return (int)dpi_uint;
- }
- }
- return 96;
- }
- /* Windows Vista-8.0 */
- {
- HDC hdc = GetDC(NULL);
- if (hdc) {
- int dpi = GetDeviceCaps(hdc, LOGPIXELSX);
- ReleaseDC(NULL, hdc);
- return dpi;
- }
- return 96;
- }
- #endif
- }
- static int SetupWindowData(_THIS, SDL_Window *window, HWND hwnd, HWND parent, SDL_bool created)
- {
- SDL_VideoData *videodata = _this->driverdata;
- SDL_WindowData *data;
- /* Allocate the window data */
- data = (SDL_WindowData *)SDL_calloc(1, sizeof(*data));
- if (data == NULL) {
- return SDL_OutOfMemory();
- }
- data->window = window;
- data->hwnd = hwnd;
- data->parent = parent;
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- data->hdc = (HDC)data->hwnd;
- #else
- data->hdc = GetDC(hwnd);
- #endif
- data->hinstance = (HINSTANCE)GetWindowLongPtr(hwnd, GWLP_HINSTANCE);
- data->created = created;
- data->high_surrogate = 0;
- data->mouse_button_flags = (WPARAM)-1;
- data->last_pointer_update = (LPARAM)-1;
- data->videodata = videodata;
- data->initializing = SDL_TRUE;
- data->last_displayID = window->last_displayID;
- data->scaling_dpi = WIN_GetScalingDPIForHWND(videodata, hwnd);
- #ifdef HIGHDPI_DEBUG
- SDL_Log("SetupWindowData: initialized data->scaling_dpi to %d", data->scaling_dpi);
- #endif
- SDL_AddHintCallback(SDL_HINT_MOUSE_RELATIVE_MODE_CENTER, WIN_MouseRelativeModeCenterChanged, data);
- window->driverdata = data;
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- /* Associate the data with the window */
- if (!SetProp(hwnd, TEXT("SDL_WindowData"), data)) {
- ReleaseDC(hwnd, data->hdc);
- SDL_free(data);
- return WIN_SetError("SetProp() failed");
- }
- #endif
- /* Set up the window proc function */
- #ifdef GWLP_WNDPROC
- data->wndproc = (WNDPROC)GetWindowLongPtr(hwnd, GWLP_WNDPROC);
- if (data->wndproc == WIN_WindowProc) {
- data->wndproc = NULL;
- } else {
- SetWindowLongPtr(hwnd, GWLP_WNDPROC, (LONG_PTR)WIN_WindowProc);
- }
- #else
- data->wndproc = (WNDPROC)GetWindowLong(hwnd, GWL_WNDPROC);
- if (data->wndproc == WIN_WindowProc) {
- data->wndproc = NULL;
- } else {
- SetWindowLong(hwnd, GWL_WNDPROC, (LONG_PTR)WIN_WindowProc);
- }
- #endif
- /* Fill in the SDL window with the window data */
- {
- RECT rect;
- if (GetClientRect(hwnd, &rect)) {
- int w = rect.right;
- int h = rect.bottom;
- WIN_ClientPointToSDL(window, &w, &h);
- if ((window->windowed.w && window->windowed.w != w) || (window->windowed.h && window->windowed.h != h)) {
- /* We tried to create a window larger than the desktop and Windows didn't allow it. Override! */
- int x, y;
- /* Figure out what the window area will be */
- WIN_AdjustWindowRect(window, &x, &y, &w, &h, SDL_FALSE);
- data->expected_resize = SDL_TRUE;
- SetWindowPos(hwnd, HWND_NOTOPMOST, x, y, w, h, SWP_NOCOPYBITS | SWP_NOZORDER | SWP_NOACTIVATE);
- data->expected_resize = SDL_FALSE;
- } else {
- window->w = w;
- window->h = h;
- }
- }
- }
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- {
- POINT point;
- point.x = 0;
- point.y = 0;
- if (ClientToScreen(hwnd, &point)) {
- int x = point.x;
- int y = point.y;
- WIN_ScreenPointToSDL(&x, &y);
- window->x = x;
- window->y = y;
- }
- }
- WIN_UpdateWindowICCProfile(window, SDL_FALSE);
- #endif
- {
- DWORD style = GetWindowLong(hwnd, GWL_STYLE);
- if (style & WS_VISIBLE) {
- window->flags &= ~SDL_WINDOW_HIDDEN;
- } else {
- window->flags |= SDL_WINDOW_HIDDEN;
- }
- if (style & WS_POPUP) {
- window->flags |= SDL_WINDOW_BORDERLESS;
- } else {
- window->flags &= ~SDL_WINDOW_BORDERLESS;
- }
- if (style & WS_THICKFRAME) {
- window->flags |= SDL_WINDOW_RESIZABLE;
- } else {
- window->flags &= ~SDL_WINDOW_RESIZABLE;
- }
- #ifdef WS_MAXIMIZE
- if (style & WS_MAXIMIZE) {
- window->flags |= SDL_WINDOW_MAXIMIZED;
- } else
- #endif
- {
- window->flags &= ~SDL_WINDOW_MAXIMIZED;
- }
- #ifdef WS_MINIMIZE
- if (style & WS_MINIMIZE) {
- window->flags |= SDL_WINDOW_MINIMIZED;
- } else
- #endif
- {
- window->flags &= ~SDL_WINDOW_MINIMIZED;
- }
- }
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- window->flags |= SDL_WINDOW_INPUT_FOCUS;
- #else
- if (GetFocus() == hwnd) {
- window->flags |= SDL_WINDOW_INPUT_FOCUS;
- SDL_SetKeyboardFocus(window);
- WIN_UpdateClipCursor(window);
- }
- #endif
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- /* Enable multi-touch */
- if (videodata->RegisterTouchWindow) {
- videodata->RegisterTouchWindow(hwnd, (TWF_FINETOUCH | TWF_WANTPALM));
- }
- #endif
- /* Force the SDL_WINDOW_ALLOW_HIGHDPI window flag if we are doing DPI scaling */
- if (videodata->dpi_scaling_enabled) {
- window->flags |= SDL_WINDOW_ALLOW_HIGHDPI;
- }
- if (data->parent && !window->parent) {
- data->destroy_parent_with_window = SDL_TRUE;
- }
- data->initializing = SDL_FALSE;
- /* All done! */
- return 0;
- }
- static void CleanupWindowData(_THIS, SDL_Window *window)
- {
- SDL_WindowData *data = window->driverdata;
- if (data) {
- SDL_DelHintCallback(SDL_HINT_MOUSE_RELATIVE_MODE_CENTER, WIN_MouseRelativeModeCenterChanged, data);
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- if (data->ICMFileName) {
- SDL_free(data->ICMFileName);
- }
- if (data->keyboard_hook) {
- UnhookWindowsHookEx(data->keyboard_hook);
- }
- ReleaseDC(data->hwnd, data->hdc);
- RemoveProp(data->hwnd, TEXT("SDL_WindowData"));
- #endif
- if (data->created) {
- DestroyWindow(data->hwnd);
- if (data->destroy_parent_with_window && data->parent) {
- DestroyWindow(data->parent);
- }
- } else {
- /* Restore any original event handler... */
- if (data->wndproc != NULL) {
- #ifdef GWLP_WNDPROC
- SetWindowLongPtr(data->hwnd, GWLP_WNDPROC,
- (LONG_PTR)data->wndproc);
- #else
- SetWindowLong(data->hwnd, GWL_WNDPROC,
- (LONG_PTR)data->wndproc);
- #endif
- }
- }
- SDL_free(data);
- }
- window->driverdata = NULL;
- }
- static void WIN_ConstrainPopup(SDL_Window *window)
- {
- /* Clamp popup windows to the output borders */
- if (SDL_WINDOW_IS_POPUP(window)) {
- SDL_Window *w;
- SDL_DisplayID displayID;
- SDL_Rect rect;
- int abs_x = window->x;
- int abs_y = window->y;
- int offset_x = 0, offset_y = 0;
- /* Calculate the total offset from the parents */
- for (w = window->parent; w->parent != NULL; w = w->parent) {
- offset_x += w->x;
- offset_y += w->y;
- }
- offset_x += w->x;
- offset_y += w->y;
- abs_x += offset_x;
- abs_y += offset_y;
- /* Constrain the popup window to the display of the toplevel parent */
- displayID = SDL_GetDisplayForWindow(w);
- SDL_GetDisplayBounds(displayID, &rect);
- if (abs_x + window->w > rect.x + rect.w) {
- abs_x -= (abs_x + window->w) - (rect.x + rect.w);
- }
- if (abs_y + window->h > rect.y + rect.h) {
- abs_y -= (abs_y + window->h) - (rect.y + rect.h);
- }
- abs_x = SDL_max(abs_x, rect.x);
- abs_y = SDL_max(abs_y, rect.y);
- window->x = window->windowed.x = abs_x - offset_x;
- window->y = window->windowed.y = abs_y - offset_y;
- }
- }
- static void WIN_SetKeyboardFocus(SDL_Window *window)
- {
- SDL_Window *topmost = window;
- /* Find the topmost parent */
- while (topmost->parent != NULL) {
- topmost = topmost->parent;
- }
- topmost->driverdata->keyboard_focus = window;
- SDL_SetKeyboardFocus(window);
- }
- int WIN_CreateWindow(_THIS, SDL_Window *window)
- {
- HWND hwnd, parent = NULL;
- DWORD style = STYLE_BASIC;
- DWORD styleEx = 0;
- int x, y;
- int w, h;
- if (SDL_WINDOW_IS_POPUP(window)) {
- parent = window->parent->driverdata->hwnd;
- } else if (window->flags & SDL_WINDOW_SKIP_TASKBAR) {
- parent = CreateWindow(SDL_Appname, TEXT(""), STYLE_BASIC, 0, 0, 32, 32, NULL, NULL, SDL_Instance, NULL);
- }
- style |= GetWindowStyle(window);
- styleEx |= GetWindowStyleEx(window);
- /* Figure out what the window area will be */
- WIN_ConstrainPopup(window);
- WIN_AdjustWindowRectWithStyle(window, style, FALSE, &x, &y, &w, &h, SDL_FALSE);
- hwnd = CreateWindowEx(styleEx, SDL_Appname, TEXT(""), style,
- x, y, w, h, parent, NULL, SDL_Instance, NULL);
- if (!hwnd) {
- return WIN_SetError("Couldn't create window");
- }
- WIN_UpdateDarkModeForHWND(hwnd);
- WIN_PumpEvents(_this);
- if (SetupWindowData(_this, window, hwnd, parent, SDL_TRUE) < 0) {
- DestroyWindow(hwnd);
- if (parent) {
- DestroyWindow(parent);
- }
- return -1;
- }
- /* Inform Windows of the frame change so we can respond to WM_NCCALCSIZE */
- SetWindowPos(hwnd, NULL, 0, 0, 0, 0, SWP_FRAMECHANGED | SWP_NOSIZE | SWP_NOZORDER | SWP_NOMOVE | SWP_NOACTIVATE);
- if (window->flags & SDL_WINDOW_MINIMIZED) {
- ShowWindow(hwnd, SW_SHOWMINNOACTIVE);
- }
- if (!(window->flags & SDL_WINDOW_OPENGL)) {
- return 0;
- }
- /* The rest of this macro mess is for OpenGL or OpenGL ES windows */
- #if SDL_VIDEO_OPENGL_ES2
- if ((_this->gl_config.profile_mask == SDL_GL_CONTEXT_PROFILE_ES ||
- SDL_GetHintBoolean(SDL_HINT_VIDEO_FORCE_EGL, SDL_FALSE)) &&
- #if SDL_VIDEO_OPENGL_WGL
- (!_this->gl_data || WIN_GL_UseEGL(_this))
- #endif /* SDL_VIDEO_OPENGL_WGL */
- ) {
- #if SDL_VIDEO_OPENGL_EGL
- if (WIN_GLES_SetupWindow(_this, window) < 0) {
- WIN_DestroyWindow(_this, window);
- return -1;
- }
- return 0;
- #else
- return SDL_SetError("Could not create GLES window surface (EGL support not configured)");
- #endif /* SDL_VIDEO_OPENGL_EGL */
- }
- #endif /* SDL_VIDEO_OPENGL_ES2 */
- #if SDL_VIDEO_OPENGL_WGL
- if (WIN_GL_SetupWindow(_this, window) < 0) {
- WIN_DestroyWindow(_this, window);
- return -1;
- }
- #else
- return SDL_SetError("Could not create GL window (WGL support not configured)");
- #endif
- return 0;
- }
- int WIN_CreateWindowFrom(_THIS, SDL_Window *window, const void *data)
- {
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- return -1;
- #else
- HWND hwnd = (HWND)data;
- LPTSTR title;
- int titleLen;
- SDL_bool isstack;
- /* Query the title from the existing window */
- titleLen = GetWindowTextLength(hwnd);
- title = SDL_small_alloc(TCHAR, titleLen + 1, &isstack);
- if (title) {
- titleLen = GetWindowText(hwnd, title, titleLen + 1);
- } else {
- titleLen = 0;
- }
- if (titleLen > 0) {
- window->title = WIN_StringToUTF8(title);
- }
- if (title) {
- SDL_small_free(title, isstack);
- }
- if (SetupWindowData(_this, window, hwnd, GetParent(hwnd), SDL_FALSE) < 0) {
- return -1;
- }
- #if SDL_VIDEO_OPENGL_WGL
- {
- const char *hint = SDL_GetHint(SDL_HINT_VIDEO_WINDOW_SHARE_PIXEL_FORMAT);
- if (hint) {
- /* This hint is a pointer (in string form) of the address of
- the window to share a pixel format with
- */
- SDL_Window *otherWindow = NULL;
- (void)SDL_sscanf(hint, "%p", (void **)&otherWindow);
- /* Do some error checking on the pointer */
- if (otherWindow != NULL && otherWindow->magic == &_this->window_magic) {
- /* If the otherWindow has SDL_WINDOW_OPENGL set, set it for the new window as well */
- if (otherWindow->flags & SDL_WINDOW_OPENGL) {
- window->flags |= SDL_WINDOW_OPENGL;
- if (!WIN_GL_SetPixelFormatFrom(_this, otherWindow, window)) {
- return -1;
- }
- }
- }
- } else if (window->flags & SDL_WINDOW_OPENGL) {
- /* Try to set up the pixel format, if it hasn't been set by the application */
- WIN_GL_SetupWindow(_this, window);
- }
- }
- #endif
- return 0;
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- }
- void WIN_SetWindowTitle(_THIS, SDL_Window *window)
- {
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- HWND hwnd = window->driverdata->hwnd;
- LPTSTR title = WIN_UTF8ToString(window->title);
- SetWindowText(hwnd, title);
- SDL_free(title);
- #endif
- }
- int WIN_SetWindowIcon(_THIS, SDL_Window *window, SDL_Surface *icon)
- {
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- HWND hwnd = window->driverdata->hwnd;
- HICON hicon = NULL;
- BYTE *icon_bmp;
- int icon_len, mask_len, row_len, y;
- BITMAPINFOHEADER *bmi;
- Uint8 *dst;
- SDL_bool isstack;
- int retval = 0;
- /* Create temporary buffer for ICONIMAGE structure */
- SDL_COMPILE_TIME_ASSERT(WIN_SetWindowIcon_uses_BITMAPINFOHEADER_to_prepare_an_ICONIMAGE, sizeof(BITMAPINFOHEADER) == 40);
- mask_len = (icon->h * (icon->w + 7) / 8);
- icon_len = sizeof(BITMAPINFOHEADER) + icon->h * icon->w * sizeof(Uint32) + mask_len;
- icon_bmp = SDL_small_alloc(BYTE, icon_len, &isstack);
- /* Write the BITMAPINFO header */
- bmi = (BITMAPINFOHEADER *)icon_bmp;
- bmi->biSize = SDL_SwapLE32(sizeof(BITMAPINFOHEADER));
- bmi->biWidth = SDL_SwapLE32(icon->w);
- bmi->biHeight = SDL_SwapLE32(icon->h * 2);
- bmi->biPlanes = SDL_SwapLE16(1);
- bmi->biBitCount = SDL_SwapLE16(32);
- bmi->biCompression = SDL_SwapLE32(BI_RGB);
- bmi->biSizeImage = SDL_SwapLE32(icon->h * icon->w * sizeof(Uint32));
- bmi->biXPelsPerMeter = SDL_SwapLE32(0);
- bmi->biYPelsPerMeter = SDL_SwapLE32(0);
- bmi->biClrUsed = SDL_SwapLE32(0);
- bmi->biClrImportant = SDL_SwapLE32(0);
- /* Write the pixels upside down into the bitmap buffer */
- SDL_assert(icon->format->format == SDL_PIXELFORMAT_ARGB8888);
- dst = &icon_bmp[sizeof(BITMAPINFOHEADER)];
- row_len = icon->w * sizeof(Uint32);
- y = icon->h;
- while (y--) {
- Uint8 *src = (Uint8 *)icon->pixels + y * icon->pitch;
- SDL_memcpy(dst, src, row_len);
- dst += row_len;
- }
- /* Write the mask */
- SDL_memset(icon_bmp + icon_len - mask_len, 0xFF, mask_len);
- hicon = CreateIconFromResource(icon_bmp, icon_len, TRUE, 0x00030000);
- SDL_small_free(icon_bmp, isstack);
- if (hicon == NULL) {
- retval = SDL_SetError("SetWindowIcon() failed, error %08X", (unsigned int)GetLastError());
- }
- /* Set the icon for the window */
- SendMessage(hwnd, WM_SETICON, ICON_SMALL, (LPARAM)hicon);
- /* Set the icon in the task manager (should we do this?) */
- SendMessage(hwnd, WM_SETICON, ICON_BIG, (LPARAM)hicon);
- return retval;
- #else
- return SDL_Unsupported();
- #endif
- }
- void WIN_SetWindowPosition(_THIS, SDL_Window *window)
- {
- /* HighDPI support: removed SWP_NOSIZE. If the move results in a DPI change, we need to allow
- * the window to resize (e.g. AdjustWindowRectExForDpi frame sizes are different).
- */
- WIN_ConstrainPopup(window);
- WIN_SetWindowPositionInternal(window, SWP_NOCOPYBITS | SWP_NOACTIVATE);
- }
- void WIN_SetWindowSize(_THIS, SDL_Window *window)
- {
- WIN_SetWindowPositionInternal(window, SWP_NOCOPYBITS | SWP_NOMOVE | SWP_NOACTIVATE);
- }
- int WIN_GetWindowBordersSize(_THIS, SDL_Window *window, int *top, int *left, int *bottom, int *right)
- {
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- HWND hwnd = window->driverdata->hwnd;
- RECT rcClient;
- /* rcClient stores the size of the inner window, while rcWindow stores the outer size relative to the top-left
- * screen position; so the top/left values of rcClient are always {0,0} and bottom/right are {height,width} */
- GetClientRect(hwnd, &rcClient);
- *top = rcClient.top;
- *left = rcClient.left;
- *bottom = rcClient.bottom;
- *right = rcClient.right;
- return 0;
- #else /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- HWND hwnd = window->driverdata->hwnd;
- RECT rcClient, rcWindow;
- POINT ptDiff;
- /* rcClient stores the size of the inner window, while rcWindow stores the outer size relative to the top-left
- * screen position; so the top/left values of rcClient are always {0,0} and bottom/right are {height,width} */
- if (!GetClientRect(hwnd, &rcClient)) {
- return SDL_SetError("GetClientRect() failed, error %08X", (unsigned int)GetLastError());
- }
- if (!GetWindowRect(hwnd, &rcWindow)) {
- return SDL_SetError("GetWindowRect() failed, error %08X", (unsigned int)GetLastError());
- }
- /* convert the top/left values to make them relative to
- * the window; they will end up being slightly negative */
- ptDiff.y = rcWindow.top;
- ptDiff.x = rcWindow.left;
- if (!ScreenToClient(hwnd, &ptDiff)) {
- return SDL_SetError("ScreenToClient() failed, error %08X", (unsigned int)GetLastError());
- }
- rcWindow.top = ptDiff.y;
- rcWindow.left = ptDiff.x;
- /* convert the bottom/right values to make them relative to the window,
- * these will be slightly bigger than the inner width/height */
- ptDiff.y = rcWindow.bottom;
- ptDiff.x = rcWindow.right;
- if (!ScreenToClient(hwnd, &ptDiff)) {
- return SDL_SetError("ScreenToClient() failed, error %08X", (unsigned int)GetLastError());
- }
- rcWindow.bottom = ptDiff.y;
- rcWindow.right = ptDiff.x;
- /* Now that both the inner and outer rects use the same coordinate system we can substract them to get the border size.
- * Keep in mind that the top/left coordinates of rcWindow are negative because the border lies slightly before {0,0},
- * so switch them around because SDL3 wants them in positive. */
- *top = rcClient.top - rcWindow.top;
- *left = rcClient.left - rcWindow.left;
- *bottom = rcWindow.bottom - rcClient.bottom;
- *right = rcWindow.right - rcClient.right;
- return 0;
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- }
- void WIN_GetWindowSizeInPixels(_THIS, SDL_Window *window, int *w, int *h)
- {
- const SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- RECT rect;
- if (GetClientRect(hwnd, &rect) && !IsRectEmpty(&rect)) {
- *w = rect.right;
- *h = rect.bottom;
- } else {
- *w = window->last_pixel_w;
- *h = window->last_pixel_h;
- }
- }
- void WIN_ShowWindow(_THIS, SDL_Window *window)
- {
- DWORD style;
- HWND hwnd;
- int nCmdShow;
- hwnd = window->driverdata->hwnd;
- nCmdShow = SDL_GetHintBoolean(SDL_HINT_WINDOW_NO_ACTIVATION_WHEN_SHOWN, SDL_FALSE) ? SW_SHOWNA : SW_SHOW;
- style = GetWindowLong(hwnd, GWL_EXSTYLE);
- if (style & WS_EX_NOACTIVATE) {
- nCmdShow = SW_SHOWNOACTIVATE;
- }
- ShowWindow(hwnd, nCmdShow);
- if (window->flags & SDL_WINDOW_POPUP_MENU) {
- if (window->parent == SDL_GetKeyboardFocus()) {
- WIN_SetKeyboardFocus(window);
- }
- }
- }
- void WIN_HideWindow(_THIS, SDL_Window *window)
- {
- HWND hwnd = window->driverdata->hwnd;
- ShowWindow(hwnd, SW_HIDE);
- /* Transfer keyboard focus back to the parent */
- if (window->flags & SDL_WINDOW_POPUP_MENU) {
- if (window == SDL_GetKeyboardFocus()) {
- SDL_Window *new_focus = window->parent;
- /* Find the highest level window that isn't being hidden or destroyed. */
- while (new_focus->parent != NULL && (new_focus->is_hiding || new_focus->is_destroying)) {
- new_focus = new_focus->parent;
- }
- WIN_SetKeyboardFocus(new_focus);
- }
- }
- }
- void WIN_RaiseWindow(_THIS, SDL_Window *window)
- {
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- /* If desired, raise the window more forcefully.
- * Technique taken from http://stackoverflow.com/questions/916259/ .
- * Specifically, http://stackoverflow.com/a/34414846 .
- *
- * The issue is that Microsoft has gone through a lot of trouble to make it
- * nearly impossible to programmatically move a window to the foreground,
- * for "security" reasons. Apparently, the following song-and-dance gets
- * around their objections. */
- SDL_bool bForce = SDL_GetHintBoolean(SDL_HINT_FORCE_RAISEWINDOW, SDL_FALSE);
- HWND hCurWnd = NULL;
- DWORD dwMyID = 0u;
- DWORD dwCurID = 0u;
- HWND hwnd = window->driverdata->hwnd;
- if (bForce) {
- hCurWnd = GetForegroundWindow();
- dwMyID = GetCurrentThreadId();
- dwCurID = GetWindowThreadProcessId(hCurWnd, NULL);
- ShowWindow(hwnd, SW_RESTORE);
- AttachThreadInput(dwCurID, dwMyID, TRUE);
- SetWindowPos(hwnd, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);
- SetWindowPos(hwnd, HWND_NOTOPMOST, 0, 0, 0, 0, SWP_NOSIZE | SWP_NOMOVE);
- }
- SetForegroundWindow(hwnd);
- if (bForce) {
- AttachThreadInput(dwCurID, dwMyID, FALSE);
- SetFocus(hwnd);
- SetActiveWindow(hwnd);
- }
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- }
- void WIN_MaximizeWindow(_THIS, SDL_Window *window)
- {
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- data->expected_resize = SDL_TRUE;
- ShowWindow(hwnd, SW_MAXIMIZE);
- data->expected_resize = SDL_FALSE;
- }
- void WIN_MinimizeWindow(_THIS, SDL_Window *window)
- {
- HWND hwnd = window->driverdata->hwnd;
- ShowWindow(hwnd, SW_MINIMIZE);
- }
- void WIN_SetWindowBordered(_THIS, SDL_Window *window, SDL_bool bordered)
- {
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- DWORD style;
- style = GetWindowLong(hwnd, GWL_STYLE);
- style &= ~STYLE_MASK;
- style |= GetWindowStyle(window);
- data->in_border_change = SDL_TRUE;
- SetWindowLong(hwnd, GWL_STYLE, style);
- WIN_SetWindowPositionInternal(window, SWP_NOCOPYBITS | SWP_FRAMECHANGED | SWP_NOZORDER | SWP_NOACTIVATE);
- data->in_border_change = SDL_FALSE;
- }
- void WIN_SetWindowResizable(_THIS, SDL_Window *window, SDL_bool resizable)
- {
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- DWORD style;
- style = GetWindowLong(hwnd, GWL_STYLE);
- style &= ~STYLE_MASK;
- style |= GetWindowStyle(window);
- SetWindowLong(hwnd, GWL_STYLE, style);
- }
- void WIN_SetWindowAlwaysOnTop(_THIS, SDL_Window *window, SDL_bool on_top)
- {
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- if (on_top) {
- SetWindowPos(hwnd, HWND_TOPMOST, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE);
- } else {
- SetWindowPos(hwnd, HWND_NOTOPMOST, 0, 0, 0, 0, SWP_NOMOVE | SWP_NOSIZE);
- }
- }
- void WIN_RestoreWindow(_THIS, SDL_Window *window)
- {
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- data->expected_resize = SDL_TRUE;
- ShowWindow(hwnd, SW_RESTORE);
- data->expected_resize = SDL_FALSE;
- }
- /**
- * Reconfigures the window to fill the given display, if fullscreen is true, otherwise restores the window.
- */
- void WIN_SetWindowFullscreen(_THIS, SDL_Window *window, SDL_VideoDisplay *display, SDL_bool fullscreen)
- {
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- SDL_DisplayData *displaydata = display->driverdata;
- SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- MONITORINFO minfo;
- DWORD style;
- HWND top;
- int x, y;
- int w, h;
- if (!fullscreen && (window->flags & SDL_WINDOW_FULLSCREEN)) {
- /* Resizing the window on hide causes problems restoring it in Wine, and it's unnecessary.
- * Also, Windows would preview the minimized window with the wrong size.
- */
- return;
- }
- #ifdef HIGHDPI_DEBUG
- SDL_Log("WIN_SetWindowFullscreen: %d", (int)fullscreen);
- #endif
- if (SDL_ShouldAllowTopmost() && (window->flags & SDL_WINDOW_ALWAYS_ON_TOP)) {
- top = HWND_TOPMOST;
- } else {
- top = HWND_NOTOPMOST;
- }
- style = GetWindowLong(hwnd, GWL_STYLE);
- style &= ~STYLE_MASK;
- style |= GetWindowStyle(window);
- /* Use GetMonitorInfo instead of WIN_GetDisplayBounds because we want the
- monitor bounds in Windows coordinates (pixels) rather than SDL coordinates (points). */
- SDL_zero(minfo);
- minfo.cbSize = sizeof(MONITORINFO);
- if (!GetMonitorInfo(displaydata->MonitorHandle, &minfo)) {
- SDL_SetError("GetMonitorInfo failed");
- return;
- }
- if (fullscreen) {
- x = minfo.rcMonitor.left;
- y = minfo.rcMonitor.top;
- w = minfo.rcMonitor.right - minfo.rcMonitor.left;
- h = minfo.rcMonitor.bottom - minfo.rcMonitor.top;
- /* Unset the maximized flag. This fixes
- https://bugzilla.libsdl.org/show_bug.cgi?id=3215
- */
- if (style & WS_MAXIMIZE) {
- data->windowed_mode_was_maximized = SDL_TRUE;
- style &= ~WS_MAXIMIZE;
- }
- } else {
- BOOL menu;
- /* Restore window-maximization state, as applicable.
- Special care is taken to *not* do this if and when we're
- alt-tab'ing away (to some other window; as indicated by
- in_window_deactivation), otherwise
- https://bugzilla.libsdl.org/show_bug.cgi?id=3215 can reproduce!
- */
- if (data->windowed_mode_was_maximized && !data->in_window_deactivation) {
- style |= WS_MAXIMIZE;
- data->windowed_mode_was_maximized = SDL_FALSE;
- }
- menu = (style & WS_CHILDWINDOW) ? FALSE : (GetMenu(hwnd) != NULL);
- WIN_AdjustWindowRectWithStyle(window, style, menu, &x, &y, &w, &h, SDL_FALSE);
- }
- SetWindowLong(hwnd, GWL_STYLE, style);
- data->expected_resize = SDL_TRUE;
- SetWindowPos(hwnd, top, x, y, w, h, SWP_NOCOPYBITS | SWP_NOACTIVATE);
- data->expected_resize = SDL_FALSE;
- #ifdef HIGHDPI_DEBUG
- SDL_Log("WIN_SetWindowFullscreen: %d finished. Set window to %d,%d, %dx%d", (int)fullscreen, x, y, w, h);
- #endif
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- }
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- void WIN_UpdateWindowICCProfile(SDL_Window *window, SDL_bool send_event)
- {
- SDL_WindowData *data = window->driverdata;
- SDL_DisplayData *displaydata = SDL_GetDisplayDriverDataForWindow(window);
- if (displaydata) {
- HDC hdc = CreateDCW(displaydata->DeviceName, NULL, NULL, NULL);
- if (hdc) {
- WCHAR fileName[MAX_PATH];
- DWORD fileNameSize = SDL_arraysize(fileName);
- if (GetICMProfileW(hdc, &fileNameSize, fileName)) {
- /* fileNameSize includes '\0' on return */
- if (!data->ICMFileName ||
- SDL_wcscmp(data->ICMFileName, fileName) != 0) {
- if (data->ICMFileName) {
- SDL_free(data->ICMFileName);
- }
- data->ICMFileName = SDL_wcsdup(fileName);
- if (send_event) {
- SDL_SendWindowEvent(window, SDL_EVENT_WINDOW_ICCPROF_CHANGED, 0, 0);
- }
- }
- }
- DeleteDC(hdc);
- }
- }
- }
- void *WIN_GetWindowICCProfile(_THIS, SDL_Window *window, size_t *size)
- {
- SDL_WindowData *data = window->driverdata;
- char *filename_utf8;
- void *iccProfileData = NULL;
- filename_utf8 = WIN_StringToUTF8(data->ICMFileName);
- if (filename_utf8) {
- iccProfileData = SDL_LoadFile(filename_utf8, size);
- if (iccProfileData == NULL) {
- SDL_SetError("Could not open ICC profile");
- }
- SDL_free(filename_utf8);
- } else {
- SDL_OutOfMemory();
- }
- return iccProfileData;
- }
- static void WIN_GrabKeyboard(SDL_Window *window)
- {
- SDL_WindowData *data = window->driverdata;
- HMODULE module;
- if (data->keyboard_hook) {
- return;
- }
- /* SetWindowsHookEx() needs to know which module contains the hook we
- want to install. This is complicated by the fact that SDL can be
- linked statically or dynamically. Fortunately XP and later provide
- this nice API that will go through the loaded modules and find the
- one containing our code.
- */
- if (!GetModuleHandleEx(GET_MODULE_HANDLE_EX_FLAG_UNCHANGED_REFCOUNT | GET_MODULE_HANDLE_EX_FLAG_FROM_ADDRESS,
- (LPTSTR)WIN_KeyboardHookProc,
- &module)) {
- return;
- }
- /* Capture a snapshot of the current keyboard state before the hook */
- if (!GetKeyboardState(data->videodata->pre_hook_key_state)) {
- return;
- }
- /* To grab the keyboard, we have to install a low-level keyboard hook to
- intercept keys that would normally be captured by the OS. Intercepting
- all key events on the system is rather invasive, but it's what Microsoft
- actually documents that you do to capture these.
- */
- data->keyboard_hook = SetWindowsHookEx(WH_KEYBOARD_LL, WIN_KeyboardHookProc, module, 0);
- }
- void WIN_UngrabKeyboard(SDL_Window *window)
- {
- SDL_WindowData *data = window->driverdata;
- if (data->keyboard_hook) {
- UnhookWindowsHookEx(data->keyboard_hook);
- data->keyboard_hook = NULL;
- }
- }
- void WIN_SetWindowMouseRect(_THIS, SDL_Window *window)
- {
- WIN_UpdateClipCursor(window);
- }
- void WIN_SetWindowMouseGrab(_THIS, SDL_Window *window, SDL_bool grabbed)
- {
- WIN_UpdateClipCursor(window);
- }
- void WIN_SetWindowKeyboardGrab(_THIS, SDL_Window *window, SDL_bool grabbed)
- {
- if (grabbed) {
- WIN_GrabKeyboard(window);
- } else {
- WIN_UngrabKeyboard(window);
- }
- }
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- void WIN_DestroyWindow(_THIS, SDL_Window *window)
- {
- if (window->shaper) {
- SDL_ShapeData *shapedata = (SDL_ShapeData *)window->shaper->driverdata;
- if (shapedata) {
- if (shapedata->mask_tree) {
- SDL_FreeShapeTree(&shapedata->mask_tree);
- }
- SDL_free(shapedata);
- }
- SDL_free(window->shaper);
- window->shaper = NULL;
- }
- CleanupWindowData(_this, window);
- }
- int WIN_GetWindowWMInfo(_THIS, SDL_Window *window, SDL_SysWMinfo *info)
- {
- const SDL_WindowData *data = (const SDL_WindowData *)window->driverdata;
- info->subsystem = SDL_SYSWM_WINDOWS;
- info->info.win.window = data->hwnd;
- info->info.win.hdc = data->hdc;
- info->info.win.hinstance = data->hinstance;
- return 0;
- }
- /*
- * Creates a HelperWindow used for DirectInput.
- */
- int SDL_HelperWindowCreate(void)
- {
- HINSTANCE hInstance = GetModuleHandle(NULL);
- WNDCLASS wce;
- /* Make sure window isn't created twice. */
- if (SDL_HelperWindow != NULL) {
- return 0;
- }
- /* Create the class. */
- SDL_zero(wce);
- wce.lpfnWndProc = DefWindowProc;
- wce.lpszClassName = SDL_HelperWindowClassName;
- wce.hInstance = hInstance;
- /* Register the class. */
- SDL_HelperWindowClass = RegisterClass(&wce);
- if (SDL_HelperWindowClass == 0 && GetLastError() != ERROR_CLASS_ALREADY_EXISTS) {
- return WIN_SetError("Unable to create Helper Window Class");
- }
- /* Create the window. */
- SDL_HelperWindow = CreateWindowEx(0, SDL_HelperWindowClassName,
- SDL_HelperWindowName,
- WS_OVERLAPPED, CW_USEDEFAULT,
- CW_USEDEFAULT, CW_USEDEFAULT,
- CW_USEDEFAULT, HWND_MESSAGE, NULL,
- hInstance, NULL);
- if (SDL_HelperWindow == NULL) {
- UnregisterClass(SDL_HelperWindowClassName, hInstance);
- return WIN_SetError("Unable to create Helper Window");
- }
- return 0;
- }
- /*
- * Destroys the HelperWindow previously created with SDL_HelperWindowCreate.
- */
- void SDL_HelperWindowDestroy(void)
- {
- HINSTANCE hInstance = GetModuleHandle(NULL);
- /* Destroy the window. */
- if (SDL_HelperWindow != NULL) {
- if (DestroyWindow(SDL_HelperWindow) == 0) {
- WIN_SetError("Unable to destroy Helper Window");
- return;
- }
- SDL_HelperWindow = NULL;
- }
- /* Unregister the class. */
- if (SDL_HelperWindowClass != 0) {
- if ((UnregisterClass(SDL_HelperWindowClassName, hInstance)) == 0) {
- WIN_SetError("Unable to destroy Helper Window Class");
- return;
- }
- SDL_HelperWindowClass = 0;
- }
- }
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- void WIN_OnWindowEnter(_THIS, SDL_Window *window)
- {
- SDL_WindowData *data = window->driverdata;
- if (data == NULL || !data->hwnd) {
- /* The window wasn't fully initialized */
- return;
- }
- if (window->flags & SDL_WINDOW_ALWAYS_ON_TOP) {
- WIN_SetWindowPositionInternal(window, SWP_NOCOPYBITS | SWP_NOSIZE | SWP_NOACTIVATE);
- }
- }
- void WIN_UpdateClipCursor(SDL_Window *window)
- {
- SDL_WindowData *data = window->driverdata;
- SDL_Mouse *mouse = SDL_GetMouse();
- RECT rect, clipped_rect;
- if (data->in_title_click || data->focus_click_pending) {
- return;
- }
- if (data->skip_update_clipcursor) {
- return;
- }
- if (!GetClipCursor(&clipped_rect)) {
- return;
- }
- if ((mouse->relative_mode || (window->flags & SDL_WINDOW_MOUSE_GRABBED) ||
- (window->mouse_rect.w > 0 && window->mouse_rect.h > 0)) &&
- (window->flags & SDL_WINDOW_INPUT_FOCUS)) {
- if (mouse->relative_mode && !mouse->relative_mode_warp && data->mouse_relative_mode_center) {
- if (GetWindowRect(data->hwnd, &rect)) {
- /* WIN_WarpCursor() jitters by +1, and remote desktop warp wobble is +/- 1 */
- LONG remote_desktop_adjustment = GetSystemMetrics(SM_REMOTESESSION) ? 2 : 0;
- LONG cx, cy;
- cx = (rect.left + rect.right) / 2;
- cy = (rect.top + rect.bottom) / 2;
- /* Make an absurdly small clip rect */
- rect.left = cx - remote_desktop_adjustment;
- rect.right = cx + 1 + remote_desktop_adjustment;
- rect.top = cy;
- rect.bottom = cy + 1;
- if (SDL_memcmp(&rect, &clipped_rect, sizeof(rect)) != 0) {
- if (ClipCursor(&rect)) {
- data->cursor_clipped_rect = rect;
- }
- }
- }
- } else {
- if (GetClientRect(data->hwnd, &rect)) {
- ClientToScreen(data->hwnd, (LPPOINT)&rect);
- ClientToScreen(data->hwnd, (LPPOINT)&rect + 1);
- if (window->mouse_rect.w > 0 && window->mouse_rect.h > 0) {
- SDL_Rect mouse_rect_win_client;
- RECT mouse_rect, intersection;
- /* mouse_rect_win_client is the mouse rect in Windows client space */
- mouse_rect_win_client = window->mouse_rect;
- WIN_ClientPointFromSDL(window, &mouse_rect_win_client.x, &mouse_rect_win_client.y);
- WIN_ClientPointFromSDL(window, &mouse_rect_win_client.w, &mouse_rect_win_client.h);
- /* mouse_rect is the rect in Windows screen space */
- mouse_rect.left = rect.left + mouse_rect_win_client.x;
- mouse_rect.top = rect.top + mouse_rect_win_client.y;
- mouse_rect.right = mouse_rect.left + mouse_rect_win_client.w;
- mouse_rect.bottom = mouse_rect.top + mouse_rect_win_client.h;
- if (IntersectRect(&intersection, &rect, &mouse_rect)) {
- SDL_memcpy(&rect, &intersection, sizeof(rect));
- } else if (window->flags & SDL_WINDOW_MOUSE_GRABBED) {
- /* Mouse rect was invalid, just do the normal grab */
- } else {
- SDL_zero(rect);
- }
- }
- if (SDL_memcmp(&rect, &clipped_rect, sizeof(rect)) != 0) {
- if (!IsRectEmpty(&rect)) {
- if (ClipCursor(&rect)) {
- data->cursor_clipped_rect = rect;
- }
- } else {
- ClipCursor(NULL);
- SDL_zero(data->cursor_clipped_rect);
- }
- }
- }
- }
- } else {
- POINT first, second;
- first.x = clipped_rect.left;
- first.y = clipped_rect.top;
- second.x = clipped_rect.right - 1;
- second.y = clipped_rect.bottom - 1;
- if (PtInRect(&data->cursor_clipped_rect, first) &&
- PtInRect(&data->cursor_clipped_rect, second)) {
- ClipCursor(NULL);
- SDL_zero(data->cursor_clipped_rect);
- }
- }
- data->last_updated_clipcursor = SDL_GetTicks();
- }
- int WIN_SetWindowHitTest(SDL_Window *window, SDL_bool enabled)
- {
- return 0; /* just succeed, the real work is done elsewhere. */
- }
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- int WIN_SetWindowOpacity(_THIS, SDL_Window *window, float opacity)
- {
- #if defined(__XBOXONE__) || defined(__XBOXSERIES__)
- return -1;
- #else
- const SDL_WindowData *data = window->driverdata;
- HWND hwnd = data->hwnd;
- const LONG style = GetWindowLong(hwnd, GWL_EXSTYLE);
- SDL_assert(style != 0);
- if (opacity == 1.0f) {
- /* want it fully opaque, just mark it unlayered if necessary. */
- if (style & WS_EX_LAYERED) {
- if (SetWindowLong(hwnd, GWL_EXSTYLE, style & ~WS_EX_LAYERED) == 0) {
- return WIN_SetError("SetWindowLong()");
- }
- }
- } else {
- const BYTE alpha = (BYTE)((int)(opacity * 255.0f));
- /* want it transparent, mark it layered if necessary. */
- if (!(style & WS_EX_LAYERED)) {
- if (SetWindowLong(hwnd, GWL_EXSTYLE, style | WS_EX_LAYERED) == 0) {
- return WIN_SetError("SetWindowLong()");
- }
- }
- if (SetLayeredWindowAttributes(hwnd, 0, alpha, LWA_ALPHA) == 0) {
- return WIN_SetError("SetLayeredWindowAttributes()");
- }
- }
- return 0;
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- }
- /**
- * Convert a point in the client area from pixels to DPI-scaled points.
- *
- * No-op if DPI scaling is not enabled.
- */
- void WIN_ClientPointToSDL(const SDL_Window *window, int *x, int *y)
- {
- const SDL_WindowData *data = window->driverdata;
- const SDL_VideoData *videodata = data->videodata;
- if (!videodata->dpi_scaling_enabled) {
- return;
- }
- *x = MulDiv(*x, 96, data->scaling_dpi);
- *y = MulDiv(*y, 96, data->scaling_dpi);
- }
- void WIN_ClientPointToSDLFloat(const SDL_Window *window, LONG x, LONG y, float *xOut, float *yOut)
- {
- const SDL_WindowData *data = window->driverdata;
- const SDL_VideoData *videodata = data->videodata;
- if (videodata->dpi_scaling_enabled) {
- *xOut = (float)(x * 96) / data->scaling_dpi;
- *yOut = (float)(y * 96) / data->scaling_dpi;
- } else {
- *xOut = (float)x;
- *yOut = (float)y;
- }
- }
- /**
- * Convert a point in the client area from DPI-scaled points to pixels.
- *
- * No-op if DPI scaling is not enabled.
- */
- void WIN_ClientPointFromSDL(const SDL_Window *window, int *x, int *y)
- {
- const SDL_WindowData *data = window->driverdata;
- const SDL_VideoData *videodata = data->videodata;
- if (!videodata->dpi_scaling_enabled) {
- return;
- }
- *x = MulDiv(*x, data->scaling_dpi, 96);
- *y = MulDiv(*y, data->scaling_dpi, 96);
- }
- void WIN_ClientPointFromSDLFloat(const SDL_Window *window, float x, float y, LONG *xOut, LONG *yOut)
- {
- const SDL_WindowData *data = window->driverdata;
- const SDL_VideoData *videodata = data->videodata;
- if (videodata->dpi_scaling_enabled) {
- *xOut = (LONG)SDL_roundf((x * data->scaling_dpi) / 96.0f);
- *yOut = (LONG)SDL_roundf((y * data->scaling_dpi) / 96.0f);
- } else {
- *xOut = (LONG)SDL_roundf(x);
- *yOut = (LONG)SDL_roundf(y);
- }
- }
- #if !defined(__XBOXONE__) && !defined(__XBOXSERIES__)
- void WIN_AcceptDragAndDrop(SDL_Window *window, SDL_bool accept)
- {
- const SDL_WindowData *data = window->driverdata;
- DragAcceptFiles(data->hwnd, accept ? TRUE : FALSE);
- }
- int WIN_FlashWindow(_THIS, SDL_Window *window, SDL_FlashOperation operation)
- {
- FLASHWINFO desc;
- SDL_zero(desc);
- desc.cbSize = sizeof(desc);
- desc.hwnd = window->driverdata->hwnd;
- switch (operation) {
- case SDL_FLASH_CANCEL:
- desc.dwFlags = FLASHW_STOP;
- break;
- case SDL_FLASH_BRIEFLY:
- desc.dwFlags = FLASHW_TRAY;
- desc.uCount = 1;
- break;
- case SDL_FLASH_UNTIL_FOCUSED:
- desc.dwFlags = (FLASHW_TRAY | FLASHW_TIMERNOFG);
- break;
- default:
- return SDL_Unsupported();
- }
- FlashWindowEx(&desc);
- return 0;
- }
- #endif /*!defined(__XBOXONE__) && !defined(__XBOXSERIES__)*/
- void WIN_UpdateDarkModeForHWND(HWND hwnd)
- {
- void *handle = SDL_LoadObject("dwmapi.dll");
- if (handle) {
- DwmSetWindowAttribute_t DwmSetWindowAttributeFunc = (DwmSetWindowAttribute_t)SDL_LoadFunction(handle, "DwmSetWindowAttribute");
- if (DwmSetWindowAttributeFunc) {
- /* FIXME: Do we need to traverse children? */
- BOOL value = (SDL_GetSystemTheme() == SDL_SYSTEM_THEME_DARK) ? TRUE : FALSE;
- DwmSetWindowAttributeFunc(hwnd, DWMWA_USE_IMMERSIVE_DARK_MODE, &value, sizeof(value));
- }
- SDL_UnloadObject(handle);
- }
- }
- #endif /* SDL_VIDEO_DRIVER_WINDOWS */
|