mirror of
git://git.sv.gnu.org/emacs.git
synced 2025-12-25 15:00:45 -08:00
494 lines
14 KiB
C
494 lines
14 KiB
C
/* Selection processing for Emacs on the Microsoft W32 API.
|
|
Copyright (C) 1993, 1994 Free Software Foundation.
|
|
|
|
This file is part of GNU Emacs.
|
|
|
|
GNU Emacs is free software; you can redistribute it and/or modify
|
|
it under the terms of the GNU General Public License as published by
|
|
the Free Software Foundation; either version 2, or (at your option)
|
|
any later version.
|
|
|
|
GNU Emacs is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
along with GNU Emacs; see the file COPYING. If not, write to
|
|
the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
Boston, MA 02111-1307, USA. */
|
|
|
|
/* Written by Kevin Gallo */
|
|
|
|
#include <config.h>
|
|
#include "lisp.h"
|
|
#include "w32term.h" /* for all of the w32 includes */
|
|
#include "dispextern.h" /* frame.h seems to want this */
|
|
#include "keyboard.h"
|
|
#include "frame.h" /* Need this to get the X window of selected_frame */
|
|
#include "blockinput.h"
|
|
#include "buffer.h"
|
|
#include "charset.h"
|
|
#include "coding.h"
|
|
#include "composite.h"
|
|
|
|
Lisp_Object QCLIPBOARD;
|
|
|
|
/* Coding system for communicating with other Windows programs via the
|
|
clipboard. */
|
|
static Lisp_Object Vselection_coding_system;
|
|
|
|
/* Coding system for the next communicating with other Windows programs. */
|
|
static Lisp_Object Vnext_selection_coding_system;
|
|
|
|
/* Sequence number, used where possible to detect when we are pasting
|
|
our own text. */
|
|
static DWORD last_clipboard_sequence_number;
|
|
extern ClipboardSequence_Proc clipboard_sequence_fn;
|
|
|
|
/* The last text we put into the clipboard. This is used when the OS
|
|
does not support sequence numbers (NT4, 95). It is undesirable to
|
|
use data put on the clipboard by Emacs because the clipboard data
|
|
could be MULEtilated by inappropriately chosen
|
|
(next-)selection-coding-system. For this reason, we must store the
|
|
text *after* it was encoded/Unix-to-DOS-converted. */
|
|
static unsigned char *last_clipboard_text = NULL;
|
|
static size_t clipboard_storage_size = 0;
|
|
|
|
#if 0
|
|
DEFUN ("w32-open-clipboard", Fw32_open_clipboard, Sw32_open_clipboard, 0, 1, 0,
|
|
doc: /* This opens the clipboard with the given frame pointer. */)
|
|
(frame)
|
|
Lisp_Object frame;
|
|
{
|
|
BOOL ok = FALSE;
|
|
|
|
if (!NILP (frame))
|
|
CHECK_LIVE_FRAME (frame);
|
|
|
|
BLOCK_INPUT;
|
|
|
|
ok = OpenClipboard ((!NILP (frame) && FRAME_W32_P (XFRAME (frame))) ? FRAME_W32_WINDOW (XFRAME (frame)) : NULL);
|
|
|
|
UNBLOCK_INPUT;
|
|
|
|
return (ok ? frame : Qnil);
|
|
}
|
|
|
|
DEFUN ("w32-empty-clipboard", Fw32_empty_clipboard,
|
|
Sw32_empty_clipboard, 0, 0, 0,
|
|
doc: /* Empty the clipboard.
|
|
Assigns ownership of the clipboard to the window which opened it. */)
|
|
()
|
|
{
|
|
BOOL ok = FALSE;
|
|
|
|
BLOCK_INPUT;
|
|
|
|
ok = EmptyClipboard ();
|
|
|
|
UNBLOCK_INPUT;
|
|
|
|
return (ok ? Qt : Qnil);
|
|
}
|
|
|
|
DEFUN ("w32-close-clipboard", Fw32_close_clipboard,
|
|
Sw32_close_clipboard, 0, 0, 0,
|
|
doc: /* Close the clipboard. */)
|
|
()
|
|
{
|
|
BOOL ok = FALSE;
|
|
|
|
BLOCK_INPUT;
|
|
|
|
ok = CloseClipboard ();
|
|
|
|
UNBLOCK_INPUT;
|
|
|
|
return (ok ? Qt : Qnil);
|
|
}
|
|
|
|
#endif
|
|
|
|
DEFUN ("w32-set-clipboard-data", Fw32_set_clipboard_data,
|
|
Sw32_set_clipboard_data, 1, 2, 0,
|
|
doc: /* This sets the clipboard data to the given text. */)
|
|
(string, frame)
|
|
Lisp_Object string, frame;
|
|
{
|
|
BOOL ok = TRUE;
|
|
HANDLE htext;
|
|
int nbytes;
|
|
int truelen, nlines = 0;
|
|
unsigned char *src;
|
|
unsigned char *dst;
|
|
|
|
CHECK_STRING (string);
|
|
|
|
if (!NILP (frame))
|
|
CHECK_LIVE_FRAME (frame);
|
|
|
|
BLOCK_INPUT;
|
|
|
|
nbytes = SBYTES (string) + 1;
|
|
src = SDATA (string);
|
|
dst = src;
|
|
|
|
/* We need to know how many lines there are, since we need CRLF line
|
|
termination for compatibility with other Windows Programs.
|
|
avoid using strchr because it recomputes the length every time */
|
|
while ((dst = memchr (dst, '\n', nbytes - (dst - src))) != NULL)
|
|
{
|
|
nlines++;
|
|
dst++;
|
|
}
|
|
|
|
{
|
|
/* Since we are now handling multilingual text, we must consider
|
|
encoding text for the clipboard. */
|
|
int charset_info = find_charset_in_text (src, SCHARS (string),
|
|
nbytes, NULL, Qnil);
|
|
|
|
if (charset_info == 0)
|
|
{
|
|
/* No multibyte character in OBJ. We need not encode it. */
|
|
|
|
/* Need to know final size after CR chars are inserted (the
|
|
standard CF_TEXT clipboard format uses CRLF line endings,
|
|
while Emacs uses just LF internally). */
|
|
|
|
truelen = nbytes + nlines;
|
|
|
|
if ((htext = GlobalAlloc (GMEM_MOVEABLE | GMEM_DDESHARE, truelen)) == NULL)
|
|
goto error;
|
|
|
|
if ((dst = (unsigned char *) GlobalLock (htext)) == NULL)
|
|
goto error;
|
|
|
|
/* convert to CRLF line endings expected by clipboard */
|
|
while (1)
|
|
{
|
|
unsigned char *next;
|
|
/* copy next line or remaining bytes including '\0' */
|
|
next = _memccpy (dst, src, '\n', nbytes);
|
|
if (next)
|
|
{
|
|
/* copied one line ending with '\n' */
|
|
int copied = next - dst;
|
|
nbytes -= copied;
|
|
src += copied;
|
|
/* insert '\r' before '\n' */
|
|
next[-1] = '\r';
|
|
next[0] = '\n';
|
|
dst = next + 1;
|
|
}
|
|
else
|
|
/* copied remaining partial line -> now finished */
|
|
break;
|
|
}
|
|
|
|
GlobalUnlock (htext);
|
|
|
|
Vlast_coding_system_used = Qraw_text;
|
|
}
|
|
else
|
|
{
|
|
/* We must encode contents of OBJ to the selection coding
|
|
system. */
|
|
int bufsize;
|
|
struct coding_system coding;
|
|
HANDLE htext2;
|
|
|
|
if (NILP (Vnext_selection_coding_system))
|
|
Vnext_selection_coding_system = Vselection_coding_system;
|
|
setup_coding_system
|
|
(Fcheck_coding_system (Vnext_selection_coding_system), &coding);
|
|
if (SYMBOLP (coding.pre_write_conversion)
|
|
&& !NILP (Ffboundp (coding.pre_write_conversion)))
|
|
{
|
|
string = run_pre_post_conversion_on_str (string, &coding, 1);
|
|
src = SDATA (string);
|
|
nbytes = SBYTES (string);
|
|
}
|
|
coding.src_multibyte = 1;
|
|
coding.dst_multibyte = 0;
|
|
Vnext_selection_coding_system = Qnil;
|
|
coding.mode |= CODING_MODE_LAST_BLOCK;
|
|
bufsize = encoding_buffer_size (&coding, nbytes);
|
|
if ((htext = GlobalAlloc (GMEM_MOVEABLE | GMEM_DDESHARE, bufsize)) == NULL)
|
|
goto error;
|
|
if ((dst = (unsigned char *) GlobalLock (htext)) == NULL)
|
|
goto error;
|
|
encode_coding (&coding, src, dst, nbytes, bufsize);
|
|
Vlast_coding_system_used = coding.symbol;
|
|
|
|
/* If clipboard sequence numbers are not supported, keep a copy for
|
|
later comparison. */
|
|
if (!clipboard_sequence_fn)
|
|
{
|
|
/* Stash away the data we are about to put into the
|
|
clipboard, so we could later check inside
|
|
Fw32_get_clipboard_data whether the clipboard still
|
|
holds our data. */
|
|
if (clipboard_storage_size < coding.produced)
|
|
{
|
|
clipboard_storage_size = coding.produced + 100;
|
|
last_clipboard_text = (char *) xrealloc (last_clipboard_text,
|
|
clipboard_storage_size);
|
|
}
|
|
if (last_clipboard_text)
|
|
memcpy (last_clipboard_text, dst, coding.produced);
|
|
}
|
|
|
|
GlobalUnlock (htext);
|
|
|
|
/* Shrink data block to actual size. */
|
|
htext2 = GlobalReAlloc (htext, coding.produced,
|
|
GMEM_MOVEABLE | GMEM_DDESHARE);
|
|
if (htext2 != NULL) htext = htext2;
|
|
}
|
|
}
|
|
|
|
if (!OpenClipboard ((!NILP (frame) && FRAME_W32_P (XFRAME (frame))) ? FRAME_W32_WINDOW (XFRAME (frame)) : NULL))
|
|
goto error;
|
|
|
|
ok = EmptyClipboard () && SetClipboardData (CF_TEXT, htext);
|
|
|
|
if (clipboard_sequence_fn)
|
|
last_clipboard_sequence_number = clipboard_sequence_fn ();
|
|
|
|
CloseClipboard ();
|
|
|
|
if (ok) goto done;
|
|
|
|
error:
|
|
|
|
ok = FALSE;
|
|
if (htext) GlobalFree (htext);
|
|
if (last_clipboard_text)
|
|
*last_clipboard_text = '\0';
|
|
|
|
last_clipboard_sequence_number = 0;
|
|
|
|
done:
|
|
UNBLOCK_INPUT;
|
|
|
|
return (ok ? string : Qnil);
|
|
}
|
|
|
|
DEFUN ("w32-get-clipboard-data", Fw32_get_clipboard_data,
|
|
Sw32_get_clipboard_data, 0, 1, 0,
|
|
doc: /* This gets the clipboard data in text format. */)
|
|
(frame)
|
|
Lisp_Object frame;
|
|
{
|
|
HANDLE htext;
|
|
Lisp_Object ret = Qnil;
|
|
|
|
if (!NILP (frame))
|
|
CHECK_LIVE_FRAME (frame);
|
|
|
|
BLOCK_INPUT;
|
|
|
|
if (!OpenClipboard ((!NILP (frame) && FRAME_W32_P (XFRAME (frame))) ? FRAME_W32_WINDOW (XFRAME (frame)) : NULL))
|
|
goto done;
|
|
|
|
if ((htext = GetClipboardData (CF_TEXT)) == NULL)
|
|
goto closeclip;
|
|
|
|
{
|
|
unsigned char *src;
|
|
unsigned char *dst;
|
|
int nbytes;
|
|
int truelen;
|
|
int require_decoding = 0;
|
|
|
|
if ((src = (unsigned char *) GlobalLock (htext)) == NULL)
|
|
goto closeclip;
|
|
|
|
nbytes = strlen (src);
|
|
|
|
/* If the text in clipboard is identical to what we put there
|
|
last time w32_set_clipboard_data was called, pretend there's no
|
|
data in the clipboard. This is so we don't pass our own text
|
|
from the clipboard (which might be troublesome if the killed
|
|
text includes null characters). */
|
|
if ((clipboard_sequence_fn
|
|
&& clipboard_sequence_fn () == last_clipboard_sequence_number)
|
|
|| (last_clipboard_text
|
|
&& clipboard_storage_size >= nbytes
|
|
&& memcmp(last_clipboard_text, src, nbytes) == 0))
|
|
goto closeclip;
|
|
|
|
{
|
|
/* If the clipboard data contains any non-ascii code, we
|
|
need to decode it. */
|
|
int i;
|
|
|
|
for (i = 0; i < nbytes; i++)
|
|
{
|
|
if (src[i] >= 0x80)
|
|
{
|
|
require_decoding = 1;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if (require_decoding)
|
|
{
|
|
int bufsize;
|
|
unsigned char *buf;
|
|
struct coding_system coding;
|
|
|
|
if (NILP (Vnext_selection_coding_system))
|
|
Vnext_selection_coding_system = Vselection_coding_system;
|
|
setup_coding_system
|
|
(Fcheck_coding_system (Vnext_selection_coding_system), &coding);
|
|
coding.src_multibyte = 0;
|
|
coding.dst_multibyte = 1;
|
|
Vnext_selection_coding_system = Qnil;
|
|
coding.mode |= CODING_MODE_LAST_BLOCK;
|
|
/* We explicitely disable composition handling because
|
|
selection data should not contain any composition
|
|
sequence. */
|
|
coding.composing = COMPOSITION_DISABLED;
|
|
bufsize = decoding_buffer_size (&coding, nbytes);
|
|
buf = (unsigned char *) xmalloc (bufsize);
|
|
decode_coding (&coding, src, buf, nbytes, bufsize);
|
|
Vlast_coding_system_used = coding.symbol;
|
|
ret = make_string_from_bytes ((char *) buf,
|
|
coding.produced_char, coding.produced);
|
|
xfree (buf);
|
|
if (SYMBOLP (coding.post_read_conversion)
|
|
&& !NILP (Ffboundp (coding.post_read_conversion)))
|
|
ret = run_pre_post_conversion_on_str (ret, &coding, 0);
|
|
}
|
|
else
|
|
{
|
|
/* Need to know final size after CR chars are removed because we
|
|
can't change the string size manually, and doing an extra
|
|
copy is silly. Note that we only remove CR when it appears
|
|
as part of CRLF. */
|
|
|
|
truelen = nbytes;
|
|
dst = src;
|
|
/* avoid using strchr because it recomputes the length everytime */
|
|
while ((dst = memchr (dst, '\r', nbytes - (dst - src))) != NULL)
|
|
{
|
|
if (dst[1] == '\n') /* safe because of trailing '\0' */
|
|
truelen--;
|
|
dst++;
|
|
}
|
|
|
|
ret = make_uninit_string (truelen);
|
|
|
|
/* Convert CRLF line endings (the standard CF_TEXT clipboard
|
|
format) to LF endings as used internally by Emacs. */
|
|
|
|
dst = SDATA (ret);
|
|
while (1)
|
|
{
|
|
unsigned char *next;
|
|
/* copy next line or remaining bytes excluding '\0' */
|
|
next = _memccpy (dst, src, '\r', nbytes);
|
|
if (next)
|
|
{
|
|
/* copied one line ending with '\r' */
|
|
int copied = next - dst;
|
|
nbytes -= copied;
|
|
dst += copied;
|
|
src += copied;
|
|
if (*src == '\n')
|
|
dst--; /* overwrite '\r' with '\n' */
|
|
}
|
|
else
|
|
/* copied remaining partial line -> now finished */
|
|
break;
|
|
}
|
|
|
|
Vlast_coding_system_used = Qraw_text;
|
|
}
|
|
|
|
GlobalUnlock (htext);
|
|
}
|
|
|
|
closeclip:
|
|
CloseClipboard ();
|
|
|
|
done:
|
|
UNBLOCK_INPUT;
|
|
|
|
return (ret);
|
|
}
|
|
|
|
/* Support checking for a clipboard selection. */
|
|
|
|
DEFUN ("x-selection-exists-p", Fx_selection_exists_p, Sx_selection_exists_p,
|
|
0, 1, 0,
|
|
doc: /* Whether there is an owner for the given X Selection.
|
|
The arg should be the name of the selection in question, typically one of
|
|
the symbols `PRIMARY', `SECONDARY', or `CLIPBOARD'.
|
|
\(Those are literal upper-case symbol names, since that's what X expects.)
|
|
For convenience, the symbol nil is the same as `PRIMARY',
|
|
and t is the same as `SECONDARY'. */)
|
|
(selection)
|
|
Lisp_Object selection;
|
|
{
|
|
CHECK_SYMBOL (selection);
|
|
|
|
/* Return nil for PRIMARY and SECONDARY selections; for CLIPBOARD, check
|
|
if the clipboard currently has valid text format contents. */
|
|
|
|
if (EQ (selection, QCLIPBOARD))
|
|
{
|
|
Lisp_Object val = Qnil;
|
|
|
|
if (OpenClipboard (NULL))
|
|
{
|
|
int format = 0;
|
|
while (format = EnumClipboardFormats (format))
|
|
if (format == CF_TEXT)
|
|
{
|
|
val = Qt;
|
|
break;
|
|
}
|
|
CloseClipboard ();
|
|
}
|
|
return val;
|
|
}
|
|
return Qnil;
|
|
}
|
|
|
|
void
|
|
syms_of_w32select ()
|
|
{
|
|
#if 0
|
|
defsubr (&Sw32_open_clipboard);
|
|
defsubr (&Sw32_empty_clipboard);
|
|
defsubr (&Sw32_close_clipboard);
|
|
#endif
|
|
defsubr (&Sw32_set_clipboard_data);
|
|
defsubr (&Sw32_get_clipboard_data);
|
|
defsubr (&Sx_selection_exists_p);
|
|
|
|
DEFVAR_LISP ("selection-coding-system", &Vselection_coding_system,
|
|
doc: /* Coding system for communicating with other programs.
|
|
When sending or receiving text via cut_buffer, selection, and clipboard,
|
|
the text is encoded or decoded by this coding system.
|
|
The default value is `iso-latin-1-dos'. */);
|
|
Vselection_coding_system = intern ("iso-latin-1-dos");
|
|
|
|
DEFVAR_LISP ("next-selection-coding-system", &Vnext_selection_coding_system,
|
|
doc: /* Coding system for the next communication with other programs.
|
|
Usually, `selection-coding-system' is used for communicating with
|
|
other programs. But, if this variable is set, it is used for the
|
|
next communication only. After the communication, this variable is
|
|
set to nil. */);
|
|
Vnext_selection_coding_system = Qnil;
|
|
|
|
QCLIPBOARD = intern ("CLIPBOARD"); staticpro (&QCLIPBOARD);
|
|
}
|
|
|
|
/* arch-tag: c96e9724-5eb1-4dad-be07-289f092fd2af
|
|
(do not change this comment) */
|