/* Simple DirectMedia Layer Copyright (C) 1997-2024 Sam Lantinga 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" #include "../SDL_dialog_utils.h" #include #include #include #include "../../core/windows/SDL_windows.h" #include "../../thread/SDL_systhread.h" /* If this number is too small, selecting too many files will give an error */ #define SELECTLIST_SIZE 65536 typedef struct { int is_save; const SDL_DialogFileFilter *filters; const char* default_file; SDL_Window* parent; DWORD flags; SDL_DialogFileCallback callback; void* userdata; } winArgs; typedef struct { SDL_Window* parent; SDL_DialogFileCallback callback; const char* default_folder; void* userdata; } winFArgs; /** Converts dialog.nFilterIndex to SDL-compatible value */ int getFilterIndex(int as_reported_by_windows, const SDL_DialogFileFilter *filters) { int filter_index = as_reported_by_windows - 1; if (filter_index < 0) { filter_index = 0; for (const SDL_DialogFileFilter *filter = filters; filter && filter->name && filter->pattern; filter++) { filter_index++; } } return filter_index; } /* TODO: The new version of file dialogs */ void windows_ShowFileDialog(void *ptr) { winArgs *args = (winArgs *) ptr; int is_save = args->is_save; const SDL_DialogFileFilter *filters = args->filters; const char* default_file = args->default_file; SDL_Window* parent = args->parent; DWORD flags = args->flags; SDL_DialogFileCallback callback = args->callback; void* userdata = args->userdata; /* GetOpenFileName and GetSaveFileName have the same signature (yes, LPOPENFILENAMEW even for the save dialog) */ typedef BOOL (WINAPI *pfnGetAnyFileNameW)(LPOPENFILENAMEW); typedef DWORD (WINAPI *pfnCommDlgExtendedError)(void); HMODULE lib = LoadLibraryW(L"Comdlg32.dll"); pfnGetAnyFileNameW pGetAnyFileName = NULL; pfnCommDlgExtendedError pCommDlgExtendedError = NULL; if (lib) { pGetAnyFileName = (pfnGetAnyFileNameW) GetProcAddress(lib, is_save ? "GetSaveFileNameW" : "GetOpenFileNameW"); pCommDlgExtendedError = (pfnCommDlgExtendedError) GetProcAddress(lib, "CommDlgExtendedError"); } else { SDL_SetError("Couldn't load Comdlg32.dll"); callback(userdata, NULL, -1); return; } if (!pGetAnyFileName) { SDL_SetError("Couldn't load GetOpenFileName/GetSaveFileName from library"); callback(userdata, NULL, -1); return; } if (!pCommDlgExtendedError) { SDL_SetError("Couldn't load CommDlgExtendedError from library"); callback(userdata, NULL, -1); return; } HWND window = NULL; if (parent) { window = (HWND) SDL_GetProperty(SDL_GetWindowProperties(parent), SDL_PROP_WINDOW_WIN32_HWND_POINTER, NULL); } wchar_t *filebuffer; /* lpstrFile */ wchar_t initfolder[MAX_PATH] = L""; /* lpstrInitialDir */ /* If SELECTLIST_SIZE is too large, putting filebuffer on the stack might cause an overflow */ filebuffer = (wchar_t *) SDL_malloc(SELECTLIST_SIZE * sizeof(wchar_t)); /* Necessary for the return code below */ SDL_memset(filebuffer, 0, SELECTLIST_SIZE * sizeof(wchar_t)); if (default_file) { /* On Windows 10, 11 and possibly others, lpstrFile can be initialized with a path and the dialog will start at that location, but *only if the path contains a filename*. If it ends with a folder (directory separator), it fails with 0x3002 (12290) FNERR_INVALIDFILENAME. For that specific case, lpstrInitialDir must be used instead, but just for that case, because lpstrInitialDir doesn't support file names. On top of that, lpstrInitialDir hides a special algorithm that decides which folder to actually use as starting point, which may or may not be the one provided, or some other unrelated folder. Also, the algorithm changes between platforms. Assuming the documentation is correct, the algorithm is there under 'lpstrInitialDir': https://learn.microsoft.com/en-us/windows/win32/api/commdlg/ns-commdlg-openfilenamew Finally, lpstrFile does not support forward slashes. lpstrInitialDir does, though. */ char last_c = default_file[SDL_strlen(default_file) - 1]; if (last_c == '\\' || last_c == '/') { MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, default_file, -1, initfolder, MAX_PATH); } else { MultiByteToWideChar(CP_UTF8, MB_ERR_INVALID_CHARS, default_file, -1, filebuffer, MAX_PATH); for (int i = 0; i < SELECTLIST_SIZE; i++) { if (filebuffer[i] == L'/') { filebuffer[i] = L'\\'; } } } } wchar_t *filter_wchar = NULL; if (filters) { /* '\x01' is used in place of a null byte */ /* suffix needs two null bytes in case the filter list is empty */ char *filterlist = convert_filters(filters, NULL, "", "", "\x01\x01", "", "\x01", "\x01", "*.", ";*.", ""); if (!filterlist) { callback(userdata, NULL, -1); SDL_free(filebuffer); return; } int filter_len = (int)SDL_strlen(filterlist); for (char *c = filterlist; *c; c++) { if (*c == '\x01') { *c = '\0'; } } int filter_wlen = MultiByteToWideChar(CP_UTF8, 0, filterlist, filter_len, NULL, 0); filter_wchar = SDL_malloc(filter_wlen * sizeof(wchar_t)); if (!filter_wchar) { SDL_free(filterlist); callback(userdata, NULL, -1); SDL_free(filebuffer); return; } MultiByteToWideChar(CP_UTF8, 0, filterlist, filter_len, filter_wchar, filter_wlen); SDL_free(filterlist); } OPENFILENAMEW dialog; dialog.lStructSize = sizeof(OPENFILENAME); dialog.hwndOwner = window; dialog.hInstance = 0; dialog.lpstrFilter = filter_wchar; dialog.lpstrCustomFilter = NULL; dialog.nMaxCustFilter = 0; dialog.nFilterIndex = 0; dialog.lpstrFile = filebuffer; dialog.nMaxFile = SELECTLIST_SIZE; dialog.lpstrFileTitle = NULL; dialog.lpstrInitialDir = *initfolder ? initfolder : NULL; dialog.lpstrTitle = NULL; dialog.Flags = flags | OFN_EXPLORER | OFN_HIDEREADONLY | OFN_NOCHANGEDIR; dialog.nFileOffset = 0; dialog.nFileExtension = 0; dialog.lpstrDefExt = NULL; dialog.lCustData = 0; dialog.lpfnHook = NULL; dialog.lpTemplateName = NULL; /* Skipped many mac-exclusive and reserved members */ dialog.FlagsEx = 0; BOOL result = pGetAnyFileName(&dialog); SDL_free(filter_wchar); if (result) { if (!(flags & OFN_ALLOWMULTISELECT)) { /* File is a C string stored in dialog.lpstrFile */ char *chosen_file = WIN_StringToUTF8W(dialog.lpstrFile); const char* opts[2] = { chosen_file, NULL }; callback(userdata, opts, getFilterIndex(dialog.nFilterIndex, filters)); SDL_free(chosen_file); } else { /* File is either a C string if the user chose a single file, else it's a series of strings formatted like: "C:\\path\\to\\folder\0filename1.ext\0filename2.ext\0\0" The code below will only stop on a double NULL in all cases, so it is important that the rest of the buffer has been zeroed. */ char chosen_folder[MAX_PATH]; char chosen_file[MAX_PATH]; wchar_t *file_ptr = dialog.lpstrFile; size_t nfiles = 0; size_t chosen_folder_size; char **chosen_files_list = (char **) SDL_malloc(sizeof(char *) * (nfiles + 1)); if (!chosen_files_list) { callback(userdata, NULL, -1); SDL_free(filebuffer); return; } chosen_files_list[nfiles] = NULL; if (WideCharToMultiByte(CP_UTF8, 0, file_ptr, -1, chosen_folder, MAX_PATH, NULL, NULL) >= MAX_PATH) { SDL_SetError("Path too long or invalid character in path"); SDL_free(chosen_files_list); callback(userdata, NULL, -1); SDL_free(filebuffer); return; } chosen_folder_size = SDL_strlen(chosen_folder); SDL_strlcpy(chosen_file, chosen_folder, MAX_PATH); chosen_file[chosen_folder_size] = '\\'; file_ptr += SDL_strlen(chosen_folder) + 1; while (*file_ptr) { nfiles++; char **new_cfl = (char **) SDL_realloc(chosen_files_list, sizeof(char*) * (nfiles + 1)); if (!new_cfl) { for (size_t i = 0; i < nfiles - 1; i++) { SDL_free(chosen_files_list[i]); } SDL_free(chosen_files_list); callback(userdata, NULL, -1); SDL_free(filebuffer); return; } chosen_files_list = new_cfl; chosen_files_list[nfiles] = NULL; int diff = ((int) chosen_folder_size) + 1; if (WideCharToMultiByte(CP_UTF8, 0, file_ptr, -1, chosen_file + diff, MAX_PATH - diff, NULL, NULL) >= MAX_PATH - diff) { SDL_SetError("Path too long or invalid character in path"); for (size_t i = 0; i < nfiles - 1; i++) { SDL_free(chosen_files_list[i]); } SDL_free(chosen_files_list); callback(userdata, NULL, -1); SDL_free(filebuffer); return; } file_ptr += SDL_strlen(chosen_file) + 1 - diff; chosen_files_list[nfiles - 1] = SDL_strdup(chosen_file); if (!chosen_files_list[nfiles - 1]) { for (size_t i = 0; i < nfiles - 1; i++) { SDL_free(chosen_files_list[i]); } SDL_free(chosen_files_list); callback(userdata, NULL, -1); SDL_free(filebuffer); return; } } /* If the user chose only one file, it's all just one string */ if (nfiles == 0) { nfiles++; char **new_cfl = (char **) SDL_realloc(chosen_files_list, sizeof(char*) * (nfiles + 1)); if (!new_cfl) { SDL_free(chosen_files_list); callback(userdata, NULL, -1); SDL_free(filebuffer); return; } chosen_files_list = new_cfl; chosen_files_list[nfiles] = NULL; chosen_files_list[nfiles - 1] = SDL_strdup(chosen_folder); if (!chosen_files_list[nfiles - 1]) { SDL_free(chosen_files_list); callback(userdata, NULL, -1); SDL_free(filebuffer); return; } } callback(userdata, (const char * const*) chosen_files_list, getFilterIndex(dialog.nFilterIndex, filters)); for (size_t i = 0; i < nfiles; i++) { SDL_free(chosen_files_list[i]); } SDL_free(chosen_files_list); } } else { DWORD error = pCommDlgExtendedError(); /* Error code 0 means the user clicked the cancel button. */ if (error == 0) { /* Unlike SDL's handling of errors, Windows does reset the error code to 0 after calling GetOpenFileName if another Windows function before set a different error code, so it's safe to check for success. */ const char* opts[1] = { NULL }; callback(userdata, opts, getFilterIndex(dialog.nFilterIndex, filters)); } else { SDL_SetError("Windows error, CommDlgExtendedError: %ld", pCommDlgExtendedError()); callback(userdata, NULL, -1); } } SDL_free(filebuffer); } int windows_file_dialog_thread(void* ptr) { windows_ShowFileDialog(ptr); SDL_free(ptr); return 0; } int CALLBACK browse_callback_proc(HWND hwnd, UINT uMsg, LPARAM lParam, LPARAM lpData) { switch (uMsg) { case BFFM_INITIALIZED: if (lpData) { SendMessage(hwnd, BFFM_SETSELECTION, TRUE, lpData); } break; case BFFM_SELCHANGED: break; case BFFM_VALIDATEFAILED: break; default: break; } return 0; } void windows_ShowFolderDialog(void* ptr) { winFArgs *args = (winFArgs *) ptr; SDL_Window *window = args->parent; SDL_DialogFileCallback callback = args->callback; void *userdata = args->userdata; HWND parent = NULL; if (window) { parent = (HWND) SDL_GetProperty(SDL_GetWindowProperties(window), SDL_PROP_WINDOW_WIN32_HWND_POINTER, NULL); } wchar_t buffer[MAX_PATH]; BROWSEINFOW dialog; dialog.hwndOwner = parent; dialog.pidlRoot = NULL; /* Windows docs say this is `LPTSTR` - apparently it's actually `LPWSTR`*/ dialog.pszDisplayName = buffer; dialog.lpszTitle = NULL; dialog.ulFlags = BIF_USENEWUI; dialog.lpfn = browse_callback_proc; dialog.lParam = (LPARAM)args->default_folder; dialog.iImage = 0; LPITEMIDLIST lpItem = SHBrowseForFolderW(&dialog); if (lpItem != NULL) { SHGetPathFromIDListW(lpItem, buffer); char *chosen_file = WIN_StringToUTF8W(buffer); const char *files[2] = { chosen_file, NULL }; callback(userdata, (const char * const*) files, -1); SDL_free(chosen_file); } else { const char *files[1] = { NULL }; callback(userdata, (const char * const*) files, -1); } } int windows_folder_dialog_thread(void* ptr) { windows_ShowFolderDialog(ptr); SDL_free(ptr); return 0; } void SDL_ShowOpenFileDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const SDL_DialogFileFilter *filters, const char* default_location, SDL_bool allow_many) { winArgs *args; SDL_Thread *thread; if (SDL_GetHint(SDL_HINT_FILE_DIALOG_DRIVER) != NULL) { SDL_Log("%s", SDL_GetHint(SDL_HINT_FILE_DIALOG_DRIVER)); SDL_SetError("File dialog driver unsupported"); callback(userdata, NULL, -1); return; } args = SDL_malloc(sizeof(winArgs)); if (args == NULL) { callback(userdata, NULL, -1); return; } args->is_save = 0; args->filters = filters; args->default_file = default_location; args->parent = window; args->flags = (allow_many == SDL_TRUE) ? OFN_ALLOWMULTISELECT : 0; args->callback = callback; args->userdata = userdata; thread = SDL_CreateThread(windows_file_dialog_thread, "SDL_ShowOpenFileDialog", (void *) args); if (thread == NULL) { callback(userdata, NULL, -1); SDL_free(args); return; } SDL_DetachThread(thread); } void SDL_ShowSaveFileDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const SDL_DialogFileFilter *filters, const char* default_location) { winArgs *args; SDL_Thread *thread; if (SDL_GetHint(SDL_HINT_FILE_DIALOG_DRIVER) != NULL) { SDL_SetError("File dialog driver unsupported"); callback(userdata, NULL, -1); return; } args = SDL_malloc(sizeof(winArgs)); if (args == NULL) { callback(userdata, NULL, -1); return; } args->is_save = 1; args->filters = filters; args->default_file = default_location; args->parent = window; args->flags = 0; args->callback = callback; args->userdata = userdata; thread = SDL_CreateThread(windows_file_dialog_thread, "SDL_ShowSaveFileDialog", (void *) args); if (thread == NULL) { callback(userdata, NULL, -1); SDL_free(args); return; } SDL_DetachThread(thread); } void SDL_ShowOpenFolderDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const char* default_location, SDL_bool allow_many) { winFArgs *args; SDL_Thread *thread; if (SDL_GetHint(SDL_HINT_FILE_DIALOG_DRIVER) != NULL) { SDL_SetError("File dialog driver unsupported"); callback(userdata, NULL, -1); return; } args = SDL_malloc(sizeof(winFArgs)); if (args == NULL) { callback(userdata, NULL, -1); return; } args->parent = window; args->callback = callback; args->default_folder = default_location; args->userdata = userdata; thread = SDL_CreateThread(windows_folder_dialog_thread, "SDL_ShowOpenFolderDialog", (void *) args); if (thread == NULL) { callback(userdata, NULL, -1); SDL_free(args); return; } SDL_DetachThread(thread); }