mirror of
https://github.com/libsdl-org/SDL.git
synced 2025-10-04 17:06:25 +00:00

Most SDL functions used to indicate success or failure using an int return code. These functions have been changed to return SDL_bool. Here is a coccinelle patch to change code that previously compared the return value to 0 and changes it to a boolean test: @ bool_return_type @ identifier func =~ "^(SDL_AddEventWatch|SDL_AddHintCallback|SDL_AddSurfaceAlternateImage|SDL_AddVulkanRenderSemaphores|SDL_BindAudioStream|SDL_BindAudioStreams|SDL_BlitSurface|SDL_BlitSurface9Grid|SDL_BlitSurfaceScaled|SDL_BlitSurfaceTiled|SDL_BlitSurfaceTiledWithScale|SDL_BlitSurfaceUnchecked|SDL_BlitSurfaceUncheckedScaled|SDL_CaptureMouse|SDL_ClearAudioStream|SDL_ClearClipboardData|SDL_ClearComposition|SDL_ClearError|SDL_ClearProperty|SDL_ClearSurface|SDL_CloseIO|SDL_CloseStorage|SDL_ConvertAudioSamples|SDL_ConvertEventToRenderCoordinates|SDL_ConvertPixels|SDL_ConvertPixelsAndColorspace|SDL_CopyFile|SDL_CopyProperties|SDL_CopyStorageFile|SDL_CreateDirectory|SDL_CreateStorageDirectory|SDL_CreateWindowAndRenderer|SDL_DateTimeToTime|SDL_DestroyWindowSurface|SDL_DetachVirtualJoystick|SDL_DisableScreenSaver|SDL_EnableScreenSaver|SDL_EnumerateDirectory|SDL_EnumerateProperties|SDL_EnumerateStorageDirectory|SDL_FillSurfaceRect|SDL_FillSurfaceRects|SDL_FlashWindow|SDL_FlipSurface|SDL_FlushAudioStream|SDL_FlushRenderer|SDL_GL_DestroyContext|SDL_GL_GetAttribute|SDL_GL_GetSwapInterval|SDL_GL_LoadLibrary|SDL_GL_MakeCurrent|SDL_GL_SetAttribute|SDL_GL_SetSwapInterval|SDL_GL_SwapWindow|SDL_GetAudioDeviceFormat|SDL_GetAudioStreamFormat|SDL_GetCameraFormat|SDL_GetClosestFullscreenDisplayMode|SDL_GetCurrentRenderOutputSize|SDL_GetCurrentTime|SDL_GetDXGIOutputInfo|SDL_GetDateTimeLocalePreferences|SDL_GetDisplayBounds|SDL_GetDisplayUsableBounds|SDL_GetGDKDefaultUser|SDL_GetGDKTaskQueue|SDL_GetGamepadSensorData|SDL_GetGamepadTouchpadFinger|SDL_GetHapticEffectStatus|SDL_GetJoystickBall|SDL_GetMasksForPixelFormat|SDL_GetPathInfo|SDL_GetRectUnion|SDL_GetRectUnionFloat|SDL_GetRenderClipRect|SDL_GetRenderColorScale|SDL_GetRenderDrawBlendMode|SDL_GetRenderDrawColor|SDL_GetRenderDrawColorFloat|SDL_GetRenderLogicalPresentation|SDL_GetRenderLogicalPresentationRect|SDL_GetRenderOutputSize|SDL_GetRenderSafeArea|SDL_GetRenderScale|SDL_GetRenderVSync|SDL_GetRenderViewport|SDL_GetSensorData|SDL_GetStorageFileSize|SDL_GetStoragePathInfo|SDL_GetSurfaceAlphaMod|SDL_GetSurfaceBlendMode|SDL_GetSurfaceClipRect|SDL_GetSurfaceColorKey|SDL_GetSurfaceColorMod|SDL_GetTextInputArea|SDL_GetTextureAlphaMod|SDL_GetTextureAlphaModFloat|SDL_GetTextureBlendMode|SDL_GetTextureColorMod|SDL_GetTextureColorModFloat|SDL_GetTextureScaleMode|SDL_GetTextureSize|SDL_GetWindowAspectRatio|SDL_GetWindowBordersSize|SDL_GetWindowMaximumSize|SDL_GetWindowMinimumSize|SDL_GetWindowPosition|SDL_GetWindowRelativeMouseMode|SDL_GetWindowSafeArea|SDL_GetWindowSize|SDL_GetWindowSizeInPixels|SDL_GetWindowSurfaceVSync|SDL_HideCursor|SDL_HideWindow|SDL_Init|SDL_InitHapticRumble|SDL_InitSubSystem|SDL_LoadWAV|SDL_LoadWAV_IO|SDL_LockAudioStream|SDL_LockProperties|SDL_LockSurface|SDL_LockTexture|SDL_LockTextureToSurface|SDL_MaximizeWindow|SDL_MinimizeWindow|SDL_MixAudio|SDL_OpenURL|SDL_OutOfMemory|SDL_PauseAudioDevice|SDL_PauseAudioStreamDevice|SDL_PauseHaptic|SDL_PlayHapticRumble|SDL_PremultiplyAlpha|SDL_PremultiplySurfaceAlpha|SDL_PushEvent|SDL_PutAudioStreamData|SDL_RaiseWindow|SDL_ReadStorageFile|SDL_ReadSurfacePixel|SDL_ReadSurfacePixelFloat|SDL_RegisterApp|SDL_ReloadGamepadMappings|SDL_RemovePath|SDL_RemoveStoragePath|SDL_RemoveTimer|SDL_RenamePath|SDL_RenameStoragePath|SDL_RenderClear|SDL_RenderCoordinatesFromWindow|SDL_RenderCoordinatesToWindow|SDL_RenderFillRect|SDL_RenderFillRects|SDL_RenderGeometry|SDL_RenderGeometryRaw|SDL_RenderLine|SDL_RenderLines|SDL_RenderPoint|SDL_RenderPoints|SDL_RenderPresent|SDL_RenderRect|SDL_RenderRects|SDL_RenderTexture|SDL_RenderTexture9Grid|SDL_RenderTextureRotated|SDL_RenderTextureTiled|SDL_RequestAndroidPermission|SDL_RestoreWindow|SDL_ResumeAudioDevice|SDL_ResumeAudioStreamDevice|SDL_ResumeHaptic|SDL_RumbleGamepad|SDL_RumbleGamepadTriggers|SDL_RumbleJoystick|SDL_RumbleJoystickTriggers|SDL_RunHapticEffect|SDL_SaveBMP|SDL_SaveBMP_IO|SDL_SendAndroidMessage|SDL_SendGamepadEffect|SDL_SendJoystickEffect|SDL_SendJoystickVirtualSensorData|SDL_SetAppMetadata|SDL_SetAppMetadataProperty|SDL_SetAudioDeviceGain|SDL_SetAudioPostmixCallback|SDL_SetAudioStreamFormat|SDL_SetAudioStreamFrequencyRatio|SDL_SetAudioStreamGain|SDL_SetAudioStreamGetCallback|SDL_SetAudioStreamInputChannelMap|SDL_SetAudioStreamOutputChannelMap|SDL_SetAudioStreamPutCallback|SDL_SetBooleanProperty|SDL_SetClipboardData|SDL_SetClipboardText|SDL_SetCursor|SDL_SetFloatProperty|SDL_SetGamepadLED|SDL_SetGamepadMapping|SDL_SetGamepadPlayerIndex|SDL_SetGamepadSensorEnabled|SDL_SetHapticAutocenter|SDL_SetHapticGain|SDL_SetJoystickLED|SDL_SetJoystickPlayerIndex|SDL_SetJoystickVirtualAxis|SDL_SetJoystickVirtualBall|SDL_SetJoystickVirtualButton|SDL_SetJoystickVirtualHat|SDL_SetJoystickVirtualTouchpad|SDL_SetLinuxThreadPriority|SDL_SetLinuxThreadPriorityAndPolicy|SDL_SetLogPriorityPrefix|SDL_SetMemoryFunctions|SDL_SetNumberProperty|SDL_SetPaletteColors|SDL_SetPointerProperty|SDL_SetPointerPropertyWithCleanup|SDL_SetPrimarySelectionText|SDL_SetRenderClipRect|SDL_SetRenderColorScale|SDL_SetRenderDrawBlendMode|SDL_SetRenderDrawColor|SDL_SetRenderDrawColorFloat|SDL_SetRenderLogicalPresentation|SDL_SetRenderScale|SDL_SetRenderTarget|SDL_SetRenderVSync|SDL_SetRenderViewport|SDL_SetScancodeName|SDL_SetStringProperty|SDL_SetSurfaceAlphaMod|SDL_SetSurfaceBlendMode|SDL_SetSurfaceColorKey|SDL_SetSurfaceColorMod|SDL_SetSurfaceColorspace|SDL_SetSurfacePalette|SDL_SetSurfaceRLE|SDL_SetTLS|SDL_SetTextInputArea|SDL_SetTextureAlphaMod|SDL_SetTextureAlphaModFloat|SDL_SetTextureBlendMode|SDL_SetTextureColorMod|SDL_SetTextureColorModFloat|SDL_SetTextureScaleMode|SDL_SetThreadPriority|SDL_SetWindowAlwaysOnTop|SDL_SetWindowAspectRatio|SDL_SetWindowBordered|SDL_SetWindowFocusable|SDL_SetWindowFullscreen|SDL_SetWindowFullscreenMode|SDL_SetWindowHitTest|SDL_SetWindowIcon|SDL_SetWindowKeyboardGrab|SDL_SetWindowMaximumSize|SDL_SetWindowMinimumSize|SDL_SetWindowModalFor|SDL_SetWindowMouseGrab|SDL_SetWindowMouseRect|SDL_SetWindowOpacity|SDL_SetWindowPosition|SDL_SetWindowRelativeMouseMode|SDL_SetWindowResizable|SDL_SetWindowShape|SDL_SetWindowSize|SDL_SetWindowSurfaceVSync|SDL_SetWindowTitle|SDL_SetiOSAnimationCallback|SDL_ShowAndroidToast|SDL_ShowCursor|SDL_ShowMessageBox|SDL_ShowSimpleMessageBox|SDL_ShowWindow|SDL_ShowWindowSystemMenu|SDL_StartTextInput|SDL_StartTextInputWithProperties|SDL_StopHapticEffect|SDL_StopHapticEffects|SDL_StopHapticRumble|SDL_StopTextInput|SDL_SyncWindow|SDL_TimeToDateTime|SDL_TryLockMutex|SDL_TryLockRWLockForReading|SDL_TryLockRWLockForWriting|SDL_TryWaitSemaphore|SDL_UnlockAudioStream|SDL_UpdateHapticEffect|SDL_UpdateNVTexture|SDL_UpdateTexture|SDL_UpdateWindowSurface|SDL_UpdateWindowSurfaceRects|SDL_UpdateYUVTexture|SDL_Vulkan_CreateSurface|SDL_Vulkan_LoadLibrary|SDL_WaitConditionTimeout|SDL_WaitSemaphoreTimeout|SDL_WarpMouseGlobal|SDL_WriteStorageFile|SDL_WriteSurfacePixel|SDL_WriteSurfacePixelFloat)$"; @@ ( func( ... ) - == 0 | - func( + !func( ... ) - < 0 | - func( + !func( ... ) - != 0 | - func( + !func( ... ) - == -1 )
527 lines
19 KiB
C
527 lines
19 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 "../../core/linux/SDL_dbus.h"
|
|
|
|
#ifdef SDL_USE_LIBDBUS
|
|
|
|
#include <errno.h>
|
|
#include <sys/types.h>
|
|
#include <sys/wait.h>
|
|
#include <unistd.h>
|
|
|
|
#define PORTAL_DESTINATION "org.freedesktop.portal.Desktop"
|
|
#define PORTAL_PATH "/org/freedesktop/portal/desktop"
|
|
#define PORTAL_INTERFACE "org.freedesktop.portal.FileChooser"
|
|
|
|
#define SIGNAL_SENDER "org.freedesktop.portal.Desktop"
|
|
#define SIGNAL_INTERFACE "org.freedesktop.portal.Request"
|
|
#define SIGNAL_NAME "Response"
|
|
#define SIGNAL_FILTER "type='signal', sender='"SIGNAL_SENDER"', interface='"SIGNAL_INTERFACE"', member='"SIGNAL_NAME"', path='"
|
|
|
|
#define HANDLE_LEN 10
|
|
|
|
#define WAYLAND_HANDLE_PREFIX "wayland:"
|
|
#define X11_HANDLE_PREFIX "x11:"
|
|
|
|
typedef struct {
|
|
SDL_DialogFileCallback callback;
|
|
void *userdata;
|
|
const char *path;
|
|
} SignalCallback;
|
|
|
|
static void DBus_AppendStringOption(SDL_DBusContext *dbus, DBusMessageIter *options, const char *key, const char *value)
|
|
{
|
|
DBusMessageIter options_pair, options_value;
|
|
|
|
dbus->message_iter_open_container(options, DBUS_TYPE_DICT_ENTRY, NULL, &options_pair);
|
|
dbus->message_iter_append_basic(&options_pair, DBUS_TYPE_STRING, &key);
|
|
dbus->message_iter_open_container(&options_pair, DBUS_TYPE_VARIANT, "s", &options_value);
|
|
dbus->message_iter_append_basic(&options_value, DBUS_TYPE_STRING, &value);
|
|
dbus->message_iter_close_container(&options_pair, &options_value);
|
|
dbus->message_iter_close_container(options, &options_pair);
|
|
}
|
|
|
|
static void DBus_AppendBoolOption(SDL_DBusContext *dbus, DBusMessageIter *options, const char *key, int value)
|
|
{
|
|
DBusMessageIter options_pair, options_value;
|
|
|
|
dbus->message_iter_open_container(options, DBUS_TYPE_DICT_ENTRY, NULL, &options_pair);
|
|
dbus->message_iter_append_basic(&options_pair, DBUS_TYPE_STRING, &key);
|
|
dbus->message_iter_open_container(&options_pair, DBUS_TYPE_VARIANT, "b", &options_value);
|
|
dbus->message_iter_append_basic(&options_value, DBUS_TYPE_BOOLEAN, &value);
|
|
dbus->message_iter_close_container(&options_pair, &options_value);
|
|
dbus->message_iter_close_container(options, &options_pair);
|
|
}
|
|
|
|
static void DBus_AppendFilter(SDL_DBusContext *dbus, DBusMessageIter *parent, const SDL_DialogFileFilter filter)
|
|
{
|
|
DBusMessageIter filter_entry, filter_array, filter_array_entry;
|
|
char *state = NULL, *patterns, *pattern, *glob_pattern;
|
|
int zero = 0;
|
|
|
|
dbus->message_iter_open_container(parent, DBUS_TYPE_STRUCT, NULL, &filter_entry);
|
|
dbus->message_iter_append_basic(&filter_entry, DBUS_TYPE_STRING, &filter.name);
|
|
dbus->message_iter_open_container(&filter_entry, DBUS_TYPE_ARRAY, "(us)", &filter_array);
|
|
|
|
patterns = SDL_strdup(filter.pattern);
|
|
if (!patterns) {
|
|
goto cleanup;
|
|
}
|
|
|
|
pattern = SDL_strtok_r(patterns, ";", &state);
|
|
while (pattern) {
|
|
size_t max_len = SDL_strlen(pattern) + 3;
|
|
|
|
dbus->message_iter_open_container(&filter_array, DBUS_TYPE_STRUCT, NULL, &filter_array_entry);
|
|
dbus->message_iter_append_basic(&filter_array_entry, DBUS_TYPE_UINT32, &zero);
|
|
|
|
glob_pattern = SDL_calloc(sizeof(char), max_len);
|
|
if (!glob_pattern) {
|
|
goto cleanup;
|
|
}
|
|
glob_pattern[0] = '*';
|
|
/* Special case: The '*' filter doesn't need to be rewritten */
|
|
if (pattern[0] != '*' || pattern[1]) {
|
|
glob_pattern[1] = '.';
|
|
SDL_strlcat(glob_pattern + 2, pattern, max_len);
|
|
}
|
|
dbus->message_iter_append_basic(&filter_array_entry, DBUS_TYPE_STRING, &glob_pattern);
|
|
SDL_free(glob_pattern);
|
|
|
|
dbus->message_iter_close_container(&filter_array, &filter_array_entry);
|
|
pattern = SDL_strtok_r(NULL, ";", &state);
|
|
}
|
|
|
|
cleanup:
|
|
SDL_free(patterns);
|
|
|
|
dbus->message_iter_close_container(&filter_entry, &filter_array);
|
|
dbus->message_iter_close_container(parent, &filter_entry);
|
|
}
|
|
|
|
static void DBus_AppendFilters(SDL_DBusContext *dbus, DBusMessageIter *options, const SDL_DialogFileFilter *filters, int nfilters)
|
|
{
|
|
DBusMessageIter options_pair, options_value, options_value_array;
|
|
static const char *filters_name = "filters";
|
|
|
|
dbus->message_iter_open_container(options, DBUS_TYPE_DICT_ENTRY, NULL, &options_pair);
|
|
dbus->message_iter_append_basic(&options_pair, DBUS_TYPE_STRING, &filters_name);
|
|
dbus->message_iter_open_container(&options_pair, DBUS_TYPE_VARIANT, "a(sa(us))", &options_value);
|
|
dbus->message_iter_open_container(&options_value, DBUS_TYPE_ARRAY, "(sa(us))", &options_value_array);
|
|
for (int i = 0; i < nfilters; i++) {
|
|
DBus_AppendFilter(dbus, &options_value_array, filters[i]);
|
|
}
|
|
dbus->message_iter_close_container(&options_value, &options_value_array);
|
|
dbus->message_iter_close_container(&options_pair, &options_value);
|
|
dbus->message_iter_close_container(options, &options_pair);
|
|
}
|
|
|
|
static void DBus_AppendByteArray(SDL_DBusContext *dbus, DBusMessageIter *options, const char *key, const char *value)
|
|
{
|
|
DBusMessageIter options_pair, options_value, options_array;
|
|
|
|
dbus->message_iter_open_container(options, DBUS_TYPE_DICT_ENTRY, NULL, &options_pair);
|
|
dbus->message_iter_append_basic(&options_pair, DBUS_TYPE_STRING, &key);
|
|
dbus->message_iter_open_container(&options_pair, DBUS_TYPE_VARIANT, "ay", &options_value);
|
|
dbus->message_iter_open_container(&options_value, DBUS_TYPE_ARRAY, "y", &options_array);
|
|
do {
|
|
dbus->message_iter_append_basic(&options_array, DBUS_TYPE_BYTE, value);
|
|
} while (*value++);
|
|
dbus->message_iter_close_container(&options_value, &options_array);
|
|
dbus->message_iter_close_container(&options_pair, &options_value);
|
|
dbus->message_iter_close_container(options, &options_pair);
|
|
}
|
|
|
|
static DBusHandlerResult DBus_MessageFilter(DBusConnection *conn, DBusMessage *msg, void *data) {
|
|
SDL_DBusContext *dbus = SDL_DBus_GetContext();
|
|
SignalCallback *signal_data = (SignalCallback *)data;
|
|
|
|
if (dbus->message_is_signal(msg, SIGNAL_INTERFACE, SIGNAL_NAME)
|
|
&& dbus->message_has_path(msg, signal_data->path)) {
|
|
DBusMessageIter signal_iter, result_array, array_entry, value_entry, uri_entry;
|
|
uint32_t result;
|
|
size_t length = 2, current = 0;
|
|
const char **path;
|
|
|
|
dbus->message_iter_init(msg, &signal_iter);
|
|
// Check if the parameters are what we expect
|
|
if (dbus->message_iter_get_arg_type(&signal_iter) != DBUS_TYPE_UINT32)
|
|
goto not_our_signal;
|
|
dbus->message_iter_get_basic(&signal_iter, &result);
|
|
|
|
if (result == 1 || result == 2) {
|
|
// cancelled
|
|
const char *result_data[] = { NULL };
|
|
signal_data->callback(signal_data->userdata, result_data, -1); // TODO: Set this to the last selected filter
|
|
goto handled;
|
|
}
|
|
else if (result) {
|
|
// some error occurred
|
|
signal_data->callback(signal_data->userdata, NULL, -1);
|
|
goto handled;
|
|
}
|
|
|
|
if (!dbus->message_iter_next(&signal_iter))
|
|
goto not_our_signal;
|
|
|
|
if (dbus->message_iter_get_arg_type(&signal_iter) != DBUS_TYPE_ARRAY)
|
|
goto not_our_signal;
|
|
dbus->message_iter_recurse(&signal_iter, &result_array);
|
|
|
|
while (dbus->message_iter_get_arg_type(&result_array) == DBUS_TYPE_DICT_ENTRY)
|
|
{
|
|
const char *method;
|
|
|
|
dbus->message_iter_recurse(&result_array, &array_entry);
|
|
if (dbus->message_iter_get_arg_type(&array_entry) != DBUS_TYPE_STRING)
|
|
goto not_our_signal;
|
|
|
|
dbus->message_iter_get_basic(&array_entry, &method);
|
|
if (!SDL_strcmp(method, "uris")) {
|
|
// we only care about the selected file paths
|
|
break;
|
|
}
|
|
|
|
if (!dbus->message_iter_next(&result_array))
|
|
goto not_our_signal;
|
|
}
|
|
|
|
if (!dbus->message_iter_next(&array_entry))
|
|
goto not_our_signal;
|
|
|
|
if (dbus->message_iter_get_arg_type(&array_entry) != DBUS_TYPE_VARIANT)
|
|
goto not_our_signal;
|
|
dbus->message_iter_recurse(&array_entry, &value_entry);
|
|
|
|
if (dbus->message_iter_get_arg_type(&value_entry) != DBUS_TYPE_ARRAY)
|
|
goto not_our_signal;
|
|
dbus->message_iter_recurse(&value_entry, &uri_entry);
|
|
|
|
path = SDL_malloc(sizeof(const char *) * length);
|
|
if (!path) {
|
|
signal_data->callback(signal_data->userdata, NULL, -1);
|
|
goto cleanup;
|
|
}
|
|
|
|
while (dbus->message_iter_get_arg_type(&uri_entry) == DBUS_TYPE_STRING)
|
|
{
|
|
const char *uri = NULL;
|
|
|
|
if (current >= length - 1) {
|
|
++length;
|
|
path = SDL_realloc(path, sizeof(const char *) * length);
|
|
if (!path) {
|
|
signal_data->callback(signal_data->userdata, NULL, -1);
|
|
goto cleanup;
|
|
}
|
|
}
|
|
|
|
dbus->message_iter_get_basic(&uri_entry, &uri);
|
|
|
|
// https://flatpak.github.io/xdg-desktop-portal/docs/doc-org.freedesktop.portal.FileChooser.html
|
|
// Returned paths will always start with 'file://'; SDL_URIToLocal() truncates it.
|
|
char *decoded_uri = SDL_malloc(SDL_strlen(uri) + 1);
|
|
if (SDL_URIToLocal(uri, decoded_uri)) {
|
|
path[current] = decoded_uri;
|
|
} else {
|
|
SDL_free(decoded_uri);
|
|
SDL_SetError("Portal dialogs: Unsupported protocol: %s", uri);
|
|
signal_data->callback(signal_data->userdata, NULL, -1);
|
|
goto cleanup;
|
|
}
|
|
|
|
dbus->message_iter_next(&uri_entry);
|
|
++current;
|
|
}
|
|
path[length - 1] = NULL;
|
|
signal_data->callback(signal_data->userdata, path, -1); // TODO: Fetch the index of the filter that was used
|
|
cleanup:
|
|
dbus->connection_remove_filter(conn, &DBus_MessageFilter, signal_data);
|
|
|
|
if (path) {
|
|
for (size_t i = 0; i < current; ++i) {
|
|
SDL_free((char *)path[i]);
|
|
}
|
|
}
|
|
|
|
SDL_free(path);
|
|
SDL_free((void *)signal_data->path);
|
|
SDL_free(signal_data);
|
|
handled:
|
|
return DBUS_HANDLER_RESULT_HANDLED;
|
|
}
|
|
not_our_signal:
|
|
return DBUS_HANDLER_RESULT_NOT_YET_HANDLED;
|
|
}
|
|
|
|
static void DBus_OpenDialog(const char *method, const char *method_title, SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const SDL_DialogFileFilter *filters, int nfilters, const char* default_location, bool allow_many, int open_folders)
|
|
{
|
|
SDL_DBusContext *dbus = SDL_DBus_GetContext();
|
|
DBusMessage *msg;
|
|
DBusMessageIter params, options;
|
|
const char *signal_id = NULL;
|
|
char *handle_str, *filter;
|
|
int filter_len;
|
|
static uint32_t handle_id = 0;
|
|
static char *default_parent_window = "";
|
|
SDL_PropertiesID props = SDL_GetWindowProperties(window);
|
|
|
|
const char *err_msg = validate_filters(filters, nfilters);
|
|
|
|
if (err_msg) {
|
|
SDL_SetError("%s", err_msg);
|
|
callback(userdata, NULL, -1);
|
|
return;
|
|
}
|
|
|
|
if (dbus == NULL) {
|
|
SDL_SetError("Failed to connect to DBus");
|
|
callback(userdata, NULL, -1);
|
|
return;
|
|
}
|
|
|
|
msg = dbus->message_new_method_call(PORTAL_DESTINATION, PORTAL_PATH, PORTAL_INTERFACE, method);
|
|
if (msg == NULL) {
|
|
SDL_SetError("Failed to send message to portal");
|
|
callback(userdata, NULL, -1);
|
|
return;
|
|
}
|
|
|
|
dbus->message_iter_init_append(msg, ¶ms);
|
|
|
|
handle_str = default_parent_window;
|
|
if (props) {
|
|
const char *parent_handle = SDL_GetStringProperty(props, SDL_PROP_WINDOW_WAYLAND_XDG_TOPLEVEL_EXPORT_HANDLE_STRING, NULL);
|
|
if (parent_handle) {
|
|
size_t len = SDL_strlen(parent_handle);
|
|
len += sizeof(WAYLAND_HANDLE_PREFIX) + 1;
|
|
handle_str = SDL_malloc(len * sizeof(char));
|
|
if (!handle_str) {
|
|
callback(userdata, NULL, -1);
|
|
return;
|
|
}
|
|
|
|
SDL_snprintf(handle_str, len, "%s%s", WAYLAND_HANDLE_PREFIX, parent_handle);
|
|
} else {
|
|
const Uint64 xid = (Uint64)SDL_GetNumberProperty(props, SDL_PROP_WINDOW_X11_WINDOW_NUMBER, 0);
|
|
if (xid) {
|
|
const size_t len = sizeof(X11_HANDLE_PREFIX) + 24; // A 64-bit number can be 20 characters max.
|
|
handle_str = SDL_malloc(len * sizeof(char));
|
|
if (!handle_str) {
|
|
callback(userdata, NULL, -1);
|
|
return;
|
|
}
|
|
|
|
// The portal wants X11 window ID numbers in hex.
|
|
SDL_snprintf(handle_str, len, "%s%" SDL_PRIx64, X11_HANDLE_PREFIX, xid);
|
|
}
|
|
}
|
|
}
|
|
|
|
dbus->message_iter_append_basic(¶ms, DBUS_TYPE_STRING, &handle_str);
|
|
if (handle_str != default_parent_window) {
|
|
SDL_free(handle_str);
|
|
}
|
|
|
|
dbus->message_iter_append_basic(¶ms, DBUS_TYPE_STRING, &method_title);
|
|
dbus->message_iter_open_container(¶ms, DBUS_TYPE_ARRAY, "{sv}", &options);
|
|
|
|
handle_str = SDL_malloc(sizeof(char) * (HANDLE_LEN + 1));
|
|
if (!handle_str) {
|
|
callback(userdata, NULL, -1);
|
|
return;
|
|
}
|
|
SDL_snprintf(handle_str, HANDLE_LEN, "%u", ++handle_id);
|
|
DBus_AppendStringOption(dbus, &options, "handle_token", handle_str);
|
|
SDL_free(handle_str);
|
|
|
|
DBus_AppendBoolOption(dbus, &options, "modal", !!window);
|
|
if (allow_many == true) {
|
|
DBus_AppendBoolOption(dbus, &options, "multiple", 1);
|
|
}
|
|
if (open_folders) {
|
|
DBus_AppendBoolOption(dbus, &options, "directory", 1);
|
|
}
|
|
if (filters) {
|
|
DBus_AppendFilters(dbus, &options, filters, nfilters);
|
|
}
|
|
if (default_location) {
|
|
DBus_AppendByteArray(dbus, &options, "current_folder", default_location);
|
|
}
|
|
dbus->message_iter_close_container(¶ms, &options);
|
|
|
|
DBusMessage *reply = dbus->connection_send_with_reply_and_block(dbus->session_conn, msg, DBUS_TIMEOUT_INFINITE, NULL);
|
|
if (reply) {
|
|
DBusMessageIter reply_iter;
|
|
dbus->message_iter_init(reply, &reply_iter);
|
|
|
|
if (dbus->message_iter_get_arg_type(&reply_iter) == DBUS_TYPE_OBJECT_PATH) {
|
|
dbus->message_iter_get_basic(&reply_iter, &signal_id);
|
|
}
|
|
}
|
|
|
|
if (!signal_id) {
|
|
SDL_SetError("Invalid response received by DBus");
|
|
callback(userdata, NULL, -1);
|
|
goto incorrect_type;
|
|
}
|
|
|
|
dbus->message_unref(msg);
|
|
|
|
filter_len = SDL_strlen(SIGNAL_FILTER) + SDL_strlen(signal_id) + 2;
|
|
filter = SDL_malloc(sizeof(char) * filter_len);
|
|
if (!filter) {
|
|
callback(userdata, NULL, -1);
|
|
goto incorrect_type;
|
|
}
|
|
|
|
SDL_snprintf(filter, filter_len, SIGNAL_FILTER"%s'", signal_id);
|
|
dbus->bus_add_match(dbus->session_conn, filter, NULL);
|
|
SDL_free(filter);
|
|
|
|
SignalCallback *data = SDL_malloc(sizeof(SignalCallback));
|
|
if (!data) {
|
|
callback(userdata, NULL, -1);
|
|
goto incorrect_type;
|
|
}
|
|
data->callback = callback;
|
|
data->userdata = userdata;
|
|
data->path = SDL_strdup(signal_id);
|
|
if (!data->path) {
|
|
SDL_free(data);
|
|
callback(userdata, NULL, -1);
|
|
goto incorrect_type;
|
|
}
|
|
|
|
/* TODO: This should be registered before opening the portal, or the filter will not catch
|
|
the message if it is sent before we register the filter.
|
|
*/
|
|
dbus->connection_add_filter(dbus->session_conn,
|
|
&DBus_MessageFilter, data, NULL);
|
|
dbus->connection_flush(dbus->session_conn);
|
|
|
|
incorrect_type:
|
|
dbus->message_unref(reply);
|
|
}
|
|
|
|
void SDL_Portal_ShowOpenFileDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const SDL_DialogFileFilter *filters, int nfilters, const char* default_location, bool allow_many)
|
|
{
|
|
DBus_OpenDialog("OpenFile", "Open File", callback, userdata, window, filters, nfilters, default_location, allow_many, 0);
|
|
}
|
|
|
|
void SDL_Portal_ShowSaveFileDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const SDL_DialogFileFilter *filters, int nfilters, const char* default_location)
|
|
{
|
|
DBus_OpenDialog("SaveFile", "Save File", callback, userdata, window, filters, nfilters, default_location, 0, 0);
|
|
}
|
|
|
|
void SDL_Portal_ShowOpenFolderDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const char* default_location, bool allow_many)
|
|
{
|
|
DBus_OpenDialog("OpenFile", "Open Folder", callback, userdata, window, NULL, 0, default_location, allow_many, 1);
|
|
}
|
|
|
|
bool SDL_Portal_detect(void)
|
|
{
|
|
SDL_DBusContext *dbus = SDL_DBus_GetContext();
|
|
DBusMessage *msg = NULL, *reply = NULL;
|
|
char *reply_str = NULL;
|
|
DBusMessageIter reply_iter;
|
|
static int portal_present = -1;
|
|
|
|
// No need for this if the result is cached.
|
|
if (portal_present != -1) {
|
|
return (portal_present > 0);
|
|
}
|
|
|
|
portal_present = 0;
|
|
|
|
if (!dbus) {
|
|
SDL_SetError("%s", "Failed to connect to DBus!");
|
|
return false;
|
|
}
|
|
|
|
// Use introspection to get the available services.
|
|
msg = dbus->message_new_method_call(PORTAL_DESTINATION, PORTAL_PATH, "org.freedesktop.DBus.Introspectable", "Introspect");
|
|
if (!msg) {
|
|
goto done;
|
|
}
|
|
|
|
reply = dbus->connection_send_with_reply_and_block(dbus->session_conn, msg, DBUS_TIMEOUT_INFINITE, NULL);
|
|
dbus->message_unref(msg);
|
|
if (!reply) {
|
|
goto done;
|
|
}
|
|
|
|
if (!dbus->message_iter_init(reply, &reply_iter)) {
|
|
goto done;
|
|
}
|
|
|
|
if (dbus->message_iter_get_arg_type(&reply_iter) != DBUS_TYPE_STRING) {
|
|
goto done;
|
|
}
|
|
|
|
/* Introspection gives us a dump of all the services on the destination in XML format, so search the
|
|
* giant string for the file chooser protocol.
|
|
*/
|
|
dbus->message_iter_get_basic(&reply_iter, &reply_str);
|
|
if (SDL_strstr(reply_str, PORTAL_INTERFACE)) {
|
|
portal_present = 1; // Found it!
|
|
}
|
|
|
|
done:
|
|
if (reply) {
|
|
dbus->message_unref(reply);
|
|
}
|
|
|
|
return (portal_present > 0);
|
|
}
|
|
|
|
#else
|
|
|
|
// Dummy implementation to avoid compilation problems
|
|
|
|
void SDL_Portal_ShowOpenFileDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const SDL_DialogFileFilter *filters, int nfilters, const char* default_location, bool allow_many)
|
|
{
|
|
SDL_Unsupported();
|
|
callback(userdata, NULL, -1);
|
|
}
|
|
|
|
void SDL_Portal_ShowSaveFileDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const SDL_DialogFileFilter *filters, int nfilters, const char* default_location)
|
|
{
|
|
SDL_Unsupported();
|
|
callback(userdata, NULL, -1);
|
|
}
|
|
|
|
void SDL_Portal_ShowOpenFolderDialog(SDL_DialogFileCallback callback, void* userdata, SDL_Window* window, const char* default_location, bool allow_many)
|
|
{
|
|
SDL_Unsupported();
|
|
callback(userdata, NULL, -1);
|
|
}
|
|
|
|
bool SDL_Portal_detect(void)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
#endif // SDL_USE_LIBDBUS
|