/* Simple DirectMedia Layer Copyright (C) 1997-2023 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_clipboard_c.h" #include "SDL_sysvideo.h" #include "../events/SDL_clipboardevents_c.h" void SDL_CancelClipboardData(Uint32 sequence) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); size_t i; if (sequence != _this->clipboard_sequence) { /* This clipboard data was already canceled */ return; } if (_this->clipboard_cleanup) { _this->clipboard_cleanup(_this->clipboard_userdata); } if (_this->clipboard_mime_types) { for (i = 0; i < _this->num_clipboard_mime_types; ++i) { SDL_free(_this->clipboard_mime_types[i]); } SDL_free(_this->clipboard_mime_types); _this->clipboard_mime_types = NULL; _this->num_clipboard_mime_types = 0; } _this->clipboard_callback = NULL; _this->clipboard_cleanup = NULL; _this->clipboard_userdata = NULL; } int SDL_SetClipboardData(SDL_ClipboardDataCallback callback, SDL_ClipboardCleanupCallback cleanup, void *userdata, const char **mime_types, size_t num_mime_types) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); size_t i; if (_this == NULL) { return SDL_SetError("Video subsystem must be initialized to set clipboard text"); } /* Parameter validation */ if (!((callback && mime_types && num_mime_types > 0) || (!callback && !mime_types && num_mime_types == 0))) { return SDL_SetError("Invalid parameters"); } if (!callback && !_this->clipboard_callback) { /* Nothing to do, don't modify the system clipboard */ return 0; } SDL_CancelClipboardData(_this->clipboard_sequence); ++_this->clipboard_sequence; if (!_this->clipboard_sequence) { _this->clipboard_sequence = 1; } _this->clipboard_callback = callback; _this->clipboard_cleanup = cleanup; _this->clipboard_userdata = userdata; if (mime_types && num_mime_types > 0) { size_t num_allocated = 0; _this->clipboard_mime_types = (char **)SDL_malloc(num_mime_types * sizeof(char *)); if (_this->clipboard_mime_types) { for (i = 0; i < num_mime_types; ++i) { _this->clipboard_mime_types[i] = SDL_strdup(mime_types[i]); if (_this->clipboard_mime_types[i]) { ++num_allocated; } } } if (num_allocated < num_mime_types) { SDL_ClearClipboardData(); return SDL_OutOfMemory(); } _this->num_clipboard_mime_types = num_mime_types; } if (_this->SetClipboardData) { if (_this->SetClipboardData(_this) < 0) { return -1; } } else if (_this->SetClipboardText) { char *text = NULL; size_t size; for (i = 0; i < num_mime_types; ++i) { const char *mime_type = _this->clipboard_mime_types[i]; if (SDL_IsTextMimeType(mime_type)) { const void *data = _this->clipboard_callback(_this->clipboard_userdata, mime_type, &size); if (data) { text = (char *)SDL_malloc(size + 1); SDL_memcpy(text, data, size); text[size] = '\0'; if (_this->SetClipboardText(_this, text) < 0) { SDL_free(text); return -1; } break; } } } if (text) { SDL_free(text); } else { if (_this->SetClipboardText(_this, "") < 0) { return -1; } } } SDL_SendClipboardUpdate(); return 0; } int SDL_ClearClipboardData(void) { return SDL_SetClipboardData(NULL, NULL, NULL, NULL, 0); } void *SDL_GetInternalClipboardData(SDL_VideoDevice *_this, const char *mime_type, size_t *size) { void *data = NULL; if (_this->clipboard_callback) { const void *provided_data = _this->clipboard_callback(_this->clipboard_userdata, mime_type, size); if (provided_data) { /* Make a copy of it for the caller and guarantee null termination */ data = SDL_malloc(*size + sizeof(Uint32)); if (data) { SDL_memcpy(data, provided_data, *size); SDL_memset((Uint8 *)data + *size, 0, sizeof(Uint32)); } else { SDL_OutOfMemory(); } } } return data; } void *SDL_GetClipboardData(const char *mime_type, size_t *size) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); if (_this == NULL) { SDL_SetError("Video subsystem must be initialized to get clipboard data"); return NULL; } if (!mime_type) { SDL_InvalidParamError("mime_type"); return NULL; } if (!size) { SDL_InvalidParamError("size"); return NULL; } /* Initialize size to empty, so implementations don't have to worry about it */ *size = 0; if (_this->GetClipboardData) { return _this->GetClipboardData(_this, mime_type, size); } else if (_this->GetClipboardText && SDL_IsTextMimeType(mime_type)) { void *data = _this->GetClipboardText(_this); if (data && *(char *)data == '\0') { SDL_free(data); data = NULL; } return data; } else { return SDL_GetInternalClipboardData(_this, mime_type, size); } } SDL_bool SDL_HasInternalClipboardData(SDL_VideoDevice *_this, const char *mime_type) { size_t i; for (i = 0; i < _this->num_clipboard_mime_types; ++i) { if (SDL_strcmp(mime_type, _this->clipboard_mime_types[i]) == 0) { return SDL_TRUE; } } return SDL_FALSE; } SDL_bool SDL_HasClipboardData(const char *mime_type) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); if (_this == NULL) { SDL_SetError("Video subsystem must be initialized to check clipboard data"); return SDL_FALSE; } if (!mime_type) { SDL_InvalidParamError("mime_type"); return SDL_FALSE; } if (_this->HasClipboardData) { return _this->HasClipboardData(_this, mime_type); } else if (_this->HasClipboardText && SDL_IsTextMimeType(mime_type)) { return _this->HasClipboardText(_this); } else { return SDL_HasInternalClipboardData(_this, mime_type); } } /* Clipboard text */ SDL_bool SDL_IsTextMimeType(const char *mime_type) { return (SDL_strncmp(mime_type, "text", 4) == 0); } static const char **SDL_GetTextMimeTypes(SDL_VideoDevice *_this, size_t *num_mime_types) { if (_this->GetTextMimeTypes) { return _this->GetTextMimeTypes(_this, num_mime_types); } else { static const char *text_mime_types[] = { "text/plain;charset=utf-8" }; *num_mime_types = SDL_arraysize(text_mime_types); return text_mime_types; } } const void *SDL_ClipboardTextCallback(void *userdata, const char *mime_type, size_t *size) { char *text = (char *)userdata; if (text) { *size = SDL_strlen(text); } else { *size = 0; } return text; } int SDL_SetClipboardText(const char *text) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); size_t num_mime_types; const char **text_mime_types; if (_this == NULL) { return SDL_SetError("Video subsystem must be initialized to set clipboard text"); } if (text && *text) { text_mime_types = SDL_GetTextMimeTypes(_this, &num_mime_types); return SDL_SetClipboardData(SDL_ClipboardTextCallback, SDL_free, SDL_strdup(text), text_mime_types, num_mime_types); } else { return SDL_ClearClipboardData(); } } char *SDL_GetClipboardText(void) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); size_t i, num_mime_types; const char **text_mime_types; size_t length; char *text = NULL; if (_this == NULL) { SDL_SetError("Video subsystem must be initialized to get clipboard text"); return SDL_strdup(""); } text_mime_types = SDL_GetTextMimeTypes(_this, &num_mime_types); for (i = 0; i < num_mime_types; ++i) { text = SDL_GetClipboardData(text_mime_types[i], &length); if (text) { break; } } if (!text) { text = SDL_strdup(""); } return text; } SDL_bool SDL_HasClipboardText(void) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); size_t i, num_mime_types; const char **text_mime_types; if (_this == NULL) { SDL_SetError("Video subsystem must be initialized to check clipboard text"); return SDL_FALSE; } text_mime_types = SDL_GetTextMimeTypes(_this, &num_mime_types); for (i = 0; i < num_mime_types; ++i) { if (SDL_HasClipboardData(text_mime_types[i])) { return SDL_TRUE; } } return SDL_FALSE; } /* Primary selection text */ int SDL_SetPrimarySelectionText(const char *text) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); if (_this == NULL) { return SDL_SetError("Video subsystem must be initialized to set primary selection text"); } if (text == NULL) { text = ""; } if (_this->SetPrimarySelectionText) { if (_this->SetPrimarySelectionText(_this, text) < 0) { return -1; } } else { SDL_free(_this->primary_selection_text); _this->primary_selection_text = SDL_strdup(text); } SDL_SendClipboardUpdate(); return 0; } char *SDL_GetPrimarySelectionText(void) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); if (_this == NULL) { SDL_SetError("Video subsystem must be initialized to get primary selection text"); return SDL_strdup(""); } if (_this->GetPrimarySelectionText) { return _this->GetPrimarySelectionText(_this); } else { const char *text = _this->primary_selection_text; if (text == NULL) { text = ""; } return SDL_strdup(text); } } SDL_bool SDL_HasPrimarySelectionText(void) { SDL_VideoDevice *_this = SDL_GetVideoDevice(); if (_this == NULL) { SDL_SetError("Video subsystem must be initialized to check primary selection text"); return SDL_FALSE; } if (_this->HasPrimarySelectionText) { return _this->HasPrimarySelectionText(_this); } else { if (_this->primary_selection_text && _this->primary_selection_text[0] != '\0') { return SDL_TRUE; } else { return SDL_FALSE; } } }