mirror of
https://github.com/tmux/tmux.git
synced 2024-11-10 13:48:48 +00:00
Fix length calculation for pasting UTF-8 characters in the status line,
GitHub issue 1753.
This commit is contained in:
parent
723010ba72
commit
a4fe7e81c8
111
status.c
111
status.c
@ -846,16 +846,79 @@ status_prompt_translate_key(struct client *c, key_code key, key_code *new_key)
|
|||||||
return (0);
|
return (0);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Paste into prompt. */
|
||||||
|
static int
|
||||||
|
status_prompt_paste(struct client *c)
|
||||||
|
{
|
||||||
|
struct paste_buffer *pb;
|
||||||
|
const char *bufdata;
|
||||||
|
size_t size, n, bufsize;
|
||||||
|
u_int i;
|
||||||
|
struct utf8_data *ud, *udp;
|
||||||
|
enum utf8_state more;
|
||||||
|
|
||||||
|
size = utf8_strlen(c->prompt_buffer);
|
||||||
|
if (c->prompt_saved != NULL) {
|
||||||
|
ud = c->prompt_saved;
|
||||||
|
n = utf8_strlen(c->prompt_saved);
|
||||||
|
} else {
|
||||||
|
if ((pb = paste_get_top(NULL)) == NULL)
|
||||||
|
return (0);
|
||||||
|
bufdata = paste_buffer_data(pb, &bufsize);
|
||||||
|
ud = xreallocarray(NULL, bufsize, sizeof *ud);
|
||||||
|
udp = ud;
|
||||||
|
for (i = 0; i != bufsize; /* nothing */) {
|
||||||
|
more = utf8_open(udp, bufdata[i]);
|
||||||
|
if (more == UTF8_MORE) {
|
||||||
|
while (++i != bufsize && more == UTF8_MORE)
|
||||||
|
more = utf8_append(udp, bufdata[i]);
|
||||||
|
if (more == UTF8_DONE) {
|
||||||
|
udp++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
i -= udp->have;
|
||||||
|
}
|
||||||
|
if (bufdata[i] <= 31 || bufdata[i] >= 127)
|
||||||
|
break;
|
||||||
|
utf8_set(udp, bufdata[i]);
|
||||||
|
udp++;
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
udp->size = 0;
|
||||||
|
n = udp - ud;
|
||||||
|
}
|
||||||
|
if (n == 0)
|
||||||
|
return (0);
|
||||||
|
|
||||||
|
c->prompt_buffer = xreallocarray(c->prompt_buffer, size + n + 1,
|
||||||
|
sizeof *c->prompt_buffer);
|
||||||
|
if (c->prompt_index == size) {
|
||||||
|
memcpy(c->prompt_buffer + c->prompt_index, ud,
|
||||||
|
n * sizeof *c->prompt_buffer);
|
||||||
|
c->prompt_index += n;
|
||||||
|
c->prompt_buffer[c->prompt_index].size = 0;
|
||||||
|
} else {
|
||||||
|
memmove(c->prompt_buffer + c->prompt_index + n,
|
||||||
|
c->prompt_buffer + c->prompt_index,
|
||||||
|
(size + 1 - c->prompt_index) * sizeof *c->prompt_buffer);
|
||||||
|
memcpy(c->prompt_buffer + c->prompt_index, ud,
|
||||||
|
n * sizeof *c->prompt_buffer);
|
||||||
|
c->prompt_index += n;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ud != c->prompt_saved)
|
||||||
|
free(ud);
|
||||||
|
return (1);
|
||||||
|
}
|
||||||
|
|
||||||
/* Handle keys in prompt. */
|
/* Handle keys in prompt. */
|
||||||
int
|
int
|
||||||
status_prompt_key(struct client *c, key_code key)
|
status_prompt_key(struct client *c, key_code key)
|
||||||
{
|
{
|
||||||
struct options *oo = c->session->options;
|
struct options *oo = c->session->options;
|
||||||
struct paste_buffer *pb;
|
|
||||||
char *s, *cp, word[64], prefix = '=';
|
char *s, *cp, word[64], prefix = '=';
|
||||||
const char *histstr, *bufdata, *ws = NULL;
|
const char *histstr, *ws = NULL;
|
||||||
u_char ch;
|
size_t size, n, off, idx, used;
|
||||||
size_t size, n, off, idx, bufsize, used;
|
|
||||||
struct utf8_data tmp, *first, *last, *ud;
|
struct utf8_data tmp, *first, *last, *ud;
|
||||||
int keys;
|
int keys;
|
||||||
|
|
||||||
@ -1107,43 +1170,9 @@ process_key:
|
|||||||
c->prompt_index = utf8_strlen(c->prompt_buffer);
|
c->prompt_index = utf8_strlen(c->prompt_buffer);
|
||||||
goto changed;
|
goto changed;
|
||||||
case '\031': /* C-y */
|
case '\031': /* C-y */
|
||||||
if (c->prompt_saved != NULL) {
|
if (status_prompt_paste(c))
|
||||||
ud = c->prompt_saved;
|
goto changed;
|
||||||
n = utf8_strlen(c->prompt_saved);
|
break;
|
||||||
} else {
|
|
||||||
if ((pb = paste_get_top(NULL)) == NULL)
|
|
||||||
break;
|
|
||||||
bufdata = paste_buffer_data(pb, &bufsize);
|
|
||||||
for (n = 0; n < bufsize; n++) {
|
|
||||||
ch = (u_char)bufdata[n];
|
|
||||||
if (ch < 32 || ch >= 127)
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
ud = xreallocarray(NULL, n, sizeof *ud);
|
|
||||||
for (idx = 0; idx < n; idx++)
|
|
||||||
utf8_set(&ud[idx], bufdata[idx]);
|
|
||||||
}
|
|
||||||
|
|
||||||
c->prompt_buffer = xreallocarray(c->prompt_buffer, size + n + 1,
|
|
||||||
sizeof *c->prompt_buffer);
|
|
||||||
if (c->prompt_index == size) {
|
|
||||||
memcpy(c->prompt_buffer + c->prompt_index, ud,
|
|
||||||
n * sizeof *c->prompt_buffer);
|
|
||||||
c->prompt_index += n;
|
|
||||||
c->prompt_buffer[c->prompt_index].size = 0;
|
|
||||||
} else {
|
|
||||||
memmove(c->prompt_buffer + c->prompt_index + n,
|
|
||||||
c->prompt_buffer + c->prompt_index,
|
|
||||||
(size + 1 - c->prompt_index) *
|
|
||||||
sizeof *c->prompt_buffer);
|
|
||||||
memcpy(c->prompt_buffer + c->prompt_index, ud,
|
|
||||||
n * sizeof *c->prompt_buffer);
|
|
||||||
c->prompt_index += n;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (ud != c->prompt_saved)
|
|
||||||
free(ud);
|
|
||||||
goto changed;
|
|
||||||
case '\024': /* C-t */
|
case '\024': /* C-t */
|
||||||
idx = c->prompt_index;
|
idx = c->prompt_index;
|
||||||
if (idx < size)
|
if (idx < size)
|
||||||
|
Loading…
Reference in New Issue
Block a user