1
0
mirror of https://git.tartarus.org/simon/putty.git synced 2025-01-10 01:48:00 +00:00

Add GUI configuration for choice of clipboards.

On all platforms, you can now configure which clipboard the mouse
pastes from, which clipboard Ctrl-Ins and Shift-Ins access, and which
Ctrl-Shift-C and Ctrl-Shift-V access. In each case, the options are:

 - nothing at all
 - a clipboard which is implicitly written by the act of mouse
   selection (the PRIMARY selection on X, CLIP_LOCAL everywhere else)
 - the standard clipboard written by explicit copy/paste UI actions
   (CLIPBOARD on X, the unique system clipboard elsewhere).

Also, you can control whether selecting text with the mouse _also_
writes to the explicitly accessed clipboard.

The wording of the various messages changes between platforms, but the
basic UI shape is the same everywhere.
This commit is contained in:
Simon Tatham 2017-12-10 17:16:50 +00:00
parent 98fa733a96
commit 0e7f0883a9
9 changed files with 405 additions and 22 deletions

View File

@ -1337,6 +1337,34 @@ static void manual_hostkey_handler(union control *ctrl, void *dlg,
} }
} }
static void clipboard_selector_handler(union control *ctrl, void *dlg,
void *data, int event)
{
Conf *conf = (Conf *)data;
int setting = ctrl->generic.context.i;
if (event == EVENT_REFRESH) {
int i, val = conf_get_int(conf, setting);
dlg_update_start(ctrl, dlg);
dlg_listbox_clear(ctrl, dlg);
dlg_listbox_addwithid(ctrl, dlg, "No action", CLIPUI_NONE);
dlg_listbox_addwithid(ctrl, dlg, CLIPNAME_IMPLICIT, CLIPUI_IMPLICIT);
dlg_listbox_addwithid(ctrl, dlg, CLIPNAME_EXPLICIT, CLIPUI_EXPLICIT);
dlg_listbox_select(ctrl, dlg, 0); /* fallback */
for (i = 0; i < 3; i++)
if (val == dlg_listbox_getid(ctrl, dlg, i))
dlg_listbox_select(ctrl, dlg, i);
dlg_update_done(ctrl, dlg);
} else if (event == EVENT_SELCHANGE) {
int index = dlg_listbox_index(ctrl, dlg);
if (index >= 0) {
int val = dlg_listbox_getid(ctrl, dlg, index);
conf_set_int(conf, setting, val);
}
}
}
void setup_config_box(struct controlbox *b, int midsession, void setup_config_box(struct controlbox *b, int midsession,
int protocol, int protcfginfo) int protocol, int protcfginfo)
{ {
@ -1860,6 +1888,25 @@ void setup_config_box(struct controlbox *b, int midsession,
"Normal", 'n', I(0), "Normal", 'n', I(0),
"Rectangular block", 'r', I(1), NULL); "Rectangular block", 'r', I(1), NULL);
s = ctrl_getset(b, "Window/Selection", "clipboards",
"Assign copy/paste actions to clipboards");
ctrl_checkbox(s, "Auto-copy selected text to "
CLIPNAME_EXPLICIT_OBJECT,
NO_SHORTCUT, HELPCTX(selection_autocopy),
conf_checkbox_handler, I(CONF_mouseautocopy));
ctrl_droplist(s, "Mouse paste action:", NO_SHORTCUT, 60,
HELPCTX(selection_clipactions),
clipboard_selector_handler,
I(CONF_mousepaste));
ctrl_droplist(s, "{Ctrl,Shift} + Ins:", NO_SHORTCUT, 60,
HELPCTX(selection_clipactions),
clipboard_selector_handler,
I(CONF_ctrlshiftins));
ctrl_droplist(s, "Ctrl + Shift + {C,V}:", NO_SHORTCUT, 60,
HELPCTX(selection_clipactions),
clipboard_selector_handler,
I(CONF_ctrlshiftcv));
/* /*
* The Window/Selection/Words panel. * The Window/Selection/Words panel.
*/ */

View File

@ -1469,6 +1469,68 @@ select a rectangular block. Using the \q{Default selection mode}
control, you can set \i{rectangular selection} as the default, and then control, you can set \i{rectangular selection} as the default, and then
you have to hold down Alt to get the \e{normal} behaviour. you have to hold down Alt to get the \e{normal} behaviour.
\S{config-clipboards} Assigning copy and paste actions to clipboards
Here you can configure which clipboard(s) are written or read by
PuTTY's various copy and paste actions.
The X Window System provides multiple clipboards (or \q{selections}),
and many applications support more than one of them by a different
user interface mechanism.
The two most commonly used selections are called \cq{PRIMARY} and
\cq{CLIPBOARD}; in applications supporting both, the usual behaviour
is that \cw{PRIMARY} is used by mouse-only actions (selecting text
automatically copies it to \cw{PRIMARY}, and middle-clicking pastes
from \cw{PRIMARY}), whereas \cw{CLIPBOARD} is used by explicit Copy
and Paste menu items or keypresses such as Ctrl-C and Ctrl-V.
On other platforms, where there is a single system clipboard, PuTTY
provides a second clipboard-like facility by permitting you to paste
the text you last selected in \e{this window}, whether or not it is
currently also in the system clipboard.
\S2{config-selection-autocopy} \q{Auto-copy selected text}
\cfg{winhelp-topic}{selection.autocopy}
The checkbox \q{Auto-copy selected text to system clipboard} controls
whether or not selecting text in the PuTTY terminal window
automatically has the side effect of copying it to the system
clipboard, without requiring a separate user interface action.
On X, the wording of this option is changed slightly so that
\cq{CLIPBOARD} is mentioned in place of the \q{system clipboard}. Text
selected in the terminal window will \e{always} be automatically
placed in the \cw{PRIMARY} selection, but if you tick this box, it
will \e{also} be placed in \cq{CLIPBOARD} at the same time.
\S2{config-selection-clipactions} Choosing a clipboard for UI actions
\cfg{winhelp-topic}{selection.clipactions}
PuTTY has three user-interface actions which can be configured to
paste into the terminal (not counting menu items). You can click
whichever mouse button (if any) is configured to paste (see
\k{config-mouse}); you can press Shift-Ins; or you can press
Ctrl-Shift-V, although that action is not enabled by default.
You can configure which of the available clipboards each of these
actions pastes from (including turning the paste action off
completely). On platforms with a single system clipboard, the
available options are to paste from that clipboard or to paste from
PuTTY's internal memory of the last selected text within that window.
On X, the options are \cw{CLIPBOARD} or \cw{PRIMARY}.
(\cw{PRIMARY} is conceptually similar in that it \e{also} refers to
the last selected text \dash just across all applications instead of
just this window.)
The two keyboard options each come with a corresponding key to copy
\e{to} the same clipboard. Whatever you configure Shift-Ins to paste
from, Ctrl-Ins will copy to the same location; similarly, Ctrl-Shift-C
will copy to whatever Ctrl-Shift-V pastes from.
\H{config-selection-words} The Words panel \H{config-selection-words} The Words panel
PuTTY will \I{word-by-word selection}select a word at a time in the PuTTY will \I{word-by-word selection}select a word at a time in the

14
putty.h
View File

@ -899,6 +899,10 @@ void cleanup_exit(int);
X(INT, NONE, rtf_paste) \ X(INT, NONE, rtf_paste) \
X(INT, NONE, mouse_override) \ X(INT, NONE, mouse_override) \
X(INT, INT, wordness) \ X(INT, INT, wordness) \
X(INT, NONE, mouseautocopy) \
X(INT, NONE, mousepaste) \
X(INT, NONE, ctrlshiftins) \
X(INT, NONE, ctrlshiftcv) \
/* translations */ \ /* translations */ \
X(INT, NONE, vtmode) \ X(INT, NONE, vtmode) \
X(STR, NONE, line_codepage) \ X(STR, NONE, line_codepage) \
@ -1433,6 +1437,16 @@ enum {
}; };
extern const char *const x11_authnames[]; /* declared in x11fwd.c */ extern const char *const x11_authnames[]; /* declared in x11fwd.c */
/*
* An enum for the copy-paste UI action configuration.
*/
enum {
CLIPUI_NONE, /* UI action has no copy/paste effect */
CLIPUI_IMPLICIT, /* use the default clipboard implicit in mouse actions */
CLIPUI_EXPLICIT, /* use the default clipboard for explicit Copy/Paste */
CLIPUI_CUSTOM, /* use a named clipboard (on systems that support it) */
};
/* /*
* Miscellaneous exports from the platform-specific code. * Miscellaneous exports from the platform-specific code.
* *

View File

@ -447,6 +447,42 @@ static void wprefs(void *sesskey, const char *name,
sfree(buf); sfree(buf);
} }
static void write_clip_setting(void *handle, const char *savekey,
Conf *conf, int confkey)
{
int val = conf_get_int(conf, confkey);
switch (val) {
case CLIPUI_NONE:
default:
write_setting_s(handle, savekey, "none");
break;
case CLIPUI_IMPLICIT:
write_setting_s(handle, savekey, "implicit");
break;
case CLIPUI_EXPLICIT:
write_setting_s(handle, savekey, "explicit");
break;
}
}
static void read_clip_setting(void *handle, const char *savekey,
int def, Conf *conf, int confkey)
{
char *setting = read_setting_s(handle, savekey);
int val;
if (!setting) {
val = def;
} else if (!strcmp(setting, "implicit")) {
val = CLIPUI_IMPLICIT;
} else if (!strcmp(setting, "explicit")) {
val = CLIPUI_EXPLICIT;
} else {
val = CLIPUI_NONE;
}
conf_set_int(conf, confkey, val);
}
char *save_settings(const char *section, Conf *conf) char *save_settings(const char *section, Conf *conf)
{ {
void *sesskey; void *sesskey;
@ -638,6 +674,11 @@ void save_open_settings(void *sesskey, Conf *conf)
} }
write_setting_s(sesskey, buf, buf2); write_setting_s(sesskey, buf, buf2);
} }
write_setting_i(sesskey, "MouseAutocopy",
conf_get_int(conf, CONF_mouseautocopy));
write_clip_setting(sesskey, "MousePaste", conf, CONF_mousepaste);
write_clip_setting(sesskey, "CtrlShiftIns", conf, CONF_ctrlshiftins);
write_clip_setting(sesskey, "CtrlShiftCV", conf, CONF_ctrlshiftcv);
write_setting_s(sesskey, "LineCodePage", conf_get_str(conf, CONF_line_codepage)); write_setting_s(sesskey, "LineCodePage", conf_get_str(conf, CONF_line_codepage));
write_setting_i(sesskey, "CJKAmbigWide", conf_get_int(conf, CONF_cjk_ambig_wide)); write_setting_i(sesskey, "CJKAmbigWide", conf_get_int(conf, CONF_cjk_ambig_wide));
write_setting_i(sesskey, "UTF8Override", conf_get_int(conf, CONF_utf8_override)); write_setting_i(sesskey, "UTF8Override", conf_get_int(conf, CONF_utf8_override));
@ -1059,6 +1100,14 @@ void load_open_settings(void *sesskey, Conf *conf)
} }
sfree(buf2); sfree(buf2);
} }
gppi(sesskey, "MouseAutocopy", CLIPUI_DEFAULT_AUTOCOPY,
conf, CONF_mouseautocopy);
read_clip_setting(sesskey, "MousePaste", CLIPUI_DEFAULT_MOUSE,
conf, CONF_mousepaste);
read_clip_setting(sesskey, "CtrlShiftIns", CLIPUI_DEFAULT_INS,
conf, CONF_ctrlshiftins);
read_clip_setting(sesskey, "CtrlShiftCV", CLIPUI_NONE,
conf, CONF_ctrlshiftcv);
/* /*
* The empty default for LineCodePage will be converted later * The empty default for LineCodePage will be converted later
* into a plausible default for the locale. * into a plausible default for the locale.

View File

@ -1022,15 +1022,98 @@ gint key_event(GtkWidget *widget, GdkEventKey *event, gpointer data)
} }
/* /*
* Neither does Shift-Ins. * Neither do Shift-Ins or Ctrl-Ins (if enabled).
*/ */
if (event->keyval == GDK_KEY_Insert && if (event->keyval == GDK_KEY_Insert &&
(event->state & GDK_SHIFT_MASK)) { (event->state & GDK_SHIFT_MASK)) {
int cfgval = conf_get_int(inst->conf, CONF_ctrlshiftins);
switch (cfgval) {
case CLIPUI_IMPLICIT:
#ifdef KEY_EVENT_DIAGNOSTICS #ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Shift-Insert paste\n")); debug((" - Shift-Insert: paste from PRIMARY\n"));
#endif #endif
term_request_paste(inst->term, SHIFT_INS_CLIPBOARD); term_request_paste(inst->term, CLIP_PRIMARY);
return TRUE; return TRUE;
case CLIPUI_EXPLICIT:
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Shift-Insert: paste from CLIPBOARD\n"));
#endif
term_request_paste(inst->term, CLIP_CLIPBOARD);
return TRUE;
default:
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Shift-Insert: no paste action\n"));
#endif
break;
}
}
if (event->keyval == GDK_KEY_Insert &&
(event->state & GDK_CONTROL_MASK)) {
static const int clips_clipboard[] = { CLIP_CLIPBOARD };
int cfgval = conf_get_int(inst->conf, CONF_ctrlshiftins);
switch (cfgval) {
case CLIPUI_IMPLICIT:
/* do nothing; re-copy to PRIMARY is not needed */
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Ctrl-Insert: non-copy to PRIMARY\n"));
#endif
return TRUE;
case CLIPUI_EXPLICIT:
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Ctrl-Insert: copy to CLIPBOARD\n"));
#endif
term_request_copy(inst->term,
clips_clipboard, lenof(clips_clipboard));
return TRUE;
default:
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Ctrl-Insert: no copy action\n"));
#endif
break;
}
}
/*
* Another pair of copy-paste keys.
*/
if ((event->state & GDK_SHIFT_MASK) &&
(event->state & GDK_CONTROL_MASK) &&
(event->keyval == GDK_KEY_C || event->keyval == GDK_KEY_c ||
event->keyval == GDK_KEY_V || event->keyval == GDK_KEY_v)) {
int cfgval = conf_get_int(inst->conf, CONF_ctrlshiftcv);
int paste = (event->keyval == GDK_KEY_V ||
event->keyval == GDK_KEY_v);
switch (cfgval) {
case CLIPUI_IMPLICIT:
if (paste) {
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Ctrl-Shift-V: paste from PRIMARY\n"));
#endif
term_request_paste(inst->term, CLIP_PRIMARY);
} else {
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Ctrl-Shift-C: non-copy to PRIMARY\n"));
#endif
}
return TRUE;
case CLIPUI_EXPLICIT:
if (paste) {
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Ctrl-Shift-V: paste from CLIPBOARD\n"));
#endif
term_request_paste(inst->term, CLIP_CLIPBOARD);
} else {
static const int clips[] = { CLIP_CLIPBOARD };
#ifdef KEY_EVENT_DIAGNOSTICS
debug((" - Ctrl-Shift-C: copy to CLIPBOARD\n"));
#endif
term_request_copy(inst->term, clips, lenof(clips));
}
return TRUE;
}
} }
special = FALSE; special = FALSE;
@ -3086,7 +3169,7 @@ void init_clipboard(struct gui_data *inst)
void paste_menu_action(void *frontend) void paste_menu_action(void *frontend)
{ {
struct gui_data *inst = (struct gui_data *)frontend; struct gui_data *inst = (struct gui_data *)frontend;
term_request_paste(inst->term, MENU_PASTE_CLIPBOARD); term_request_paste(inst->term, CLIP_CLIPBOARD);
} }
static void set_window_titles(struct gui_data *inst) static void set_window_titles(struct gui_data *inst)
@ -4169,6 +4252,32 @@ void event_log_menuitem(GtkMenuItem *item, gpointer data)
showeventlog(inst->eventlogstuff, inst->window); showeventlog(inst->eventlogstuff, inst->window);
} }
void setup_clipboards(Terminal *term, Conf *conf)
{
assert(term->mouse_select_clipboards[0] == CLIP_LOCAL);
term->n_mouse_select_clipboards = 1;
term->mouse_select_clipboards[
term->n_mouse_select_clipboards++] = MOUSE_SELECT_CLIPBOARD;
if (conf_get_int(conf, CONF_mouseautocopy)) {
term->mouse_select_clipboards[
term->n_mouse_select_clipboards++] = CLIP_CLIPBOARD;
}
switch (conf_get_int(conf, CONF_mousepaste)) {
case CLIPUI_IMPLICIT:
term->mouse_paste_clipboard = MOUSE_PASTE_CLIPBOARD;
break;
case CLIPUI_EXPLICIT:
term->mouse_paste_clipboard = CLIP_CLIPBOARD;
break;
default:
term->mouse_paste_clipboard = CLIP_NULL;
break;
}
}
struct after_change_settings_dialog_ctx { struct after_change_settings_dialog_ctx {
struct gui_data *inst; struct gui_data *inst;
Conf *newconf; Conf *newconf;
@ -4243,6 +4352,7 @@ static void after_change_settings_dialog(void *vctx, int retval)
} }
/* Pass new config data to the terminal */ /* Pass new config data to the terminal */
term_reconfig(inst->term, inst->conf); term_reconfig(inst->term, inst->conf);
setup_clipboards(inst->term, inst->conf);
/* Pass new config data to the back end */ /* Pass new config data to the back end */
if (inst->back) if (inst->back)
inst->back->reconfig(inst->backhandle, inst->conf); inst->back->reconfig(inst->backhandle, inst->conf);
@ -5021,9 +5131,7 @@ void new_session_window(Conf *conf, const char *geometry_string)
inst->eventlogstuff = eventlogstuff_new(); inst->eventlogstuff = eventlogstuff_new();
inst->term = term_init(inst->conf, &inst->ucsdata, inst); inst->term = term_init(inst->conf, &inst->ucsdata, inst);
inst->term->mouse_select_clipboards[ setup_clipboards(inst->term, inst->conf);
inst->term->n_mouse_select_clipboards++] = MOUSE_SELECT_CLIPBOARD;
inst->term->mouse_paste_clipboard = MOUSE_PASTE_CLIPBOARD;
inst->logctx = log_init(inst, inst->conf); inst->logctx = log_init(inst, inst->conf);
term_provide_logctx(inst->term, inst->logctx); term_provide_logctx(inst->term, inst->logctx);

View File

@ -125,15 +125,31 @@ unsigned long getticks(void);
#ifdef OSX_GTK #ifdef OSX_GTK
/* OS X has no PRIMARY selection */ /* OS X has no PRIMARY selection */
#define SHIFT_INS_CLIPBOARD CLIP_CLIPBOARD #define MOUSE_SELECT_CLIPBOARD CLIP_NULL
#define MOUSE_SELECT_CLIPBOARD CLIP_CLIPBOARD #define MOUSE_PASTE_CLIPBOARD CLIP_LOCAL
#define MOUSE_PASTE_CLIPBOARD CLIP_CLIPBOARD #define CLIPNAME_IMPLICIT "Last selected text"
#define CLIPNAME_EXPLICIT "System clipboard"
#define CLIPNAME_EXPLICIT_OBJECT "system clipboard"
/* These defaults are the ones that more or less comply with the OS X
* Human Interface Guidelines, i.e. copy/paste to the system clipboard
* is _not_ implicit but requires a specific UI action. This is at
* odds with all other PuTTY front ends' defaults, but on OS X there
* is no multi-decade precedent for PuTTY working the other way. */
#define CLIPUI_DEFAULT_AUTOCOPY FALSE
#define CLIPUI_DEFAULT_MOUSE CLIPUI_IMPLICIT
#define CLIPUI_DEFAULT_INS CLIPUI_EXPLICIT
#else #else
#define SHIFT_INS_CLIPBOARD CLIP_PRIMARY
#define MOUSE_SELECT_CLIPBOARD CLIP_PRIMARY #define MOUSE_SELECT_CLIPBOARD CLIP_PRIMARY
#define MOUSE_PASTE_CLIPBOARD CLIP_PRIMARY #define MOUSE_PASTE_CLIPBOARD CLIP_PRIMARY
#define CLIPNAME_IMPLICIT "PRIMARY"
#define CLIPNAME_EXPLICIT "CLIPBOARD"
#define CLIPNAME_EXPLICIT_OBJECT "CLIPBOARD"
/* These defaults are the ones Unix PuTTY has historically had since
* it was first thought of in 2002 */
#define CLIPUI_DEFAULT_AUTOCOPY FALSE
#define CLIPUI_DEFAULT_MOUSE CLIPUI_IMPLICIT
#define CLIPUI_DEFAULT_INS CLIPUI_IMPLICIT
#endif #endif
#define MENU_PASTE_CLIPBOARD CLIP_CLIPBOARD
/* The per-session frontend structure managed by gtkwin.c */ /* The per-session frontend structure managed by gtkwin.c */
struct gui_data; struct gui_data;

View File

@ -52,7 +52,8 @@
#define IDM_SAVEDSESS 0x0160 #define IDM_SAVEDSESS 0x0160
#define IDM_COPYALL 0x0170 #define IDM_COPYALL 0x0170
#define IDM_FULLSCREEN 0x0180 #define IDM_FULLSCREEN 0x0180
#define IDM_PASTE 0x0190 #define IDM_COPY 0x0190
#define IDM_PASTE 0x01A0
#define IDM_SPECIALSEP 0x0200 #define IDM_SPECIALSEP 0x0200
#define IDM_SPECIAL_MIN 0x0400 #define IDM_SPECIAL_MIN 0x0400
@ -106,6 +107,7 @@ static void make_full_screen(void);
static void clear_full_screen(void); static void clear_full_screen(void);
static void flip_full_screen(void); static void flip_full_screen(void);
static void process_clipdata(HGLOBAL clipdata, int unicode); static void process_clipdata(HGLOBAL clipdata, int unicode);
static void setup_clipboards(Terminal *, Conf *);
/* Window layout information */ /* Window layout information */
static void reset_window(int); static void reset_window(int);
@ -637,9 +639,7 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
* timer_change_notify() which will expect hwnd to exist.) * timer_change_notify() which will expect hwnd to exist.)
*/ */
term = term_init(conf, &ucsdata, NULL); term = term_init(conf, &ucsdata, NULL);
term->mouse_select_clipboards[ setup_clipboards(term, conf);
term->n_mouse_select_clipboards++] = CLIP_SYSTEM;
term->mouse_paste_clipboard = CLIP_SYSTEM;
logctx = log_init(NULL, conf); logctx = log_init(NULL, conf);
term_provide_logctx(term, logctx); term_provide_logctx(term, logctx);
term_size(term, conf_get_int(conf, CONF_height), term_size(term, conf_get_int(conf, CONF_height),
@ -712,6 +712,7 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
popup_menus[SYSMENU].menu = GetSystemMenu(hwnd, FALSE); popup_menus[SYSMENU].menu = GetSystemMenu(hwnd, FALSE);
popup_menus[CTXMENU].menu = CreatePopupMenu(); popup_menus[CTXMENU].menu = CreatePopupMenu();
AppendMenu(popup_menus[CTXMENU].menu, MF_ENABLED, IDM_COPY, "&Copy");
AppendMenu(popup_menus[CTXMENU].menu, MF_ENABLED, IDM_PASTE, "&Paste"); AppendMenu(popup_menus[CTXMENU].menu, MF_ENABLED, IDM_PASTE, "&Paste");
savedsess_menu = CreateMenu(); savedsess_menu = CreateMenu();
@ -853,6 +854,30 @@ int WINAPI WinMain(HINSTANCE inst, HINSTANCE prev, LPSTR cmdline, int show)
return msg.wParam; /* ... but optimiser doesn't know */ return msg.wParam; /* ... but optimiser doesn't know */
} }
static void setup_clipboards(Terminal *term, Conf *conf)
{
assert(term->mouse_select_clipboards[0] == CLIP_LOCAL);
term->n_mouse_select_clipboards = 1;
if (conf_get_int(conf, CONF_mouseautocopy)) {
term->mouse_select_clipboards[
term->n_mouse_select_clipboards++] = CLIP_SYSTEM;
}
switch (conf_get_int(conf, CONF_mousepaste)) {
case CLIPUI_IMPLICIT:
term->mouse_paste_clipboard = CLIP_LOCAL;
break;
case CLIPUI_EXPLICIT:
term->mouse_paste_clipboard = CLIP_SYSTEM;
break;
default:
term->mouse_paste_clipboard = CLIP_NULL;
break;
}
}
/* /*
* Clean up and exit. * Clean up and exit.
*/ */
@ -1998,6 +2023,8 @@ static void conf_cache_data(void)
vtmode = conf_get_int(conf, CONF_vtmode); vtmode = conf_get_int(conf, CONF_vtmode);
} }
static const int clips_system[] = { CLIP_SYSTEM };
static LRESULT CALLBACK WndProc(HWND hwnd, UINT message, static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
WPARAM wParam, LPARAM lParam) WPARAM wParam, LPARAM lParam)
{ {
@ -2210,6 +2237,7 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
/* Pass new config data to the terminal */ /* Pass new config data to the terminal */
term_reconfig(term, conf); term_reconfig(term, conf);
setup_clipboards(term, conf);
/* Pass new config data to the back end */ /* Pass new config data to the back end */
if (back) if (back)
@ -2331,10 +2359,10 @@ static LRESULT CALLBACK WndProc(HWND hwnd, UINT message,
} }
break; break;
case IDM_COPYALL: case IDM_COPYALL:
{ term_copyall(term, clips_system, lenof(clips_system));
static const int clips[] = { CLIP_SYSTEM }; break;
term_copyall(term, clips, lenof(clips)); case IDM_COPY:
} term_request_copy(term, clips_system, lenof(clips_system));
break; break;
case IDM_PASTE: case IDM_PASTE:
term_request_paste(term, CLIP_SYSTEM); term_request_paste(term, CLIP_SYSTEM);
@ -4151,8 +4179,54 @@ static int TranslateKey(UINT message, WPARAM wParam, LPARAM lParam,
term_scroll_to_selection(term, (wParam == VK_PRIOR ? 0 : 1)); term_scroll_to_selection(term, (wParam == VK_PRIOR ? 0 : 1));
return 0; return 0;
} }
if (wParam == VK_INSERT && shift_state == 2) {
switch (conf_get_int(conf, CONF_ctrlshiftins)) {
case CLIPUI_IMPLICIT:
break; /* no need to re-copy to CLIP_LOCAL */
case CLIPUI_EXPLICIT:
term_request_copy(term, clips_system, lenof(clips_system));
break;
default:
break;
}
return 0;
}
if (wParam == VK_INSERT && shift_state == 1) { if (wParam == VK_INSERT && shift_state == 1) {
term_request_paste(term, CLIP_SYSTEM); switch (conf_get_int(conf, CONF_ctrlshiftins)) {
case CLIPUI_IMPLICIT:
term_request_paste(term, CLIP_LOCAL);
break;
case CLIPUI_EXPLICIT:
term_request_paste(term, CLIP_SYSTEM);
break;
default:
break;
}
return 0;
}
if (wParam == 'C' && shift_state == 3) {
switch (conf_get_int(conf, CONF_ctrlshiftcv)) {
case CLIPUI_IMPLICIT:
break; /* no need to re-copy to CLIP_LOCAL */
case CLIPUI_EXPLICIT:
term_request_copy(term, clips_system, lenof(clips_system));
break;
default:
break;
}
return 0;
}
if (wParam == 'V' && shift_state == 3) {
switch (conf_get_int(conf, CONF_ctrlshiftcv)) {
case CLIPUI_IMPLICIT:
term_request_paste(term, CLIP_LOCAL);
break;
case CLIPUI_EXPLICIT:
term_request_paste(term, CLIP_SYSTEM);
break;
default:
break;
}
return 0; return 0;
} }
if (left_alt && wParam == VK_F4 && conf_get_int(conf, CONF_alt_f4)) { if (left_alt && wParam == VK_F4 && conf_get_int(conf, CONF_alt_f4)) {

View File

@ -123,6 +123,8 @@
#define WINHELP_CTX_selection_charclasses "selection.charclasses:config-charclasses" #define WINHELP_CTX_selection_charclasses "selection.charclasses:config-charclasses"
#define WINHELP_CTX_selection_linedraw "selection.linedraw:config-linedrawpaste" #define WINHELP_CTX_selection_linedraw "selection.linedraw:config-linedrawpaste"
#define WINHELP_CTX_selection_rtf "selection.rtf:config-rtfpaste" #define WINHELP_CTX_selection_rtf "selection.rtf:config-rtfpaste"
#define WINHELP_CTX_selection_autocopy "selection.autocopy:config-selection-autocopy"
#define WINHELP_CTX_selection_clipactions "selection.clipactions:config-selection-clipactions"
#define WINHELP_CTX_colours_ansi "colours.ansi:config-ansicolour" #define WINHELP_CTX_colours_ansi "colours.ansi:config-ansicolour"
#define WINHELP_CTX_colours_xterm256 "colours.xterm256:config-xtermcolour" #define WINHELP_CTX_colours_xterm256 "colours.xterm256:config-xtermcolour"
#define WINHELP_CTX_colours_truecolour "colours.truecolour:config-truecolour" #define WINHELP_CTX_colours_truecolour "colours.truecolour:config-truecolour"

View File

@ -634,4 +634,15 @@ int remove_from_jumplist_registry(const char *item);
* empty one. */ * empty one. */
char *get_jumplist_registry_entries(void); char *get_jumplist_registry_entries(void);
/*
* Windows clipboard-UI wording.
*/
#define CLIPNAME_IMPLICIT "Last selected text"
#define CLIPNAME_EXPLICIT "System clipboard"
#define CLIPNAME_EXPLICIT_OBJECT "system clipboard"
/* These defaults are the ones PuTTY has historically had */
#define CLIPUI_DEFAULT_AUTOCOPY TRUE
#define CLIPUI_DEFAULT_MOUSE CLIPUI_EXPLICIT
#define CLIPUI_DEFAULT_INS CLIPUI_EXPLICIT
#endif #endif