blob: fed5844bfba74a64030f9fd67e0199a659808f23 [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 },
50 { NULL, VTERM_KEY_NONE },
51 };
52 int i;
53
54 for(i = 0; keys[i].name; i++) {
55 if(streq(str, keys[i].name))
56 return keys[i].key;
57 }
58
59 return VTERM_KEY_NONE;
60}
61
62static VTerm *vt;
63static VTermState *state;
64static VTermScreen *screen;
65
66static VTermEncodingInstance encoding;
67
68static int parser_text(const char bytes[], size_t len, void *user)
69{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +020070 size_t i;
71 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +020072
73 printf("text ");
74 for(i = 0; i < len; i++) {
75 unsigned char b = bytes[i];
76 if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
77 break;
78 printf(i ? ",%x" : "%x", b);
79 }
80 printf("\n");
81
82 return i;
83}
84
85static int parser_control(unsigned char control, void *user)
86{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +020087 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +020088 printf("control %02x\n", control);
89
90 return 1;
91}
92
93static int parser_escape(const char bytes[], size_t len, void *user)
94{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +020095 size_t i;
96 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +020097
98 if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
99 if(len < 2)
100 return -1;
101 len = 2;
102 }
103 else {
104 len = 1;
105 }
106
107 printf("escape ");
108 for(i = 0; i < len; i++)
109 printf("%02x", bytes[i]);
110 printf("\n");
111
112 return len;
113}
114
115static int parser_csi(const char *leader, const long args[], int argcount, const char *intermed, char command, void *user)
116{
117 int i;
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200118 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200119 printf("csi %02x", command);
120
121 if(leader && leader[0]) {
122 printf(" L=");
123 for(i = 0; leader[i]; i++)
124 printf("%02x", leader[i]);
125 }
126
127 for(i = 0; i < argcount; i++) {
128 char sep = i ? ',' : ' ';
129
130 if(args[i] == CSI_ARG_MISSING)
131 printf("%c*", sep);
132 else
133 printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
134 }
135
136 if(intermed && intermed[0]) {
137 printf(" I=");
138 for(i = 0; intermed[i]; i++)
139 printf("%02x", intermed[i]);
140 }
141
142 printf("\n");
143
144 return 1;
145}
146
147static int parser_osc(const char *command, size_t cmdlen, void *user)
148{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200149 size_t i;
150 (void)user;
151
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200152 printf("osc ");
153 for(i = 0; i < cmdlen; i++)
154 printf("%02x", command[i]);
155 printf("\n");
156
157 return 1;
158}
159
160static int parser_dcs(const char *command, size_t cmdlen, void *user)
161{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200162 size_t i;
163 (void)user;
164
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200165 printf("dcs ");
166 for(i = 0; i < cmdlen; i++)
167 printf("%02x", command[i]);
168 printf("\n");
169
170 return 1;
171}
172
173static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100174 parser_text, // text
175 parser_control, // control
176 parser_escape, // escape
177 parser_csi, // csi
178 parser_osc, // osc
179 parser_dcs, // dcs
180 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200181};
182
Bram Moolenaar707d2262019-12-04 22:16:54 +0100183// These callbacks are shared by State and Screen
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200184
185static int want_movecursor = 0;
186static VTermPos state_pos;
187static int movecursor(VTermPos pos, VTermPos oldpos, int visible, void *user)
188{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200189 (void)oldpos, (void)visible, (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200190 state_pos = pos;
191
192 if(want_movecursor)
193 printf("movecursor %d,%d\n", pos.row, pos.col);
194
195 return 1;
196}
197
198static int want_scrollrect = 0;
199static int scrollrect(VTermRect rect, int downward, int rightward, void *user)
200{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200201 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200202 if(!want_scrollrect)
203 return 0;
204
205 printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
206 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
207 downward, rightward);
208
209 return 1;
210}
211
212static int want_moverect = 0;
213static int moverect(VTermRect dest, VTermRect src, void *user)
214{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200215 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200216 if(!want_moverect)
217 return 0;
218
219 printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
220 src.start_row, src.end_row, src.start_col, src.end_col,
221 dest.start_row, dest.end_row, dest.start_col, dest.end_col);
222
223 return 1;
224}
225
226static int want_settermprop = 0;
227static int settermprop(VTermProp prop, VTermValue *val, void *user)
228{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200229 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200230 VTermValueType type;
231 if(!want_settermprop)
232 return 1;
233
234 type = vterm_get_prop_type(prop);
235 switch(type) {
236 case VTERM_VALUETYPE_BOOL:
237 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
238 return 1;
239 case VTERM_VALUETYPE_INT:
240 printf("settermprop %d %d\n", prop, val->number);
241 return 1;
242 case VTERM_VALUETYPE_STRING:
243 printf("settermprop %d \"%s\"\n", prop, val->string);
244 return 1;
245 case VTERM_VALUETYPE_COLOR:
246 printf("settermprop %d rgb(%d,%d,%d)\n", prop, val->color.red, val->color.green, val->color.blue);
247 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200248
249 case VTERM_N_VALUETYPES:
250 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200251 }
252
253 return 0;
254}
255
Bram Moolenaar707d2262019-12-04 22:16:54 +0100256// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200257
258static int want_state_putglyph = 0;
259static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user)
260{
261 int i;
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200262 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200263 if(!want_state_putglyph)
264 return 1;
265
266 printf("putglyph ");
267 for(i = 0; info->chars[i]; i++)
268 printf(i ? ",%x" : "%x", info->chars[i]);
269 printf(" %d %d,%d", info->width, pos.row, pos.col);
270 if(info->protected_cell)
271 printf(" prot");
272 if(info->dwl)
273 printf(" dwl");
274 if(info->dhl)
275 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
276 printf("\n");
277
278 return 1;
279}
280
281static int want_state_erase = 0;
282static int state_erase(VTermRect rect, int selective, void *user)
283{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200284 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200285 if(!want_state_erase)
286 return 1;
287
288 printf("erase %d..%d,%d..%d%s\n",
289 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
290 selective ? " selective" : "");
291
292 return 1;
293}
294
295static struct {
296 int bold;
297 int underline;
298 int italic;
299 int blink;
300 int reverse;
301 int strike;
302 int font;
303 VTermColor foreground;
304 VTermColor background;
305} state_pen;
306static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user)
307{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200308 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200309 switch(attr) {
310 case VTERM_ATTR_BOLD:
311 state_pen.bold = val->boolean;
312 break;
313 case VTERM_ATTR_UNDERLINE:
314 state_pen.underline = val->number;
315 break;
316 case VTERM_ATTR_ITALIC:
317 state_pen.italic = val->boolean;
318 break;
319 case VTERM_ATTR_BLINK:
320 state_pen.blink = val->boolean;
321 break;
322 case VTERM_ATTR_REVERSE:
323 state_pen.reverse = val->boolean;
324 break;
325 case VTERM_ATTR_STRIKE:
326 state_pen.strike = val->boolean;
327 break;
328 case VTERM_ATTR_FONT:
329 state_pen.font = val->number;
330 break;
331 case VTERM_ATTR_FOREGROUND:
332 state_pen.foreground = val->color;
333 break;
334 case VTERM_ATTR_BACKGROUND:
335 state_pen.background = val->color;
336 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200337
338 case VTERM_N_ATTRS:
339 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200340 }
341
342 return 1;
343}
344
345static int state_setlineinfo(int row, const VTermLineInfo *newinfo, const VTermLineInfo *oldinfo, void *user)
346{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200347 (void)row, (void)newinfo, (void)oldinfo, (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200348 return 1;
349}
350
351VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100352 state_putglyph, // putglyph
353 movecursor, // movecursor
354 scrollrect, // scrollrect
355 moverect, // moverect
356 state_erase, // erase
357 NULL, // initpen
358 state_setpenattr, // setpenattr
359 settermprop, // settermprop
360 NULL, // bell
361 NULL, // resize
362 state_setlineinfo, // setlineinfo
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200363};
364
365static int want_screen_damage = 0;
366static int want_screen_damage_cells = 0;
367static int screen_damage(VTermRect rect, void *user)
368{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200369 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200370 if(!want_screen_damage)
371 return 1;
372
373 printf("damage %d..%d,%d..%d",
374 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
375
376 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200377 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200378 int row;
379 int col;
380
381 for(row = rect.start_row; row < rect.end_row; row++) {
382 int eol = rect.end_col;
383 while(eol > rect.start_col) {
384 VTermScreenCell cell;
385 VTermPos pos;
386 pos.row = row;
387 pos.col = eol-1;
388 vterm_screen_get_cell(screen, pos, &cell);
389 if(cell.chars[0])
390 break;
391
392 eol--;
393 }
394
395 if(eol == rect.start_col)
396 break;
397
398 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200399 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200400
401 printf(" %d<", row);
402 for(col = rect.start_col; col < eol; col++) {
403 VTermScreenCell cell;
404 VTermPos pos;
405 pos.row = row;
406 pos.col = col;
407 vterm_screen_get_cell(screen, pos, &cell);
408 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
409 }
410 printf(">");
411 }
412 }
413
414 printf("\n");
415
416 return 1;
417}
418
419static int want_screen_scrollback = 0;
420static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user)
421{
422 int eol;
423 int c;
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200424 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200425
426 if(!want_screen_scrollback)
427 return 1;
428
429 eol = cols;
430 while(eol && !cells[eol-1].chars[0])
431 eol--;
432
433 printf("sb_pushline %d =", cols);
434 for(c = 0; c < eol; c++)
435 printf(" %02X", cells[c].chars[0]);
436 printf("\n");
437
438 return 1;
439}
440
441static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user)
442{
443 int col;
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200444 (void)user;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200445
446 if(!want_screen_scrollback)
447 return 0;
448
Bram Moolenaar707d2262019-12-04 22:16:54 +0100449 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200450 for(col = 0; col < cols; col++) {
451 if(col < 5)
452 cells[col].chars[0] = 'A' + col;
453 else
454 cells[col].chars[0] = 0;
455
456 cells[col].width = 1;
457 }
458
459 printf("sb_popline %d\n", cols);
460 return 1;
461}
462
463VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100464 screen_damage, // damage
465 moverect, // moverect
466 movecursor, // movecursor
467 settermprop, // settermprop
468 NULL, // bell
469 NULL, // resize
470 screen_sb_pushline, // sb_pushline
471 screen_sb_popline // sb_popline
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200472};
473
474int main(int argc, char **argv)
475{
476 char line[1024] = {0};
477 int flag;
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200478 (void)argc, (void)argv;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200479
480 int err;
481
482 setvbuf(stdout, NULL, _IONBF, 0);
483
484 while(fgets(line, sizeof line, stdin)) {
485 char *nl;
486 size_t outlen;
487 err = 0;
488
489 if((nl = strchr(line, '\n')))
490 *nl = '\0';
491
492 if(streq(line, "INIT")) {
493 if(!vt)
494 vt = vterm_new(25, 80);
495 }
496
497 else if(streq(line, "WANTPARSER")) {
498 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
499 }
500
501 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
502 int i = 9;
503 int sense = 1;
504 if(!state) {
505 state = vterm_obtain_state(vt);
506 vterm_state_set_callbacks(state, &state_cbs, NULL);
507 vterm_state_set_bold_highbright(state, 1);
508 vterm_state_reset(state, 1);
509 }
510
511 while(line[i] == ' ')
512 i++;
513 for( ; line[i]; i++)
514 switch(line[i]) {
515 case '+':
516 sense = 1;
517 break;
518 case '-':
519 sense = 0;
520 break;
521 case 'g':
522 want_state_putglyph = sense;
523 break;
524 case 's':
525 want_scrollrect = sense;
526 break;
527 case 'm':
528 want_moverect = sense;
529 break;
530 case 'e':
531 want_state_erase = sense;
532 break;
533 case 'p':
534 want_settermprop = sense;
535 break;
536 case 'f':
537 vterm_state_set_unrecognised_fallbacks(state, sense ? &parser_cbs : NULL, NULL);
538 break;
539 default:
540 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
541 }
542 }
543
544 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
545 int i = 10;
546 int sense = 1;
547 if(!screen)
548 screen = vterm_obtain_screen(vt);
549 vterm_screen_enable_altscreen(screen, 1);
550 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
551
552 while(line[i] == ' ')
553 i++;
554 for( ; line[i]; i++)
555 switch(line[i]) {
556 case '-':
557 sense = 0;
558 break;
559 case 'd':
560 want_screen_damage = sense;
561 break;
562 case 'D':
563 want_screen_damage = sense;
564 want_screen_damage_cells = sense;
565 break;
566 case 'm':
567 want_moverect = sense;
568 break;
569 case 'c':
570 want_movecursor = sense;
571 break;
572 case 'p':
573 want_settermprop = 1;
574 break;
575 case 'b':
576 want_screen_scrollback = sense;
577 break;
578 default:
579 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
580 }
581 }
582
583 else if(sscanf(line, "UTF8 %d", &flag)) {
584 vterm_set_utf8(vt, flag);
585 }
586
587 else if(streq(line, "RESET")) {
588 if(state) {
589 vterm_state_reset(state, 1);
590 vterm_state_get_cursorpos(state, &state_pos);
591 }
592 if(screen) {
593 vterm_screen_reset(screen, 1);
594 }
595 }
596
597 else if(strstartswith(line, "RESIZE ")) {
598 int rows, cols;
599 char *linep = line + 7;
600 while(linep[0] == ' ')
601 linep++;
602 sscanf(linep, "%d, %d", &rows, &cols);
603 vterm_set_size(vt, rows, cols);
604 }
605
606 else if(strstartswith(line, "PUSH ")) {
607 char *bytes = line + 5;
608 size_t len = inplace_hex2bytes(bytes);
609 size_t written = vterm_input_write(vt, bytes, len);
610 if(written < len)
611 fprintf(stderr, "! short write\n");
612 }
613
614 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100615 // This isn't really external API but it's hard to get this out any
616 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200617 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
618 if(encoding.enc->init)
619 (*encoding.enc->init)(encoding.enc, encoding.data);
620 }
621
622 else if(strstartswith(line, "ENCIN ")) {
623 char *bytes = line + 6;
624 size_t len = inplace_hex2bytes(bytes);
625
626 uint32_t cp[1024];
627 int cpi = 0;
628 size_t pos = 0;
629
630 (*encoding.enc->decode)(encoding.enc, encoding.data,
631 cp, &cpi, len, bytes, &pos, len);
632
633 if(cpi > 0) {
634 int i;
635 printf("encout ");
636 for(i = 0; i < cpi; i++) {
637 printf(i ? ",%x" : "%x", cp[i]);
638 }
639 printf("\n");
640 }
641 }
642
643 else if(strstartswith(line, "INCHAR ")) {
644 char *linep = line + 7;
645 unsigned int c = 0;
646 VTermModifier mod;
647 while(linep[0] == ' ')
648 linep++;
649 mod = strpe_modifiers(&linep);
650 sscanf(linep, " %x", &c);
651
652 vterm_keyboard_unichar(vt, c, mod);
653 }
654
655 else if(strstartswith(line, "INKEY ")) {
656 VTermModifier mod;
657 VTermKey key;
658 char *linep = line + 6;
659 while(linep[0] == ' ')
660 linep++;
661 mod = strpe_modifiers(&linep);
662 while(linep[0] == ' ')
663 linep++;
664 key = strp_key(linep);
665
666 vterm_keyboard_key(vt, key, mod);
667 }
668
669 else if(strstartswith(line, "PASTE ")) {
670 char *linep = line + 6;
671 if(streq(linep, "START"))
672 vterm_keyboard_start_paste(vt);
673 else if(streq(linep, "END"))
674 vterm_keyboard_end_paste(vt);
675 else
676 goto abort_line;
677 }
678
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200679 else if(strstartswith(line, "FOCUS ")) {
680 char *linep = line + 6;
681 if(streq(linep, "IN"))
682 vterm_state_focus_in(state);
683 else if(streq(linep, "OUT"))
684 vterm_state_focus_out(state);
685 else
686 goto abort_line;
687 }
688
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200689 else if(strstartswith(line, "MOUSEMOVE ")) {
690 char *linep = line + 10;
691 int row, col, len;
692 VTermModifier mod;
693 while(linep[0] == ' ')
694 linep++;
695 sscanf(linep, "%d,%d%n", &row, &col, &len);
696 linep += len;
697 while(linep[0] == ' ')
698 linep++;
699 mod = strpe_modifiers(&linep);
700 vterm_mouse_move(vt, row, col, mod);
701 }
702
703 else if(strstartswith(line, "MOUSEBTN ")) {
704 char *linep = line + 9;
705 char press;
706 int button, len;
707 VTermModifier mod;
708 while(linep[0] == ' ')
709 linep++;
710 sscanf(linep, "%c %d%n", &press, &button, &len);
711 linep += len;
712 while(linep[0] == ' ')
713 linep++;
714 mod = strpe_modifiers(&linep);
715 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
716 }
717
718 else if(strstartswith(line, "DAMAGEMERGE ")) {
719 char *linep = line + 12;
720 while(linep[0] == ' ')
721 linep++;
722 if(streq(linep, "CELL"))
723 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
724 else if(streq(linep, "ROW"))
725 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
726 else if(streq(linep, "SCREEN"))
727 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
728 else if(streq(linep, "SCROLL"))
729 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
730 }
731
732 else if(strstartswith(line, "DAMAGEFLUSH")) {
733 vterm_screen_flush_damage(screen);
734 }
735
736 else if(line[0] == '?') {
737 if(streq(line, "?cursor")) {
738 VTermPos pos;
739 vterm_state_get_cursorpos(state, &pos);
740 if(pos.row != state_pos.row)
741 printf("! row mismatch: state=%d,%d event=%d,%d\n",
742 pos.row, pos.col, state_pos.row, state_pos.col);
743 else if(pos.col != state_pos.col)
744 printf("! col mismatch: state=%d,%d event=%d,%d\n",
745 pos.row, pos.col, state_pos.row, state_pos.col);
746 else
747 printf("%d,%d\n", state_pos.row, state_pos.col);
748 }
749 else if(strstartswith(line, "?pen ")) {
750 VTermValue val;
751 char *linep = line + 5;
752 while(linep[0] == ' ')
753 linep++;
754
755#define BOOLSTR(v) ((v) ? "on" : "off")
756
757 if(streq(linep, "bold")) {
758 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
759 if(val.boolean != state_pen.bold)
760 printf("! pen bold mismatch; state=%s, event=%s\n",
761 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
762 else
763 printf("%s\n", BOOLSTR(state_pen.bold));
764 }
765 else if(streq(linep, "underline")) {
766 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
767 if(val.boolean != state_pen.underline)
768 printf("! pen underline mismatch; state=%d, event=%d\n",
769 val.boolean, state_pen.underline);
770 else
771 printf("%d\n", state_pen.underline);
772 }
773 else if(streq(linep, "italic")) {
774 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
775 if(val.boolean != state_pen.italic)
776 printf("! pen italic mismatch; state=%s, event=%s\n",
777 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
778 else
779 printf("%s\n", BOOLSTR(state_pen.italic));
780 }
781 else if(streq(linep, "blink")) {
782 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
783 if(val.boolean != state_pen.blink)
784 printf("! pen blink mismatch; state=%s, event=%s\n",
785 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
786 else
787 printf("%s\n", BOOLSTR(state_pen.blink));
788 }
789 else if(streq(linep, "reverse")) {
790 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
791 if(val.boolean != state_pen.reverse)
792 printf("! pen reverse mismatch; state=%s, event=%s\n",
793 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
794 else
795 printf("%s\n", BOOLSTR(state_pen.reverse));
796 }
797 else if(streq(linep, "font")) {
798 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
799 if(val.boolean != state_pen.font)
800 printf("! pen font mismatch; state=%d, event=%d\n",
801 val.boolean, state_pen.font);
802 else
803 printf("%d\n", state_pen.font);
804 }
805 else if(streq(linep, "foreground")) {
806 printf("rgb(%d,%d,%d)\n", state_pen.foreground.red, state_pen.foreground.green, state_pen.foreground.blue);
807 }
808 else if(streq(linep, "background")) {
809 printf("rgb(%d,%d,%d)\n", state_pen.background.red, state_pen.background.green, state_pen.background.blue);
810 }
811 else
812 printf("?\n");
813 }
814 else if(strstartswith(line, "?screen_chars ")) {
815 char *linep = line + 13;
816 VTermRect rect;
817 size_t len;
818 while(linep[0] == ' ')
819 linep++;
820 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
821 printf("! screen_chars unrecognised input\n");
822 goto abort_line;
823 }
824 len = vterm_screen_get_chars(screen, NULL, 0, rect);
825 if(len == (size_t)-1)
826 printf("! screen_chars error\n");
827 else if(len == 0)
828 printf("\n");
829 else {
830 uint32_t *chars = malloc(sizeof(uint32_t) * len);
831 size_t i;
832 vterm_screen_get_chars(screen, chars, len, rect);
833 for(i = 0; i < len; i++) {
834 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
835 }
836 free(chars);
837 }
838 }
839 else if(strstartswith(line, "?screen_text ")) {
840 char *linep = line + 12;
841 VTermRect rect;
842 size_t len;
843 while(linep[0] == ' ')
844 linep++;
845 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
846 printf("! screen_text unrecognised input\n");
847 goto abort_line;
848 }
849 len = vterm_screen_get_text(screen, NULL, 0, rect);
850 if(len == (size_t)-1)
851 printf("! screen_text error\n");
852 else if(len == 0)
853 printf("\n");
854 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100855 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200856 unsigned char *buffer = malloc(len + 4);
857 unsigned char *text = buffer + 2;
858 text[-2] = 0x55; text[-1] = 0xAA;
859 text[len] = 0x55; text[len+1] = 0xAA;
860
861 vterm_screen_get_text(screen, (char *)text, len, rect);
862
863 if(text[-2] != 0x55 || text[-1] != 0xAA)
864 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
865 else if(text[len] != 0x55 || text[len+1] != 0xAA)
866 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
867 else
868 {
869 size_t i;
870 for(i = 0; i < len; i++) {
871 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
872 }
873 }
874
875 free(buffer);
876 }
877 }
878 else if(strstartswith(line, "?screen_cell ")) {
879 char *linep = line + 12;
880 int i;
881 VTermPos pos;
882 VTermScreenCell cell;
883 while(linep[0] == ' ')
884 linep++;
885 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
886 printf("! screen_cell unrecognised input\n");
887 goto abort_line;
888 }
889 if(!vterm_screen_get_cell(screen, pos, &cell))
890 goto abort_line;
891 printf("{");
892 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
893 printf("%s0x%x", i ? "," : "", cell.chars[i]);
894 }
895 printf("} width=%d attrs={", cell.width);
896 if(cell.attrs.bold) printf("B");
897 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
898 if(cell.attrs.italic) printf("I");
899 if(cell.attrs.blink) printf("K");
900 if(cell.attrs.reverse) printf("R");
901 if(cell.attrs.font) printf("F%d", cell.attrs.font);
902 printf("} ");
903 if(cell.attrs.dwl) printf("dwl ");
904 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
905 printf("fg=rgb(%d,%d,%d) ", cell.fg.red, cell.fg.green, cell.fg.blue);
906 printf("bg=rgb(%d,%d,%d)\n", cell.bg.red, cell.bg.green, cell.bg.blue);
907 }
908 else if(strstartswith(line, "?screen_eol ")) {
909 VTermPos pos;
910 char *linep = line + 12;
911 while(linep[0] == ' ')
912 linep++;
913 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
914 printf("! screen_eol unrecognised input\n");
915 goto abort_line;
916 }
917 printf("%d\n", vterm_screen_is_eol(screen, pos));
918 }
919 else if(strstartswith(line, "?screen_attrs_extent ")) {
920 VTermPos pos;
921 VTermRect rect;
922 char *linep = line + 21;
923 while(linep[0] == ' ')
924 linep++;
925 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
926 printf("! screen_attrs_extent unrecognised input\n");
927 goto abort_line;
928 }
929 rect.start_col = 0;
930 rect.end_col = -1;
931 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
932 printf("! screen_attrs_extent failed\n");
933 goto abort_line;
934 }
935 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
936 }
937 else
938 printf("?\n");
939
940 memset(line, 0, sizeof line);
941 continue;
942 }
943
944 else
945 abort_line: err = 1;
946
947 outlen = vterm_output_get_buffer_current(vt);
948 if(outlen > 0) {
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200949 size_t i;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200950 char outbuff[1024];
951 vterm_output_read(vt, outbuff, outlen);
952
953 printf("output ");
954 for(i = 0; i < outlen; i++)
955 printf("%x%s", (unsigned char)outbuff[i], i < outlen-1 ? "," : "\n");
956 }
957
958 printf(err ? "?\n" : "DONE\n");
959 }
960
961 vterm_free(vt);
962
963 return 0;
964}