blob: 5285d94ec0ba005047146a555178fa07c973785c [file] [log] [blame]
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001#include "vterm.h"
Bram Moolenaar707d2262019-12-04 22:16:54 +01002#include "../src/vterm_internal.h" // We pull in some internal bits too
Bram Moolenaare4f25e42017-07-07 11:54:15 +02003
Bram Moolenaar6a12d262022-10-16 19:26:52 +01004#include <assert.h>
Bram Moolenaare4f25e42017-07-07 11:54:15 +02005#include <stdio.h>
6#include <string.h>
7
8#define streq(a,b) (!strcmp(a,b))
9#define strstartswith(a,b) (!strncmp(a,b,strlen(b)))
10
11static size_t inplace_hex2bytes(char *s)
12{
13 char *inpos = s, *outpos = s;
14
15 while(*inpos) {
16 unsigned int ch;
Bram Moolenaar7da34152021-11-24 19:30:55 +000017 if(sscanf(inpos, "%2x", &ch) < 1)
18 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +020019 *outpos = ch;
20 outpos += 1; inpos += 2;
21 }
22
23 return outpos - s;
24}
25
26static VTermModifier strpe_modifiers(char **strp)
27{
28 VTermModifier state = 0;
29
30 while((*strp)[0]) {
31 switch(((*strp)++)[0]) {
32 case 'S': state |= VTERM_MOD_SHIFT; break;
33 case 'C': state |= VTERM_MOD_CTRL; break;
34 case 'A': state |= VTERM_MOD_ALT; break;
35 default: return state;
36 }
37 }
38
39 return state;
40}
41
42static VTermKey strp_key(char *str)
43{
44 static struct {
45 char *name;
46 VTermKey key;
47 } keys[] = {
48 { "Up", VTERM_KEY_UP },
49 { "Tab", VTERM_KEY_TAB },
50 { "Enter", VTERM_KEY_ENTER },
51 { "KP0", VTERM_KEY_KP_0 },
Bram Moolenaarc4c9f7e2020-05-17 20:52:45 +020052 { "F1", VTERM_KEY_FUNCTION(1) },
Bram Moolenaare4f25e42017-07-07 11:54:15 +020053 { NULL, VTERM_KEY_NONE },
54 };
55 int i;
56
57 for(i = 0; keys[i].name; i++) {
58 if(streq(str, keys[i].name))
59 return keys[i].key;
60 }
61
62 return VTERM_KEY_NONE;
63}
64
Bram Moolenaare5886cc2020-05-21 20:10:04 +020065static void print_color(const VTermColor *col)
66{
67 if (VTERM_COLOR_IS_RGB(col)) {
68 printf("rgb(%d,%d,%d", col->red, col->green, col->blue);
69 }
70 else if (VTERM_COLOR_IS_INDEXED(col)) {
71 printf("idx(%d", col->index);
72 }
73 else {
74 printf("invalid(%d", col->type);
75 }
76 if (VTERM_COLOR_IS_DEFAULT_FG(col)) {
77 printf(",is_default_fg");
78 }
79 if (VTERM_COLOR_IS_DEFAULT_BG(col)) {
80 printf(",is_default_bg");
81 }
82 printf(")");
83}
84
zeertzjqb00df7a2023-08-08 11:03:00 +080085static VTermColor strpe_color(char **strp)
86{
87 uint8_t r, g, b, idx;
88 int len = 0;
89 VTermColor col;
90
91 if(sscanf(*strp, "rgb(%hhu,%hhu,%hhu)%n", &r, &g, &b, &len) == 3 && len > 0) {
92 *strp += len;
93 vterm_color_rgb(&col, r, g, b);
94 }
95 else if(sscanf(*strp, "idx(%hhu)%n", &idx, &len) == 1 && len > 0) {
96 *strp += len;
97 vterm_color_indexed(&col, idx);
98 }
99 else
100 vterm_color_rgb(&col, 127, 127, 127);
101
102 return col;
103}
104
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200105static VTerm *vt;
106static VTermState *state;
107static VTermScreen *screen;
108
109static VTermEncodingInstance encoding;
110
Bram Moolenaarcd630be2020-04-12 14:50:26 +0200111static void term_output(const char *s, size_t len, void *user UNUSED)
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200112{
113 size_t i;
114
115 printf("output ");
116 for(i = 0; i < len; i++)
117 printf("%x%s", (unsigned char)s[i], i < len-1 ? "," : "\n");
118}
119
120static void printhex(const char *s, size_t len)
121{
122 while(len--)
Bram Moolenaar7da34152021-11-24 19:30:55 +0000123 printf("%02x", (uint8_t)(s++)[0]);
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200124}
125
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200126static int parser_text(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200127{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200128 size_t i;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200129
130 printf("text ");
131 for(i = 0; i < len; i++) {
132 unsigned char b = bytes[i];
133 if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
134 break;
135 printf(i ? ",%x" : "%x", b);
136 }
137 printf("\n");
138
139 return i;
140}
141
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200142static int parser_control(unsigned char control, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200143{
144 printf("control %02x\n", control);
145
146 return 1;
147}
148
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200149static int parser_escape(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200150{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200151 if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
152 if(len < 2)
153 return -1;
154 len = 2;
155 }
156 else {
157 len = 1;
158 }
159
160 printf("escape ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200161 printhex(bytes, len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200162 printf("\n");
163
164 return len;
165}
166
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200167static int parser_csi(const char *leader, const long args[], int argcount, const char *intermed, char command, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200168{
169 int i;
170 printf("csi %02x", command);
171
172 if(leader && leader[0]) {
173 printf(" L=");
174 for(i = 0; leader[i]; i++)
175 printf("%02x", leader[i]);
176 }
177
178 for(i = 0; i < argcount; i++) {
179 char sep = i ? ',' : ' ';
180
181 if(args[i] == CSI_ARG_MISSING)
182 printf("%c*", sep);
183 else
184 printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
185 }
186
187 if(intermed && intermed[0]) {
188 printf(" I=");
189 for(i = 0; intermed[i]; i++)
190 printf("%02x", intermed[i]);
191 }
192
193 printf("\n");
194
195 return 1;
196}
197
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200198static int parser_osc(int command, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200199{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200200
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200201 printf("osc ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200202
203 if(frag.initial) {
204 if(command == -1)
205 printf("[");
206 else
207 printf("[%d;", command);
208 }
209
210 printhex(frag.str, frag.len);
211
212 if(frag.final)
213 printf("]");
214
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200215 printf("\n");
216
217 return 1;
218}
219
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200220static int parser_dcs(const char *command, size_t commandlen, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200221{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200222 printf("dcs ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200223
224 if(frag.initial) {
225 size_t i;
226 printf("[");
227 for(i = 0; i < commandlen; i++)
228 printf("%02x", command[i]);
229 }
230
231 printhex(frag.str, frag.len);
232
233 if(frag.final)
234 printf("]");
235
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200236 printf("\n");
237
238 return 1;
239}
240
Bram Moolenaar7da34152021-11-24 19:30:55 +0000241static int parser_apc(VTermStringFragment frag, void *user UNUSED)
242{
243 printf("apc ");
244
245 if(frag.initial)
246 printf("[");
247
248 printhex(frag.str, frag.len);
249
250 if(frag.final)
251 printf("]");
252
253 printf("\n");
254
255 return 1;
256}
257
258static int parser_pm(VTermStringFragment frag, void *user UNUSED)
259{
260 printf("pm ");
261
262 if(frag.initial)
263 printf("[");
264
265 printhex(frag.str, frag.len);
266
267 if(frag.final)
268 printf("]");
269
270 printf("\n");
271
272 return 1;
273}
274
275static int parser_sos(VTermStringFragment frag, void *user UNUSED)
276{
277 printf("sos ");
278
279 if(frag.initial)
280 printf("[");
281
282 printhex(frag.str, frag.len);
283
284 if(frag.final)
285 printf("]");
286
287 printf("\n");
288
289 return 1;
290}
291
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200292static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100293 parser_text, // text
294 parser_control, // control
295 parser_escape, // escape
296 parser_csi, // csi
297 parser_osc, // osc
298 parser_dcs, // dcs
Bram Moolenaar7da34152021-11-24 19:30:55 +0000299 parser_apc, // apc
300 parser_pm, // pm
301 parser_sos, // sos
Bram Moolenaar707d2262019-12-04 22:16:54 +0100302 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200303};
304
Bram Moolenaard8637282020-05-20 18:41:41 +0200305static VTermStateFallbacks fallbacks = {
306 parser_control, // control
307 parser_csi, // csi
308 parser_osc, // osc
Bram Moolenaar7da34152021-11-24 19:30:55 +0000309 parser_dcs, // dcs
310 parser_apc, // dcs
311 parser_pm, // pm
312 parser_sos // sos
Bram Moolenaard8637282020-05-20 18:41:41 +0200313};
314
315/* These callbacks are shared by State and Screen */
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200316
317static int want_movecursor = 0;
318static VTermPos state_pos;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200319static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200320{
321 state_pos = pos;
322
323 if(want_movecursor)
324 printf("movecursor %d,%d\n", pos.row, pos.col);
325
326 return 1;
327}
328
329static int want_scrollrect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200330static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200331{
332 if(!want_scrollrect)
333 return 0;
334
335 printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
336 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
337 downward, rightward);
338
339 return 1;
340}
341
342static int want_moverect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200343static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200344{
345 if(!want_moverect)
346 return 0;
347
348 printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
349 src.start_row, src.end_row, src.start_col, src.end_col,
350 dest.start_row, dest.end_row, dest.start_col, dest.end_col);
351
352 return 1;
353}
354
355static int want_settermprop = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200356static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200357{
358 VTermValueType type;
359 if(!want_settermprop)
360 return 1;
361
362 type = vterm_get_prop_type(prop);
363 switch(type) {
364 case VTERM_VALUETYPE_BOOL:
365 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
366 return 1;
367 case VTERM_VALUETYPE_INT:
368 printf("settermprop %d %d\n", prop, val->number);
369 return 1;
370 case VTERM_VALUETYPE_STRING:
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200371 printf("settermprop %d %s\"%.*s\"%s\n", prop,
372 val->string.initial ? "[" : "", val->string.len, val->string.str, val->string.final ? "]" : "");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200373 return 1;
374 case VTERM_VALUETYPE_COLOR:
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200375 printf("settermprop %d ", prop);
376 print_color(&val->color);
377 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200378 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200379
380 case VTERM_N_VALUETYPES:
381 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200382 }
383
384 return 0;
385}
386
Bram Moolenaar707d2262019-12-04 22:16:54 +0100387// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200388
389static int want_state_putglyph = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200390static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200391{
392 int i;
393 if(!want_state_putglyph)
394 return 1;
395
396 printf("putglyph ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200397 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && info->chars[i]; i++)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200398 printf(i ? ",%x" : "%x", info->chars[i]);
399 printf(" %d %d,%d", info->width, pos.row, pos.col);
400 if(info->protected_cell)
401 printf(" prot");
402 if(info->dwl)
403 printf(" dwl");
404 if(info->dhl)
405 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
406 printf("\n");
407
408 return 1;
409}
410
411static int want_state_erase = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200412static int state_erase(VTermRect rect, int selective, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200413{
414 if(!want_state_erase)
415 return 1;
416
417 printf("erase %d..%d,%d..%d%s\n",
418 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
419 selective ? " selective" : "");
420
421 return 1;
422}
423
424static struct {
425 int bold;
426 int underline;
427 int italic;
428 int blink;
429 int reverse;
Bram Moolenaard8637282020-05-20 18:41:41 +0200430 int conceal;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200431 int strike;
432 int font;
Bram Moolenaare6a16e92022-10-17 14:51:36 +0100433 int small;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100434 int baseline;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200435 VTermColor foreground;
436 VTermColor background;
437} state_pen;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200438static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200439{
440 switch(attr) {
441 case VTERM_ATTR_BOLD:
442 state_pen.bold = val->boolean;
443 break;
444 case VTERM_ATTR_UNDERLINE:
445 state_pen.underline = val->number;
446 break;
447 case VTERM_ATTR_ITALIC:
448 state_pen.italic = val->boolean;
449 break;
450 case VTERM_ATTR_BLINK:
451 state_pen.blink = val->boolean;
452 break;
453 case VTERM_ATTR_REVERSE:
454 state_pen.reverse = val->boolean;
455 break;
Bram Moolenaard8637282020-05-20 18:41:41 +0200456 case VTERM_ATTR_CONCEAL:
457 state_pen.conceal = val->boolean;
458 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200459 case VTERM_ATTR_STRIKE:
460 state_pen.strike = val->boolean;
461 break;
462 case VTERM_ATTR_FONT:
463 state_pen.font = val->number;
464 break;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100465 case VTERM_ATTR_SMALL:
Bram Moolenaare6a16e92022-10-17 14:51:36 +0100466 state_pen.small = val->boolean;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100467 break;
468 case VTERM_ATTR_BASELINE:
469 state_pen.baseline = val->number;
470 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200471 case VTERM_ATTR_FOREGROUND:
472 state_pen.foreground = val->color;
473 break;
474 case VTERM_ATTR_BACKGROUND:
475 state_pen.background = val->color;
476 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200477
478 case VTERM_N_ATTRS:
479 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200480 }
481
482 return 1;
483}
484
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200485static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200486{
487 return 1;
488}
489
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100490static int want_state_scrollback = 0;
491static int state_sb_clear(void *user UNUSED) {
492 if(!want_state_scrollback)
493 return 1;
494
495 printf("sb_clear\n");
496 return 0;
497}
498
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200499VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100500 state_putglyph, // putglyph
501 movecursor, // movecursor
502 scrollrect, // scrollrect
503 moverect, // moverect
504 state_erase, // erase
505 NULL, // initpen
506 state_setpenattr, // setpenattr
507 settermprop, // settermprop
508 NULL, // bell
509 NULL, // resize
510 state_setlineinfo, // setlineinfo
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100511 state_sb_clear, // sb_clear
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200512};
513
Bram Moolenaar7da34152021-11-24 19:30:55 +0000514static int selection_set(VTermSelectionMask mask, VTermStringFragment frag, void *user UNUSED)
515{
516 printf("selection-set mask=%04X ", mask);
517 if(frag.initial)
518 printf("[");
519 printhex(frag.str, frag.len);
520 if(frag.final)
521 printf("]");
522 printf("\n");
523
524 return 1;
525}
526
527static int selection_query(VTermSelectionMask mask, void *user UNUSED)
528{
529 printf("selection-query mask=%04X\n", mask);
530
531 return 1;
532}
533
534VTermSelectionCallbacks selection_cbs = {
535 .set = selection_set,
536 .query = selection_query,
537};
538
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200539static int want_screen_damage = 0;
540static int want_screen_damage_cells = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200541static int screen_damage(VTermRect rect, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200542{
543 if(!want_screen_damage)
544 return 1;
545
546 printf("damage %d..%d,%d..%d",
547 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
548
549 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200550 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200551 int row;
552 int col;
553
554 for(row = rect.start_row; row < rect.end_row; row++) {
555 int eol = rect.end_col;
556 while(eol > rect.start_col) {
557 VTermScreenCell cell;
558 VTermPos pos;
559 pos.row = row;
560 pos.col = eol-1;
561 vterm_screen_get_cell(screen, pos, &cell);
562 if(cell.chars[0])
563 break;
564
565 eol--;
566 }
567
568 if(eol == rect.start_col)
569 break;
570
571 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200572 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200573
574 printf(" %d<", row);
575 for(col = rect.start_col; col < eol; col++) {
576 VTermScreenCell cell;
577 VTermPos pos;
578 pos.row = row;
579 pos.col = col;
580 vterm_screen_get_cell(screen, pos, &cell);
581 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
582 }
583 printf(">");
584 }
585 }
586
587 printf("\n");
588
589 return 1;
590}
591
592static int want_screen_scrollback = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200593static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200594{
595 int eol;
596 int c;
597
598 if(!want_screen_scrollback)
599 return 1;
600
601 eol = cols;
602 while(eol && !cells[eol-1].chars[0])
603 eol--;
604
605 printf("sb_pushline %d =", cols);
606 for(c = 0; c < eol; c++)
607 printf(" %02X", cells[c].chars[0]);
608 printf("\n");
609
610 return 1;
611}
612
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200613static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200614{
615 int col;
616
617 if(!want_screen_scrollback)
618 return 0;
619
Bram Moolenaar707d2262019-12-04 22:16:54 +0100620 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200621 for(col = 0; col < cols; col++) {
622 if(col < 5)
623 cells[col].chars[0] = 'A' + col;
624 else
625 cells[col].chars[0] = 0;
626
627 cells[col].width = 1;
628 }
629
630 printf("sb_popline %d\n", cols);
631 return 1;
632}
633
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100634static int screen_sb_clear(void *user UNUSED)
635{
636 if(!want_screen_scrollback)
637 return 1;
638
639 printf("sb_clear\n");
640 return 0;
641}
642
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200643VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100644 screen_damage, // damage
645 moverect, // moverect
646 movecursor, // movecursor
647 settermprop, // settermprop
648 NULL, // bell
649 NULL, // resize
650 screen_sb_pushline, // sb_pushline
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100651 screen_sb_popline, // sb_popline
652 screen_sb_clear, // sb_clear
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200653};
654
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200655int main(int argc UNUSED, char **argv UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200656{
657 char line[1024] = {0};
658 int flag;
659
660 int err;
661
662 setvbuf(stdout, NULL, _IONBF, 0);
663
664 while(fgets(line, sizeof line, stdin)) {
665 char *nl;
666 size_t outlen;
667 err = 0;
668
669 if((nl = strchr(line, '\n')))
670 *nl = '\0';
671
672 if(streq(line, "INIT")) {
673 if(!vt)
674 vt = vterm_new(25, 80);
Bram Moolenaar94d729c2020-05-17 21:50:16 +0200675
676 // Somehow this makes tests fail
677 // vterm_output_set_callback(vt, term_output, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200678 }
679
680 else if(streq(line, "WANTPARSER")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100681 assert(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200682 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
683 }
684
685 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
686 int i = 9;
687 int sense = 1;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100688 assert(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200689 if(!state) {
690 state = vterm_obtain_state(vt);
691 vterm_state_set_callbacks(state, &state_cbs, NULL);
zeertzjqb00df7a2023-08-08 11:03:00 +0800692 /* In some tests we want to check the behaviour of overflowing the
693 * buffer, so make it nicely small
694 */
695 vterm_state_set_selection_callbacks(state, &selection_cbs, NULL, NULL, 16);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200696 vterm_state_set_bold_highbright(state, 1);
697 vterm_state_reset(state, 1);
698 }
699
700 while(line[i] == ' ')
701 i++;
702 for( ; line[i]; i++)
703 switch(line[i]) {
704 case '+':
705 sense = 1;
706 break;
707 case '-':
708 sense = 0;
709 break;
710 case 'g':
711 want_state_putglyph = sense;
712 break;
713 case 's':
714 want_scrollrect = sense;
715 break;
716 case 'm':
717 want_moverect = sense;
718 break;
719 case 'e':
720 want_state_erase = sense;
721 break;
722 case 'p':
723 want_settermprop = sense;
724 break;
725 case 'f':
Bram Moolenaard8637282020-05-20 18:41:41 +0200726 vterm_state_set_unrecognised_fallbacks(state, sense ? &fallbacks : NULL, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200727 break;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100728 case 'b':
729 want_state_scrollback = sense;
730 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200731 default:
732 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
733 }
734 }
735
736 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
737 int i = 10;
738 int sense = 1;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100739 assert(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200740 if(!screen)
741 screen = vterm_obtain_screen(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200742 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
743
744 while(line[i] == ' ')
745 i++;
746 for( ; line[i]; i++)
747 switch(line[i]) {
748 case '-':
749 sense = 0;
750 break;
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200751 case 'a':
752 vterm_screen_enable_altscreen(screen, 1);
753 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200754 case 'd':
755 want_screen_damage = sense;
756 break;
757 case 'D':
758 want_screen_damage = sense;
759 want_screen_damage_cells = sense;
760 break;
761 case 'm':
762 want_moverect = sense;
763 break;
764 case 'c':
765 want_movecursor = sense;
766 break;
767 case 'p':
768 want_settermprop = 1;
769 break;
770 case 'b':
771 want_screen_scrollback = sense;
772 break;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100773 case 'r':
774 vterm_screen_enable_reflow(screen, sense);
775 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200776 default:
777 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
778 }
779 }
780
781 else if(sscanf(line, "UTF8 %d", &flag)) {
782 vterm_set_utf8(vt, flag);
783 }
784
785 else if(streq(line, "RESET")) {
786 if(state) {
787 vterm_state_reset(state, 1);
788 vterm_state_get_cursorpos(state, &state_pos);
789 }
790 if(screen) {
791 vterm_screen_reset(screen, 1);
792 }
793 }
794
795 else if(strstartswith(line, "RESIZE ")) {
796 int rows, cols;
797 char *linep = line + 7;
798 while(linep[0] == ' ')
799 linep++;
800 sscanf(linep, "%d, %d", &rows, &cols);
801 vterm_set_size(vt, rows, cols);
802 }
803
804 else if(strstartswith(line, "PUSH ")) {
805 char *bytes = line + 5;
806 size_t len = inplace_hex2bytes(bytes);
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100807 assert(len);
808
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200809 size_t written = vterm_input_write(vt, bytes, len);
810 if(written < len)
811 fprintf(stderr, "! short write\n");
812 }
813
814 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100815 // This isn't really external API but it's hard to get this out any
816 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200817 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
818 if(encoding.enc->init)
819 (*encoding.enc->init)(encoding.enc, encoding.data);
820 }
821
822 else if(strstartswith(line, "ENCIN ")) {
823 char *bytes = line + 6;
824 size_t len = inplace_hex2bytes(bytes);
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100825 assert(len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200826
827 uint32_t cp[1024];
828 int cpi = 0;
829 size_t pos = 0;
830
831 (*encoding.enc->decode)(encoding.enc, encoding.data,
832 cp, &cpi, len, bytes, &pos, len);
833
834 if(cpi > 0) {
835 int i;
836 printf("encout ");
837 for(i = 0; i < cpi; i++) {
838 printf(i ? ",%x" : "%x", cp[i]);
839 }
840 printf("\n");
841 }
842 }
843
844 else if(strstartswith(line, "INCHAR ")) {
845 char *linep = line + 7;
846 unsigned int c = 0;
847 VTermModifier mod;
848 while(linep[0] == ' ')
849 linep++;
850 mod = strpe_modifiers(&linep);
851 sscanf(linep, " %x", &c);
852
853 vterm_keyboard_unichar(vt, c, mod);
854 }
855
856 else if(strstartswith(line, "INKEY ")) {
857 VTermModifier mod;
858 VTermKey key;
859 char *linep = line + 6;
860 while(linep[0] == ' ')
861 linep++;
862 mod = strpe_modifiers(&linep);
863 while(linep[0] == ' ')
864 linep++;
865 key = strp_key(linep);
866
867 vterm_keyboard_key(vt, key, mod);
868 }
869
870 else if(strstartswith(line, "PASTE ")) {
871 char *linep = line + 6;
872 if(streq(linep, "START"))
873 vterm_keyboard_start_paste(vt);
874 else if(streq(linep, "END"))
875 vterm_keyboard_end_paste(vt);
876 else
877 goto abort_line;
878 }
879
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200880 else if(strstartswith(line, "FOCUS ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100881 assert(state);
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200882 char *linep = line + 6;
883 if(streq(linep, "IN"))
884 vterm_state_focus_in(state);
885 else if(streq(linep, "OUT"))
886 vterm_state_focus_out(state);
887 else
888 goto abort_line;
889 }
890
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200891 else if(strstartswith(line, "MOUSEMOVE ")) {
892 char *linep = line + 10;
893 int row, col, len;
894 VTermModifier mod;
895 while(linep[0] == ' ')
896 linep++;
897 sscanf(linep, "%d,%d%n", &row, &col, &len);
898 linep += len;
899 while(linep[0] == ' ')
900 linep++;
901 mod = strpe_modifiers(&linep);
902 vterm_mouse_move(vt, row, col, mod);
903 }
904
905 else if(strstartswith(line, "MOUSEBTN ")) {
906 char *linep = line + 9;
907 char press;
908 int button, len;
909 VTermModifier mod;
910 while(linep[0] == ' ')
911 linep++;
912 sscanf(linep, "%c %d%n", &press, &button, &len);
913 linep += len;
914 while(linep[0] == ' ')
915 linep++;
916 mod = strpe_modifiers(&linep);
917 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
918 }
919
Bram Moolenaar7da34152021-11-24 19:30:55 +0000920 else if(strstartswith(line, "SELECTION ")) {
921 char *linep = line + 10;
922 unsigned int mask;
923 int len;
924 VTermStringFragment frag = { 0 };
925 sscanf(linep, "%x%n", &mask, &len);
926 linep += len;
927 while(linep[0] == ' ')
928 linep++;
929 if(linep[0] == '[') {
930 frag.initial = TRUE;
931 linep++;
932 while(linep[0] == ' ')
933 linep++;
934 }
935 frag.len = inplace_hex2bytes(linep);
936 frag.str = linep;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100937 assert(frag.len);
938
Bram Moolenaar7da34152021-11-24 19:30:55 +0000939 linep += frag.len * 2;
940 while(linep[0] == ' ')
941 linep++;
942 if(linep[0] == ']') {
943 frag.final = TRUE;
944 }
945 vterm_state_send_selection(state, mask, frag);
946 }
947
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200948 else if(strstartswith(line, "DAMAGEMERGE ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100949 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200950 char *linep = line + 12;
951 while(linep[0] == ' ')
952 linep++;
953 if(streq(linep, "CELL"))
954 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
955 else if(streq(linep, "ROW"))
956 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
957 else if(streq(linep, "SCREEN"))
958 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
959 else if(streq(linep, "SCROLL"))
960 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
961 }
962
963 else if(strstartswith(line, "DAMAGEFLUSH")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100964 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200965 vterm_screen_flush_damage(screen);
966 }
967
zeertzjqb00df7a2023-08-08 11:03:00 +0800968 else if(strstartswith(line, "SETDEFAULTCOL ")) {
969 assert(screen);
970 char *linep = line + 14;
971 while(linep[0] == ' ')
972 linep++;
973 VTermColor fg = strpe_color(&linep);
974 if(linep[0]) {
975 while(linep[0] == ' ')
976 linep++;
977 VTermColor bg = strpe_color(&linep);
978
979 vterm_screen_set_default_colors(screen, &fg, &bg);
980 }
981 else
982 vterm_screen_set_default_colors(screen, &fg, NULL);
983 }
984
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200985 else if(line[0] == '?') {
986 if(streq(line, "?cursor")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100987 assert(state);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200988 VTermPos pos;
989 vterm_state_get_cursorpos(state, &pos);
990 if(pos.row != state_pos.row)
991 printf("! row mismatch: state=%d,%d event=%d,%d\n",
992 pos.row, pos.col, state_pos.row, state_pos.col);
993 else if(pos.col != state_pos.col)
994 printf("! col mismatch: state=%d,%d event=%d,%d\n",
995 pos.row, pos.col, state_pos.row, state_pos.col);
996 else
997 printf("%d,%d\n", state_pos.row, state_pos.col);
998 }
999 else if(strstartswith(line, "?pen ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001000 assert(state);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001001 VTermValue val;
1002 char *linep = line + 5;
1003 while(linep[0] == ' ')
1004 linep++;
1005
1006#define BOOLSTR(v) ((v) ? "on" : "off")
1007
1008 if(streq(linep, "bold")) {
1009 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
1010 if(val.boolean != state_pen.bold)
1011 printf("! pen bold mismatch; state=%s, event=%s\n",
1012 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
1013 else
1014 printf("%s\n", BOOLSTR(state_pen.bold));
1015 }
1016 else if(streq(linep, "underline")) {
1017 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
1018 if(val.boolean != state_pen.underline)
1019 printf("! pen underline mismatch; state=%d, event=%d\n",
1020 val.boolean, state_pen.underline);
1021 else
1022 printf("%d\n", state_pen.underline);
1023 }
1024 else if(streq(linep, "italic")) {
1025 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
1026 if(val.boolean != state_pen.italic)
1027 printf("! pen italic mismatch; state=%s, event=%s\n",
1028 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
1029 else
1030 printf("%s\n", BOOLSTR(state_pen.italic));
1031 }
1032 else if(streq(linep, "blink")) {
1033 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
1034 if(val.boolean != state_pen.blink)
1035 printf("! pen blink mismatch; state=%s, event=%s\n",
1036 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
1037 else
1038 printf("%s\n", BOOLSTR(state_pen.blink));
1039 }
1040 else if(streq(linep, "reverse")) {
1041 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
1042 if(val.boolean != state_pen.reverse)
1043 printf("! pen reverse mismatch; state=%s, event=%s\n",
1044 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
1045 else
1046 printf("%s\n", BOOLSTR(state_pen.reverse));
1047 }
1048 else if(streq(linep, "font")) {
1049 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
1050 if(val.boolean != state_pen.font)
1051 printf("! pen font mismatch; state=%d, event=%d\n",
1052 val.boolean, state_pen.font);
1053 else
1054 printf("%d\n", state_pen.font);
1055 }
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001056 else if(streq(linep, "small")) {
1057 vterm_state_get_penattr(state, VTERM_ATTR_SMALL, &val);
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001058 if(val.boolean != state_pen.small)
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001059 printf("! pen small mismatch; state=%s, event=%s\n",
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001060 BOOLSTR(val.boolean), BOOLSTR(state_pen.small));
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001061 else
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001062 printf("%s\n", BOOLSTR(state_pen.small));
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001063 }
1064 else if(streq(linep, "baseline")) {
1065 vterm_state_get_penattr(state, VTERM_ATTR_BASELINE, &val);
1066 if(val.number != state_pen.baseline)
1067 printf("! pen baseline mismatch: state=%d, event=%d\n",
1068 val.number, state_pen.baseline);
1069 else
1070 printf("%s\n", state_pen.baseline == VTERM_BASELINE_RAISE ? "raise"
1071 : state_pen.baseline == VTERM_BASELINE_LOWER ? "lower"
1072 : "normal");
1073 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001074 else if(streq(linep, "foreground")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +02001075 print_color(&state_pen.foreground);
1076 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001077 }
1078 else if(streq(linep, "background")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +02001079 print_color(&state_pen.background);
1080 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001081 }
1082 else
1083 printf("?\n");
1084 }
Bram Moolenaar88d68de2020-05-18 21:51:01 +02001085 else if(strstartswith(line, "?lineinfo ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001086 assert(state);
Bram Moolenaar88d68de2020-05-18 21:51:01 +02001087 char *linep = line + 10;
1088 int row;
1089 const VTermLineInfo *info;
1090 while(linep[0] == ' ')
1091 linep++;
1092 if(sscanf(linep, "%d", &row) < 1) {
1093 printf("! lineinfo unrecognised input\n");
1094 goto abort_line;
1095 }
1096 info = vterm_state_get_lineinfo(state, row);
1097 if(info->doublewidth)
1098 printf("dwl ");
1099 if(info->doubleheight)
1100 printf("dhl ");
1101 if(info->continuation)
1102 printf("cont ");
1103 printf("\n");
1104 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001105 else if(strstartswith(line, "?screen_chars ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001106 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001107 char *linep = line + 13;
1108 VTermRect rect;
1109 size_t len;
1110 while(linep[0] == ' ')
1111 linep++;
Bram Moolenaar501e7772022-10-16 14:35:46 +01001112 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) == 4)
1113 ; // fine
1114 else if(sscanf(linep, "%d", &rect.start_row) == 1) {
1115 rect.end_row = rect.start_row + 1;
1116 rect.start_col = 0;
1117 vterm_get_size(vt, NULL, &rect.end_col);
1118 }
1119 else {
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001120 printf("! screen_chars unrecognised input\n");
1121 goto abort_line;
1122 }
1123 len = vterm_screen_get_chars(screen, NULL, 0, rect);
1124 if(len == (size_t)-1)
1125 printf("! screen_chars error\n");
1126 else if(len == 0)
1127 printf("\n");
1128 else {
1129 uint32_t *chars = malloc(sizeof(uint32_t) * len);
1130 size_t i;
1131 vterm_screen_get_chars(screen, chars, len, rect);
1132 for(i = 0; i < len; i++) {
1133 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
1134 }
1135 free(chars);
1136 }
1137 }
1138 else if(strstartswith(line, "?screen_text ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001139 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001140 char *linep = line + 12;
1141 VTermRect rect;
1142 size_t len;
1143 while(linep[0] == ' ')
1144 linep++;
1145 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
1146 printf("! screen_text unrecognised input\n");
1147 goto abort_line;
1148 }
1149 len = vterm_screen_get_text(screen, NULL, 0, rect);
1150 if(len == (size_t)-1)
1151 printf("! screen_text error\n");
1152 else if(len == 0)
1153 printf("\n");
1154 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +01001155 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001156 unsigned char *buffer = malloc(len + 4);
1157 unsigned char *text = buffer + 2;
1158 text[-2] = 0x55; text[-1] = 0xAA;
1159 text[len] = 0x55; text[len+1] = 0xAA;
1160
1161 vterm_screen_get_text(screen, (char *)text, len, rect);
1162
1163 if(text[-2] != 0x55 || text[-1] != 0xAA)
1164 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
1165 else if(text[len] != 0x55 || text[len+1] != 0xAA)
1166 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
1167 else
1168 {
1169 size_t i;
1170 for(i = 0; i < len; i++) {
1171 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
1172 }
1173 }
1174
1175 free(buffer);
1176 }
1177 }
1178 else if(strstartswith(line, "?screen_cell ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001179 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001180 char *linep = line + 12;
1181 int i;
1182 VTermPos pos;
1183 VTermScreenCell cell;
1184 while(linep[0] == ' ')
1185 linep++;
1186 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1187 printf("! screen_cell unrecognised input\n");
1188 goto abort_line;
1189 }
1190 if(!vterm_screen_get_cell(screen, pos, &cell))
1191 goto abort_line;
1192 printf("{");
1193 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
1194 printf("%s0x%x", i ? "," : "", cell.chars[i]);
1195 }
1196 printf("} width=%d attrs={", cell.width);
1197 if(cell.attrs.bold) printf("B");
1198 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
1199 if(cell.attrs.italic) printf("I");
1200 if(cell.attrs.blink) printf("K");
1201 if(cell.attrs.reverse) printf("R");
1202 if(cell.attrs.font) printf("F%d", cell.attrs.font);
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001203 if(cell.attrs.small) printf("S");
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001204 if(cell.attrs.baseline) printf(
1205 cell.attrs.baseline == VTERM_BASELINE_RAISE ? "^" :
1206 "_");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001207 printf("} ");
1208 if(cell.attrs.dwl) printf("dwl ");
1209 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
Bram Moolenaare5886cc2020-05-21 20:10:04 +02001210 printf("fg=");
1211 vterm_screen_convert_color_to_rgb(screen, &cell.fg);
1212 print_color(&cell.fg);
1213 printf(" bg=");
1214 vterm_screen_convert_color_to_rgb(screen, &cell.bg);
1215 print_color(&cell.bg);
1216 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001217 }
1218 else if(strstartswith(line, "?screen_eol ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001219 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001220 VTermPos pos;
1221 char *linep = line + 12;
1222 while(linep[0] == ' ')
1223 linep++;
1224 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1225 printf("! screen_eol unrecognised input\n");
1226 goto abort_line;
1227 }
1228 printf("%d\n", vterm_screen_is_eol(screen, pos));
1229 }
1230 else if(strstartswith(line, "?screen_attrs_extent ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001231 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001232 VTermPos pos;
1233 VTermRect rect;
1234 char *linep = line + 21;
1235 while(linep[0] == ' ')
1236 linep++;
1237 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1238 printf("! screen_attrs_extent unrecognised input\n");
1239 goto abort_line;
1240 }
1241 rect.start_col = 0;
1242 rect.end_col = -1;
1243 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
1244 printf("! screen_attrs_extent failed\n");
1245 goto abort_line;
1246 }
1247 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
1248 }
1249 else
1250 printf("?\n");
1251
1252 memset(line, 0, sizeof line);
1253 continue;
1254 }
1255
1256 else
1257 abort_line: err = 1;
1258
1259 outlen = vterm_output_get_buffer_current(vt);
1260 if(outlen > 0) {
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001261 char outbuff[1024];
1262 vterm_output_read(vt, outbuff, outlen);
1263
Bram Moolenaar88c1ee82020-04-12 13:38:57 +02001264 term_output(outbuff, outlen, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001265 }
1266
1267 printf(err ? "?\n" : "DONE\n");
1268 }
1269
1270 vterm_free(vt);
1271
1272 return 0;
1273}