Commit | Line | Data |
---|---|---|
767079a8 | 1 | /* 16-bit Windows Selection processing for emacs on MS-Windows |
21cfcccf EZ |
2 | Copyright (C) 1996, 1997 Free Software Foundation. |
3 | ||
4 | This file is part of GNU Emacs. | |
5 | ||
6 | GNU Emacs is free software; you can redistribute it and/or modify | |
7 | it under the terms of the GNU General Public License as published by | |
8 | the Free Software Foundation; either version 2, or (at your option) | |
9 | any later version. | |
10 | ||
11 | GNU Emacs is distributed in the hope that it will be useful, | |
12 | but WITHOUT ANY WARRANTY; without even the implied warranty of | |
13 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
14 | GNU General Public License for more details. | |
15 | ||
16 | You should have received a copy of the GNU General Public License | |
17 | along with GNU Emacs; see the file COPYING. If not, write to | |
18 | the Free Software Foundation, Inc., 59 Temple Place - Suite 330, | |
19 | Boston, MA 02111-1307, USA. */ | |
20 | ||
21 | /* These functions work by using WinOldAp interface. WinOldAp | |
22 | (WINOLDAP.MOD) is a Microsoft Windows extension supporting | |
23 | "old" (character-mode) application access to Dynamic Data Exchange, | |
24 | menus, and the Windows clipboard. */ | |
25 | ||
26 | /* Written by Dale P. Smith <dpsm@en.com> */ | |
27 | /* Adapted to DJGPP v1 by Eli Zaretskii <eliz@is.elta.co.il> */ | |
28 | ||
29 | #ifdef MSDOS | |
30 | ||
31 | #include <config.h> | |
32 | #include <string.h> | |
33 | #include <dpmi.h> | |
34 | #include <go32.h> | |
35 | #include <sys/farptr.h> | |
36 | #include "lisp.h" | |
37 | #include "dispextern.h" /* frame.h seems to want this */ | |
38 | #include "frame.h" /* Need this to get the X window of selected_frame */ | |
39 | #include "blockinput.h" | |
583f5d53 EZ |
40 | #include "buffer.h" |
41 | #include "charset.h" | |
42 | #include "coding.h" | |
21cfcccf EZ |
43 | |
44 | /* If ever some function outside this file will need to call any | |
45 | clipboard-related function, the following prototypes and constants | |
46 | should be put on a header file. Right now, nobody else uses them. */ | |
47 | ||
48 | #define CF_TEXT 0x01 | |
49 | #define CF_BITMAP 0x02 | |
50 | #define CF_METAFILE 0x03 | |
51 | #define CF_SYLK 0x04 | |
52 | #define CF_DIF 0x05 | |
53 | #define CF_TIFF 0x06 | |
54 | #define CF_OEMTEXT 0x07 | |
55 | #define CF_DIBBITMAP 0x08 | |
56 | #define CF_WINWRITE 0x80 | |
57 | #define CF_DSPTEXT 0x81 | |
58 | #define CF_DSPBITMAP 0x82 | |
59 | ||
60 | unsigned identify_winoldap_version (void); | |
61 | unsigned open_clipboard (void); | |
62 | unsigned empty_clipboard (void); | |
583f5d53 | 63 | unsigned set_clipboard_data (unsigned, void *, unsigned, int); |
21cfcccf | 64 | unsigned get_clipboard_data_size (unsigned); |
583f5d53 | 65 | unsigned get_clipboard_data (unsigned, void *, unsigned, int); |
21cfcccf EZ |
66 | unsigned close_clipboard (void); |
67 | unsigned clipboard_compact (unsigned); | |
68 | ||
69 | Lisp_Object QCLIPBOARD, QPRIMARY; | |
70 | ||
583f5d53 EZ |
71 | /* Coding system for communicating with other Windows programs via the |
72 | clipboard. */ | |
199c7c44 | 73 | static Lisp_Object Vselection_coding_system; |
583f5d53 | 74 | |
9963e859 EZ |
75 | /* Coding system for the next communicating with other Windows programs. */ |
76 | static Lisp_Object Vnext_selection_coding_system; | |
77 | ||
21cfcccf EZ |
78 | /* The segment address and the size of the buffer in low |
79 | memory used to move data between us and WinOldAp module. */ | |
80 | ||
81 | static struct { | |
82 | unsigned long size; | |
83 | unsigned short rm_segment; | |
84 | } clipboard_xfer_buf_info; | |
85 | \f | |
86 | /* Emulation of `__dpmi_int' and friends for DJGPP v1.x */ | |
87 | ||
88 | #if __DJGPP__ < 2 | |
89 | ||
90 | typedef _go32_dpmi_registers __dpmi_regs; | |
91 | #define __tb _go32_info_block.linear_address_of_transfer_buffer | |
92 | #define _dos_ds _go32_info_block.selector_for_linear_memory | |
93 | ||
94 | static int | |
95 | __dpmi_int (intno, regs) | |
96 | int intno; | |
97 | __dpmi_regs *regs; | |
98 | { | |
99 | regs->x.ss = regs->x.sp = regs->x.flags = 0; | |
100 | return _go32_dpmi_simulate_int (intno, regs); | |
101 | } | |
102 | ||
103 | #endif /* __DJGPP__ < 2 */ | |
104 | \f | |
105 | /* C functions to access the Windows 3.1x clipboard from DOS apps. | |
106 | ||
107 | The information was obtained from the Microsoft Knowledge Base, | |
108 | article Q67675 and can be found at: | |
109 | http://www.microsoft.com/kb/developr/win_dk/q67675.htm */ | |
110 | ||
111 | /* See also Ralf Brown's Interrupt List. | |
112 | ||
113 | I also seem to remember reading about this in Dr. Dobbs Journal a | |
114 | while ago, but if you knew my memory... :-) | |
115 | ||
116 | Dale P. Smith <dpsm@en.com> */ | |
117 | ||
118 | /* Return the WinOldAp support version, or 0x1700 if not supported. */ | |
119 | unsigned | |
120 | identify_winoldap_version () | |
121 | { | |
122 | __dpmi_regs regs; | |
123 | ||
124 | /* Calls Int 2Fh/AX=1700h | |
125 | Return Values AX == 1700H: Clipboard functions not available | |
126 | <> 1700H: AL = Major version number | |
127 | AH = Minor version number */ | |
128 | regs.x.ax = 0x1700; | |
129 | __dpmi_int(0x2f, ®s); | |
130 | return regs.x.ax; | |
131 | } | |
132 | ||
133 | /* Open the clipboard, return non-zero if successfull. */ | |
134 | unsigned | |
135 | open_clipboard () | |
136 | { | |
137 | __dpmi_regs regs; | |
138 | ||
139 | /* Is WINOLDAP supported? */ | |
140 | /* Kludge alert!! If WinOldAp is not supported, we return a 0, | |
141 | which is the same as ``Clipboard already open''. Currently, | |
142 | this is taken as an error by all the functions that use | |
143 | `open_clipboard', but if somebody someday will use that ``open'' | |
144 | clipboard, they will have interesting time debugging it... */ | |
145 | if (identify_winoldap_version () == 0x1700) | |
146 | return 0; | |
147 | ||
148 | /* Calls Int 2Fh/AX=1701h | |
149 | Return Values AX == 0: Clipboard already open | |
150 | <> 0: Clipboard opened */ | |
151 | regs.x.ax = 0x1701; | |
152 | __dpmi_int(0x2f, ®s); | |
153 | return regs.x.ax; | |
154 | } | |
155 | ||
156 | /* Empty clipboard, return non-zero if successfull. */ | |
157 | unsigned | |
158 | empty_clipboard () | |
159 | { | |
160 | __dpmi_regs regs; | |
161 | ||
162 | /* Calls Int 2Fh/AX=1702h | |
163 | Return Values AX == 0: Error occurred | |
164 | <> 0: OK, Clipboard emptied */ | |
165 | regs.x.ax = 0x1702; | |
166 | __dpmi_int(0x2f, ®s); | |
167 | return regs.x.ax; | |
168 | } | |
169 | ||
170 | /* Ensure we have a buffer in low memory with enough memory for data | |
171 | of size WANT_SIZE. Return the linear address of the buffer. */ | |
172 | static unsigned long | |
173 | alloc_xfer_buf (want_size) | |
174 | unsigned want_size; | |
175 | { | |
176 | __dpmi_regs regs; | |
177 | ||
178 | /* If the usual DJGPP transfer buffer is large enough, use that. */ | |
179 | if (want_size <= _go32_info_block.size_of_transfer_buffer) | |
180 | return __tb & 0xfffff; | |
181 | ||
767079a8 EZ |
182 | /* Don't even try to allocate more than 1MB of memory: DOS cannot |
183 | possibly handle that (it will overflow the BX register below). */ | |
184 | if (want_size > 0xfffff) | |
185 | return 0; | |
186 | ||
21cfcccf EZ |
187 | /* Need size rounded up to the nearest paragraph, and in |
188 | paragraph units (1 paragraph = 16 bytes). */ | |
189 | clipboard_xfer_buf_info.size = (want_size + 15) >> 4; | |
190 | ||
191 | /* The NT DPMI host crashes us if we free DOS memory via the | |
192 | DPMI service. Work around by calling DOS allocate/free block. */ | |
193 | regs.h.ah = 0x48; | |
194 | regs.x.bx = clipboard_xfer_buf_info.size; | |
195 | __dpmi_int (0x21, ®s); | |
196 | if (regs.x.flags & 1) | |
197 | { | |
198 | clipboard_xfer_buf_info.size = 0; | |
199 | return 0; | |
200 | } | |
201 | ||
202 | clipboard_xfer_buf_info.rm_segment = regs.x.ax; | |
203 | return (((int)clipboard_xfer_buf_info.rm_segment) << 4) & 0xfffff; | |
204 | } | |
205 | ||
206 | /* Free our clipboard buffer. We always free it after use, because | |
207 | keeping it leaves less free conventional memory for subprocesses. | |
208 | The clipboard buffer tends to be large in size, because for small | |
209 | clipboard data sizes we use the DJGPP transfer buffer. */ | |
210 | static void | |
211 | free_xfer_buf () | |
212 | { | |
213 | /* If the size is 0, we used DJGPP transfer buffer, so don't free. */ | |
214 | if (clipboard_xfer_buf_info.size) | |
215 | { | |
216 | __dpmi_regs regs; | |
217 | ||
218 | /* The NT DPMI host crashes us if we free DOS memory via | |
219 | the DPMI service. Work around by calling DOS free block. */ | |
220 | regs.h.ah = 0x49; | |
221 | regs.x.es = clipboard_xfer_buf_info.rm_segment; | |
222 | __dpmi_int (0x21, ®s); | |
223 | clipboard_xfer_buf_info.size = 0; | |
224 | } | |
225 | } | |
226 | ||
0265f89f | 227 | /* Copy data into the clipboard, return zero if successfull. */ |
21cfcccf | 228 | unsigned |
583f5d53 | 229 | set_clipboard_data (Format, Data, Size, Raw) |
21cfcccf EZ |
230 | unsigned Format; |
231 | void *Data; | |
232 | unsigned Size; | |
583f5d53 | 233 | int Raw; |
21cfcccf EZ |
234 | { |
235 | __dpmi_regs regs; | |
236 | unsigned truelen; | |
237 | unsigned long xbuf_addr, buf_offset; | |
238 | unsigned char *dp = Data, *dstart = dp; | |
239 | ||
538d05bb | 240 | if (Format != CF_OEMTEXT) |
21cfcccf EZ |
241 | return 0; |
242 | ||
243 | /* need to know final size after '\r' chars are inserted (the | |
538d05bb | 244 | standard CF_OEMTEXT clipboard format uses CRLF line endings, |
21cfcccf | 245 | while Emacs uses just LF internally). */ |
0265f89f | 246 | truelen = Size + 1; /* +1 for the terminating null */ |
583f5d53 EZ |
247 | |
248 | if (!Raw) | |
21cfcccf | 249 | { |
583f5d53 EZ |
250 | /* avoid using strchr because it recomputes the length everytime */ |
251 | while ((dp = memchr (dp, '\n', Size - (dp - dstart))) != 0) | |
252 | { | |
253 | truelen++; | |
254 | dp++; | |
255 | } | |
21cfcccf EZ |
256 | } |
257 | ||
258 | if (clipboard_compact (truelen) < truelen) | |
259 | return 0; | |
260 | ||
261 | if ((xbuf_addr = alloc_xfer_buf (truelen)) == 0) | |
262 | return 0; | |
263 | ||
583f5d53 EZ |
264 | /* Move the buffer into the low memory, convert LF into CR-LF if needed. */ |
265 | if (Raw) | |
266 | dosmemput (Data, truelen, __tb); | |
267 | else | |
21cfcccf | 268 | { |
583f5d53 EZ |
269 | dp = Data; |
270 | buf_offset = xbuf_addr; | |
271 | _farsetsel (_dos_ds); | |
272 | while (Size--) | |
273 | { | |
0265f89f EZ |
274 | /* Don't allow them to put binary data into the clipboard, since |
275 | it will cause yanked data to be truncated at the first null. */ | |
276 | if (*dp == '\0') | |
277 | return 2; | |
583f5d53 EZ |
278 | if (*dp == '\n') |
279 | _farnspokeb (buf_offset++, '\r'); | |
280 | _farnspokeb (buf_offset++, *dp++); | |
281 | } | |
21cfcccf EZ |
282 | } |
283 | ||
0265f89f EZ |
284 | /* Terminate with a null, otherwise Windows does strange things when |
285 | the text size is an integral multiple of 32 bytes. */ | |
286 | _farnspokeb (buf_offset, *dp); | |
287 | ||
21cfcccf EZ |
288 | /* Calls Int 2Fh/AX=1703h with: |
289 | DX = WinOldAp-Supported Clipboard format | |
290 | ES:BX = Pointer to data | |
291 | SI:CX = Size of data in bytes | |
292 | Return Values AX == 0: Error occurred | |
293 | <> 0: OK. Data copied into the Clipboard. */ | |
294 | regs.x.ax = 0x1703; | |
295 | regs.x.dx = Format; | |
296 | regs.x.si = truelen >> 16; | |
297 | regs.x.cx = truelen & 0xffff; | |
298 | regs.x.es = xbuf_addr >> 4; | |
299 | regs.x.bx = xbuf_addr & 15; | |
300 | __dpmi_int(0x2f, ®s); | |
301 | ||
302 | free_xfer_buf (); | |
303 | ||
0265f89f EZ |
304 | /* Zero means success, otherwise (1 or 2) it's an error. */ |
305 | return regs.x.ax > 0 ? 0 : 1; | |
21cfcccf EZ |
306 | } |
307 | ||
308 | /* Return the size of the clipboard data of format FORMAT. */ | |
309 | unsigned | |
310 | get_clipboard_data_size (Format) | |
311 | unsigned Format; | |
312 | { | |
313 | __dpmi_regs regs; | |
314 | ||
315 | /* Calls Int 2Fh/AX=1704h with: | |
316 | DX = WinOldAp-Supported Clipboard format | |
317 | Return Values DX:AX == Size of the data in bytes, including any | |
318 | headers. | |
319 | == 0 If data in this format is not in | |
320 | the clipboard. */ | |
321 | regs.x.ax = 0x1704; | |
322 | regs.x.dx = Format; | |
323 | __dpmi_int(0x2f, ®s); | |
324 | return ( (((unsigned)regs.x.dx) << 16) | regs.x.ax); | |
325 | } | |
326 | ||
327 | /* Get clipboard data, return its length. | |
328 | Warning: this doesn't check whether DATA has enough space to hold | |
329 | SIZE bytes. */ | |
330 | unsigned | |
583f5d53 | 331 | get_clipboard_data (Format, Data, Size, Raw) |
21cfcccf EZ |
332 | unsigned Format; |
333 | void *Data; | |
334 | unsigned Size; | |
583f5d53 | 335 | int Raw; |
21cfcccf EZ |
336 | { |
337 | __dpmi_regs regs; | |
21cfcccf EZ |
338 | unsigned long xbuf_addr; |
339 | unsigned char *dp = Data; | |
0265f89f EZ |
340 | /* The last 32-byte aligned block of data. See commentary below. */ |
341 | unsigned char *last_block = dp + ((Size & 0x1f) | |
342 | ? (Size & 0x20) | |
343 | : Size - 0x20); | |
21cfcccf | 344 | |
538d05bb | 345 | if (Format != CF_OEMTEXT) |
21cfcccf EZ |
346 | return 0; |
347 | ||
348 | if (Size == 0) | |
349 | return 0; | |
350 | ||
351 | if ((xbuf_addr = alloc_xfer_buf (Size)) == 0) | |
352 | return 0; | |
353 | ||
354 | /* Calls Int 2Fh/AX=1705h with: | |
355 | DX = WinOldAp-Supported Clipboard format | |
356 | ES:BX = Pointer to data buffer to hold data | |
357 | Return Values AX == 0: Error occurred (or data in this format is not | |
358 | in the clipboard) | |
359 | <> 0: OK */ | |
360 | regs.x.ax = 0x1705; | |
361 | regs.x.dx = Format; | |
362 | regs.x.es = xbuf_addr >> 4; | |
363 | regs.x.bx = xbuf_addr & 15; | |
364 | __dpmi_int(0x2f, ®s); | |
365 | if (regs.x.ax != 0) | |
366 | { | |
583f5d53 EZ |
367 | /* Copy data from low memory, remove CR |
368 | characters before LF if needed. */ | |
21cfcccf EZ |
369 | _farsetsel (_dos_ds); |
370 | while (Size--) | |
371 | { | |
372 | register unsigned char c = _farnspeekb (xbuf_addr++); | |
373 | ||
583f5d53 | 374 | if ((*dp++ = c) == '\r' && !Raw && _farnspeekb (xbuf_addr) == '\n') |
21cfcccf EZ |
375 | { |
376 | dp--; | |
377 | *dp++ = '\n'; | |
378 | xbuf_addr++; | |
0265f89f | 379 | last_block--; /* adjust the beginning of the last 32 bytes */ |
21cfcccf EZ |
380 | } |
381 | /* Windows reportedly rounds up the size of clipboard data | |
382 | (passed in SIZE) to a multiple of 32. We therefore bail | |
0265f89f EZ |
383 | out when we see the first null character in the last 32-byte |
384 | block. */ | |
385 | else if (c == '\0' && dp > last_block) | |
386 | break; | |
21cfcccf EZ |
387 | } |
388 | } | |
389 | ||
390 | free_xfer_buf (); | |
391 | ||
0265f89f | 392 | return (unsigned) (dp - (unsigned char *)Data - 1); |
21cfcccf EZ |
393 | } |
394 | ||
395 | /* Close clipboard, return non-zero if successfull. */ | |
396 | unsigned | |
397 | close_clipboard () | |
398 | { | |
399 | __dpmi_regs regs; | |
400 | ||
401 | /* Calls Int 2Fh/AX=1708h | |
402 | Return Values AX == 0: Error occurred | |
403 | <> 0: OK */ | |
404 | regs.x.ax = 0x1708; | |
405 | __dpmi_int(0x2f, ®s); | |
406 | return regs.x.ax; | |
407 | } | |
408 | ||
409 | /* Compact clipboard data so that at least SIZE bytes is available. */ | |
410 | unsigned | |
411 | clipboard_compact (Size) | |
412 | unsigned Size; | |
413 | { | |
414 | __dpmi_regs regs; | |
415 | ||
416 | /* Calls Int 2Fh/AX=1709H with: | |
417 | SI:CX = Desired memory size in bytes. | |
418 | Return Values DX:AX == Number of bytes of largest block of free memory. | |
419 | == 0 if error or no memory */ | |
420 | regs.x.ax = 0x1709; | |
421 | regs.x.si = Size >> 16; | |
422 | regs.x.cx = Size & 0xffff; | |
423 | __dpmi_int(0x2f, ®s); | |
424 | return ((unsigned)regs.x.dx << 16) | regs.x.ax; | |
425 | } | |
426 | \f | |
427 | static char no_mem_msg[] = | |
428 | "(Not enough DOS memory to put saved text into clipboard.)"; | |
0265f89f EZ |
429 | static char binary_msg[] = |
430 | "(Binary characters in saved text; clipboard data not set.)"; | |
21cfcccf | 431 | |
767079a8 | 432 | DEFUN ("w16-set-clipboard-data", Fw16_set_clipboard_data, Sw16_set_clipboard_data, 1, 2, 0, |
21cfcccf EZ |
433 | "This sets the clipboard data to the given text.") |
434 | (string, frame) | |
435 | Lisp_Object string, frame; | |
436 | { | |
0265f89f | 437 | unsigned ok = 1, put_status = 0; |
21cfcccf | 438 | int nbytes; |
583f5d53 EZ |
439 | unsigned char *src, *dst = NULL; |
440 | int charsets[MAX_CHARSET + 1]; | |
441 | int num; | |
442 | int no_crlf_conversion; | |
21cfcccf EZ |
443 | |
444 | CHECK_STRING (string, 0); | |
445 | ||
446 | if (NILP (frame)) | |
447 | frame = Fselected_frame (); | |
448 | ||
449 | CHECK_LIVE_FRAME (frame, 0); | |
450 | if ( !FRAME_MSDOS_P (XFRAME (frame))) | |
451 | goto done; | |
452 | ||
453 | BLOCK_INPUT; | |
454 | ||
199c7c44 | 455 | nbytes = STRING_BYTES (XSTRING (string)); |
21cfcccf | 456 | src = XSTRING (string)->data; |
583f5d53 EZ |
457 | |
458 | /* Since we are now handling multilingual text, we must consider | |
459 | encoding text for the clipboard. */ | |
583f5d53 | 460 | bzero (charsets, (MAX_CHARSET + 1) * sizeof (int)); |
199c7c44 | 461 | num = ((nbytes <= 1 /* Check the possibility of short cut. */ |
9963e859 EZ |
462 | || !STRING_MULTIBYTE (string) |
463 | || nbytes == XSTRING (string)->size) | |
583f5d53 | 464 | ? 0 |
42912be3 | 465 | : find_charset_in_str (src, nbytes, charsets, Qnil, 0, 1)); |
583f5d53 EZ |
466 | |
467 | if (!num || (num == 1 && charsets[CHARSET_ASCII])) | |
468 | { | |
469 | /* No multibyte character in OBJ. We need not encode it, but we | |
470 | will have to convert it to DOS CR-LF style. */ | |
471 | no_crlf_conversion = 0; | |
472 | } | |
473 | else | |
474 | { | |
475 | /* We must encode contents of STRING according to what | |
476 | clipboard-coding-system specifies. */ | |
477 | int bufsize; | |
478 | struct coding_system coding; | |
479 | unsigned char *htext2; | |
480 | ||
9963e859 EZ |
481 | if (NILP (Vnext_selection_coding_system)) |
482 | Vnext_selection_coding_system = Vselection_coding_system; | |
583f5d53 | 483 | setup_coding_system |
9963e859 EZ |
484 | (Fcheck_coding_system (Vnext_selection_coding_system), &coding); |
485 | Vnext_selection_coding_system = Qnil; | |
583f5d53 EZ |
486 | coding.mode |= CODING_MODE_LAST_BLOCK; |
487 | Vlast_coding_system_used = coding.symbol; | |
488 | bufsize = encoding_buffer_size (&coding, nbytes); | |
489 | dst = (unsigned char *) xmalloc (bufsize); | |
490 | encode_coding (&coding, src, dst, nbytes, bufsize); | |
491 | no_crlf_conversion = 1; | |
492 | } | |
493 | ||
21cfcccf EZ |
494 | if (!open_clipboard ()) |
495 | goto error; | |
496 | ||
583f5d53 | 497 | ok = empty_clipboard () |
0265f89f EZ |
498 | && ((put_status |
499 | = set_clipboard_data (CF_OEMTEXT, src, nbytes, no_crlf_conversion)) | |
500 | == 0); | |
583f5d53 EZ |
501 | |
502 | if (!no_crlf_conversion) | |
503 | Vlast_coding_system_used = Qraw_text; | |
21cfcccf EZ |
504 | close_clipboard (); |
505 | ||
31354c30 | 506 | if (ok) goto unblock; |
21cfcccf EZ |
507 | |
508 | error: | |
509 | ||
510 | ok = 0; | |
511 | ||
31354c30 | 512 | unblock: |
583f5d53 EZ |
513 | if (dst) |
514 | xfree (dst); | |
31354c30 EZ |
515 | UNBLOCK_INPUT; |
516 | ||
21cfcccf EZ |
517 | /* Notify user if the text is too large to fit into DOS memory. |
518 | (This will happen somewhere after 600K bytes (470K in DJGPP v1.x), | |
519 | depending on user system configuration.) If we just silently | |
520 | fail the function, people might wonder why their text sometimes | |
521 | doesn't make it to the clipboard. */ | |
0265f89f | 522 | if (put_status) |
21cfcccf | 523 | { |
0265f89f EZ |
524 | switch (put_status) |
525 | { | |
526 | case 1: | |
527 | message2 (no_mem_msg, sizeof (no_mem_msg) - 1, 0); | |
528 | break; | |
529 | case 2: | |
530 | message2 (binary_msg, sizeof (binary_msg) - 1, 0); | |
531 | break; | |
532 | } | |
ad66baa4 | 533 | sit_for (2, 0, 0, 1, 1); |
21cfcccf EZ |
534 | } |
535 | ||
536 | done: | |
21cfcccf | 537 | |
0265f89f | 538 | return (ok && put_status == 0 ? string : Qnil); |
21cfcccf EZ |
539 | } |
540 | ||
767079a8 | 541 | DEFUN ("w16-get-clipboard-data", Fw16_get_clipboard_data, Sw16_get_clipboard_data, 0, 1, 0, |
21cfcccf EZ |
542 | "This gets the clipboard data in text format.") |
543 | (frame) | |
544 | Lisp_Object frame; | |
545 | { | |
546 | unsigned data_size, truelen; | |
547 | unsigned char *htext; | |
548 | Lisp_Object ret = Qnil; | |
583f5d53 EZ |
549 | int no_crlf_conversion; |
550 | int require_encoding = 0; | |
551 | ||
21cfcccf EZ |
552 | if (NILP (frame)) |
553 | frame = Fselected_frame (); | |
554 | ||
555 | CHECK_LIVE_FRAME (frame, 0); | |
556 | if ( !FRAME_MSDOS_P (XFRAME (frame))) | |
557 | goto done; | |
558 | ||
559 | BLOCK_INPUT; | |
560 | ||
561 | if (!open_clipboard ()) | |
31354c30 | 562 | goto unblock; |
21cfcccf | 563 | |
538d05bb | 564 | if ((data_size = get_clipboard_data_size (CF_OEMTEXT)) == 0 || |
21cfcccf EZ |
565 | (htext = (unsigned char *)xmalloc (data_size)) == 0) |
566 | goto closeclip; | |
567 | ||
568 | /* need to know final size after '\r' chars are removed because | |
569 | we can't change the string size manually, and doing an extra | |
570 | copy is silly */ | |
538d05bb | 571 | if ((truelen = get_clipboard_data (CF_OEMTEXT, htext, data_size, 0)) == 0) |
21cfcccf EZ |
572 | goto closeclip; |
573 | ||
583f5d53 | 574 | /* Do we need to decode it? */ |
9963e859 EZ |
575 | if ( |
576 | #if 1 | |
577 | 1 | |
578 | #else | |
579 | ! NILP (buffer_defaults.enable_multibyte_characters) | |
580 | #endif | |
581 | ) | |
583f5d53 EZ |
582 | { |
583 | /* If the clipboard data contains any 8-bit Latin-1 code, we | |
584 | need to decode it. */ | |
585 | int i; | |
586 | ||
587 | for (i = 0; i < truelen; i++) | |
588 | { | |
589 | if (htext[i] >= 0x80) | |
590 | { | |
591 | require_encoding = 1; | |
592 | break; | |
593 | } | |
594 | } | |
595 | } | |
596 | if (require_encoding) | |
597 | { | |
598 | int bufsize; | |
599 | unsigned char *buf; | |
600 | struct coding_system coding; | |
601 | ||
9963e859 EZ |
602 | if (NILP (Vnext_selection_coding_system)) |
603 | Vnext_selection_coding_system = Vselection_coding_system; | |
583f5d53 | 604 | setup_coding_system |
9963e859 EZ |
605 | (Fcheck_coding_system (Vnext_selection_coding_system), &coding); |
606 | Vnext_selection_coding_system = Qnil; | |
583f5d53 | 607 | coding.mode |= CODING_MODE_LAST_BLOCK; |
538d05bb | 608 | truelen = get_clipboard_data (CF_OEMTEXT, htext, data_size, 1); |
583f5d53 EZ |
609 | bufsize = decoding_buffer_size (&coding, truelen); |
610 | buf = (unsigned char *) xmalloc (bufsize); | |
611 | decode_coding (&coding, htext, buf, truelen, bufsize); | |
612 | truelen = (coding.fake_multibyte | |
613 | ? multibyte_chars_in_text (buf, coding.produced) | |
614 | : coding.produced_char); | |
615 | ret = make_string_from_bytes ((char *) buf, truelen, coding.produced); | |
616 | xfree (buf); | |
617 | Vlast_coding_system_used = coding.symbol; | |
618 | } | |
619 | else | |
620 | { | |
621 | ret = make_unibyte_string ((char *) htext, truelen); | |
622 | Vlast_coding_system_used = Qraw_text; | |
623 | } | |
624 | ||
21cfcccf EZ |
625 | xfree (htext); |
626 | ||
627 | closeclip: | |
628 | close_clipboard (); | |
31354c30 EZ |
629 | |
630 | unblock: | |
631 | UNBLOCK_INPUT; | |
21cfcccf EZ |
632 | |
633 | done: | |
21cfcccf EZ |
634 | |
635 | return (ret); | |
636 | } | |
637 | ||
638 | /* Support checking for a clipboard selection. */ | |
639 | ||
640 | DEFUN ("x-selection-exists-p", Fx_selection_exists_p, Sx_selection_exists_p, | |
641 | 0, 1, 0, | |
642 | "Whether there is an owner for the given X Selection.\n\ | |
643 | The arg should be the name of the selection in question, typically one of\n\ | |
644 | the symbols `PRIMARY', `SECONDARY', or `CLIPBOARD'.\n\ | |
645 | \(Those are literal upper-case symbol names, since that's what X expects.)\n\ | |
646 | For convenience, the symbol nil is the same as `PRIMARY',\n\ | |
647 | and t is the same as `SECONDARY'.") | |
648 | (selection) | |
649 | Lisp_Object selection; | |
650 | { | |
651 | CHECK_SYMBOL (selection, 0); | |
652 | ||
653 | /* Return nil for SECONDARY selection. For PRIMARY (or nil) | |
654 | selection, check if there is some text on the kill-ring; | |
655 | for CLIPBOARD, check if the clipboard currently has valid | |
656 | text format contents. | |
657 | ||
658 | The test for killed text on the kill-ring emulates the Emacs | |
659 | behavior on X, where killed text is also put into X selection | |
660 | by the X interface code. (On MSDOS, killed text is only put | |
661 | into the clipboard if we run under Windows, so we cannot check | |
662 | the clipboard alone.) */ | |
663 | if ((EQ (selection, Qnil) || EQ (selection, QPRIMARY)) | |
664 | && ! NILP (XSYMBOL (Fintern_soft (build_string ("kill-ring"), | |
665 | Qnil))->value)) | |
666 | return Qt; | |
667 | ||
668 | if (EQ (selection, QCLIPBOARD)) | |
669 | { | |
670 | Lisp_Object val = Qnil; | |
671 | ||
672 | if (open_clipboard ()) | |
673 | { | |
538d05bb | 674 | if (get_clipboard_data_size (CF_OEMTEXT)) |
21cfcccf EZ |
675 | val = Qt; |
676 | close_clipboard (); | |
677 | } | |
678 | return val; | |
679 | } | |
680 | return Qnil; | |
681 | } | |
682 | ||
683 | void | |
684 | syms_of_win16select () | |
685 | { | |
767079a8 EZ |
686 | defsubr (&Sw16_set_clipboard_data); |
687 | defsubr (&Sw16_get_clipboard_data); | |
21cfcccf EZ |
688 | defsubr (&Sx_selection_exists_p); |
689 | ||
199c7c44 | 690 | DEFVAR_LISP ("selection-coding-system", &Vselection_coding_system, |
583f5d53 EZ |
691 | "Coding system for communicating with other X clients.\n\ |
692 | When sending or receiving text via cut_buffer, selection, and clipboard,\n\ | |
693 | the text is encoded or decoded by this coding system.\n\ | |
694 | A default value is `iso-latin-1-dos'"); | |
199c7c44 | 695 | Vselection_coding_system=intern ("iso-latin-1-dos"); |
9963e859 EZ |
696 | |
697 | DEFVAR_LISP ("next-selection-coding-system", &Vnext_selection_coding_system, | |
698 | "Coding system for the next communication with other X clients.\n\ | |
699 | Usually, `selection-coding-system' is used for communicating with\n\ | |
700 | other X clients. But, if this variable is set, it is used for the\n\ | |
701 | next communication only. After the communication, this variable is\n\ | |
702 | set to nil."); | |
703 | Vnext_selection_coding_system = Qnil; | |
583f5d53 | 704 | |
21cfcccf EZ |
705 | QPRIMARY = intern ("PRIMARY"); staticpro (&QPRIMARY); |
706 | QCLIPBOARD = intern ("CLIPBOARD"); staticpro (&QCLIPBOARD); | |
707 | } | |
708 | ||
709 | #endif /* MSDOS */ |