Merge branch 'obsd-master'

This commit is contained in:
Thomas Adam 2017-01-07 16:01:17 +00:00
commit 29e64a8c64
9 changed files with 185 additions and 20 deletions

View File

@ -112,8 +112,10 @@ cmd_send_keys_exec(struct cmd *self, struct cmdq_item *item)
return (CMD_RETURN_NORMAL); return (CMD_RETURN_NORMAL);
} }
if (args_has(args, 'R')) if (args_has(args, 'R')) {
window_pane_reset_palette(wp);
input_reset(wp, 1); input_reset(wp, 1);
}
for (; np != 0; np--) { for (; np != 0; np--) {
for (i = 0; i < args->argc; i++) { for (i = 0; i < args->argc; i++) {
@ -128,7 +130,8 @@ cmd_send_keys_exec(struct cmd *self, struct cmdq_item *item)
if (literal) { if (literal) {
ud = utf8_fromcstr(args->argv[i]); ud = utf8_fromcstr(args->argv[i]);
for (uc = ud; uc->size != 0; uc++) { for (uc = ud; uc->size != 0; uc++) {
if (utf8_combine(uc, &wc) == UTF8_DONE) if (utf8_combine(uc, &wc) != UTF8_DONE)
continue;
window_pane_key(wp, NULL, s, wc, NULL); window_pane_key(wp, NULL, s, wc, NULL);
} }
free(ud); free(ud);

81
input.c
View File

@ -105,6 +105,9 @@ static void input_set_state(struct window_pane *,
const struct input_transition *); const struct input_transition *);
static void input_reset_cell(struct input_ctx *); static void input_reset_cell(struct input_ctx *);
static void input_osc_4(struct window_pane *, const char *);
static void input_osc_104(struct window_pane *, const char *);
/* Transition entry/exit handlers. */ /* Transition entry/exit handlers. */
static void input_clear(struct input_ctx *); static void input_clear(struct input_ctx *);
static void input_ground(struct input_ctx *); static void input_ground(struct input_ctx *);
@ -162,6 +165,7 @@ enum input_esc_type {
INPUT_ESC_SCSG0_ON, INPUT_ESC_SCSG0_ON,
INPUT_ESC_SCSG1_OFF, INPUT_ESC_SCSG1_OFF,
INPUT_ESC_SCSG1_ON, INPUT_ESC_SCSG1_ON,
INPUT_ESC_ST,
}; };
/* Escape command table. */ /* Escape command table. */
@ -179,6 +183,7 @@ static const struct input_table_entry input_esc_table[] = {
{ 'E', "", INPUT_ESC_NEL }, { 'E', "", INPUT_ESC_NEL },
{ 'H', "", INPUT_ESC_HTS }, { 'H', "", INPUT_ESC_HTS },
{ 'M', "", INPUT_ESC_RI }, { 'M', "", INPUT_ESC_RI },
{ '\\', "", INPUT_ESC_ST },
{ 'c', "", INPUT_ESC_RIS }, { 'c', "", INPUT_ESC_RIS },
}; };
@ -1141,6 +1146,7 @@ input_esc_dispatch(struct input_ctx *ictx)
switch (entry->type) { switch (entry->type) {
case INPUT_ESC_RIS: case INPUT_ESC_RIS:
window_pane_reset_palette(ictx->wp);
input_reset_cell(ictx); input_reset_cell(ictx);
screen_write_reset(sctx); screen_write_reset(sctx);
break; break;
@ -1188,6 +1194,9 @@ input_esc_dispatch(struct input_ctx *ictx)
case INPUT_ESC_SCSG1_OFF: case INPUT_ESC_SCSG1_OFF:
ictx->cell.g1set = 0; ictx->cell.g1set = 0;
break; break;
case INPUT_ESC_ST:
/* ST terminates OSC but the state transition already did it. */
break;
} }
return (0); return (0);
@ -1850,10 +1859,16 @@ input_exit_osc(struct input_ctx *ictx)
screen_set_title(ictx->ctx.s, p); screen_set_title(ictx->ctx.s, p);
server_status_window(ictx->wp->window); server_status_window(ictx->wp->window);
break; break;
case 4:
input_osc_4(ictx->wp, p);
break;
case 12: case 12:
if (*p != '?') /* ? is colour request */ if (*p != '?') /* ? is colour request */
screen_set_cursor_colour(ictx->ctx.s, p); screen_set_cursor_colour(ictx->ctx.s, p);
break; break;
case 104:
input_osc_104(ictx->wp, p);
break;
case 112: case 112:
if (*p == '\0') /* no arguments allowed */ if (*p == '\0') /* no arguments allowed */
screen_set_cursor_colour(ictx->ctx.s, ""); screen_set_cursor_colour(ictx->ctx.s, "");
@ -1961,3 +1976,69 @@ input_utf8_close(struct input_ctx *ictx)
return (0); return (0);
} }
/* Handle the OSC 4 sequence for setting (multiple) palette entries. */
static void
input_osc_4(struct window_pane *wp, const char *p)
{
char *copy, *s, *next = NULL;
long idx;
u_int r, g, b;
copy = s = xstrdup(p);
while (s != NULL && *s != '\0') {
idx = strtol(s, &next, 10);
if (*next++ != ';')
goto bad;
if (idx < 0 || idx >= 0x100)
goto bad;
s = strsep(&next, ";");
if (sscanf(s, "rgb:%2x/%2x/%2x", &r, &g, &b) != 3) {
s = next;
continue;
}
window_pane_set_palette(wp, idx, colour_join_rgb(r, g, b));
s = next;
}
free(copy);
return;
bad:
log_debug("bad OSC 4: %s", p);
free(copy);
}
/* Handle the OSC 104 sequence for unsetting (multiple) palette entries. */
static void
input_osc_104(struct window_pane *wp, const char *p)
{
char *copy, *s;
long idx;
if (*p == '\0') {
window_pane_reset_palette(wp);
return;
}
copy = s = xstrdup(p);
while (*s != '\0') {
idx = strtol(s, &s, 10);
if (*s != '\0' && *s != ';')
goto bad;
if (idx < 0 || idx >= 0x100)
goto bad;
window_pane_unset_palette(wp, idx);
if (*s == ';')
s++;
}
free(copy);
return;
bad:
log_debug("bad OSC 104: %s", p);
free(copy);
}

View File

@ -588,6 +588,8 @@ screen_redraw_draw_number(struct client *c, struct window_pane *wp, u_int top)
gc.bg = active_colour; gc.bg = active_colour;
else else
gc.bg = colour; gc.bg = colour;
gc.flags |= GRID_FLAG_NOPALETTE;
tty_attributes(tty, &gc, wp); tty_attributes(tty, &gc, wp);
for (ptr = buf; *ptr != '\0'; ptr++) { for (ptr = buf; *ptr != '\0'; ptr++) {
if (*ptr < '0' || *ptr > '9') if (*ptr < '0' || *ptr > '9')
@ -615,6 +617,8 @@ draw_text:
gc.fg = active_colour; gc.fg = active_colour;
else else
gc.fg = colour; gc.fg = colour;
gc.flags |= GRID_FLAG_NOPALETTE;
tty_attributes(tty, &gc, wp); tty_attributes(tty, &gc, wp);
tty_puts(tty, buf); tty_puts(tty, buf);

11
tmux.h
View File

@ -578,6 +578,7 @@ enum utf8_state {
#define GRID_FLAG_PADDING 0x4 #define GRID_FLAG_PADDING 0x4
#define GRID_FLAG_EXTENDED 0x8 #define GRID_FLAG_EXTENDED 0x8
#define GRID_FLAG_SELECTED 0x10 #define GRID_FLAG_SELECTED 0x10
#define GRID_FLAG_NOPALETTE 0x20
/* Grid line flags. */ /* Grid line flags. */
#define GRID_LINE_WRAPPED 0x1 #define GRID_LINE_WRAPPED 0x1
@ -831,6 +832,8 @@ struct window_pane {
struct grid_cell colgc; struct grid_cell colgc;
int *palette;
int pipe_fd; int pipe_fd;
struct bufferevent *pipe_event; struct bufferevent *pipe_event;
size_t pipe_off; size_t pipe_off;
@ -859,6 +862,11 @@ struct window_pane {
TAILQ_HEAD(window_panes, window_pane); TAILQ_HEAD(window_panes, window_pane);
RB_HEAD(window_pane_tree, window_pane); RB_HEAD(window_pane_tree, window_pane);
#define WINDOW_PANE_PALETTE_HAS(wp, c) \
((wp) != NULL && (wp)->palette != NULL && \
((c) < 0x100 || (c) & COLOUR_FLAG_256) && \
(wp)->palette[(c) & 0xff] != 0)
/* Window structure. */ /* Window structure. */
struct window { struct window {
u_int id; u_int id;
@ -2128,6 +2136,9 @@ void window_pane_alternate_on(struct window_pane *,
struct grid_cell *, int); struct grid_cell *, int);
void window_pane_alternate_off(struct window_pane *, void window_pane_alternate_off(struct window_pane *,
struct grid_cell *, int); struct grid_cell *, int);
void window_pane_set_palette(struct window_pane *, u_int, int);
void window_pane_unset_palette(struct window_pane *, u_int);
void window_pane_reset_palette(struct window_pane *);
int window_pane_set_mode(struct window_pane *, int window_pane_set_mode(struct window_pane *,
const struct window_mode *); const struct window_mode *);
void window_pane_reset_mode(struct window_pane *); void window_pane_reset_mode(struct window_pane *);

36
tty.c
View File

@ -48,8 +48,10 @@ static void tty_cursor_pane(struct tty *, const struct tty_ctx *, u_int,
u_int); u_int);
static void tty_colours(struct tty *, const struct grid_cell *); static void tty_colours(struct tty *, const struct grid_cell *);
static void tty_check_fg(struct tty *, struct grid_cell *); static void tty_check_fg(struct tty *, const struct window_pane *,
static void tty_check_bg(struct tty *, struct grid_cell *); struct grid_cell *);
static void tty_check_bg(struct tty *, const struct window_pane *,
struct grid_cell *);
static void tty_colours_fg(struct tty *, const struct grid_cell *); static void tty_colours_fg(struct tty *, const struct grid_cell *);
static void tty_colours_bg(struct tty *, const struct grid_cell *); static void tty_colours_bg(struct tty *, const struct grid_cell *);
@ -1507,8 +1509,8 @@ tty_attributes(struct tty *tty, const struct grid_cell *gc,
} }
/* Fix up the colours if necessary. */ /* Fix up the colours if necessary. */
tty_check_fg(tty, &gc2); tty_check_fg(tty, wp, &gc2);
tty_check_bg(tty, &gc2); tty_check_bg(tty, wp, &gc2);
/* If any bits are being cleared, reset everything. */ /* If any bits are being cleared, reset everything. */
if (tc->attr & ~gc2.attr) if (tc->attr & ~gc2.attr)
@ -1604,12 +1606,18 @@ tty_colours(struct tty *tty, const struct grid_cell *gc)
tty_colours_bg(tty, gc); tty_colours_bg(tty, gc);
} }
void static void
tty_check_fg(struct tty *tty, struct grid_cell *gc) tty_check_fg(struct tty *tty, const struct window_pane *wp,
struct grid_cell *gc)
{ {
u_char r, g, b; u_char r, g, b;
u_int colours; u_int colours;
/* Perform substitution if this pane has a palette */
if ((~gc->flags & GRID_FLAG_NOPALETTE) &&
gc->fg != 8 && WINDOW_PANE_PALETTE_HAS(wp, gc->fg))
gc->fg = wp->palette[gc->fg & 0xff];
/* Is this a 24-bit colour? */ /* Is this a 24-bit colour? */
if (gc->fg & COLOUR_FLAG_RGB) { if (gc->fg & COLOUR_FLAG_RGB) {
/* Not a 24-bit terminal? Translate to 256-colour palette. */ /* Not a 24-bit terminal? Translate to 256-colour palette. */
@ -1646,12 +1654,18 @@ tty_check_fg(struct tty *tty, struct grid_cell *gc)
} }
} }
void static void
tty_check_bg(struct tty *tty, struct grid_cell *gc) tty_check_bg(struct tty *tty, const struct window_pane *wp,
struct grid_cell *gc)
{ {
u_char r, g, b; u_char r, g, b;
u_int colours; u_int colours;
/* Perform substitution if this pane has a palette */
if ((~gc->flags & GRID_FLAG_NOPALETTE) &&
gc->bg != 8 && WINDOW_PANE_PALETTE_HAS(wp, gc->bg))
gc->bg = wp->palette[gc->bg & 0xff];
/* Is this a 24-bit colour? */ /* Is this a 24-bit colour? */
if (gc->bg & COLOUR_FLAG_RGB) { if (gc->bg & COLOUR_FLAG_RGB) {
/* Not a 24-bit terminal? Translate to 256-colour palette. */ /* Not a 24-bit terminal? Translate to 256-colour palette. */
@ -1826,6 +1840,9 @@ tty_default_colours(struct grid_cell *gc, const struct window_pane *wp)
gc->fg = agc->fg; gc->fg = agc->fg;
else else
gc->fg = wgc->fg; gc->fg = wgc->fg;
if (gc->fg != 8 && WINDOW_PANE_PALETTE_HAS(wp, gc->fg))
gc->fg = wp->palette[gc->fg & 0xff];
} }
if (gc->bg == 8) { if (gc->bg == 8) {
@ -1835,6 +1852,9 @@ tty_default_colours(struct grid_cell *gc, const struct window_pane *wp)
gc->bg = agc->bg; gc->bg = agc->bg;
else else
gc->bg = wgc->bg; gc->bg = wgc->bg;
if (gc->bg != 8 && WINDOW_PANE_PALETTE_HAS(wp, gc->bg))
gc->bg = wp->palette[gc->bg & 0xff];
} }
} }

View File

@ -796,6 +796,7 @@ window_choose_write_line(struct window_pane *wp, struct screen_write_ctx *ctx,
last = screen_size_y(s) - 1; last = screen_size_y(s) - 1;
memcpy(&gc, &grid_default_cell, sizeof gc); memcpy(&gc, &grid_default_cell, sizeof gc);
gc.flags |= GRID_FLAG_NOPALETTE;
if (data->selected == data->top + py) if (data->selected == data->top + py)
style_apply(&gc, oo, "mode-style"); style_apply(&gc, oo, "mode-style");

View File

@ -230,6 +230,7 @@ window_clock_draw_screen(struct window_pane *wp)
screen_write_cursormove(&ctx, x, y); screen_write_cursormove(&ctx, x, y);
memcpy(&gc, &grid_default_cell, sizeof gc); memcpy(&gc, &grid_default_cell, sizeof gc);
gc.flags |= GRID_FLAG_NOPALETTE;
gc.fg = colour; gc.fg = colour;
screen_write_puts(&ctx, &gc, "%s", tim); screen_write_puts(&ctx, &gc, "%s", tim);
} }
@ -242,6 +243,7 @@ window_clock_draw_screen(struct window_pane *wp)
y = (screen_size_y(s) / 2) - 3; y = (screen_size_y(s) / 2) - 3;
memcpy(&gc, &grid_default_cell, sizeof gc); memcpy(&gc, &grid_default_cell, sizeof gc);
gc.flags |= GRID_FLAG_NOPALETTE;
gc.bg = colour; gc.bg = colour;
for (ptr = tim; *ptr != '\0'; ptr++) { for (ptr = tim; *ptr != '\0'; ptr++) {
if (*ptr >= '0' && *ptr <= '9') if (*ptr >= '0' && *ptr <= '9')

View File

@ -1253,6 +1253,7 @@ window_copy_write_line(struct window_pane *wp, struct screen_write_ctx *ctx,
size_t size = 0; size_t size = 0;
style_apply(&gc, oo, "mode-style"); style_apply(&gc, oo, "mode-style");
gc.flags |= GRID_FLAG_NOPALETTE;
if (py == 0) { if (py == 0) {
size = xsnprintf(hdr, sizeof hdr, size = xsnprintf(hdr, sizeof hdr,
@ -1455,6 +1456,7 @@ window_copy_update_selection(struct window_pane *wp, int may_redraw)
/* Set colours and selection. */ /* Set colours and selection. */
style_apply(&gc, oo, "mode-style"); style_apply(&gc, oo, "mode-style");
gc.flags |= GRID_FLAG_NOPALETTE;
screen_set_selection(s, sx, sy, endsx, endsy, data->rectflag, &gc); screen_set_selection(s, sx, sy, endsx, endsy, data->rectflag, &gc);
if (data->rectflag && may_redraw) { if (data->rectflag && may_redraw) {

View File

@ -447,24 +447,30 @@ window_set_active_pane(struct window *w, struct window_pane *wp)
void void
window_redraw_active_switch(struct window *w, struct window_pane *wp) window_redraw_active_switch(struct window *w, struct window_pane *wp)
{ {
const struct grid_cell *agc, *wgc; const struct grid_cell *gc;
if (wp == w->active) if (wp == w->active)
return; return;
/* /*
* If window-style and window-active-style are the same, we don't need * If window-style and window-active-style are the same, we don't need
* to redraw panes when switching active panes. Otherwise, if the * to redraw panes when switching active panes.
* active or inactive pane do not have a custom style, they will need
* to be redrawn.
*/ */
agc = options_get_style(w->options, "window-active-style"); gc = options_get_style(w->options, "window-active-style");
wgc = options_get_style(w->options, "window-style"); if (style_equal(gc, options_get_style(w->options, "window-style")))
if (style_equal(agc, wgc))
return; return;
if (style_equal(&grid_default_cell, &w->active->colgc))
/*
* If the now active or inactive pane do not have a custom style or if
* the palette is different, they need to be redrawn.
*/
if (WINDOW_PANE_PALETTE_HAS(w->active, w->active->colgc.fg) ||
WINDOW_PANE_PALETTE_HAS(w->active, w->active->colgc.bg) ||
style_equal(&grid_default_cell, &w->active->colgc))
w->active->flags |= PANE_REDRAW; w->active->flags |= PANE_REDRAW;
if (style_equal(&grid_default_cell, &wp->colgc)) if (WINDOW_PANE_PALETTE_HAS(wp, wp->colgc.fg) ||
WINDOW_PANE_PALETTE_HAS(wp, wp->colgc.bg) ||
style_equal(&grid_default_cell, &wp->colgc))
wp->flags |= PANE_REDRAW; wp->flags |= PANE_REDRAW;
} }
@ -832,6 +838,7 @@ window_pane_destroy(struct window_pane *wp)
free((void *)wp->cwd); free((void *)wp->cwd);
free(wp->shell); free(wp->shell);
cmd_free_argv(wp->argc, wp->argv); cmd_free_argv(wp->argc, wp->argv);
free(wp->palette);
free(wp); free(wp);
} }
@ -1107,6 +1114,40 @@ window_pane_alternate_off(struct window_pane *wp, struct grid_cell *gc,
wp->flags |= PANE_REDRAW; wp->flags |= PANE_REDRAW;
} }
void
window_pane_set_palette(struct window_pane *wp, u_int n, int colour)
{
if (n > 0xff)
return;
if (wp->palette == NULL)
wp->palette = xcalloc(0x100, sizeof *wp->palette);
wp->palette[n] = colour;
wp->flags |= PANE_REDRAW;
}
void
window_pane_unset_palette(struct window_pane *wp, u_int n)
{
if (n > 0xff || wp->palette == NULL)
return;
wp->palette[n] = 0;
wp->flags |= PANE_REDRAW;
}
void
window_pane_reset_palette(struct window_pane *wp)
{
if (wp->palette == NULL)
return;
free(wp->palette);
wp->palette = NULL;
wp->flags |= PANE_REDRAW;
}
static void static void
window_pane_mode_timer(__unused int fd, __unused short events, void *arg) window_pane_mode_timer(__unused int fd, __unused short events, void *arg)
{ {