blob: 10afbfd768b4431415f3b3e3178f069d81993796 [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
4#include <stdio.h>
5#include <string.h>
6
7#define streq(a,b) (!strcmp(a,b))
8#define strstartswith(a,b) (!strncmp(a,b,strlen(b)))
9
10static size_t inplace_hex2bytes(char *s)
11{
12 char *inpos = s, *outpos = s;
13
14 while(*inpos) {
15 unsigned int ch;
16 sscanf(inpos, "%2x", &ch);
17 *outpos = ch;
18 outpos += 1; inpos += 2;
19 }
20
21 return outpos - s;
22}
23
24static VTermModifier strpe_modifiers(char **strp)
25{
26 VTermModifier state = 0;
27
28 while((*strp)[0]) {
29 switch(((*strp)++)[0]) {
30 case 'S': state |= VTERM_MOD_SHIFT; break;
31 case 'C': state |= VTERM_MOD_CTRL; break;
32 case 'A': state |= VTERM_MOD_ALT; break;
33 default: return state;
34 }
35 }
36
37 return state;
38}
39
40static VTermKey strp_key(char *str)
41{
42 static struct {
43 char *name;
44 VTermKey key;
45 } keys[] = {
46 { "Up", VTERM_KEY_UP },
47 { "Tab", VTERM_KEY_TAB },
48 { "Enter", VTERM_KEY_ENTER },
49 { "KP0", VTERM_KEY_KP_0 },
Bram Moolenaarc4c9f7e2020-05-17 20:52:45 +020050 { "F1", VTERM_KEY_FUNCTION(1) },
Bram Moolenaare4f25e42017-07-07 11:54:15 +020051 { NULL, VTERM_KEY_NONE },
52 };
53 int i;
54
55 for(i = 0; keys[i].name; i++) {
56 if(streq(str, keys[i].name))
57 return keys[i].key;
58 }
59
60 return VTERM_KEY_NONE;
61}
62
Bram Moolenaare5886cc2020-05-21 20:10:04 +020063static void print_color(const VTermColor *col)
64{
65 if (VTERM_COLOR_IS_RGB(col)) {
66 printf("rgb(%d,%d,%d", col->red, col->green, col->blue);
67 }
68 else if (VTERM_COLOR_IS_INDEXED(col)) {
69 printf("idx(%d", col->index);
70 }
71 else {
72 printf("invalid(%d", col->type);
73 }
74 if (VTERM_COLOR_IS_DEFAULT_FG(col)) {
75 printf(",is_default_fg");
76 }
77 if (VTERM_COLOR_IS_DEFAULT_BG(col)) {
78 printf(",is_default_bg");
79 }
80 printf(")");
81}
82
Bram Moolenaare4f25e42017-07-07 11:54:15 +020083static VTerm *vt;
84static VTermState *state;
85static VTermScreen *screen;
86
87static VTermEncodingInstance encoding;
88
Bram Moolenaarcd630be2020-04-12 14:50:26 +020089static void term_output(const char *s, size_t len, void *user UNUSED)
Bram Moolenaar88c1ee82020-04-12 13:38:57 +020090{
91 size_t i;
92
93 printf("output ");
94 for(i = 0; i < len; i++)
95 printf("%x%s", (unsigned char)s[i], i < len-1 ? "," : "\n");
96}
97
98static void printhex(const char *s, size_t len)
99{
100 while(len--)
101 printf("%02x", (s++)[0]);
102}
103
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200104static int parser_text(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200105{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200106 size_t i;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200107
108 printf("text ");
109 for(i = 0; i < len; i++) {
110 unsigned char b = bytes[i];
111 if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
112 break;
113 printf(i ? ",%x" : "%x", b);
114 }
115 printf("\n");
116
117 return i;
118}
119
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200120static int parser_control(unsigned char control, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200121{
122 printf("control %02x\n", control);
123
124 return 1;
125}
126
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200127static int parser_escape(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200128{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200129 if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
130 if(len < 2)
131 return -1;
132 len = 2;
133 }
134 else {
135 len = 1;
136 }
137
138 printf("escape ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200139 printhex(bytes, len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200140 printf("\n");
141
142 return len;
143}
144
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200145static 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 +0200146{
147 int i;
148 printf("csi %02x", command);
149
150 if(leader && leader[0]) {
151 printf(" L=");
152 for(i = 0; leader[i]; i++)
153 printf("%02x", leader[i]);
154 }
155
156 for(i = 0; i < argcount; i++) {
157 char sep = i ? ',' : ' ';
158
159 if(args[i] == CSI_ARG_MISSING)
160 printf("%c*", sep);
161 else
162 printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
163 }
164
165 if(intermed && intermed[0]) {
166 printf(" I=");
167 for(i = 0; intermed[i]; i++)
168 printf("%02x", intermed[i]);
169 }
170
171 printf("\n");
172
173 return 1;
174}
175
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200176static int parser_osc(int command, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200177{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200178
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200179 printf("osc ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200180
181 if(frag.initial) {
182 if(command == -1)
183 printf("[");
184 else
185 printf("[%d;", command);
186 }
187
188 printhex(frag.str, frag.len);
189
190 if(frag.final)
191 printf("]");
192
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200193 printf("\n");
194
195 return 1;
196}
197
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200198static int parser_dcs(const char *command, size_t commandlen, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200199{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200200 printf("dcs ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200201
202 if(frag.initial) {
203 size_t i;
204 printf("[");
205 for(i = 0; i < commandlen; i++)
206 printf("%02x", command[i]);
207 }
208
209 printhex(frag.str, frag.len);
210
211 if(frag.final)
212 printf("]");
213
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200214 printf("\n");
215
216 return 1;
217}
218
219static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100220 parser_text, // text
221 parser_control, // control
222 parser_escape, // escape
223 parser_csi, // csi
224 parser_osc, // osc
225 parser_dcs, // dcs
226 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200227};
228
Bram Moolenaard8637282020-05-20 18:41:41 +0200229static VTermStateFallbacks fallbacks = {
230 parser_control, // control
231 parser_csi, // csi
232 parser_osc, // osc
233 parser_dcs // dcs
234};
235
236/* These callbacks are shared by State and Screen */
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200237
238static int want_movecursor = 0;
239static VTermPos state_pos;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200240static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200241{
242 state_pos = pos;
243
244 if(want_movecursor)
245 printf("movecursor %d,%d\n", pos.row, pos.col);
246
247 return 1;
248}
249
250static int want_scrollrect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200251static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200252{
253 if(!want_scrollrect)
254 return 0;
255
256 printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
257 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
258 downward, rightward);
259
260 return 1;
261}
262
263static int want_moverect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200264static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200265{
266 if(!want_moverect)
267 return 0;
268
269 printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
270 src.start_row, src.end_row, src.start_col, src.end_col,
271 dest.start_row, dest.end_row, dest.start_col, dest.end_col);
272
273 return 1;
274}
275
276static int want_settermprop = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200277static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200278{
279 VTermValueType type;
280 if(!want_settermprop)
281 return 1;
282
283 type = vterm_get_prop_type(prop);
284 switch(type) {
285 case VTERM_VALUETYPE_BOOL:
286 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
287 return 1;
288 case VTERM_VALUETYPE_INT:
289 printf("settermprop %d %d\n", prop, val->number);
290 return 1;
291 case VTERM_VALUETYPE_STRING:
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200292 printf("settermprop %d %s\"%.*s\"%s\n", prop,
293 val->string.initial ? "[" : "", val->string.len, val->string.str, val->string.final ? "]" : "");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200294 return 1;
295 case VTERM_VALUETYPE_COLOR:
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200296 printf("settermprop %d ", prop);
297 print_color(&val->color);
298 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200299 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200300
301 case VTERM_N_VALUETYPES:
302 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200303 }
304
305 return 0;
306}
307
Bram Moolenaar707d2262019-12-04 22:16:54 +0100308// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200309
310static int want_state_putglyph = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200311static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200312{
313 int i;
314 if(!want_state_putglyph)
315 return 1;
316
317 printf("putglyph ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200318 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && info->chars[i]; i++)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200319 printf(i ? ",%x" : "%x", info->chars[i]);
320 printf(" %d %d,%d", info->width, pos.row, pos.col);
321 if(info->protected_cell)
322 printf(" prot");
323 if(info->dwl)
324 printf(" dwl");
325 if(info->dhl)
326 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
327 printf("\n");
328
329 return 1;
330}
331
332static int want_state_erase = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200333static int state_erase(VTermRect rect, int selective, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200334{
335 if(!want_state_erase)
336 return 1;
337
338 printf("erase %d..%d,%d..%d%s\n",
339 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
340 selective ? " selective" : "");
341
342 return 1;
343}
344
345static struct {
346 int bold;
347 int underline;
348 int italic;
349 int blink;
350 int reverse;
Bram Moolenaard8637282020-05-20 18:41:41 +0200351 int conceal;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200352 int strike;
353 int font;
354 VTermColor foreground;
355 VTermColor background;
356} state_pen;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200357static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200358{
359 switch(attr) {
360 case VTERM_ATTR_BOLD:
361 state_pen.bold = val->boolean;
362 break;
363 case VTERM_ATTR_UNDERLINE:
364 state_pen.underline = val->number;
365 break;
366 case VTERM_ATTR_ITALIC:
367 state_pen.italic = val->boolean;
368 break;
369 case VTERM_ATTR_BLINK:
370 state_pen.blink = val->boolean;
371 break;
372 case VTERM_ATTR_REVERSE:
373 state_pen.reverse = val->boolean;
374 break;
Bram Moolenaard8637282020-05-20 18:41:41 +0200375 case VTERM_ATTR_CONCEAL:
376 state_pen.conceal = val->boolean;
377 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200378 case VTERM_ATTR_STRIKE:
379 state_pen.strike = val->boolean;
380 break;
381 case VTERM_ATTR_FONT:
382 state_pen.font = val->number;
383 break;
384 case VTERM_ATTR_FOREGROUND:
385 state_pen.foreground = val->color;
386 break;
387 case VTERM_ATTR_BACKGROUND:
388 state_pen.background = val->color;
389 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200390
391 case VTERM_N_ATTRS:
392 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200393 }
394
395 return 1;
396}
397
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200398static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200399{
400 return 1;
401}
402
403VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100404 state_putglyph, // putglyph
405 movecursor, // movecursor
406 scrollrect, // scrollrect
407 moverect, // moverect
408 state_erase, // erase
409 NULL, // initpen
410 state_setpenattr, // setpenattr
411 settermprop, // settermprop
412 NULL, // bell
413 NULL, // resize
414 state_setlineinfo, // setlineinfo
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200415};
416
417static int want_screen_damage = 0;
418static int want_screen_damage_cells = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200419static int screen_damage(VTermRect rect, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200420{
421 if(!want_screen_damage)
422 return 1;
423
424 printf("damage %d..%d,%d..%d",
425 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
426
427 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200428 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200429 int row;
430 int col;
431
432 for(row = rect.start_row; row < rect.end_row; row++) {
433 int eol = rect.end_col;
434 while(eol > rect.start_col) {
435 VTermScreenCell cell;
436 VTermPos pos;
437 pos.row = row;
438 pos.col = eol-1;
439 vterm_screen_get_cell(screen, pos, &cell);
440 if(cell.chars[0])
441 break;
442
443 eol--;
444 }
445
446 if(eol == rect.start_col)
447 break;
448
449 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200450 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200451
452 printf(" %d<", row);
453 for(col = rect.start_col; col < eol; col++) {
454 VTermScreenCell cell;
455 VTermPos pos;
456 pos.row = row;
457 pos.col = col;
458 vterm_screen_get_cell(screen, pos, &cell);
459 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
460 }
461 printf(">");
462 }
463 }
464
465 printf("\n");
466
467 return 1;
468}
469
470static int want_screen_scrollback = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200471static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200472{
473 int eol;
474 int c;
475
476 if(!want_screen_scrollback)
477 return 1;
478
479 eol = cols;
480 while(eol && !cells[eol-1].chars[0])
481 eol--;
482
483 printf("sb_pushline %d =", cols);
484 for(c = 0; c < eol; c++)
485 printf(" %02X", cells[c].chars[0]);
486 printf("\n");
487
488 return 1;
489}
490
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200491static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200492{
493 int col;
494
495 if(!want_screen_scrollback)
496 return 0;
497
Bram Moolenaar707d2262019-12-04 22:16:54 +0100498 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200499 for(col = 0; col < cols; col++) {
500 if(col < 5)
501 cells[col].chars[0] = 'A' + col;
502 else
503 cells[col].chars[0] = 0;
504
505 cells[col].width = 1;
506 }
507
508 printf("sb_popline %d\n", cols);
509 return 1;
510}
511
512VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100513 screen_damage, // damage
514 moverect, // moverect
515 movecursor, // movecursor
516 settermprop, // settermprop
517 NULL, // bell
518 NULL, // resize
519 screen_sb_pushline, // sb_pushline
520 screen_sb_popline // sb_popline
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200521};
522
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200523int main(int argc UNUSED, char **argv UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200524{
525 char line[1024] = {0};
526 int flag;
527
528 int err;
529
530 setvbuf(stdout, NULL, _IONBF, 0);
531
532 while(fgets(line, sizeof line, stdin)) {
533 char *nl;
534 size_t outlen;
535 err = 0;
536
537 if((nl = strchr(line, '\n')))
538 *nl = '\0';
539
540 if(streq(line, "INIT")) {
541 if(!vt)
542 vt = vterm_new(25, 80);
Bram Moolenaar94d729c2020-05-17 21:50:16 +0200543
544 // Somehow this makes tests fail
545 // vterm_output_set_callback(vt, term_output, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200546 }
547
548 else if(streq(line, "WANTPARSER")) {
549 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
550 }
551
552 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
553 int i = 9;
554 int sense = 1;
555 if(!state) {
556 state = vterm_obtain_state(vt);
557 vterm_state_set_callbacks(state, &state_cbs, NULL);
558 vterm_state_set_bold_highbright(state, 1);
559 vterm_state_reset(state, 1);
560 }
561
562 while(line[i] == ' ')
563 i++;
564 for( ; line[i]; i++)
565 switch(line[i]) {
566 case '+':
567 sense = 1;
568 break;
569 case '-':
570 sense = 0;
571 break;
572 case 'g':
573 want_state_putglyph = sense;
574 break;
575 case 's':
576 want_scrollrect = sense;
577 break;
578 case 'm':
579 want_moverect = sense;
580 break;
581 case 'e':
582 want_state_erase = sense;
583 break;
584 case 'p':
585 want_settermprop = sense;
586 break;
587 case 'f':
Bram Moolenaard8637282020-05-20 18:41:41 +0200588 vterm_state_set_unrecognised_fallbacks(state, sense ? &fallbacks : NULL, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200589 break;
590 default:
591 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
592 }
593 }
594
595 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
596 int i = 10;
597 int sense = 1;
598 if(!screen)
599 screen = vterm_obtain_screen(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200600 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
601
602 while(line[i] == ' ')
603 i++;
604 for( ; line[i]; i++)
605 switch(line[i]) {
606 case '-':
607 sense = 0;
608 break;
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200609 case 'a':
610 vterm_screen_enable_altscreen(screen, 1);
611 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200612 case 'd':
613 want_screen_damage = sense;
614 break;
615 case 'D':
616 want_screen_damage = sense;
617 want_screen_damage_cells = sense;
618 break;
619 case 'm':
620 want_moverect = sense;
621 break;
622 case 'c':
623 want_movecursor = sense;
624 break;
625 case 'p':
626 want_settermprop = 1;
627 break;
628 case 'b':
629 want_screen_scrollback = sense;
630 break;
631 default:
632 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
633 }
634 }
635
636 else if(sscanf(line, "UTF8 %d", &flag)) {
637 vterm_set_utf8(vt, flag);
638 }
639
640 else if(streq(line, "RESET")) {
641 if(state) {
642 vterm_state_reset(state, 1);
643 vterm_state_get_cursorpos(state, &state_pos);
644 }
645 if(screen) {
646 vterm_screen_reset(screen, 1);
647 }
648 }
649
650 else if(strstartswith(line, "RESIZE ")) {
651 int rows, cols;
652 char *linep = line + 7;
653 while(linep[0] == ' ')
654 linep++;
655 sscanf(linep, "%d, %d", &rows, &cols);
656 vterm_set_size(vt, rows, cols);
657 }
658
659 else if(strstartswith(line, "PUSH ")) {
660 char *bytes = line + 5;
661 size_t len = inplace_hex2bytes(bytes);
662 size_t written = vterm_input_write(vt, bytes, len);
663 if(written < len)
664 fprintf(stderr, "! short write\n");
665 }
666
667 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100668 // This isn't really external API but it's hard to get this out any
669 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200670 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
671 if(encoding.enc->init)
672 (*encoding.enc->init)(encoding.enc, encoding.data);
673 }
674
675 else if(strstartswith(line, "ENCIN ")) {
676 char *bytes = line + 6;
677 size_t len = inplace_hex2bytes(bytes);
678
679 uint32_t cp[1024];
680 int cpi = 0;
681 size_t pos = 0;
682
683 (*encoding.enc->decode)(encoding.enc, encoding.data,
684 cp, &cpi, len, bytes, &pos, len);
685
686 if(cpi > 0) {
687 int i;
688 printf("encout ");
689 for(i = 0; i < cpi; i++) {
690 printf(i ? ",%x" : "%x", cp[i]);
691 }
692 printf("\n");
693 }
694 }
695
696 else if(strstartswith(line, "INCHAR ")) {
697 char *linep = line + 7;
698 unsigned int c = 0;
699 VTermModifier mod;
700 while(linep[0] == ' ')
701 linep++;
702 mod = strpe_modifiers(&linep);
703 sscanf(linep, " %x", &c);
704
705 vterm_keyboard_unichar(vt, c, mod);
706 }
707
708 else if(strstartswith(line, "INKEY ")) {
709 VTermModifier mod;
710 VTermKey key;
711 char *linep = line + 6;
712 while(linep[0] == ' ')
713 linep++;
714 mod = strpe_modifiers(&linep);
715 while(linep[0] == ' ')
716 linep++;
717 key = strp_key(linep);
718
719 vterm_keyboard_key(vt, key, mod);
720 }
721
722 else if(strstartswith(line, "PASTE ")) {
723 char *linep = line + 6;
724 if(streq(linep, "START"))
725 vterm_keyboard_start_paste(vt);
726 else if(streq(linep, "END"))
727 vterm_keyboard_end_paste(vt);
728 else
729 goto abort_line;
730 }
731
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200732 else if(strstartswith(line, "FOCUS ")) {
733 char *linep = line + 6;
734 if(streq(linep, "IN"))
735 vterm_state_focus_in(state);
736 else if(streq(linep, "OUT"))
737 vterm_state_focus_out(state);
738 else
739 goto abort_line;
740 }
741
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200742 else if(strstartswith(line, "MOUSEMOVE ")) {
743 char *linep = line + 10;
744 int row, col, len;
745 VTermModifier mod;
746 while(linep[0] == ' ')
747 linep++;
748 sscanf(linep, "%d,%d%n", &row, &col, &len);
749 linep += len;
750 while(linep[0] == ' ')
751 linep++;
752 mod = strpe_modifiers(&linep);
753 vterm_mouse_move(vt, row, col, mod);
754 }
755
756 else if(strstartswith(line, "MOUSEBTN ")) {
757 char *linep = line + 9;
758 char press;
759 int button, len;
760 VTermModifier mod;
761 while(linep[0] == ' ')
762 linep++;
763 sscanf(linep, "%c %d%n", &press, &button, &len);
764 linep += len;
765 while(linep[0] == ' ')
766 linep++;
767 mod = strpe_modifiers(&linep);
768 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
769 }
770
771 else if(strstartswith(line, "DAMAGEMERGE ")) {
772 char *linep = line + 12;
773 while(linep[0] == ' ')
774 linep++;
775 if(streq(linep, "CELL"))
776 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
777 else if(streq(linep, "ROW"))
778 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
779 else if(streq(linep, "SCREEN"))
780 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
781 else if(streq(linep, "SCROLL"))
782 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
783 }
784
785 else if(strstartswith(line, "DAMAGEFLUSH")) {
786 vterm_screen_flush_damage(screen);
787 }
788
789 else if(line[0] == '?') {
790 if(streq(line, "?cursor")) {
791 VTermPos pos;
792 vterm_state_get_cursorpos(state, &pos);
793 if(pos.row != state_pos.row)
794 printf("! row mismatch: state=%d,%d event=%d,%d\n",
795 pos.row, pos.col, state_pos.row, state_pos.col);
796 else if(pos.col != state_pos.col)
797 printf("! col mismatch: state=%d,%d event=%d,%d\n",
798 pos.row, pos.col, state_pos.row, state_pos.col);
799 else
800 printf("%d,%d\n", state_pos.row, state_pos.col);
801 }
802 else if(strstartswith(line, "?pen ")) {
803 VTermValue val;
804 char *linep = line + 5;
805 while(linep[0] == ' ')
806 linep++;
807
808#define BOOLSTR(v) ((v) ? "on" : "off")
809
810 if(streq(linep, "bold")) {
811 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
812 if(val.boolean != state_pen.bold)
813 printf("! pen bold mismatch; state=%s, event=%s\n",
814 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
815 else
816 printf("%s\n", BOOLSTR(state_pen.bold));
817 }
818 else if(streq(linep, "underline")) {
819 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
820 if(val.boolean != state_pen.underline)
821 printf("! pen underline mismatch; state=%d, event=%d\n",
822 val.boolean, state_pen.underline);
823 else
824 printf("%d\n", state_pen.underline);
825 }
826 else if(streq(linep, "italic")) {
827 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
828 if(val.boolean != state_pen.italic)
829 printf("! pen italic mismatch; state=%s, event=%s\n",
830 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
831 else
832 printf("%s\n", BOOLSTR(state_pen.italic));
833 }
834 else if(streq(linep, "blink")) {
835 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
836 if(val.boolean != state_pen.blink)
837 printf("! pen blink mismatch; state=%s, event=%s\n",
838 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
839 else
840 printf("%s\n", BOOLSTR(state_pen.blink));
841 }
842 else if(streq(linep, "reverse")) {
843 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
844 if(val.boolean != state_pen.reverse)
845 printf("! pen reverse mismatch; state=%s, event=%s\n",
846 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
847 else
848 printf("%s\n", BOOLSTR(state_pen.reverse));
849 }
850 else if(streq(linep, "font")) {
851 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
852 if(val.boolean != state_pen.font)
853 printf("! pen font mismatch; state=%d, event=%d\n",
854 val.boolean, state_pen.font);
855 else
856 printf("%d\n", state_pen.font);
857 }
858 else if(streq(linep, "foreground")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200859 print_color(&state_pen.foreground);
860 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200861 }
862 else if(streq(linep, "background")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200863 print_color(&state_pen.background);
864 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200865 }
866 else
867 printf("?\n");
868 }
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200869 else if(strstartswith(line, "?lineinfo ")) {
870 char *linep = line + 10;
871 int row;
872 const VTermLineInfo *info;
873 while(linep[0] == ' ')
874 linep++;
875 if(sscanf(linep, "%d", &row) < 1) {
876 printf("! lineinfo unrecognised input\n");
877 goto abort_line;
878 }
879 info = vterm_state_get_lineinfo(state, row);
880 if(info->doublewidth)
881 printf("dwl ");
882 if(info->doubleheight)
883 printf("dhl ");
884 if(info->continuation)
885 printf("cont ");
886 printf("\n");
887 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200888 else if(strstartswith(line, "?screen_chars ")) {
889 char *linep = line + 13;
890 VTermRect rect;
891 size_t len;
892 while(linep[0] == ' ')
893 linep++;
894 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
895 printf("! screen_chars unrecognised input\n");
896 goto abort_line;
897 }
898 len = vterm_screen_get_chars(screen, NULL, 0, rect);
899 if(len == (size_t)-1)
900 printf("! screen_chars error\n");
901 else if(len == 0)
902 printf("\n");
903 else {
904 uint32_t *chars = malloc(sizeof(uint32_t) * len);
905 size_t i;
906 vterm_screen_get_chars(screen, chars, len, rect);
907 for(i = 0; i < len; i++) {
908 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
909 }
910 free(chars);
911 }
912 }
913 else if(strstartswith(line, "?screen_text ")) {
914 char *linep = line + 12;
915 VTermRect rect;
916 size_t len;
917 while(linep[0] == ' ')
918 linep++;
919 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
920 printf("! screen_text unrecognised input\n");
921 goto abort_line;
922 }
923 len = vterm_screen_get_text(screen, NULL, 0, rect);
924 if(len == (size_t)-1)
925 printf("! screen_text error\n");
926 else if(len == 0)
927 printf("\n");
928 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100929 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200930 unsigned char *buffer = malloc(len + 4);
931 unsigned char *text = buffer + 2;
932 text[-2] = 0x55; text[-1] = 0xAA;
933 text[len] = 0x55; text[len+1] = 0xAA;
934
935 vterm_screen_get_text(screen, (char *)text, len, rect);
936
937 if(text[-2] != 0x55 || text[-1] != 0xAA)
938 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
939 else if(text[len] != 0x55 || text[len+1] != 0xAA)
940 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
941 else
942 {
943 size_t i;
944 for(i = 0; i < len; i++) {
945 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
946 }
947 }
948
949 free(buffer);
950 }
951 }
952 else if(strstartswith(line, "?screen_cell ")) {
953 char *linep = line + 12;
954 int i;
955 VTermPos pos;
956 VTermScreenCell cell;
957 while(linep[0] == ' ')
958 linep++;
959 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
960 printf("! screen_cell unrecognised input\n");
961 goto abort_line;
962 }
963 if(!vterm_screen_get_cell(screen, pos, &cell))
964 goto abort_line;
965 printf("{");
966 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
967 printf("%s0x%x", i ? "," : "", cell.chars[i]);
968 }
969 printf("} width=%d attrs={", cell.width);
970 if(cell.attrs.bold) printf("B");
971 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
972 if(cell.attrs.italic) printf("I");
973 if(cell.attrs.blink) printf("K");
974 if(cell.attrs.reverse) printf("R");
975 if(cell.attrs.font) printf("F%d", cell.attrs.font);
976 printf("} ");
977 if(cell.attrs.dwl) printf("dwl ");
978 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200979 printf("fg=");
980 vterm_screen_convert_color_to_rgb(screen, &cell.fg);
981 print_color(&cell.fg);
982 printf(" bg=");
983 vterm_screen_convert_color_to_rgb(screen, &cell.bg);
984 print_color(&cell.bg);
985 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200986 }
987 else if(strstartswith(line, "?screen_eol ")) {
988 VTermPos pos;
989 char *linep = line + 12;
990 while(linep[0] == ' ')
991 linep++;
992 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
993 printf("! screen_eol unrecognised input\n");
994 goto abort_line;
995 }
996 printf("%d\n", vterm_screen_is_eol(screen, pos));
997 }
998 else if(strstartswith(line, "?screen_attrs_extent ")) {
999 VTermPos pos;
1000 VTermRect rect;
1001 char *linep = line + 21;
1002 while(linep[0] == ' ')
1003 linep++;
1004 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1005 printf("! screen_attrs_extent unrecognised input\n");
1006 goto abort_line;
1007 }
1008 rect.start_col = 0;
1009 rect.end_col = -1;
1010 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
1011 printf("! screen_attrs_extent failed\n");
1012 goto abort_line;
1013 }
1014 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
1015 }
1016 else
1017 printf("?\n");
1018
1019 memset(line, 0, sizeof line);
1020 continue;
1021 }
1022
1023 else
1024 abort_line: err = 1;
1025
1026 outlen = vterm_output_get_buffer_current(vt);
1027 if(outlen > 0) {
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001028 char outbuff[1024];
1029 vterm_output_read(vt, outbuff, outlen);
1030
Bram Moolenaar88c1ee82020-04-12 13:38:57 +02001031 term_output(outbuff, outlen, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001032 }
1033
1034 printf(err ? "?\n" : "DONE\n");
1035 }
1036
1037 vterm_free(vt);
1038
1039 return 0;
1040}