blob: d12c1205ad09a712146365b6af5e76ad8192a430 [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
Bram Moolenaare4f25e42017-07-07 11:54:15 +020085static VTerm *vt;
86static VTermState *state;
87static VTermScreen *screen;
88
89static VTermEncodingInstance encoding;
90
Bram Moolenaarcd630be2020-04-12 14:50:26 +020091static void term_output(const char *s, size_t len, void *user UNUSED)
Bram Moolenaar88c1ee82020-04-12 13:38:57 +020092{
93 size_t i;
94
95 printf("output ");
96 for(i = 0; i < len; i++)
97 printf("%x%s", (unsigned char)s[i], i < len-1 ? "," : "\n");
98}
99
100static void printhex(const char *s, size_t len)
101{
102 while(len--)
Bram Moolenaar7da34152021-11-24 19:30:55 +0000103 printf("%02x", (uint8_t)(s++)[0]);
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200104}
105
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200106static int parser_text(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200107{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200108 size_t i;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200109
110 printf("text ");
111 for(i = 0; i < len; i++) {
112 unsigned char b = bytes[i];
113 if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
114 break;
115 printf(i ? ",%x" : "%x", b);
116 }
117 printf("\n");
118
119 return i;
120}
121
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200122static int parser_control(unsigned char control, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200123{
124 printf("control %02x\n", control);
125
126 return 1;
127}
128
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200129static int parser_escape(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200130{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200131 if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
132 if(len < 2)
133 return -1;
134 len = 2;
135 }
136 else {
137 len = 1;
138 }
139
140 printf("escape ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200141 printhex(bytes, len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200142 printf("\n");
143
144 return len;
145}
146
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200147static 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 +0200148{
149 int i;
150 printf("csi %02x", command);
151
152 if(leader && leader[0]) {
153 printf(" L=");
154 for(i = 0; leader[i]; i++)
155 printf("%02x", leader[i]);
156 }
157
158 for(i = 0; i < argcount; i++) {
159 char sep = i ? ',' : ' ';
160
161 if(args[i] == CSI_ARG_MISSING)
162 printf("%c*", sep);
163 else
164 printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
165 }
166
167 if(intermed && intermed[0]) {
168 printf(" I=");
169 for(i = 0; intermed[i]; i++)
170 printf("%02x", intermed[i]);
171 }
172
173 printf("\n");
174
175 return 1;
176}
177
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200178static int parser_osc(int command, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200179{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200180
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200181 printf("osc ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200182
183 if(frag.initial) {
184 if(command == -1)
185 printf("[");
186 else
187 printf("[%d;", command);
188 }
189
190 printhex(frag.str, frag.len);
191
192 if(frag.final)
193 printf("]");
194
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200195 printf("\n");
196
197 return 1;
198}
199
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200200static int parser_dcs(const char *command, size_t commandlen, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200201{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200202 printf("dcs ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200203
204 if(frag.initial) {
205 size_t i;
206 printf("[");
207 for(i = 0; i < commandlen; i++)
208 printf("%02x", command[i]);
209 }
210
211 printhex(frag.str, frag.len);
212
213 if(frag.final)
214 printf("]");
215
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200216 printf("\n");
217
218 return 1;
219}
220
Bram Moolenaar7da34152021-11-24 19:30:55 +0000221static int parser_apc(VTermStringFragment frag, void *user UNUSED)
222{
223 printf("apc ");
224
225 if(frag.initial)
226 printf("[");
227
228 printhex(frag.str, frag.len);
229
230 if(frag.final)
231 printf("]");
232
233 printf("\n");
234
235 return 1;
236}
237
238static int parser_pm(VTermStringFragment frag, void *user UNUSED)
239{
240 printf("pm ");
241
242 if(frag.initial)
243 printf("[");
244
245 printhex(frag.str, frag.len);
246
247 if(frag.final)
248 printf("]");
249
250 printf("\n");
251
252 return 1;
253}
254
255static int parser_sos(VTermStringFragment frag, void *user UNUSED)
256{
257 printf("sos ");
258
259 if(frag.initial)
260 printf("[");
261
262 printhex(frag.str, frag.len);
263
264 if(frag.final)
265 printf("]");
266
267 printf("\n");
268
269 return 1;
270}
271
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200272static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100273 parser_text, // text
274 parser_control, // control
275 parser_escape, // escape
276 parser_csi, // csi
277 parser_osc, // osc
278 parser_dcs, // dcs
Bram Moolenaar7da34152021-11-24 19:30:55 +0000279 parser_apc, // apc
280 parser_pm, // pm
281 parser_sos, // sos
Bram Moolenaar707d2262019-12-04 22:16:54 +0100282 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200283};
284
Bram Moolenaard8637282020-05-20 18:41:41 +0200285static VTermStateFallbacks fallbacks = {
286 parser_control, // control
287 parser_csi, // csi
288 parser_osc, // osc
Bram Moolenaar7da34152021-11-24 19:30:55 +0000289 parser_dcs, // dcs
290 parser_apc, // dcs
291 parser_pm, // pm
292 parser_sos // sos
Bram Moolenaard8637282020-05-20 18:41:41 +0200293};
294
295/* These callbacks are shared by State and Screen */
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200296
297static int want_movecursor = 0;
298static VTermPos state_pos;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200299static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200300{
301 state_pos = pos;
302
303 if(want_movecursor)
304 printf("movecursor %d,%d\n", pos.row, pos.col);
305
306 return 1;
307}
308
309static int want_scrollrect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200310static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200311{
312 if(!want_scrollrect)
313 return 0;
314
315 printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
316 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
317 downward, rightward);
318
319 return 1;
320}
321
322static int want_moverect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200323static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200324{
325 if(!want_moverect)
326 return 0;
327
328 printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
329 src.start_row, src.end_row, src.start_col, src.end_col,
330 dest.start_row, dest.end_row, dest.start_col, dest.end_col);
331
332 return 1;
333}
334
335static int want_settermprop = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200336static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200337{
338 VTermValueType type;
339 if(!want_settermprop)
340 return 1;
341
342 type = vterm_get_prop_type(prop);
343 switch(type) {
344 case VTERM_VALUETYPE_BOOL:
345 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
346 return 1;
347 case VTERM_VALUETYPE_INT:
348 printf("settermprop %d %d\n", prop, val->number);
349 return 1;
350 case VTERM_VALUETYPE_STRING:
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200351 printf("settermprop %d %s\"%.*s\"%s\n", prop,
352 val->string.initial ? "[" : "", val->string.len, val->string.str, val->string.final ? "]" : "");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200353 return 1;
354 case VTERM_VALUETYPE_COLOR:
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200355 printf("settermprop %d ", prop);
356 print_color(&val->color);
357 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200358 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200359
360 case VTERM_N_VALUETYPES:
361 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200362 }
363
364 return 0;
365}
366
Bram Moolenaar707d2262019-12-04 22:16:54 +0100367// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200368
369static int want_state_putglyph = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200370static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200371{
372 int i;
373 if(!want_state_putglyph)
374 return 1;
375
376 printf("putglyph ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200377 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && info->chars[i]; i++)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200378 printf(i ? ",%x" : "%x", info->chars[i]);
379 printf(" %d %d,%d", info->width, pos.row, pos.col);
380 if(info->protected_cell)
381 printf(" prot");
382 if(info->dwl)
383 printf(" dwl");
384 if(info->dhl)
385 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
386 printf("\n");
387
388 return 1;
389}
390
391static int want_state_erase = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200392static int state_erase(VTermRect rect, int selective, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200393{
394 if(!want_state_erase)
395 return 1;
396
397 printf("erase %d..%d,%d..%d%s\n",
398 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
399 selective ? " selective" : "");
400
401 return 1;
402}
403
404static struct {
405 int bold;
406 int underline;
407 int italic;
408 int blink;
409 int reverse;
Bram Moolenaard8637282020-05-20 18:41:41 +0200410 int conceal;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200411 int strike;
412 int font;
Bram Moolenaare6a16e92022-10-17 14:51:36 +0100413 int small;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100414 int baseline;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200415 VTermColor foreground;
416 VTermColor background;
417} state_pen;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200418static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200419{
420 switch(attr) {
421 case VTERM_ATTR_BOLD:
422 state_pen.bold = val->boolean;
423 break;
424 case VTERM_ATTR_UNDERLINE:
425 state_pen.underline = val->number;
426 break;
427 case VTERM_ATTR_ITALIC:
428 state_pen.italic = val->boolean;
429 break;
430 case VTERM_ATTR_BLINK:
431 state_pen.blink = val->boolean;
432 break;
433 case VTERM_ATTR_REVERSE:
434 state_pen.reverse = val->boolean;
435 break;
Bram Moolenaard8637282020-05-20 18:41:41 +0200436 case VTERM_ATTR_CONCEAL:
437 state_pen.conceal = val->boolean;
438 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200439 case VTERM_ATTR_STRIKE:
440 state_pen.strike = val->boolean;
441 break;
442 case VTERM_ATTR_FONT:
443 state_pen.font = val->number;
444 break;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100445 case VTERM_ATTR_SMALL:
Bram Moolenaare6a16e92022-10-17 14:51:36 +0100446 state_pen.small = val->boolean;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100447 break;
448 case VTERM_ATTR_BASELINE:
449 state_pen.baseline = val->number;
450 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200451 case VTERM_ATTR_FOREGROUND:
452 state_pen.foreground = val->color;
453 break;
454 case VTERM_ATTR_BACKGROUND:
455 state_pen.background = val->color;
456 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200457
458 case VTERM_N_ATTRS:
459 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200460 }
461
462 return 1;
463}
464
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200465static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200466{
467 return 1;
468}
469
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100470static int want_state_scrollback = 0;
471static int state_sb_clear(void *user UNUSED) {
472 if(!want_state_scrollback)
473 return 1;
474
475 printf("sb_clear\n");
476 return 0;
477}
478
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200479VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100480 state_putglyph, // putglyph
481 movecursor, // movecursor
482 scrollrect, // scrollrect
483 moverect, // moverect
484 state_erase, // erase
485 NULL, // initpen
486 state_setpenattr, // setpenattr
487 settermprop, // settermprop
488 NULL, // bell
489 NULL, // resize
490 state_setlineinfo, // setlineinfo
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100491 state_sb_clear, // sb_clear
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200492};
493
Bram Moolenaar7da34152021-11-24 19:30:55 +0000494static int selection_set(VTermSelectionMask mask, VTermStringFragment frag, void *user UNUSED)
495{
496 printf("selection-set mask=%04X ", mask);
497 if(frag.initial)
498 printf("[");
499 printhex(frag.str, frag.len);
500 if(frag.final)
501 printf("]");
502 printf("\n");
503
504 return 1;
505}
506
507static int selection_query(VTermSelectionMask mask, void *user UNUSED)
508{
509 printf("selection-query mask=%04X\n", mask);
510
511 return 1;
512}
513
514VTermSelectionCallbacks selection_cbs = {
515 .set = selection_set,
516 .query = selection_query,
517};
518
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200519static int want_screen_damage = 0;
520static int want_screen_damage_cells = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200521static int screen_damage(VTermRect rect, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200522{
523 if(!want_screen_damage)
524 return 1;
525
526 printf("damage %d..%d,%d..%d",
527 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
528
529 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200530 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200531 int row;
532 int col;
533
534 for(row = rect.start_row; row < rect.end_row; row++) {
535 int eol = rect.end_col;
536 while(eol > rect.start_col) {
537 VTermScreenCell cell;
538 VTermPos pos;
539 pos.row = row;
540 pos.col = eol-1;
541 vterm_screen_get_cell(screen, pos, &cell);
542 if(cell.chars[0])
543 break;
544
545 eol--;
546 }
547
548 if(eol == rect.start_col)
549 break;
550
551 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200552 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200553
554 printf(" %d<", row);
555 for(col = rect.start_col; col < eol; col++) {
556 VTermScreenCell cell;
557 VTermPos pos;
558 pos.row = row;
559 pos.col = col;
560 vterm_screen_get_cell(screen, pos, &cell);
561 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
562 }
563 printf(">");
564 }
565 }
566
567 printf("\n");
568
569 return 1;
570}
571
572static int want_screen_scrollback = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200573static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200574{
575 int eol;
576 int c;
577
578 if(!want_screen_scrollback)
579 return 1;
580
581 eol = cols;
582 while(eol && !cells[eol-1].chars[0])
583 eol--;
584
585 printf("sb_pushline %d =", cols);
586 for(c = 0; c < eol; c++)
587 printf(" %02X", cells[c].chars[0]);
588 printf("\n");
589
590 return 1;
591}
592
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200593static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200594{
595 int col;
596
597 if(!want_screen_scrollback)
598 return 0;
599
Bram Moolenaar707d2262019-12-04 22:16:54 +0100600 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200601 for(col = 0; col < cols; col++) {
602 if(col < 5)
603 cells[col].chars[0] = 'A' + col;
604 else
605 cells[col].chars[0] = 0;
606
607 cells[col].width = 1;
608 }
609
610 printf("sb_popline %d\n", cols);
611 return 1;
612}
613
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100614static int screen_sb_clear(void *user UNUSED)
615{
616 if(!want_screen_scrollback)
617 return 1;
618
619 printf("sb_clear\n");
620 return 0;
621}
622
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200623VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100624 screen_damage, // damage
625 moverect, // moverect
626 movecursor, // movecursor
627 settermprop, // settermprop
628 NULL, // bell
629 NULL, // resize
630 screen_sb_pushline, // sb_pushline
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100631 screen_sb_popline, // sb_popline
632 screen_sb_clear, // sb_clear
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200633};
634
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200635int main(int argc UNUSED, char **argv UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200636{
637 char line[1024] = {0};
638 int flag;
639
640 int err;
641
642 setvbuf(stdout, NULL, _IONBF, 0);
643
644 while(fgets(line, sizeof line, stdin)) {
645 char *nl;
646 size_t outlen;
647 err = 0;
648
649 if((nl = strchr(line, '\n')))
650 *nl = '\0';
651
652 if(streq(line, "INIT")) {
653 if(!vt)
654 vt = vterm_new(25, 80);
Bram Moolenaar94d729c2020-05-17 21:50:16 +0200655
656 // Somehow this makes tests fail
657 // vterm_output_set_callback(vt, term_output, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200658 }
659
660 else if(streq(line, "WANTPARSER")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100661 assert(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200662 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
663 }
664
665 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
666 int i = 9;
667 int sense = 1;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100668 assert(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200669 if(!state) {
670 state = vterm_obtain_state(vt);
671 vterm_state_set_callbacks(state, &state_cbs, NULL);
Bram Moolenaar7da34152021-11-24 19:30:55 +0000672 vterm_state_set_selection_callbacks(state, &selection_cbs, NULL, NULL, 1024);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200673 vterm_state_set_bold_highbright(state, 1);
674 vterm_state_reset(state, 1);
675 }
676
677 while(line[i] == ' ')
678 i++;
679 for( ; line[i]; i++)
680 switch(line[i]) {
681 case '+':
682 sense = 1;
683 break;
684 case '-':
685 sense = 0;
686 break;
687 case 'g':
688 want_state_putglyph = sense;
689 break;
690 case 's':
691 want_scrollrect = sense;
692 break;
693 case 'm':
694 want_moverect = sense;
695 break;
696 case 'e':
697 want_state_erase = sense;
698 break;
699 case 'p':
700 want_settermprop = sense;
701 break;
702 case 'f':
Bram Moolenaard8637282020-05-20 18:41:41 +0200703 vterm_state_set_unrecognised_fallbacks(state, sense ? &fallbacks : NULL, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200704 break;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100705 case 'b':
706 want_state_scrollback = sense;
707 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200708 default:
709 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
710 }
711 }
712
713 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
714 int i = 10;
715 int sense = 1;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100716 assert(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200717 if(!screen)
718 screen = vterm_obtain_screen(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200719 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
720
721 while(line[i] == ' ')
722 i++;
723 for( ; line[i]; i++)
724 switch(line[i]) {
725 case '-':
726 sense = 0;
727 break;
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200728 case 'a':
729 vterm_screen_enable_altscreen(screen, 1);
730 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200731 case 'd':
732 want_screen_damage = sense;
733 break;
734 case 'D':
735 want_screen_damage = sense;
736 want_screen_damage_cells = sense;
737 break;
738 case 'm':
739 want_moverect = sense;
740 break;
741 case 'c':
742 want_movecursor = sense;
743 break;
744 case 'p':
745 want_settermprop = 1;
746 break;
747 case 'b':
748 want_screen_scrollback = sense;
749 break;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100750 case 'r':
751 vterm_screen_enable_reflow(screen, sense);
752 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200753 default:
754 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
755 }
756 }
757
758 else if(sscanf(line, "UTF8 %d", &flag)) {
759 vterm_set_utf8(vt, flag);
760 }
761
762 else if(streq(line, "RESET")) {
763 if(state) {
764 vterm_state_reset(state, 1);
765 vterm_state_get_cursorpos(state, &state_pos);
766 }
767 if(screen) {
768 vterm_screen_reset(screen, 1);
769 }
770 }
771
772 else if(strstartswith(line, "RESIZE ")) {
773 int rows, cols;
774 char *linep = line + 7;
775 while(linep[0] == ' ')
776 linep++;
777 sscanf(linep, "%d, %d", &rows, &cols);
778 vterm_set_size(vt, rows, cols);
779 }
780
781 else if(strstartswith(line, "PUSH ")) {
782 char *bytes = line + 5;
783 size_t len = inplace_hex2bytes(bytes);
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100784 assert(len);
785
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200786 size_t written = vterm_input_write(vt, bytes, len);
787 if(written < len)
788 fprintf(stderr, "! short write\n");
789 }
790
791 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100792 // This isn't really external API but it's hard to get this out any
793 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200794 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
795 if(encoding.enc->init)
796 (*encoding.enc->init)(encoding.enc, encoding.data);
797 }
798
799 else if(strstartswith(line, "ENCIN ")) {
800 char *bytes = line + 6;
801 size_t len = inplace_hex2bytes(bytes);
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100802 assert(len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200803
804 uint32_t cp[1024];
805 int cpi = 0;
806 size_t pos = 0;
807
808 (*encoding.enc->decode)(encoding.enc, encoding.data,
809 cp, &cpi, len, bytes, &pos, len);
810
811 if(cpi > 0) {
812 int i;
813 printf("encout ");
814 for(i = 0; i < cpi; i++) {
815 printf(i ? ",%x" : "%x", cp[i]);
816 }
817 printf("\n");
818 }
819 }
820
821 else if(strstartswith(line, "INCHAR ")) {
822 char *linep = line + 7;
823 unsigned int c = 0;
824 VTermModifier mod;
825 while(linep[0] == ' ')
826 linep++;
827 mod = strpe_modifiers(&linep);
828 sscanf(linep, " %x", &c);
829
830 vterm_keyboard_unichar(vt, c, mod);
831 }
832
833 else if(strstartswith(line, "INKEY ")) {
834 VTermModifier mod;
835 VTermKey key;
836 char *linep = line + 6;
837 while(linep[0] == ' ')
838 linep++;
839 mod = strpe_modifiers(&linep);
840 while(linep[0] == ' ')
841 linep++;
842 key = strp_key(linep);
843
844 vterm_keyboard_key(vt, key, mod);
845 }
846
847 else if(strstartswith(line, "PASTE ")) {
848 char *linep = line + 6;
849 if(streq(linep, "START"))
850 vterm_keyboard_start_paste(vt);
851 else if(streq(linep, "END"))
852 vterm_keyboard_end_paste(vt);
853 else
854 goto abort_line;
855 }
856
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200857 else if(strstartswith(line, "FOCUS ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100858 assert(state);
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200859 char *linep = line + 6;
860 if(streq(linep, "IN"))
861 vterm_state_focus_in(state);
862 else if(streq(linep, "OUT"))
863 vterm_state_focus_out(state);
864 else
865 goto abort_line;
866 }
867
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200868 else if(strstartswith(line, "MOUSEMOVE ")) {
869 char *linep = line + 10;
870 int row, col, len;
871 VTermModifier mod;
872 while(linep[0] == ' ')
873 linep++;
874 sscanf(linep, "%d,%d%n", &row, &col, &len);
875 linep += len;
876 while(linep[0] == ' ')
877 linep++;
878 mod = strpe_modifiers(&linep);
879 vterm_mouse_move(vt, row, col, mod);
880 }
881
882 else if(strstartswith(line, "MOUSEBTN ")) {
883 char *linep = line + 9;
884 char press;
885 int button, len;
886 VTermModifier mod;
887 while(linep[0] == ' ')
888 linep++;
889 sscanf(linep, "%c %d%n", &press, &button, &len);
890 linep += len;
891 while(linep[0] == ' ')
892 linep++;
893 mod = strpe_modifiers(&linep);
894 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
895 }
896
Bram Moolenaar7da34152021-11-24 19:30:55 +0000897 else if(strstartswith(line, "SELECTION ")) {
898 char *linep = line + 10;
899 unsigned int mask;
900 int len;
901 VTermStringFragment frag = { 0 };
902 sscanf(linep, "%x%n", &mask, &len);
903 linep += len;
904 while(linep[0] == ' ')
905 linep++;
906 if(linep[0] == '[') {
907 frag.initial = TRUE;
908 linep++;
909 while(linep[0] == ' ')
910 linep++;
911 }
912 frag.len = inplace_hex2bytes(linep);
913 frag.str = linep;
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100914 assert(frag.len);
915
Bram Moolenaar7da34152021-11-24 19:30:55 +0000916 linep += frag.len * 2;
917 while(linep[0] == ' ')
918 linep++;
919 if(linep[0] == ']') {
920 frag.final = TRUE;
921 }
922 vterm_state_send_selection(state, mask, frag);
923 }
924
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200925 else if(strstartswith(line, "DAMAGEMERGE ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100926 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200927 char *linep = line + 12;
928 while(linep[0] == ' ')
929 linep++;
930 if(streq(linep, "CELL"))
931 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
932 else if(streq(linep, "ROW"))
933 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
934 else if(streq(linep, "SCREEN"))
935 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
936 else if(streq(linep, "SCROLL"))
937 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
938 }
939
940 else if(strstartswith(line, "DAMAGEFLUSH")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100941 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200942 vterm_screen_flush_damage(screen);
943 }
944
945 else if(line[0] == '?') {
946 if(streq(line, "?cursor")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100947 assert(state);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200948 VTermPos pos;
949 vterm_state_get_cursorpos(state, &pos);
950 if(pos.row != state_pos.row)
951 printf("! row mismatch: state=%d,%d event=%d,%d\n",
952 pos.row, pos.col, state_pos.row, state_pos.col);
953 else if(pos.col != state_pos.col)
954 printf("! col mismatch: state=%d,%d event=%d,%d\n",
955 pos.row, pos.col, state_pos.row, state_pos.col);
956 else
957 printf("%d,%d\n", state_pos.row, state_pos.col);
958 }
959 else if(strstartswith(line, "?pen ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +0100960 assert(state);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200961 VTermValue val;
962 char *linep = line + 5;
963 while(linep[0] == ' ')
964 linep++;
965
966#define BOOLSTR(v) ((v) ? "on" : "off")
967
968 if(streq(linep, "bold")) {
969 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
970 if(val.boolean != state_pen.bold)
971 printf("! pen bold mismatch; state=%s, event=%s\n",
972 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
973 else
974 printf("%s\n", BOOLSTR(state_pen.bold));
975 }
976 else if(streq(linep, "underline")) {
977 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
978 if(val.boolean != state_pen.underline)
979 printf("! pen underline mismatch; state=%d, event=%d\n",
980 val.boolean, state_pen.underline);
981 else
982 printf("%d\n", state_pen.underline);
983 }
984 else if(streq(linep, "italic")) {
985 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
986 if(val.boolean != state_pen.italic)
987 printf("! pen italic mismatch; state=%s, event=%s\n",
988 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
989 else
990 printf("%s\n", BOOLSTR(state_pen.italic));
991 }
992 else if(streq(linep, "blink")) {
993 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
994 if(val.boolean != state_pen.blink)
995 printf("! pen blink mismatch; state=%s, event=%s\n",
996 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
997 else
998 printf("%s\n", BOOLSTR(state_pen.blink));
999 }
1000 else if(streq(linep, "reverse")) {
1001 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
1002 if(val.boolean != state_pen.reverse)
1003 printf("! pen reverse mismatch; state=%s, event=%s\n",
1004 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
1005 else
1006 printf("%s\n", BOOLSTR(state_pen.reverse));
1007 }
1008 else if(streq(linep, "font")) {
1009 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
1010 if(val.boolean != state_pen.font)
1011 printf("! pen font mismatch; state=%d, event=%d\n",
1012 val.boolean, state_pen.font);
1013 else
1014 printf("%d\n", state_pen.font);
1015 }
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001016 else if(streq(linep, "small")) {
1017 vterm_state_get_penattr(state, VTERM_ATTR_SMALL, &val);
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001018 if(val.boolean != state_pen.small)
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001019 printf("! pen small mismatch; state=%s, event=%s\n",
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001020 BOOLSTR(val.boolean), BOOLSTR(state_pen.small));
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001021 else
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001022 printf("%s\n", BOOLSTR(state_pen.small));
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001023 }
1024 else if(streq(linep, "baseline")) {
1025 vterm_state_get_penattr(state, VTERM_ATTR_BASELINE, &val);
1026 if(val.number != state_pen.baseline)
1027 printf("! pen baseline mismatch: state=%d, event=%d\n",
1028 val.number, state_pen.baseline);
1029 else
1030 printf("%s\n", state_pen.baseline == VTERM_BASELINE_RAISE ? "raise"
1031 : state_pen.baseline == VTERM_BASELINE_LOWER ? "lower"
1032 : "normal");
1033 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001034 else if(streq(linep, "foreground")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +02001035 print_color(&state_pen.foreground);
1036 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001037 }
1038 else if(streq(linep, "background")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +02001039 print_color(&state_pen.background);
1040 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001041 }
1042 else
1043 printf("?\n");
1044 }
Bram Moolenaar88d68de2020-05-18 21:51:01 +02001045 else if(strstartswith(line, "?lineinfo ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001046 assert(state);
Bram Moolenaar88d68de2020-05-18 21:51:01 +02001047 char *linep = line + 10;
1048 int row;
1049 const VTermLineInfo *info;
1050 while(linep[0] == ' ')
1051 linep++;
1052 if(sscanf(linep, "%d", &row) < 1) {
1053 printf("! lineinfo unrecognised input\n");
1054 goto abort_line;
1055 }
1056 info = vterm_state_get_lineinfo(state, row);
1057 if(info->doublewidth)
1058 printf("dwl ");
1059 if(info->doubleheight)
1060 printf("dhl ");
1061 if(info->continuation)
1062 printf("cont ");
1063 printf("\n");
1064 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001065 else if(strstartswith(line, "?screen_chars ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001066 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001067 char *linep = line + 13;
1068 VTermRect rect;
1069 size_t len;
1070 while(linep[0] == ' ')
1071 linep++;
Bram Moolenaar501e7772022-10-16 14:35:46 +01001072 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) == 4)
1073 ; // fine
1074 else if(sscanf(linep, "%d", &rect.start_row) == 1) {
1075 rect.end_row = rect.start_row + 1;
1076 rect.start_col = 0;
1077 vterm_get_size(vt, NULL, &rect.end_col);
1078 }
1079 else {
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001080 printf("! screen_chars unrecognised input\n");
1081 goto abort_line;
1082 }
1083 len = vterm_screen_get_chars(screen, NULL, 0, rect);
1084 if(len == (size_t)-1)
1085 printf("! screen_chars error\n");
1086 else if(len == 0)
1087 printf("\n");
1088 else {
1089 uint32_t *chars = malloc(sizeof(uint32_t) * len);
1090 size_t i;
1091 vterm_screen_get_chars(screen, chars, len, rect);
1092 for(i = 0; i < len; i++) {
1093 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
1094 }
1095 free(chars);
1096 }
1097 }
1098 else if(strstartswith(line, "?screen_text ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001099 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001100 char *linep = line + 12;
1101 VTermRect rect;
1102 size_t len;
1103 while(linep[0] == ' ')
1104 linep++;
1105 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
1106 printf("! screen_text unrecognised input\n");
1107 goto abort_line;
1108 }
1109 len = vterm_screen_get_text(screen, NULL, 0, rect);
1110 if(len == (size_t)-1)
1111 printf("! screen_text error\n");
1112 else if(len == 0)
1113 printf("\n");
1114 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +01001115 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001116 unsigned char *buffer = malloc(len + 4);
1117 unsigned char *text = buffer + 2;
1118 text[-2] = 0x55; text[-1] = 0xAA;
1119 text[len] = 0x55; text[len+1] = 0xAA;
1120
1121 vterm_screen_get_text(screen, (char *)text, len, rect);
1122
1123 if(text[-2] != 0x55 || text[-1] != 0xAA)
1124 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
1125 else if(text[len] != 0x55 || text[len+1] != 0xAA)
1126 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
1127 else
1128 {
1129 size_t i;
1130 for(i = 0; i < len; i++) {
1131 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
1132 }
1133 }
1134
1135 free(buffer);
1136 }
1137 }
1138 else if(strstartswith(line, "?screen_cell ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001139 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001140 char *linep = line + 12;
1141 int i;
1142 VTermPos pos;
1143 VTermScreenCell cell;
1144 while(linep[0] == ' ')
1145 linep++;
1146 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1147 printf("! screen_cell unrecognised input\n");
1148 goto abort_line;
1149 }
1150 if(!vterm_screen_get_cell(screen, pos, &cell))
1151 goto abort_line;
1152 printf("{");
1153 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
1154 printf("%s0x%x", i ? "," : "", cell.chars[i]);
1155 }
1156 printf("} width=%d attrs={", cell.width);
1157 if(cell.attrs.bold) printf("B");
1158 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
1159 if(cell.attrs.italic) printf("I");
1160 if(cell.attrs.blink) printf("K");
1161 if(cell.attrs.reverse) printf("R");
1162 if(cell.attrs.font) printf("F%d", cell.attrs.font);
Bram Moolenaare6a16e92022-10-17 14:51:36 +01001163 if(cell.attrs.small) printf("S");
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001164 if(cell.attrs.baseline) printf(
1165 cell.attrs.baseline == VTERM_BASELINE_RAISE ? "^" :
1166 "_");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001167 printf("} ");
1168 if(cell.attrs.dwl) printf("dwl ");
1169 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
Bram Moolenaare5886cc2020-05-21 20:10:04 +02001170 printf("fg=");
1171 vterm_screen_convert_color_to_rgb(screen, &cell.fg);
1172 print_color(&cell.fg);
1173 printf(" bg=");
1174 vterm_screen_convert_color_to_rgb(screen, &cell.bg);
1175 print_color(&cell.bg);
1176 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001177 }
1178 else if(strstartswith(line, "?screen_eol ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001179 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001180 VTermPos pos;
1181 char *linep = line + 12;
1182 while(linep[0] == ' ')
1183 linep++;
1184 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1185 printf("! screen_eol unrecognised input\n");
1186 goto abort_line;
1187 }
1188 printf("%d\n", vterm_screen_is_eol(screen, pos));
1189 }
1190 else if(strstartswith(line, "?screen_attrs_extent ")) {
Bram Moolenaar6a12d262022-10-16 19:26:52 +01001191 assert(screen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001192 VTermPos pos;
1193 VTermRect rect;
1194 char *linep = line + 21;
1195 while(linep[0] == ' ')
1196 linep++;
1197 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1198 printf("! screen_attrs_extent unrecognised input\n");
1199 goto abort_line;
1200 }
1201 rect.start_col = 0;
1202 rect.end_col = -1;
1203 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
1204 printf("! screen_attrs_extent failed\n");
1205 goto abort_line;
1206 }
1207 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
1208 }
1209 else
1210 printf("?\n");
1211
1212 memset(line, 0, sizeof line);
1213 continue;
1214 }
1215
1216 else
1217 abort_line: err = 1;
1218
1219 outlen = vterm_output_get_buffer_current(vt);
1220 if(outlen > 0) {
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001221 char outbuff[1024];
1222 vterm_output_read(vt, outbuff, outlen);
1223
Bram Moolenaar88c1ee82020-04-12 13:38:57 +02001224 term_output(outbuff, outlen, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001225 }
1226
1227 printf(err ? "?\n" : "DONE\n");
1228 }
1229
1230 vterm_free(vt);
1231
1232 return 0;
1233}