28#include <winpr/wlog.h>
29#include <winpr/image.h>
31#include "sdl_clip.hpp"
32#include "sdl_context.hpp"
34#define TAG CLIENT_TAG("sdl.cliprdr")
36#define mime_text_plain "text/plain"
38const char mime_text_utf8[] = mime_text_plain
";charset=utf-8";
40[[nodiscard]]
static const std::vector<const char*>& s_mime_text()
42 static std::vector<const char*> values;
45 values = std::vector<const char*>(
46 { mime_text_plain, mime_text_utf8,
"UTF8_STRING",
"COMPOUND_TEXT",
"TEXT",
"STRING" });
51static const char s_mime_png[] =
"image/png";
52static const char s_mime_webp[] =
"image/webp";
53static const char s_mime_jpg[] =
"image/jpeg";
55static const char s_mime_tiff[] =
"image/tiff";
56static const char s_mime_uri_list[] =
"text/uri-list";
57static const char s_mime_html[] =
"text/html";
59#define BMP_MIME_LIST "image/bmp", "image/x-bmp", "image/x-MS-bmp", "image/x-win-bitmap"
61[[nodiscard]]
static const std::vector<const char*>& s_mime_bitmap()
63 static std::vector<const char*> values;
66 values = std::vector<const char*>({ BMP_MIME_LIST });
71[[nodiscard]]
static const std::vector<const char*>& s_mime_image()
73 static std::vector<const char*> values;
76 if (winpr_image_format_is_supported(WINPR_IMAGE_WEBP))
77 values.push_back(s_mime_webp);
79 if (winpr_image_format_is_supported(WINPR_IMAGE_PNG))
80 values.push_back(s_mime_png);
82 if (winpr_image_format_is_supported(WINPR_IMAGE_JPEG))
83 values.push_back(s_mime_jpg);
85 auto bmp = std::vector<const char*>({ s_mime_tiff, BMP_MIME_LIST });
86 values.insert(values.end(), bmp.begin(), bmp.end());
91static const char s_mime_gnome_copied_files[] =
"x-special/gnome-copied-files";
92static const char s_mime_mate_copied_files[] =
"x-special/mate-copied-files";
94static const char s_mime_freerdp_update[] =
"x-special/freerdp-clipboard-update";
96static const char* s_type_HtmlFormat =
"HTML Format";
97static const char* s_type_FileGroupDescriptorW =
"FileGroupDescriptorW";
99class ClipboardLockGuard
102 explicit ClipboardLockGuard(wClipboard* clipboard) : _clipboard(clipboard)
104 ClipboardLock(_clipboard);
106 ClipboardLockGuard(
const ClipboardLockGuard& other) =
delete;
107 ClipboardLockGuard(ClipboardLockGuard&& other) =
delete;
109 ClipboardLockGuard& operator=(
const ClipboardLockGuard& rhs) =
delete;
110 ClipboardLockGuard& operator=(ClipboardLockGuard&& rhs) =
delete;
112 ~ClipboardLockGuard()
114 ClipboardUnlock(_clipboard);
118 wClipboard* _clipboard;
123 return (lhs.formatId < rhs.formatId);
128 return (lhs.formatId == rhs.formatId);
132 : _sdl(
sdl), _file(cliprdr_file_context_new(this)), _log(WLog_Get(TAG)),
133 _system(ClipboardCreate()), _event(CreateEventA(nullptr, TRUE, FALSE, nullptr)),
134 _uuid(
sdl::utils::generate_uuid_v4())
138 std::stringstream ss;
139 ss << s_mime_freerdp_update <<
"-" << _uuid;
140 _mime_uuid = ss.str();
142 std::ignore = cliprdr_file_context_set_locally_available(_file, TRUE);
147 cliprdr_file_context_free(_file);
148 ClipboardDestroy(_system);
149 std::ignore = CloseHandle(_event);
152bool sdlClip::init(CliprdrClientContext* clip)
157 _ctx->MonitorReady = sdlClip::MonitorReady;
158 _ctx->ServerCapabilities = sdlClip::ReceiveServerCapabilities;
159 _ctx->ServerFormatList = sdlClip::ReceiveServerFormatList;
160 _ctx->ServerFormatListResponse = sdlClip::ReceiveFormatListResponse;
161 _ctx->ServerFormatDataRequest = sdlClip::ReceiveFormatDataRequest;
162 _ctx->ServerFormatDataResponse = sdlClip::ReceiveFormatDataResponse;
164 return cliprdr_file_context_init(_file, _ctx);
167bool sdlClip::uninit(CliprdrClientContext* clip)
170 if (!cliprdr_file_context_uninit(_file, _ctx))
173 clip->custom =
nullptr;
177bool sdlClip::contains(
const char** mime_types, Sint32 count)
179 for (Sint32 x = 0; x < count; x++)
181 const auto mime = mime_types[x];
182 if (mime && (strcmp(_mime_uuid.c_str(), mime) == 0))
188bool sdlClip::handleEvent(
const SDL_ClipboardEvent& ev)
190 if (!_ctx || !_sync || ev.owner)
192 _last_timestamp = ev.timestamp;
193 if (!_current_mimetypes.empty())
197 SDL_SetClipboardData(sdlClip::ClipDataCb, sdlClip::ClipCleanCb,
this, ev.mime_types,
198 WINPR_ASSERTING_INT_CAST(
size_t, ev.num_mime_types));
199 _current_mimetypes.clear();
205 if (ev.timestamp == _last_timestamp)
210 if (contains(ev.mime_types, ev.num_mime_types))
215 clearServerFormats();
217 std::string mime_html = s_mime_html;
219 std::vector<std::string> mime_bitmap = { BMP_MIME_LIST };
220 std::string mime_webp = s_mime_webp;
221 std::string mime_png = s_mime_png;
222 std::string mime_jpeg = s_mime_jpg;
223 std::string mime_tiff = s_mime_tiff;
224 std::vector<std::string> mime_images = { mime_webp, mime_png, mime_jpeg, mime_tiff };
226 std::vector<std::string> clientFormatNames;
227 std::vector<CLIPRDR_FORMAT> clientFormats;
229 size_t nformats = WINPR_ASSERTING_INT_CAST(
size_t, ev.num_mime_types);
230 const char** clipboard_mime_formats = ev.mime_types;
232 WLog_Print(_log, WLOG_TRACE,
"SDL has %" PRIuz
" formats", nformats);
234 bool textPushed =
false;
235 bool imgPushed =
false;
236 bool filePushed =
false;
238 for (
size_t i = 0; i < nformats; i++)
240 std::string local_mime = clipboard_mime_formats[i];
241 WLog_Print(_log, WLOG_TRACE,
" - %s", local_mime.c_str());
243 if (std::find(s_mime_text().begin(), s_mime_text().end(), local_mime) !=
249 clientFormats.push_back({ CF_TEXT,
nullptr });
250 clientFormats.push_back({ CF_OEMTEXT,
nullptr });
251 clientFormats.push_back({ CF_UNICODETEXT,
nullptr });
255 else if (local_mime == mime_html)
257 clientFormatNames.emplace_back(s_type_HtmlFormat);
258 else if ((std::find(mime_bitmap.begin(), mime_bitmap.end(), local_mime) !=
259 mime_bitmap.end()) ||
260 (std::find(mime_images.begin(), mime_images.end(), local_mime) !=
266 clientFormats.push_back({ CF_DIB,
nullptr });
267#if defined(WINPR_UTILS_IMAGE_DIBv5)
268 clientFormats.push_back({ CF_DIBV5,
nullptr });
271 for (
auto& bmp : mime_bitmap)
272 clientFormatNames.push_back(bmp);
274 for (
auto& img : mime_images)
275 clientFormatNames.push_back(img);
277 clientFormatNames.emplace_back(s_type_HtmlFormat);
281 else if (mime_is_file(local_mime))
285 clientFormatNames.emplace_back(s_type_FileGroupDescriptorW);
291 for (
auto& name : clientFormatNames)
293 clientFormats.push_back({ ClipboardRegisterFormat(_system, name.c_str()), name.data() });
296 std::sort(clientFormats.begin(), clientFormats.end(),
297 [](
const auto& a,
const auto& b) { return a < b; });
298 auto u = std::unique(clientFormats.begin(), clientFormats.end());
299 clientFormats.erase(u, clientFormats.end());
302 { CB_FORMAT_LIST, 0, 0 },
303 static_cast<UINT32
>(clientFormats.size()),
304 clientFormats.data(),
307 WLog_Print(_log, WLOG_TRACE,
308 "-------------- client format list [%" PRIu32
"] ------------------",
309 formatList.numFormats);
310 for (UINT32 x = 0; x < formatList.numFormats; x++)
312 auto format = &formatList.formats[x];
313 WLog_Print(_log, WLOG_TRACE,
"client announces %" PRIu32
" [%s][%s]", format->formatId,
314 ClipboardGetFormatIdString(format->formatId), format->formatName);
317 if (cliprdr_file_context_notify_new_client_format_list(_file) != CHANNEL_RC_OK)
321 WINPR_ASSERT(_ctx->ClientFormatList);
322 return _ctx->ClientFormatList(_ctx, &formatList) == CHANNEL_RC_OK;
327 WINPR_UNUSED(monitorReady);
328 WINPR_ASSERT(context);
329 WINPR_ASSERT(monitorReady);
331 auto clipboard =
static_cast<sdlClip*
>(
332 cliprdr_file_context_get_context(
static_cast<CliprdrFileContext*
>(context->custom)));
333 WINPR_ASSERT(clipboard);
335 auto ret = clipboard->SendClientCapabilities();
336 if (ret != CHANNEL_RC_OK)
339 clipboard->_sync =
true;
340 if (!sdl_push_user_event(SDL_EVENT_CLIPBOARD_UPDATE))
341 return ERROR_INTERNAL_ERROR;
343 return CHANNEL_RC_OK;
346UINT sdlClip::SendClientCapabilities()
349 CB_CAPSTYPE_GENERAL, 12, CB_CAPS_VERSION_2,
350 CB_USE_LONG_FORMAT_NAMES | cliprdr_file_context_current_flags(_file)
357 WINPR_ASSERT(_ctx->ClientCapabilities);
358 return _ctx->ClientCapabilities(_ctx, &capabilities);
361void sdlClip::clearServerFormats()
363 _serverFormats.clear();
365 cliprdr_file_context_clear(_file);
368UINT sdlClip::SendFormatListResponse(BOOL status)
371 { CB_FORMAT_LIST_RESPONSE,
static_cast<UINT16
>(status ? CB_RESPONSE_OK : CB_RESPONSE_FAIL),
375 WINPR_ASSERT(_ctx->ClientFormatListResponse);
376 return _ctx->ClientFormatListResponse(_ctx, &formatListResponse);
379UINT sdlClip::SendDataResponse(
const BYTE* data,
size_t size)
383 if (size > UINT32_MAX)
384 return ERROR_INVALID_PARAMETER;
386 response.common.msgFlags = (data) ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
387 response.common.dataLen =
static_cast<UINT32
>(size);
388 response.requestedFormatData = data;
391 WINPR_ASSERT(_ctx->ClientFormatDataResponse);
392 return _ctx->ClientFormatDataResponse(_ctx, &response);
395UINT sdlClip::SendDataRequest(uint32_t formatID,
const std::string& mime)
399 _request_queue.emplace(formatID, mime);
402 WINPR_ASSERT(_ctx->ClientFormatDataRequest);
403 UINT ret = _ctx->ClientFormatDataRequest(_ctx, &request);
404 if (ret != CHANNEL_RC_OK)
406 WLog_Print(_log, WLOG_ERROR,
"error sending ClientFormatDataRequest, cancelling request");
407 _request_queue.pop();
413std::string sdlClip::getServerFormat(uint32_t
id)
415 for (
auto& fmt : _serverFormats)
417 if (fmt.formatId() ==
id)
419 if (fmt.formatName())
420 return fmt.formatName();
428uint32_t sdlClip::serverIdForMime(
const std::string& mime)
430 std::string cmp = mime;
431 if (mime_is_html(mime))
432 cmp = s_type_HtmlFormat;
433 if (mime_is_file(mime))
434 cmp = s_type_FileGroupDescriptorW;
436 for (
auto& format : _serverFormats)
438 if (!format.formatName())
440 if (cmp == format.formatName())
441 return format.formatId();
444 if (mime_is_image(mime))
446 if (mime_is_text(mime))
447 return CF_UNICODETEXT;
452UINT sdlClip::ReceiveServerCapabilities(CliprdrClientContext* context,
455 WINPR_ASSERT(context);
456 WINPR_ASSERT(capabilities);
458 auto capsPtr =
reinterpret_cast<const BYTE*
>(capabilities->capabilitySets);
459 WINPR_ASSERT(capsPtr);
461 auto clipboard =
static_cast<sdlClip*
>(
462 cliprdr_file_context_get_context(
static_cast<CliprdrFileContext*
>(context->custom)));
463 WINPR_ASSERT(clipboard);
465 if (!cliprdr_file_context_remote_set_flags(clipboard->_file, 0))
466 return ERROR_INTERNAL_ERROR;
468 for (UINT32 i = 0; i < capabilities->cCapabilitiesSets; i++)
472 if (caps->capabilitySetType == CB_CAPSTYPE_GENERAL)
476 if (!cliprdr_file_context_remote_set_flags(clipboard->_file, generalCaps->generalFlags))
477 return ERROR_INTERNAL_ERROR;
480 capsPtr += caps->capabilitySetLength;
483 return CHANNEL_RC_OK;
486UINT sdlClip::ReceiveServerFormatList(CliprdrClientContext* context,
494 if (!context || !context->custom)
495 return ERROR_INVALID_PARAMETER;
497 auto filecontext =
static_cast<CliprdrFileContext*
>(context->custom);
498 auto clipboard =
static_cast<sdlClip*
>(cliprdr_file_context_get_context(filecontext));
499 WINPR_ASSERT(clipboard);
501 clipboard->clearServerFormats();
503 for (UINT32 i = 0; i < formatList->numFormats; i++)
507 clipboard->_serverFormats.emplace_back(format->formatId, format->formatName);
509 if (format->formatName)
511 if (strcmp(format->formatName, s_type_HtmlFormat) == 0)
516 else if (strcmp(format->formatName, s_type_FileGroupDescriptorW) == 0)
524 switch (format->formatId)
542 clipboard->_current_mimetypes.clear();
545 ClipboardLockGuard systemlock(clipboard->_system);
546 std::scoped_lock lock(clipboard->_lock);
547 auto res = cliprdr_file_context_notify_new_server_format_list(filecontext);
548 if (res != CHANNEL_RC_OK)
554 clipboard->_current_mimetypes.insert(clipboard->_current_mimetypes.end(),
555 s_mime_text().begin(), s_mime_text().end());
559 clipboard->_current_mimetypes.insert(clipboard->_current_mimetypes.end(),
560 s_mime_bitmap().begin(), s_mime_bitmap().end());
561 clipboard->_current_mimetypes.insert(clipboard->_current_mimetypes.end(),
562 s_mime_image().begin(), s_mime_image().end());
566 clipboard->_current_mimetypes.push_back(s_mime_html);
570 clipboard->_current_mimetypes.push_back(s_mime_uri_list);
571 clipboard->_current_mimetypes.push_back(s_mime_gnome_copied_files);
572 clipboard->_current_mimetypes.push_back(s_mime_mate_copied_files);
574 clipboard->_current_mimetypes.push_back(clipboard->_mime_uuid.c_str());
576 auto s = clipboard->_current_mimetypes.size();
577 SDL_Event ev = { SDL_EVENT_CLIPBOARD_UPDATE };
578 ev.clipboard.owner =
true;
579 ev.clipboard.timestamp = SDL_GetTicksNS();
580 ev.clipboard.num_mime_types = WINPR_ASSERTING_INT_CAST(Sint32, s);
581 ev.clipboard.mime_types = clipboard->_current_mimetypes.data();
583 auto rc = (SDL_PushEvent(&ev) == 1);
584 return clipboard->SendFormatListResponse(rc);
587UINT sdlClip::ReceiveFormatListResponse(WINPR_ATTR_UNUSED CliprdrClientContext* context,
590 WINPR_ASSERT(context);
591 WINPR_ASSERT(formatListResponse);
593 if (formatListResponse->common.msgFlags & CB_RESPONSE_FAIL)
594 WLog_WARN(TAG,
"format list update failed");
595 return CHANNEL_RC_OK;
598std::shared_ptr<BYTE> sdlClip::ReceiveFormatDataRequestHandle(
601 const char* mime =
nullptr;
606 std::shared_ptr<BYTE> data;
608 WINPR_ASSERT(clipboard);
609 WINPR_ASSERT(formatDataRequest);
612 auto localFormatId = formatId = formatDataRequest->requestedFormatId;
613 WLog_Print(clipboard->_log, WLOG_DEBUG,
"Requesting format %s [0x%08" PRIx32
"] [%s]",
614 ClipboardGetFormatIdString(localFormatId), localFormatId,
615 ClipboardGetFormatName(clipboard->_system, localFormatId));
617 ClipboardLockGuard systemlock(clipboard->_system);
618 std::scoped_lock lock(clipboard->_lock);
620 const UINT32 fileFormatId =
621 ClipboardGetFormatId(clipboard->_system, s_type_FileGroupDescriptorW);
622 const UINT32 htmlFormatId = ClipboardGetFormatId(clipboard->_system, s_type_HtmlFormat);
629 localFormatId = ClipboardGetFormatId(clipboard->_system, mime_text_plain);
630 mime = mime_text_utf8;
635 mime = s_mime_bitmap().at(0);
636 localFormatId = ClipboardGetFormatId(clipboard->_system, mime);
644 if (formatId == fileFormatId)
646 localFormatId = ClipboardGetFormatId(clipboard->_system, s_mime_uri_list);
647 mime = s_mime_uri_list;
649 else if (formatId == htmlFormatId)
652 if (!SDL_HasClipboardData(s_mime_html))
654 for (
const auto& cmime : s_mime_image())
656 if (SDL_HasClipboardData(cmime))
658 localFormatId = ClipboardGetFormatId(clipboard->_system, cmime);
666 localFormatId = ClipboardGetFormatId(clipboard->_system, s_mime_html);
676 auto sdldata = std::shared_ptr<void>(SDL_GetClipboardData(mime, &size), SDL_free);
680 if (fileFormatId == formatId)
682 auto bdata =
static_cast<const char*
>(sdldata.get());
683 if (!cliprdr_file_context_update_client_data(clipboard->_file, bdata, size))
687 res = ClipboardSetData(clipboard->_system, localFormatId, sdldata.get(),
688 static_cast<uint32_t
>(size));
695 auto ptr =
static_cast<BYTE*
>(ClipboardGetData(clipboard->_system, formatId, &ptrlen));
696 data = std::shared_ptr<BYTE>(ptr, free);
701 if (fileFormatId == formatId)
703 BYTE* ddata =
nullptr;
705 const UINT32 flags = cliprdr_file_context_remote_get_flags(clipboard->_file);
706 const UINT32 error = cliprdr_serialize_file_list_ex(
710 auto tmp = std::shared_ptr<BYTE>(ddata, free);
722UINT sdlClip::ReceiveFormatDataRequest(CliprdrClientContext* context,
725 WINPR_ASSERT(context);
726 WINPR_ASSERT(formatDataRequest);
728 auto clipboard =
static_cast<sdlClip*
>(
729 cliprdr_file_context_get_context(
static_cast<CliprdrFileContext*
>(context->custom)));
730 WINPR_ASSERT(clipboard);
733 auto rc = ReceiveFormatDataRequestHandle(clipboard, formatDataRequest, len);
734 return clipboard->SendDataResponse(rc.get(), len);
737UINT sdlClip::ReceiveFormatDataResponse(CliprdrClientContext* context,
740 WINPR_ASSERT(context);
741 WINPR_ASSERT(formatDataResponse);
743 const UINT32 size = formatDataResponse->common.dataLen;
744 const BYTE* data = formatDataResponse->requestedFormatData;
746 auto clipboard =
static_cast<sdlClip*
>(
747 cliprdr_file_context_get_context(
static_cast<CliprdrFileContext*
>(context->custom)));
748 WINPR_ASSERT(clipboard);
750 ClipboardLockGuard systemlock(clipboard->_system);
751 std::scoped_lock lock(clipboard->_lock);
752 if (clipboard->_request_queue.empty())
754 WLog_Print(clipboard->_log, WLOG_ERROR,
"no pending format request");
755 return ERROR_INTERNAL_ERROR;
760 UINT32 srcFormatId = 0;
761 auto& request = clipboard->_request_queue.front();
762 bool success = (formatDataResponse->common.msgFlags & CB_RESPONSE_OK) &&
763 !(formatDataResponse->common.msgFlags & CB_RESPONSE_FAIL);
764 request.setSuccess(success);
768 WLog_Print(clipboard->_log, WLOG_WARN,
769 "clipboard data request for format %" PRIu32
" [%s], mime %s failed",
770 request.format(), request.formatstr().c_str(), request.mime().c_str());
774 switch (request.format())
779 srcFormatId = request.format();
784 srcFormatId = request.format();
789 auto name = clipboard->getServerFormat(request.format());
792 if (name == s_type_FileGroupDescriptorW)
795 ClipboardGetFormatId(clipboard->_system, s_type_FileGroupDescriptorW);
797 if (!cliprdr_file_context_update_server_data(
798 clipboard->_file, clipboard->_system, data, size))
799 return ERROR_INTERNAL_ERROR;
801 else if (name == s_type_HtmlFormat)
803 srcFormatId = ClipboardGetFormatId(clipboard->_system, s_type_HtmlFormat);
810 if (!ClipboardSetData(clipboard->_system, srcFormatId, data, size))
812 WLog_Print(clipboard->_log, WLOG_ERROR,
"error when setting clipboard data");
813 return ERROR_INTERNAL_ERROR;
815 WLog_Print(clipboard->_log, WLOG_DEBUG,
"updated clipboard data %s [0x%08" PRIx32
"]",
816 ClipboardGetFormatName(clipboard->_system, srcFormatId), srcFormatId);
819 if (!SetEvent(clipboard->_event))
821 WLog_Print(clipboard->_log, WLOG_ERROR,
"error when setting clipboard event");
822 return ERROR_INTERNAL_ERROR;
825 return CHANNEL_RC_OK;
828const void* sdlClip::ClipDataCb(
void* userdata,
const char* mime_type,
size_t* size)
830 auto clip =
static_cast<sdlClip*
>(userdata);
833 WINPR_ASSERT(mime_type);
838 if (mime_is_text(mime_type))
839 mime_type =
"text/plain";
842 ClipboardLockGuard systemlock(clip->_system);
843 std::scoped_lock lock(clip->_lock);
846 auto cache = clip->_cache_data.find(mime_type);
847 if (cache != clip->_cache_data.end())
849 *size = cache->second.size;
850 return cache->second.ptr.get();
853 auto formatID = clip->serverIdForMime(mime_type);
857 auto mimeFormatID = ClipboardRegisterFormat(clip->_system, mime_type);
858 auto fptr = ClipboardGetData(clip->_system, mimeFormatID, &fsize);
861 auto ptr = std::shared_ptr<void>(fptr, free);
862 clip->_cache_data.insert({ mime_type, { fsize, ptr } });
864 auto fcache = clip->_cache_data.find(mime_type);
865 if (fcache != clip->_cache_data.end())
867 *size = fcache->second.size;
868 return fcache->second.ptr.get();
872 WLog_Print(clip->_log, WLOG_DEBUG,
"requesting format %s [%s 0x%08" PRIx32
"]", mime_type,
873 ClipboardGetFormatName(clip->_system, formatID), formatID);
874 if (clip->SendDataRequest(formatID, mime_type))
878 HANDLE hdl[2] = { freerdp_abort_event(clip->_sdl->context()), clip->_event };
880 DWORD status = WaitForMultipleObjects(ARRAYSIZE(hdl), hdl, FALSE, 10 * 1000);
882 if (status != WAIT_OBJECT_0 + 1)
884 std::scoped_lock lock(clip->_lock);
885 clip->_request_queue.pop();
887 if (status == WAIT_TIMEOUT)
888 WLog_Print(clip->_log, WLOG_ERROR,
889 "no reply in 10 seconds, returning empty content");
896 ClipboardLockGuard systemlock(clip->_system);
897 std::scoped_lock lock(clip->_lock);
898 auto request = clip->_request_queue.front();
899 clip->_request_queue.pop();
901 if (clip->_request_queue.empty())
902 std::ignore = ResetEvent(clip->_event);
904 if (request.success())
906 auto formatID = ClipboardRegisterFormat(clip->_system, mime_type);
907 auto data = ClipboardGetData(clip->_system, formatID, &len);
910 WLog_Print(clip->_log, WLOG_ERROR,
"error retrieving clipboard data");
914 auto ptr = std::shared_ptr<void>(data, free);
915 clip->_cache_data.insert({ mime_type, { len, ptr } });
924void sdlClip::ClipCleanCb(
void* userdata)
926 auto clip =
static_cast<sdlClip*
>(userdata);
928 ClipboardLockGuard give_me_a_name(clip->_system);
929 std::scoped_lock lock(clip->_lock);
930 ClipboardEmpty(clip->_system);
933bool sdlClip::mime_is_file(
const std::string& mime)
935 if (strncmp(s_mime_uri_list, mime.c_str(),
sizeof(s_mime_uri_list)) == 0)
937 if (strncmp(s_mime_gnome_copied_files, mime.c_str(),
sizeof(s_mime_gnome_copied_files)) == 0)
939 if (strncmp(s_mime_mate_copied_files, mime.c_str(),
sizeof(s_mime_mate_copied_files)) == 0)
944bool sdlClip::mime_is_text(
const std::string& mime)
946 for (
const auto& tmime : s_mime_text())
948 assert(tmime !=
nullptr);
956bool sdlClip::mime_is_image(
const std::string& mime)
958 for (
const auto& imime : s_mime_image())
960 assert(imime !=
nullptr);
968bool sdlClip::mime_is_bmp(
const std::string& mime)
970 for (
const auto& imime : s_mime_bitmap())
972 assert(imime !=
nullptr);
980bool sdlClip::mime_is_html(
const std::string& mime)
982 return mime.compare(s_mime_html) == 0;
985ClipRequest::ClipRequest(UINT32 format,
const std::string& mime)
986 : _format(format), _mime(mime), _success(false)
990uint32_t ClipRequest::format()
const
995std::string ClipRequest::formatstr()
const
997 return ClipboardGetFormatIdString(_format);
1000std::string ClipRequest::mime()
const
1005bool ClipRequest::success()
const
1010void ClipRequest::setSuccess(
bool status)
1015CliprdrFormat::CliprdrFormat(uint32_t formatID,
const char* formatName) : _formatID(formatID)
1018 _formatName = formatName;
1021uint32_t CliprdrFormat::formatId()
const
1026const char* CliprdrFormat::formatName()
const
1028 if (_formatName.empty())
1030 return _formatName.c_str();
object that handles clipboard context for the SDL3 client