blob: 92882fdfd9a3291fadc3a475e3bd16b796a9fede [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
63static VTerm *vt;
64static VTermState *state;
65static VTermScreen *screen;
66
67static VTermEncodingInstance encoding;
68
Bram Moolenaarcd630be2020-04-12 14:50:26 +020069static void term_output(const char *s, size_t len, void *user UNUSED)
Bram Moolenaar88c1ee82020-04-12 13:38:57 +020070{
71 size_t i;
72
73 printf("output ");
74 for(i = 0; i < len; i++)
75 printf("%x%s", (unsigned char)s[i], i < len-1 ? "," : "\n");
76}
77
78static void printhex(const char *s, size_t len)
79{
80 while(len--)
81 printf("%02x", (s++)[0]);
82}
83
Bram Moolenaard19a8f92020-04-11 21:42:48 +020084static int parser_text(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +020085{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +020086 size_t i;
Bram Moolenaare4f25e42017-07-07 11:54:15 +020087
88 printf("text ");
89 for(i = 0; i < len; i++) {
90 unsigned char b = bytes[i];
91 if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
92 break;
93 printf(i ? ",%x" : "%x", b);
94 }
95 printf("\n");
96
97 return i;
98}
99
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200100static int parser_control(unsigned char control, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200101{
102 printf("control %02x\n", control);
103
104 return 1;
105}
106
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200107static int parser_escape(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200108{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200109 if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
110 if(len < 2)
111 return -1;
112 len = 2;
113 }
114 else {
115 len = 1;
116 }
117
118 printf("escape ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200119 printhex(bytes, len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200120 printf("\n");
121
122 return len;
123}
124
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200125static 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 +0200126{
127 int i;
128 printf("csi %02x", command);
129
130 if(leader && leader[0]) {
131 printf(" L=");
132 for(i = 0; leader[i]; i++)
133 printf("%02x", leader[i]);
134 }
135
136 for(i = 0; i < argcount; i++) {
137 char sep = i ? ',' : ' ';
138
139 if(args[i] == CSI_ARG_MISSING)
140 printf("%c*", sep);
141 else
142 printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
143 }
144
145 if(intermed && intermed[0]) {
146 printf(" I=");
147 for(i = 0; intermed[i]; i++)
148 printf("%02x", intermed[i]);
149 }
150
151 printf("\n");
152
153 return 1;
154}
155
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200156static int parser_osc(int command, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200157{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200158
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200159 printf("osc ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200160
161 if(frag.initial) {
162 if(command == -1)
163 printf("[");
164 else
165 printf("[%d;", command);
166 }
167
168 printhex(frag.str, frag.len);
169
170 if(frag.final)
171 printf("]");
172
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200173 printf("\n");
174
175 return 1;
176}
177
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200178static int parser_dcs(const char *command, size_t commandlen, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200179{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200180 printf("dcs ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200181
182 if(frag.initial) {
183 size_t i;
184 printf("[");
185 for(i = 0; i < commandlen; i++)
186 printf("%02x", command[i]);
187 }
188
189 printhex(frag.str, frag.len);
190
191 if(frag.final)
192 printf("]");
193
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200194 printf("\n");
195
196 return 1;
197}
198
199static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100200 parser_text, // text
201 parser_control, // control
202 parser_escape, // escape
203 parser_csi, // csi
204 parser_osc, // osc
205 parser_dcs, // dcs
206 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200207};
208
Bram Moolenaar707d2262019-12-04 22:16:54 +0100209// These callbacks are shared by State and Screen
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200210
211static int want_movecursor = 0;
212static VTermPos state_pos;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200213static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200214{
215 state_pos = pos;
216
217 if(want_movecursor)
218 printf("movecursor %d,%d\n", pos.row, pos.col);
219
220 return 1;
221}
222
223static int want_scrollrect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200224static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200225{
226 if(!want_scrollrect)
227 return 0;
228
229 printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
230 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
231 downward, rightward);
232
233 return 1;
234}
235
236static int want_moverect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200237static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200238{
239 if(!want_moverect)
240 return 0;
241
242 printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
243 src.start_row, src.end_row, src.start_col, src.end_col,
244 dest.start_row, dest.end_row, dest.start_col, dest.end_col);
245
246 return 1;
247}
248
249static int want_settermprop = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200250static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200251{
252 VTermValueType type;
253 if(!want_settermprop)
254 return 1;
255
256 type = vterm_get_prop_type(prop);
257 switch(type) {
258 case VTERM_VALUETYPE_BOOL:
259 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
260 return 1;
261 case VTERM_VALUETYPE_INT:
262 printf("settermprop %d %d\n", prop, val->number);
263 return 1;
264 case VTERM_VALUETYPE_STRING:
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200265 printf("settermprop %d %s\"%.*s\"%s\n", prop,
266 val->string.initial ? "[" : "", val->string.len, val->string.str, val->string.final ? "]" : "");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200267 return 1;
268 case VTERM_VALUETYPE_COLOR:
269 printf("settermprop %d rgb(%d,%d,%d)\n", prop, val->color.red, val->color.green, val->color.blue);
270 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200271
272 case VTERM_N_VALUETYPES:
273 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200274 }
275
276 return 0;
277}
278
Bram Moolenaar707d2262019-12-04 22:16:54 +0100279// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200280
281static int want_state_putglyph = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200282static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200283{
284 int i;
285 if(!want_state_putglyph)
286 return 1;
287
288 printf("putglyph ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200289 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && info->chars[i]; i++)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200290 printf(i ? ",%x" : "%x", info->chars[i]);
291 printf(" %d %d,%d", info->width, pos.row, pos.col);
292 if(info->protected_cell)
293 printf(" prot");
294 if(info->dwl)
295 printf(" dwl");
296 if(info->dhl)
297 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
298 printf("\n");
299
300 return 1;
301}
302
303static int want_state_erase = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200304static int state_erase(VTermRect rect, int selective, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200305{
306 if(!want_state_erase)
307 return 1;
308
309 printf("erase %d..%d,%d..%d%s\n",
310 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
311 selective ? " selective" : "");
312
313 return 1;
314}
315
316static struct {
317 int bold;
318 int underline;
319 int italic;
320 int blink;
321 int reverse;
322 int strike;
323 int font;
324 VTermColor foreground;
325 VTermColor background;
326} state_pen;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200327static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200328{
329 switch(attr) {
330 case VTERM_ATTR_BOLD:
331 state_pen.bold = val->boolean;
332 break;
333 case VTERM_ATTR_UNDERLINE:
334 state_pen.underline = val->number;
335 break;
336 case VTERM_ATTR_ITALIC:
337 state_pen.italic = val->boolean;
338 break;
339 case VTERM_ATTR_BLINK:
340 state_pen.blink = val->boolean;
341 break;
342 case VTERM_ATTR_REVERSE:
343 state_pen.reverse = val->boolean;
344 break;
345 case VTERM_ATTR_STRIKE:
346 state_pen.strike = val->boolean;
347 break;
348 case VTERM_ATTR_FONT:
349 state_pen.font = val->number;
350 break;
351 case VTERM_ATTR_FOREGROUND:
352 state_pen.foreground = val->color;
353 break;
354 case VTERM_ATTR_BACKGROUND:
355 state_pen.background = val->color;
356 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200357
358 case VTERM_N_ATTRS:
359 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200360 }
361
362 return 1;
363}
364
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200365static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200366{
367 return 1;
368}
369
370VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100371 state_putglyph, // putglyph
372 movecursor, // movecursor
373 scrollrect, // scrollrect
374 moverect, // moverect
375 state_erase, // erase
376 NULL, // initpen
377 state_setpenattr, // setpenattr
378 settermprop, // settermprop
379 NULL, // bell
380 NULL, // resize
381 state_setlineinfo, // setlineinfo
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200382};
383
384static int want_screen_damage = 0;
385static int want_screen_damage_cells = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200386static int screen_damage(VTermRect rect, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200387{
388 if(!want_screen_damage)
389 return 1;
390
391 printf("damage %d..%d,%d..%d",
392 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
393
394 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200395 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200396 int row;
397 int col;
398
399 for(row = rect.start_row; row < rect.end_row; row++) {
400 int eol = rect.end_col;
401 while(eol > rect.start_col) {
402 VTermScreenCell cell;
403 VTermPos pos;
404 pos.row = row;
405 pos.col = eol-1;
406 vterm_screen_get_cell(screen, pos, &cell);
407 if(cell.chars[0])
408 break;
409
410 eol--;
411 }
412
413 if(eol == rect.start_col)
414 break;
415
416 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200417 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200418
419 printf(" %d<", row);
420 for(col = rect.start_col; col < eol; col++) {
421 VTermScreenCell cell;
422 VTermPos pos;
423 pos.row = row;
424 pos.col = col;
425 vterm_screen_get_cell(screen, pos, &cell);
426 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
427 }
428 printf(">");
429 }
430 }
431
432 printf("\n");
433
434 return 1;
435}
436
437static int want_screen_scrollback = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200438static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200439{
440 int eol;
441 int c;
442
443 if(!want_screen_scrollback)
444 return 1;
445
446 eol = cols;
447 while(eol && !cells[eol-1].chars[0])
448 eol--;
449
450 printf("sb_pushline %d =", cols);
451 for(c = 0; c < eol; c++)
452 printf(" %02X", cells[c].chars[0]);
453 printf("\n");
454
455 return 1;
456}
457
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200458static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200459{
460 int col;
461
462 if(!want_screen_scrollback)
463 return 0;
464
Bram Moolenaar707d2262019-12-04 22:16:54 +0100465 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200466 for(col = 0; col < cols; col++) {
467 if(col < 5)
468 cells[col].chars[0] = 'A' + col;
469 else
470 cells[col].chars[0] = 0;
471
472 cells[col].width = 1;
473 }
474
475 printf("sb_popline %d\n", cols);
476 return 1;
477}
478
479VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100480 screen_damage, // damage
481 moverect, // moverect
482 movecursor, // movecursor
483 settermprop, // settermprop
484 NULL, // bell
485 NULL, // resize
486 screen_sb_pushline, // sb_pushline
487 screen_sb_popline // sb_popline
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200488};
489
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200490int main(int argc UNUSED, char **argv UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200491{
492 char line[1024] = {0};
493 int flag;
494
495 int err;
496
497 setvbuf(stdout, NULL, _IONBF, 0);
498
499 while(fgets(line, sizeof line, stdin)) {
500 char *nl;
501 size_t outlen;
502 err = 0;
503
504 if((nl = strchr(line, '\n')))
505 *nl = '\0';
506
507 if(streq(line, "INIT")) {
508 if(!vt)
509 vt = vterm_new(25, 80);
Bram Moolenaar94d729c2020-05-17 21:50:16 +0200510
511 // Somehow this makes tests fail
512 // vterm_output_set_callback(vt, term_output, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200513 }
514
515 else if(streq(line, "WANTPARSER")) {
516 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
517 }
518
519 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
520 int i = 9;
521 int sense = 1;
522 if(!state) {
523 state = vterm_obtain_state(vt);
524 vterm_state_set_callbacks(state, &state_cbs, NULL);
525 vterm_state_set_bold_highbright(state, 1);
526 vterm_state_reset(state, 1);
527 }
528
529 while(line[i] == ' ')
530 i++;
531 for( ; line[i]; i++)
532 switch(line[i]) {
533 case '+':
534 sense = 1;
535 break;
536 case '-':
537 sense = 0;
538 break;
539 case 'g':
540 want_state_putglyph = sense;
541 break;
542 case 's':
543 want_scrollrect = sense;
544 break;
545 case 'm':
546 want_moverect = sense;
547 break;
548 case 'e':
549 want_state_erase = sense;
550 break;
551 case 'p':
552 want_settermprop = sense;
553 break;
554 case 'f':
555 vterm_state_set_unrecognised_fallbacks(state, sense ? &parser_cbs : NULL, NULL);
556 break;
557 default:
558 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
559 }
560 }
561
562 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
563 int i = 10;
564 int sense = 1;
565 if(!screen)
566 screen = vterm_obtain_screen(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200567 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
568
569 while(line[i] == ' ')
570 i++;
571 for( ; line[i]; i++)
572 switch(line[i]) {
573 case '-':
574 sense = 0;
575 break;
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200576 case 'a':
577 vterm_screen_enable_altscreen(screen, 1);
578 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200579 case 'd':
580 want_screen_damage = sense;
581 break;
582 case 'D':
583 want_screen_damage = sense;
584 want_screen_damage_cells = sense;
585 break;
586 case 'm':
587 want_moverect = sense;
588 break;
589 case 'c':
590 want_movecursor = sense;
591 break;
592 case 'p':
593 want_settermprop = 1;
594 break;
595 case 'b':
596 want_screen_scrollback = sense;
597 break;
598 default:
599 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
600 }
601 }
602
603 else if(sscanf(line, "UTF8 %d", &flag)) {
604 vterm_set_utf8(vt, flag);
605 }
606
607 else if(streq(line, "RESET")) {
608 if(state) {
609 vterm_state_reset(state, 1);
610 vterm_state_get_cursorpos(state, &state_pos);
611 }
612 if(screen) {
613 vterm_screen_reset(screen, 1);
614 }
615 }
616
617 else if(strstartswith(line, "RESIZE ")) {
618 int rows, cols;
619 char *linep = line + 7;
620 while(linep[0] == ' ')
621 linep++;
622 sscanf(linep, "%d, %d", &rows, &cols);
623 vterm_set_size(vt, rows, cols);
624 }
625
626 else if(strstartswith(line, "PUSH ")) {
627 char *bytes = line + 5;
628 size_t len = inplace_hex2bytes(bytes);
629 size_t written = vterm_input_write(vt, bytes, len);
630 if(written < len)
631 fprintf(stderr, "! short write\n");
632 }
633
634 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100635 // This isn't really external API but it's hard to get this out any
636 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200637 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
638 if(encoding.enc->init)
639 (*encoding.enc->init)(encoding.enc, encoding.data);
640 }
641
642 else if(strstartswith(line, "ENCIN ")) {
643 char *bytes = line + 6;
644 size_t len = inplace_hex2bytes(bytes);
645
646 uint32_t cp[1024];
647 int cpi = 0;
648 size_t pos = 0;
649
650 (*encoding.enc->decode)(encoding.enc, encoding.data,
651 cp, &cpi, len, bytes, &pos, len);
652
653 if(cpi > 0) {
654 int i;
655 printf("encout ");
656 for(i = 0; i < cpi; i++) {
657 printf(i ? ",%x" : "%x", cp[i]);
658 }
659 printf("\n");
660 }
661 }
662
663 else if(strstartswith(line, "INCHAR ")) {
664 char *linep = line + 7;
665 unsigned int c = 0;
666 VTermModifier mod;
667 while(linep[0] == ' ')
668 linep++;
669 mod = strpe_modifiers(&linep);
670 sscanf(linep, " %x", &c);
671
672 vterm_keyboard_unichar(vt, c, mod);
673 }
674
675 else if(strstartswith(line, "INKEY ")) {
676 VTermModifier mod;
677 VTermKey key;
678 char *linep = line + 6;
679 while(linep[0] == ' ')
680 linep++;
681 mod = strpe_modifiers(&linep);
682 while(linep[0] == ' ')
683 linep++;
684 key = strp_key(linep);
685
686 vterm_keyboard_key(vt, key, mod);
687 }
688
689 else if(strstartswith(line, "PASTE ")) {
690 char *linep = line + 6;
691 if(streq(linep, "START"))
692 vterm_keyboard_start_paste(vt);
693 else if(streq(linep, "END"))
694 vterm_keyboard_end_paste(vt);
695 else
696 goto abort_line;
697 }
698
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200699 else if(strstartswith(line, "FOCUS ")) {
700 char *linep = line + 6;
701 if(streq(linep, "IN"))
702 vterm_state_focus_in(state);
703 else if(streq(linep, "OUT"))
704 vterm_state_focus_out(state);
705 else
706 goto abort_line;
707 }
708
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200709 else if(strstartswith(line, "MOUSEMOVE ")) {
710 char *linep = line + 10;
711 int row, col, len;
712 VTermModifier mod;
713 while(linep[0] == ' ')
714 linep++;
715 sscanf(linep, "%d,%d%n", &row, &col, &len);
716 linep += len;
717 while(linep[0] == ' ')
718 linep++;
719 mod = strpe_modifiers(&linep);
720 vterm_mouse_move(vt, row, col, mod);
721 }
722
723 else if(strstartswith(line, "MOUSEBTN ")) {
724 char *linep = line + 9;
725 char press;
726 int button, len;
727 VTermModifier mod;
728 while(linep[0] == ' ')
729 linep++;
730 sscanf(linep, "%c %d%n", &press, &button, &len);
731 linep += len;
732 while(linep[0] == ' ')
733 linep++;
734 mod = strpe_modifiers(&linep);
735 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
736 }
737
738 else if(strstartswith(line, "DAMAGEMERGE ")) {
739 char *linep = line + 12;
740 while(linep[0] == ' ')
741 linep++;
742 if(streq(linep, "CELL"))
743 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
744 else if(streq(linep, "ROW"))
745 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
746 else if(streq(linep, "SCREEN"))
747 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
748 else if(streq(linep, "SCROLL"))
749 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
750 }
751
752 else if(strstartswith(line, "DAMAGEFLUSH")) {
753 vterm_screen_flush_damage(screen);
754 }
755
756 else if(line[0] == '?') {
757 if(streq(line, "?cursor")) {
758 VTermPos pos;
759 vterm_state_get_cursorpos(state, &pos);
760 if(pos.row != state_pos.row)
761 printf("! row mismatch: state=%d,%d event=%d,%d\n",
762 pos.row, pos.col, state_pos.row, state_pos.col);
763 else if(pos.col != state_pos.col)
764 printf("! col mismatch: state=%d,%d event=%d,%d\n",
765 pos.row, pos.col, state_pos.row, state_pos.col);
766 else
767 printf("%d,%d\n", state_pos.row, state_pos.col);
768 }
769 else if(strstartswith(line, "?pen ")) {
770 VTermValue val;
771 char *linep = line + 5;
772 while(linep[0] == ' ')
773 linep++;
774
775#define BOOLSTR(v) ((v) ? "on" : "off")
776
777 if(streq(linep, "bold")) {
778 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
779 if(val.boolean != state_pen.bold)
780 printf("! pen bold mismatch; state=%s, event=%s\n",
781 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
782 else
783 printf("%s\n", BOOLSTR(state_pen.bold));
784 }
785 else if(streq(linep, "underline")) {
786 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
787 if(val.boolean != state_pen.underline)
788 printf("! pen underline mismatch; state=%d, event=%d\n",
789 val.boolean, state_pen.underline);
790 else
791 printf("%d\n", state_pen.underline);
792 }
793 else if(streq(linep, "italic")) {
794 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
795 if(val.boolean != state_pen.italic)
796 printf("! pen italic mismatch; state=%s, event=%s\n",
797 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
798 else
799 printf("%s\n", BOOLSTR(state_pen.italic));
800 }
801 else if(streq(linep, "blink")) {
802 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
803 if(val.boolean != state_pen.blink)
804 printf("! pen blink mismatch; state=%s, event=%s\n",
805 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
806 else
807 printf("%s\n", BOOLSTR(state_pen.blink));
808 }
809 else if(streq(linep, "reverse")) {
810 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
811 if(val.boolean != state_pen.reverse)
812 printf("! pen reverse mismatch; state=%s, event=%s\n",
813 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
814 else
815 printf("%s\n", BOOLSTR(state_pen.reverse));
816 }
817 else if(streq(linep, "font")) {
818 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
819 if(val.boolean != state_pen.font)
820 printf("! pen font mismatch; state=%d, event=%d\n",
821 val.boolean, state_pen.font);
822 else
823 printf("%d\n", state_pen.font);
824 }
825 else if(streq(linep, "foreground")) {
826 printf("rgb(%d,%d,%d)\n", state_pen.foreground.red, state_pen.foreground.green, state_pen.foreground.blue);
827 }
828 else if(streq(linep, "background")) {
829 printf("rgb(%d,%d,%d)\n", state_pen.background.red, state_pen.background.green, state_pen.background.blue);
830 }
831 else
832 printf("?\n");
833 }
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200834 else if(strstartswith(line, "?lineinfo ")) {
835 char *linep = line + 10;
836 int row;
837 const VTermLineInfo *info;
838 while(linep[0] == ' ')
839 linep++;
840 if(sscanf(linep, "%d", &row) < 1) {
841 printf("! lineinfo unrecognised input\n");
842 goto abort_line;
843 }
844 info = vterm_state_get_lineinfo(state, row);
845 if(info->doublewidth)
846 printf("dwl ");
847 if(info->doubleheight)
848 printf("dhl ");
849 if(info->continuation)
850 printf("cont ");
851 printf("\n");
852 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200853 else if(strstartswith(line, "?screen_chars ")) {
854 char *linep = line + 13;
855 VTermRect rect;
856 size_t len;
857 while(linep[0] == ' ')
858 linep++;
859 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
860 printf("! screen_chars unrecognised input\n");
861 goto abort_line;
862 }
863 len = vterm_screen_get_chars(screen, NULL, 0, rect);
864 if(len == (size_t)-1)
865 printf("! screen_chars error\n");
866 else if(len == 0)
867 printf("\n");
868 else {
869 uint32_t *chars = malloc(sizeof(uint32_t) * len);
870 size_t i;
871 vterm_screen_get_chars(screen, chars, len, rect);
872 for(i = 0; i < len; i++) {
873 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
874 }
875 free(chars);
876 }
877 }
878 else if(strstartswith(line, "?screen_text ")) {
879 char *linep = line + 12;
880 VTermRect rect;
881 size_t len;
882 while(linep[0] == ' ')
883 linep++;
884 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
885 printf("! screen_text unrecognised input\n");
886 goto abort_line;
887 }
888 len = vterm_screen_get_text(screen, NULL, 0, rect);
889 if(len == (size_t)-1)
890 printf("! screen_text error\n");
891 else if(len == 0)
892 printf("\n");
893 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100894 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200895 unsigned char *buffer = malloc(len + 4);
896 unsigned char *text = buffer + 2;
897 text[-2] = 0x55; text[-1] = 0xAA;
898 text[len] = 0x55; text[len+1] = 0xAA;
899
900 vterm_screen_get_text(screen, (char *)text, len, rect);
901
902 if(text[-2] != 0x55 || text[-1] != 0xAA)
903 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
904 else if(text[len] != 0x55 || text[len+1] != 0xAA)
905 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
906 else
907 {
908 size_t i;
909 for(i = 0; i < len; i++) {
910 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
911 }
912 }
913
914 free(buffer);
915 }
916 }
917 else if(strstartswith(line, "?screen_cell ")) {
918 char *linep = line + 12;
919 int i;
920 VTermPos pos;
921 VTermScreenCell cell;
922 while(linep[0] == ' ')
923 linep++;
924 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
925 printf("! screen_cell unrecognised input\n");
926 goto abort_line;
927 }
928 if(!vterm_screen_get_cell(screen, pos, &cell))
929 goto abort_line;
930 printf("{");
931 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
932 printf("%s0x%x", i ? "," : "", cell.chars[i]);
933 }
934 printf("} width=%d attrs={", cell.width);
935 if(cell.attrs.bold) printf("B");
936 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
937 if(cell.attrs.italic) printf("I");
938 if(cell.attrs.blink) printf("K");
939 if(cell.attrs.reverse) printf("R");
940 if(cell.attrs.font) printf("F%d", cell.attrs.font);
941 printf("} ");
942 if(cell.attrs.dwl) printf("dwl ");
943 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
944 printf("fg=rgb(%d,%d,%d) ", cell.fg.red, cell.fg.green, cell.fg.blue);
945 printf("bg=rgb(%d,%d,%d)\n", cell.bg.red, cell.bg.green, cell.bg.blue);
946 }
947 else if(strstartswith(line, "?screen_eol ")) {
948 VTermPos pos;
949 char *linep = line + 12;
950 while(linep[0] == ' ')
951 linep++;
952 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
953 printf("! screen_eol unrecognised input\n");
954 goto abort_line;
955 }
956 printf("%d\n", vterm_screen_is_eol(screen, pos));
957 }
958 else if(strstartswith(line, "?screen_attrs_extent ")) {
959 VTermPos pos;
960 VTermRect rect;
961 char *linep = line + 21;
962 while(linep[0] == ' ')
963 linep++;
964 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
965 printf("! screen_attrs_extent unrecognised input\n");
966 goto abort_line;
967 }
968 rect.start_col = 0;
969 rect.end_col = -1;
970 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
971 printf("! screen_attrs_extent failed\n");
972 goto abort_line;
973 }
974 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
975 }
976 else
977 printf("?\n");
978
979 memset(line, 0, sizeof line);
980 continue;
981 }
982
983 else
984 abort_line: err = 1;
985
986 outlen = vterm_output_get_buffer_current(vt);
987 if(outlen > 0) {
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200988 char outbuff[1024];
989 vterm_output_read(vt, outbuff, outlen);
990
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200991 term_output(outbuff, outlen, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200992 }
993
994 printf(err ? "?\n" : "DONE\n");
995 }
996
997 vterm_free(vt);
998
999 return 0;
1000}