mirror of
https://github.com/tmux/tmux.git
synced 2025-01-12 03:08:46 +00:00
Extend the menu drawing function to support custom characters and
styles, from Alexis Hildebrandt.
This commit is contained in:
parent
2b535bc173
commit
1071ef8fc5
3
menu.c
3
menu.c
@ -203,7 +203,8 @@ menu_draw_cb(struct client *c, void *data,
|
|||||||
|
|
||||||
screen_write_start(&ctx, s);
|
screen_write_start(&ctx, s);
|
||||||
screen_write_clearscreen(&ctx, 8);
|
screen_write_clearscreen(&ctx, 8);
|
||||||
screen_write_menu(&ctx, menu, md->choice, &gc);
|
screen_write_menu(&ctx, menu, md->choice, BOX_LINES_DEFAULT,
|
||||||
|
&grid_default_cell, &grid_default_cell, &gc);
|
||||||
screen_write_stop(&ctx);
|
screen_write_stop(&ctx);
|
||||||
|
|
||||||
for (i = 0; i < screen_size_y(&md->s); i++) {
|
for (i = 0; i < screen_size_y(&md->s); i++) {
|
||||||
|
130
screen-write.c
130
screen-write.c
@ -30,7 +30,6 @@ static void screen_write_collect_clear(struct screen_write_ctx *, u_int,
|
|||||||
static void screen_write_collect_scroll(struct screen_write_ctx *, u_int);
|
static void screen_write_collect_scroll(struct screen_write_ctx *, u_int);
|
||||||
static void screen_write_collect_flush(struct screen_write_ctx *, int,
|
static void screen_write_collect_flush(struct screen_write_ctx *, int,
|
||||||
const char *);
|
const char *);
|
||||||
|
|
||||||
static int screen_write_overwrite(struct screen_write_ctx *,
|
static int screen_write_overwrite(struct screen_write_ctx *,
|
||||||
struct grid_cell *, u_int);
|
struct grid_cell *, u_int);
|
||||||
static const struct grid_cell *screen_write_combine(struct screen_write_ctx *,
|
static const struct grid_cell *screen_write_combine(struct screen_write_ctx *,
|
||||||
@ -592,9 +591,46 @@ screen_write_fast_copy(struct screen_write_ctx *ctx, struct screen *src,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Select character set for drawing border lines. */
|
||||||
|
static void
|
||||||
|
screen_write_box_border_set(enum box_lines lines, int cell_type,
|
||||||
|
struct grid_cell *gc)
|
||||||
|
{
|
||||||
|
switch (lines) {
|
||||||
|
case BOX_LINES_NONE:
|
||||||
|
break;
|
||||||
|
case BOX_LINES_DOUBLE:
|
||||||
|
gc->attr &= ~GRID_ATTR_CHARSET;
|
||||||
|
utf8_copy(&gc->data, tty_acs_double_borders(cell_type));
|
||||||
|
break;
|
||||||
|
case BOX_LINES_HEAVY:
|
||||||
|
gc->attr &= ~GRID_ATTR_CHARSET;
|
||||||
|
utf8_copy(&gc->data, tty_acs_heavy_borders(cell_type));
|
||||||
|
break;
|
||||||
|
case BOX_LINES_ROUNDED:
|
||||||
|
gc->attr &= ~GRID_ATTR_CHARSET;
|
||||||
|
utf8_copy(&gc->data, tty_acs_rounded_borders(cell_type));
|
||||||
|
break;
|
||||||
|
case BOX_LINES_SIMPLE:
|
||||||
|
gc->attr &= ~GRID_ATTR_CHARSET;
|
||||||
|
utf8_set(&gc->data, SIMPLE_BORDERS[cell_type]);
|
||||||
|
break;
|
||||||
|
case BOX_LINES_PADDED:
|
||||||
|
gc->attr &= ~GRID_ATTR_CHARSET;
|
||||||
|
utf8_set(&gc->data, PADDED_BORDERS[cell_type]);
|
||||||
|
break;
|
||||||
|
case BOX_LINES_SINGLE:
|
||||||
|
case BOX_LINES_DEFAULT:
|
||||||
|
gc->attr |= GRID_ATTR_CHARSET;
|
||||||
|
utf8_set(&gc->data, CELL_BORDERS[cell_type]);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Draw a horizontal line on screen. */
|
/* Draw a horizontal line on screen. */
|
||||||
void
|
void
|
||||||
screen_write_hline(struct screen_write_ctx *ctx, u_int nx, int left, int right)
|
screen_write_hline(struct screen_write_ctx *ctx, u_int nx, int left, int right,
|
||||||
|
enum box_lines lines, const struct grid_cell *border_gc)
|
||||||
{
|
{
|
||||||
struct screen *s = ctx->s;
|
struct screen *s = ctx->s;
|
||||||
struct grid_cell gc;
|
struct grid_cell gc;
|
||||||
@ -603,13 +639,27 @@ screen_write_hline(struct screen_write_ctx *ctx, u_int nx, int left, int right)
|
|||||||
cx = s->cx;
|
cx = s->cx;
|
||||||
cy = s->cy;
|
cy = s->cy;
|
||||||
|
|
||||||
|
if (border_gc != NULL)
|
||||||
|
memcpy(&gc, border_gc, sizeof gc);
|
||||||
|
else
|
||||||
memcpy(&gc, &grid_default_cell, sizeof gc);
|
memcpy(&gc, &grid_default_cell, sizeof gc);
|
||||||
gc.attr |= GRID_ATTR_CHARSET;
|
gc.attr |= GRID_ATTR_CHARSET;
|
||||||
|
|
||||||
screen_write_putc(ctx, &gc, left ? 't' : 'q');
|
if (left)
|
||||||
|
screen_write_box_border_set(lines, CELL_LEFTJOIN, &gc);
|
||||||
|
else
|
||||||
|
screen_write_box_border_set(lines, CELL_LEFTRIGHT, &gc);
|
||||||
|
screen_write_cell(ctx, &gc);
|
||||||
|
|
||||||
|
screen_write_box_border_set(lines, CELL_LEFTRIGHT, &gc);
|
||||||
for (i = 1; i < nx - 1; i++)
|
for (i = 1; i < nx - 1; i++)
|
||||||
screen_write_putc(ctx, &gc, 'q');
|
screen_write_cell(ctx, &gc);
|
||||||
screen_write_putc(ctx, &gc, right ? 'u' : 'q');
|
|
||||||
|
if (right)
|
||||||
|
screen_write_box_border_set(lines, CELL_RIGHTJOIN, &gc);
|
||||||
|
else
|
||||||
|
screen_write_box_border_set(lines, CELL_LEFTRIGHT, &gc);
|
||||||
|
screen_write_cell(ctx, &gc);
|
||||||
|
|
||||||
screen_write_set_cursor(ctx, cx, cy);
|
screen_write_set_cursor(ctx, cx, cy);
|
||||||
}
|
}
|
||||||
@ -641,86 +691,54 @@ screen_write_vline(struct screen_write_ctx *ctx, u_int ny, int top, int bottom)
|
|||||||
|
|
||||||
/* Draw a menu on screen. */
|
/* Draw a menu on screen. */
|
||||||
void
|
void
|
||||||
screen_write_menu(struct screen_write_ctx *ctx, struct menu *menu,
|
screen_write_menu(struct screen_write_ctx *ctx, struct menu *menu, int choice,
|
||||||
int choice, const struct grid_cell *choice_gc)
|
enum box_lines lines, const struct grid_cell *menu_gc,
|
||||||
|
const struct grid_cell *border_gc, const struct grid_cell *choice_gc)
|
||||||
{
|
{
|
||||||
struct screen *s = ctx->s;
|
struct screen *s = ctx->s;
|
||||||
struct grid_cell default_gc;
|
struct grid_cell default_gc;
|
||||||
const struct grid_cell *gc = &default_gc;
|
const struct grid_cell *gc = &default_gc;
|
||||||
u_int cx, cy, i, j;
|
u_int cx, cy, i, j, width = menu->width;
|
||||||
const char *name;
|
const char *name;
|
||||||
|
|
||||||
cx = s->cx;
|
cx = s->cx;
|
||||||
cy = s->cy;
|
cy = s->cy;
|
||||||
|
|
||||||
memcpy(&default_gc, &grid_default_cell, sizeof default_gc);
|
memcpy(&default_gc, menu_gc, sizeof default_gc);
|
||||||
|
|
||||||
screen_write_box(ctx, menu->width + 4, menu->count + 2,
|
screen_write_box(ctx, menu->width + 4, menu->count + 2, lines,
|
||||||
BOX_LINES_DEFAULT, &default_gc, menu->title);
|
border_gc, menu->title);
|
||||||
|
|
||||||
for (i = 0; i < menu->count; i++) {
|
for (i = 0; i < menu->count; i++) {
|
||||||
name = menu->items[i].name;
|
name = menu->items[i].name;
|
||||||
if (name == NULL) {
|
if (name == NULL) {
|
||||||
screen_write_cursormove(ctx, cx, cy + 1 + i, 0);
|
screen_write_cursormove(ctx, cx, cy + 1 + i, 0);
|
||||||
screen_write_hline(ctx, menu->width + 4, 1, 1);
|
screen_write_hline(ctx, width + 4, 1, 1, lines, gc);
|
||||||
} else {
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if (choice >= 0 && i == (u_int)choice && *name != '-')
|
if (choice >= 0 && i == (u_int)choice && *name != '-')
|
||||||
gc = choice_gc;
|
gc = choice_gc;
|
||||||
|
|
||||||
screen_write_cursormove(ctx, cx + 2, cy + 1 + i, 0);
|
screen_write_cursormove(ctx, cx + 2, cy + 1 + i, 0);
|
||||||
for (j = 0; j < menu->width; j++)
|
for (j = 0; j < width; j++)
|
||||||
screen_write_putc(ctx, gc, ' ');
|
screen_write_putc(ctx, gc, ' ');
|
||||||
|
|
||||||
screen_write_cursormove(ctx, cx + 2, cy + 1 + i, 0);
|
screen_write_cursormove(ctx, cx + 2, cy + 1 + i, 0);
|
||||||
if (*name == '-') {
|
if (*name == '-') {
|
||||||
name++;
|
|
||||||
default_gc.attr |= GRID_ATTR_DIM;
|
default_gc.attr |= GRID_ATTR_DIM;
|
||||||
format_draw(ctx, gc, menu->width, name, NULL,
|
format_draw(ctx, gc, width, name + 1, NULL, 0);
|
||||||
0);
|
|
||||||
default_gc.attr &= ~GRID_ATTR_DIM;
|
default_gc.attr &= ~GRID_ATTR_DIM;
|
||||||
} else
|
continue;
|
||||||
format_draw(ctx, gc, menu->width, name, NULL,
|
|
||||||
gc == choice_gc);
|
|
||||||
gc = &default_gc;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
format_draw(ctx, gc, width, name, NULL, gc == choice_gc);
|
||||||
|
gc = &default_gc;
|
||||||
}
|
}
|
||||||
|
|
||||||
screen_write_set_cursor(ctx, cx, cy);
|
screen_write_set_cursor(ctx, cx, cy);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void
|
|
||||||
screen_write_box_border_set(enum box_lines box_lines, int cell_type,
|
|
||||||
struct grid_cell *gc)
|
|
||||||
{
|
|
||||||
switch (box_lines) {
|
|
||||||
case BOX_LINES_NONE:
|
|
||||||
break;
|
|
||||||
case BOX_LINES_DOUBLE:
|
|
||||||
gc->attr &= ~GRID_ATTR_CHARSET;
|
|
||||||
utf8_copy(&gc->data, tty_acs_double_borders(cell_type));
|
|
||||||
break;
|
|
||||||
case BOX_LINES_HEAVY:
|
|
||||||
gc->attr &= ~GRID_ATTR_CHARSET;
|
|
||||||
utf8_copy(&gc->data, tty_acs_heavy_borders(cell_type));
|
|
||||||
break;
|
|
||||||
case BOX_LINES_ROUNDED:
|
|
||||||
gc->attr &= ~GRID_ATTR_CHARSET;
|
|
||||||
utf8_copy(&gc->data, tty_acs_rounded_borders(cell_type));
|
|
||||||
break;
|
|
||||||
case BOX_LINES_SIMPLE:
|
|
||||||
gc->attr &= ~GRID_ATTR_CHARSET;
|
|
||||||
utf8_set(&gc->data, SIMPLE_BORDERS[cell_type]);
|
|
||||||
break;
|
|
||||||
case BOX_LINES_PADDED:
|
|
||||||
gc->attr &= ~GRID_ATTR_CHARSET;
|
|
||||||
utf8_set(&gc->data, PADDED_BORDERS[cell_type]);
|
|
||||||
break;
|
|
||||||
case BOX_LINES_SINGLE:
|
|
||||||
case BOX_LINES_DEFAULT:
|
|
||||||
gc->attr |= GRID_ATTR_CHARSET;
|
|
||||||
utf8_set(&gc->data, CELL_BORDERS[cell_type]);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Draw a box on screen. */
|
/* Draw a box on screen. */
|
||||||
void
|
void
|
||||||
screen_write_box(struct screen_write_ctx *ctx, u_int nx, u_int ny,
|
screen_write_box(struct screen_write_ctx *ctx, u_int nx, u_int ny,
|
||||||
|
4
tmux.h
4
tmux.h
@ -2889,9 +2889,11 @@ void screen_write_putc(struct screen_write_ctx *, const struct grid_cell *,
|
|||||||
u_char);
|
u_char);
|
||||||
void screen_write_fast_copy(struct screen_write_ctx *, struct screen *,
|
void screen_write_fast_copy(struct screen_write_ctx *, struct screen *,
|
||||||
u_int, u_int, u_int, u_int);
|
u_int, u_int, u_int, u_int);
|
||||||
void screen_write_hline(struct screen_write_ctx *, u_int, int, int);
|
void screen_write_hline(struct screen_write_ctx *, u_int, int, int,
|
||||||
|
enum box_lines, const struct grid_cell *);
|
||||||
void screen_write_vline(struct screen_write_ctx *, u_int, int, int);
|
void screen_write_vline(struct screen_write_ctx *, u_int, int, int);
|
||||||
void screen_write_menu(struct screen_write_ctx *, struct menu *, int,
|
void screen_write_menu(struct screen_write_ctx *, struct menu *, int,
|
||||||
|
enum box_lines, const struct grid_cell *, const struct grid_cell *,
|
||||||
const struct grid_cell *);
|
const struct grid_cell *);
|
||||||
void screen_write_box(struct screen_write_ctx *, u_int, u_int,
|
void screen_write_box(struct screen_write_ctx *, u_int, u_int,
|
||||||
enum box_lines, const struct grid_cell *, const char *);
|
enum box_lines, const struct grid_cell *, const char *);
|
||||||
|
@ -242,7 +242,7 @@ window_client_draw(__unused void *modedata, void *itemdata,
|
|||||||
screen_write_cursormove(ctx, cx, cy + 2, 0);
|
screen_write_cursormove(ctx, cx, cy + 2, 0);
|
||||||
else
|
else
|
||||||
screen_write_cursormove(ctx, cx, cy + sy - 1 - lines, 0);
|
screen_write_cursormove(ctx, cx, cy + sy - 1 - lines, 0);
|
||||||
screen_write_hline(ctx, sx, 0, 0);
|
screen_write_hline(ctx, sx, 0, 0, BOX_LINES_DEFAULT, NULL);
|
||||||
|
|
||||||
if (at != 0)
|
if (at != 0)
|
||||||
screen_write_cursormove(ctx, cx, cy, 0);
|
screen_write_cursormove(ctx, cx, cy, 0);
|
||||||
|
Loading…
Reference in New Issue
Block a user