-
Notifications
You must be signed in to change notification settings - Fork 263
/
ui-terminal-vt100.c
223 lines (192 loc) · 6.36 KB
/
ui-terminal-vt100.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
/* This file is included from ui-terminal.c
*
* The goal is *not* to reimplement curses. Instead we aim to provide the
* simplest possible drawing backend for VT-100 compatible terminals.
* This is useful for debugging and fuzzing purposes as well as for environments
* with no curses support.
*
* Currently no attempt is made to optimize terminal output. The amount of
* flickering will depend on the smartness of your terminal emulator.
*
* The following terminal escape sequences are used:
*
* - CSI ? 1049 h Save cursor and use Alternate Screen Buffer (DECSET)
* - CSI ? 1049 l Use Normal Screen Buffer and restore cursor (DECRST)
* - CSI ? 25 l Hide Cursor (DECTCEM)
* - CSI ? 25 h Show Cursor (DECTCEM)
* - CSI 2 J Erase in Display (ED)
* - CSI row ; column H Cursor Position (CUP)
* - CSI ... m Character Attributes (SGR)
* - CSI 0 m Normal
* - CSI 1 m Bold
* - CSI 3 m Italicized
* - CSI 4 m Underlined
* - CSI 5 m Blink
* - CSI 7 m Inverse
* - CSI 22 m Normal (not bold)
* - CSI 23 m Not italicized
* - CSI 24 m Not underlined
* - CSI 25 m Not blinking
* - CSI 27 m Not inverse
* - CSI 30-37,39 Set foreground color
* - CSI 38 ; 2 ; R ; G ; B m Set RGB foreground color
* - CSI 40-47,49 Set background color
* - CSI 48 ; 2 ; R ; G ; B m Set RGB background color
*
* See http://invisible-island.net/xterm/ctlseqs/ctlseqs.txt
* for further information.
*/
#include <stdio.h>
#include "buffer.h"
#define UI_TERMKEY_FLAGS TERMKEY_FLAG_UTF8
#define CELL_COLOR_BLACK { .index = 0 }
#define CELL_COLOR_RED { .index = 1 }
#define CELL_COLOR_GREEN { .index = 2 }
#define CELL_COLOR_YELLOW { .index = 3 }
#define CELL_COLOR_BLUE { .index = 4 }
#define CELL_COLOR_MAGENTA { .index = 5 }
#define CELL_COLOR_CYAN { .index = 6 }
#define CELL_COLOR_WHITE { .index = 7 }
#define CELL_COLOR_DEFAULT { .index = 9 }
#define CELL_ATTR_NORMAL 0
#define CELL_ATTR_UNDERLINE (1 << 0)
#define CELL_ATTR_REVERSE (1 << 1)
#define CELL_ATTR_BLINK (1 << 2)
#define CELL_ATTR_BOLD (1 << 3)
#define CELL_ATTR_ITALIC (1 << 4)
#define CELL_ATTR_DIM (1 << 5)
static inline bool cell_color_equal(CellColor c1, CellColor c2) {
if (c1.index != (uint8_t)-1 || c2.index != (uint8_t)-1)
return c1.index == c2.index;
return c1.r == c2.r && c1.g == c2.g && c1.b == c2.b;
}
static CellColor color_rgb(Ui *ui, uint8_t r, uint8_t g, uint8_t b) {
return (CellColor){ .r = r, .g = g, .b = b, .index = (uint8_t)-1 };
}
static CellColor color_terminal(Ui *ui, uint8_t index) {
return (CellColor){ .r = 0, .g = 0, .b = 0, .index = index };
}
static void output(const char *data, size_t len) {
fwrite(data, len, 1, stderr);
fflush(stderr);
}
static void output_literal(const char *data) {
output(data, strlen(data));
}
static void screen_alternate(bool alternate) {
output_literal(alternate ? "\x1b[?1049h" : "\x1b[0m" "\x1b[?1049l" "\x1b[0m" );
}
static void cursor_visible(bool visible) {
output_literal(visible ? "\x1b[?25h" : "\x1b[?25l");
}
static void ui_term_backend_blit(Ui *tui) {
Buffer *buf = tui->ctx;
buffer_clear(buf);
CellAttr attr = CELL_ATTR_NORMAL;
CellColor fg = CELL_COLOR_DEFAULT, bg = CELL_COLOR_DEFAULT;
int w = tui->width, h = tui->height;
Cell *cell = tui->cells;
/* reposition cursor, erase screen, reset attributes */
buffer_append0(buf, "\x1b[H" "\x1b[J" "\x1b[0m");
for (int y = 0; y < h; y++) {
for (int x = 0; x < w; x++) {
CellStyle *style = &cell->style;
if (style->attr != attr) {
static const struct {
CellAttr attr;
char on[4], off[4];
} cell_attrs[] = {
{ CELL_ATTR_BOLD, "1", "22" },
{ CELL_ATTR_DIM, "2", "22" },
{ CELL_ATTR_ITALIC, "3", "23" },
{ CELL_ATTR_UNDERLINE, "4", "24" },
{ CELL_ATTR_BLINK, "5", "25" },
{ CELL_ATTR_REVERSE, "7", "27" },
};
for (size_t i = 0; i < LENGTH(cell_attrs); i++) {
CellAttr a = cell_attrs[i].attr;
if ((style->attr & a) == (attr & a))
continue;
buffer_appendf(buf, "\x1b[%sm",
style->attr & a ?
cell_attrs[i].on :
cell_attrs[i].off);
}
attr = style->attr;
}
if (!cell_color_equal(fg, style->fg)) {
fg = style->fg;
if (fg.index != (uint8_t)-1) {
buffer_appendf(buf, "\x1b[%dm", 30 + fg.index);
} else {
buffer_appendf(buf, "\x1b[38;2;%d;%d;%dm",
fg.r, fg.g, fg.b);
}
}
if (!cell_color_equal(bg, style->bg)) {
bg = style->bg;
if (bg.index != (uint8_t)-1) {
buffer_appendf(buf, "\x1b[%dm", 40 + bg.index);
} else {
buffer_appendf(buf, "\x1b[48;2;%d;%d;%dm",
bg.r, bg.g, bg.b);
}
}
buffer_append0(buf, cell->data);
cell++;
}
}
output(buffer_content(buf), buffer_length0(buf));
}
static void ui_term_backend_clear(Ui *tui) { }
static bool ui_term_backend_resize(Ui *tui, int width, int height) {
return true;
}
static void ui_term_backend_save(Ui *tui, bool fscr) {
cursor_visible(true);
}
static void ui_term_backend_restore(Ui *tui) {
cursor_visible(false);
}
int ui_terminal_colors(void) {
char *term = getenv("TERM");
return (term && strstr(term, "-256color")) ? 256 : 16;
}
static void ui_term_backend_suspend(Ui *tui) {
if (!tui->termkey) return;
termkey_stop(tui->termkey);
cursor_visible(true);
screen_alternate(false);
}
void ui_terminal_resume(Ui *tui) {
screen_alternate(true);
cursor_visible(false);
termkey_start(tui->termkey);
}
static bool ui_term_backend_init(Ui *tui, char *term) {
ui_terminal_resume(tui);
return true;
}
static bool ui_backend_init(Ui *ui) {
Buffer *buf = calloc(1, sizeof(Buffer));
if (!buf)
return false;
buffer_init(buf);
ui->ctx = buf;
return true;
}
static void ui_term_backend_free(Ui *tui) {
Buffer *buf = tui->ctx;
ui_term_backend_suspend(tui);
buffer_release(buf);
free(buf);
}
static bool is_default_color(CellColor c) {
return c.index == ((CellColor) CELL_COLOR_DEFAULT).index;
}
static bool is_default_fg(CellColor c) {
return is_default_color(c);
}
static bool is_default_bg(CellColor c) {
return is_default_color(c);
}