FreeRDP
Loading...
Searching...
No Matches
Clipboard.m
1
21#import "Clipboard.h"
22#import "MRDPView.h"
23
24int mac_cliprdr_send_client_format_list(CliprdrClientContext *cliprdr)
25{
26 UINT32 formatId;
27 UINT32 numFormats;
28 UINT32 *pFormatIds;
29 const char *formatName;
30 CLIPRDR_FORMAT *formats;
31 CLIPRDR_FORMAT_LIST formatList = { 0 };
32
33 WINPR_ASSERT(cliprdr);
34 mfContext *mfc = (mfContext *)cliprdr->custom;
35 WINPR_ASSERT(mfc);
36
37 pFormatIds = NULL;
38 numFormats = ClipboardGetFormatIds(mfc->clipboard, &pFormatIds);
39
40 formats = (CLIPRDR_FORMAT *)calloc(numFormats, sizeof(CLIPRDR_FORMAT));
41
42 if (!formats)
43 return -1;
44
45 for (UINT32 index = 0; index < numFormats; index++)
46 {
47 formatId = pFormatIds[index];
48 formatName = ClipboardGetFormatName(mfc->clipboard, formatId);
49
50 formats[index].formatId = formatId;
51 formats[index].formatName = NULL;
52
53 if ((formatId > CF_MAX) && formatName)
54 formats[index].formatName = _strdup(formatName);
55 }
56
57 formatList.common.msgFlags = 0;
58 formatList.numFormats = numFormats;
59 formatList.formats = formats;
60 formatList.common.msgType = CB_FORMAT_LIST;
61
62 mfc->cliprdr->ClientFormatList(mfc->cliprdr, &formatList);
63
64 for (UINT32 index = 0; index < numFormats; index++)
65 {
66 free(formats[index].formatName);
67 }
68
69 free(pFormatIds);
70 free(formats);
71
72 return 1;
73}
74
75static int mac_cliprdr_send_client_format_list_response(CliprdrClientContext *cliprdr, BOOL status)
76{
77 CLIPRDR_FORMAT_LIST_RESPONSE formatListResponse;
78
79 formatListResponse.common.msgType = CB_FORMAT_LIST_RESPONSE;
80 formatListResponse.common.msgFlags = status ? CB_RESPONSE_OK : CB_RESPONSE_FAIL;
81 formatListResponse.common.dataLen = 0;
82
83 cliprdr->ClientFormatListResponse(cliprdr, &formatListResponse);
84
85 return 1;
86}
87
88static UINT mac_cliprdr_send_client_format_data_request(CliprdrClientContext *cliprdr,
89 UINT32 formatId)
90{
91 CLIPRDR_FORMAT_DATA_REQUEST formatDataRequest = { 0 };
92 WINPR_ASSERT(cliprdr);
93
94 if (formatId == 0)
95 return CHANNEL_RC_OK;
96
97 mfContext *mfc = (mfContext *)cliprdr->custom;
98 WINPR_ASSERT(mfc);
99
100 formatDataRequest.common.msgType = CB_FORMAT_DATA_REQUEST;
101 formatDataRequest.common.msgFlags = 0;
102
103 formatDataRequest.requestedFormatId = formatId;
104 mfc->requestedFormatId = formatId;
105 (void)ResetEvent(mfc->clipboardRequestEvent);
106
107 return cliprdr->ClientFormatDataRequest(cliprdr, &formatDataRequest);
108}
109
110static int mac_cliprdr_send_client_capabilities(CliprdrClientContext *cliprdr)
111{
112 CLIPRDR_CAPABILITIES capabilities;
113 CLIPRDR_GENERAL_CAPABILITY_SET generalCapabilitySet;
114
115 capabilities.cCapabilitiesSets = 1;
116 capabilities.capabilitySets = (CLIPRDR_CAPABILITY_SET *)&(generalCapabilitySet);
117
118 generalCapabilitySet.capabilitySetType = CB_CAPSTYPE_GENERAL;
119 generalCapabilitySet.capabilitySetLength = 12;
120
121 generalCapabilitySet.version = CB_CAPS_VERSION_2;
122 generalCapabilitySet.generalFlags = CB_USE_LONG_FORMAT_NAMES;
123
124 cliprdr->ClientCapabilities(cliprdr, &capabilities);
125
126 return 1;
127}
128
134static UINT mac_cliprdr_monitor_ready(CliprdrClientContext *cliprdr,
135 const CLIPRDR_MONITOR_READY *monitorReady)
136{
137 mfContext *mfc = (mfContext *)cliprdr->custom;
138
139 mfc->clipboardSync = TRUE;
140 mac_cliprdr_send_client_capabilities(cliprdr);
141 mac_cliprdr_send_client_format_list(cliprdr);
142
143 return CHANNEL_RC_OK;
144}
145
151static UINT mac_cliprdr_server_capabilities(CliprdrClientContext *cliprdr,
152 const CLIPRDR_CAPABILITIES *capabilities)
153{
154 CLIPRDR_CAPABILITY_SET *capabilitySet;
155 mfContext *mfc = (mfContext *)cliprdr->custom;
156
157 for (UINT32 index = 0; index < capabilities->cCapabilitiesSets; index++)
158 {
159 capabilitySet = &(capabilities->capabilitySets[index]);
160
161 if ((capabilitySet->capabilitySetType == CB_CAPSTYPE_GENERAL) &&
162 (capabilitySet->capabilitySetLength >= CB_CAPSTYPE_GENERAL_LEN))
163 {
164 CLIPRDR_GENERAL_CAPABILITY_SET *generalCapabilitySet =
165 (CLIPRDR_GENERAL_CAPABILITY_SET *)capabilitySet;
166
167 mfc->clipboardCapabilities = generalCapabilitySet->generalFlags;
168 break;
169 }
170 }
171
172 return CHANNEL_RC_OK;
173}
174
180static UINT mac_cliprdr_server_format_list(CliprdrClientContext *cliprdr,
181 const CLIPRDR_FORMAT_LIST *formatList)
182{
183 WINPR_ASSERT(cliprdr);
184
185 mfContext *mfc = (mfContext *)cliprdr->custom;
186 WINPR_ASSERT(mfc);
187
188 if (mfc->serverFormats)
189 {
190 for (UINT32 index = 0; index < mfc->numServerFormats; index++)
191 {
192 free(mfc->serverFormats[index].formatName);
193 }
194
195 free(mfc->serverFormats);
196 mfc->serverFormats = NULL;
197 mfc->numServerFormats = 0;
198 }
199
200 if (formatList->numFormats < 1)
201 return CHANNEL_RC_OK;
202
203 mfc->numServerFormats = formatList->numFormats;
204 mfc->serverFormats = (CLIPRDR_FORMAT *)calloc(mfc->numServerFormats, sizeof(CLIPRDR_FORMAT));
205
206 if (!mfc->serverFormats)
207 return CHANNEL_RC_NO_MEMORY;
208
209 for (UINT32 index = 0; index < mfc->numServerFormats; index++)
210 {
211 mfc->serverFormats[index].formatId = formatList->formats[index].formatId;
212 mfc->serverFormats[index].formatName = NULL;
213
214 if (formatList->formats[index].formatName)
215 mfc->serverFormats[index].formatName = _strdup(formatList->formats[index].formatName);
216 }
217
218 mac_cliprdr_send_client_format_list_response(cliprdr, TRUE);
219
220 uint32_t formatId = 0;
221 for (UINT32 index = 0; index < mfc->numServerFormats; index++)
222 {
223 const CLIPRDR_FORMAT *format = &(mfc->serverFormats[index]);
224
225 if (format->formatId == CF_UNICODETEXT)
226 formatId = format->formatId;
227 else if (format->formatId == CF_OEMTEXT)
228 {
229 if (formatId == 0)
230 formatId == CF_OEMTEXT;
231 }
232 else if (format->formatId == CF_TEXT)
233 {
234 if (formatId == 0)
235 formatId == CF_TEXT;
236 }
237 }
238
239 return mac_cliprdr_send_client_format_data_request(cliprdr, formatId);
240}
241
247static UINT
248mac_cliprdr_server_format_list_response(CliprdrClientContext *cliprdr,
249 const CLIPRDR_FORMAT_LIST_RESPONSE *formatListResponse)
250{
251 return CHANNEL_RC_OK;
252}
253
259static UINT
260mac_cliprdr_server_lock_clipboard_data(CliprdrClientContext *cliprdr,
261 const CLIPRDR_LOCK_CLIPBOARD_DATA *lockClipboardData)
262{
263 return CHANNEL_RC_OK;
264}
265
271static UINT
272mac_cliprdr_server_unlock_clipboard_data(CliprdrClientContext *cliprdr,
273 const CLIPRDR_UNLOCK_CLIPBOARD_DATA *unlockClipboardData)
274{
275 return CHANNEL_RC_OK;
276}
277
283static UINT
284mac_cliprdr_server_format_data_request(CliprdrClientContext *cliprdr,
285 const CLIPRDR_FORMAT_DATA_REQUEST *formatDataRequest)
286{
287 BYTE *data;
288 UINT32 size;
289 UINT32 formatId;
290 CLIPRDR_FORMAT_DATA_RESPONSE response = { 0 };
291
292 WINPR_ASSERT(cliprdr);
293
294 mfContext *mfc = (mfContext *)cliprdr->custom;
295 WINPR_ASSERT(mfc);
296
297 formatId = formatDataRequest->requestedFormatId;
298 data = (BYTE *)ClipboardGetData(mfc->clipboard, formatId, &size);
299
300 response.common.msgFlags = CB_RESPONSE_OK;
301 response.common.dataLen = size;
302 response.requestedFormatData = data;
303
304 if (!data)
305 {
306 response.common.msgFlags = CB_RESPONSE_FAIL;
307 response.common.dataLen = 0;
308 response.requestedFormatData = NULL;
309 }
310
311 cliprdr->ClientFormatDataResponse(cliprdr, &response);
312
313 free(data);
314
315 return CHANNEL_RC_OK;
316}
317
323static UINT
324mac_cliprdr_server_format_data_response(CliprdrClientContext *cliprdr,
325 const CLIPRDR_FORMAT_DATA_RESPONSE *formatDataResponse)
326{
327 UINT32 formatId;
328 CLIPRDR_FORMAT *format = NULL;
329 mfContext *mfc = (mfContext *)cliprdr->custom;
330 MRDPView *view = (MRDPView *)mfc->view;
331
332 if (formatDataResponse->common.msgFlags & CB_RESPONSE_FAIL)
333 {
334 (void)SetEvent(mfc->clipboardRequestEvent);
335 return ERROR_INTERNAL_ERROR;
336 }
337
338 for (UINT32 index = 0; index < mfc->numServerFormats; index++)
339 {
340 if (mfc->requestedFormatId == mfc->serverFormats[index].formatId)
341 format = &(mfc->serverFormats[index]);
342 }
343
344 if (!format)
345 {
346 (void)SetEvent(mfc->clipboardRequestEvent);
347 return ERROR_INTERNAL_ERROR;
348 }
349
350 if (format->formatName)
351 formatId = ClipboardRegisterFormat(mfc->clipboard, format->formatName);
352 else
353 formatId = format->formatId;
354
355 const size_t size = formatDataResponse->common.dataLen;
356
357 ClipboardSetData(mfc->clipboard, formatId, formatDataResponse->requestedFormatData, size);
358
359 (void)SetEvent(mfc->clipboardRequestEvent);
360
361 if ((formatId == CF_TEXT) || (formatId == CF_OEMTEXT) || (formatId == CF_UNICODETEXT))
362 {
363 formatId = ClipboardRegisterFormat(mfc->clipboard, "text/plain");
364
365 UINT32 dstSize = 0;
366 char *data = ClipboardGetData(mfc->clipboard, formatId, &dstSize);
367
368 dstSize = strnlen(data, dstSize); /* we need the size without the null terminator */
369
370 NSString *str = [[NSString alloc] initWithBytes:(void *)data
371 length:dstSize
372 encoding:NSUTF8StringEncoding];
373 free(data);
374
375 NSArray *types = [[NSArray alloc] initWithObjects:NSPasteboardTypeString, nil];
376 [view->pasteboard_wr declareTypes:types owner:view];
377 [view->pasteboard_wr setString:str forType:NSPasteboardTypeString];
378 }
379
380 return CHANNEL_RC_OK;
381}
382
388static UINT
389mac_cliprdr_server_file_contents_request(CliprdrClientContext *cliprdr,
390 const CLIPRDR_FILE_CONTENTS_REQUEST *fileContentsRequest)
391{
392 return CHANNEL_RC_OK;
393}
394
400static UINT mac_cliprdr_server_file_contents_response(
401 CliprdrClientContext *cliprdr, const CLIPRDR_FILE_CONTENTS_RESPONSE *fileContentsResponse)
402{
403 return CHANNEL_RC_OK;
404}
405
406void mac_cliprdr_init(mfContext *mfc, CliprdrClientContext *cliprdr)
407{
408 cliprdr->custom = (void *)mfc;
409 mfc->cliprdr = cliprdr;
410
411 mfc->clipboard = ClipboardCreate();
412 mfc->clipboardRequestEvent = CreateEvent(NULL, TRUE, FALSE, NULL);
413
414 cliprdr->MonitorReady = mac_cliprdr_monitor_ready;
415 cliprdr->ServerCapabilities = mac_cliprdr_server_capabilities;
416 cliprdr->ServerFormatList = mac_cliprdr_server_format_list;
417 cliprdr->ServerFormatListResponse = mac_cliprdr_server_format_list_response;
418 cliprdr->ServerLockClipboardData = mac_cliprdr_server_lock_clipboard_data;
419 cliprdr->ServerUnlockClipboardData = mac_cliprdr_server_unlock_clipboard_data;
420 cliprdr->ServerFormatDataRequest = mac_cliprdr_server_format_data_request;
421 cliprdr->ServerFormatDataResponse = mac_cliprdr_server_format_data_response;
422 cliprdr->ServerFileContentsRequest = mac_cliprdr_server_file_contents_request;
423 cliprdr->ServerFileContentsResponse = mac_cliprdr_server_file_contents_response;
424}
425
426void mac_cliprdr_uninit(mfContext *mfc, CliprdrClientContext *cliprdr)
427{
428 cliprdr->custom = NULL;
429 mfc->cliprdr = NULL;
430
431 ClipboardDestroy(mfc->clipboard);
432 (void)CloseHandle(mfc->clipboardRequestEvent);
433}