mirror of
https://github.com/libsdl-org/SDL.git
synced 2025-05-28 15:39:10 +00:00
540 lines
18 KiB
C
540 lines
18 KiB
C
/*
|
|
Simple DirectMedia Layer
|
|
Copyright (C) 1997-2024 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"
|
|
#include "../SDL_dialog_utils.h"
|
|
|
|
#include <windows.h>
|
|
#include <commdlg.h>
|
|
#include <shlobj.h>
|
|
#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);
|
|
}
|