blob: 5fa4cfbca41bfc4af629e581f7ebc956ce78239d [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
Bram Moolenaar88c1ee82020-04-12 13:38:57 +020068static void term_output(const char *s, size_t len, void *user)
69{
70 size_t i;
71
72 printf("output ");
73 for(i = 0; i < len; i++)
74 printf("%x%s", (unsigned char)s[i], i < len-1 ? "," : "\n");
75}
76
77static void printhex(const char *s, size_t len)
78{
79 while(len--)
80 printf("%02x", (s++)[0]);
81}
82
Bram Moolenaard19a8f92020-04-11 21:42:48 +020083static int parser_text(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +020084{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +020085 size_t i;
Bram Moolenaare4f25e42017-07-07 11:54:15 +020086
87 printf("text ");
88 for(i = 0; i < len; i++) {
89 unsigned char b = bytes[i];
90 if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
91 break;
92 printf(i ? ",%x" : "%x", b);
93 }
94 printf("\n");
95
96 return i;
97}
98
Bram Moolenaard19a8f92020-04-11 21:42:48 +020099static int parser_control(unsigned char control, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200100{
101 printf("control %02x\n", control);
102
103 return 1;
104}
105
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200106static int parser_escape(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200107{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200108 if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
109 if(len < 2)
110 return -1;
111 len = 2;
112 }
113 else {
114 len = 1;
115 }
116
117 printf("escape ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200118 printhex(bytes, len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200119 printf("\n");
120
121 return len;
122}
123
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200124static 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 +0200125{
126 int i;
127 printf("csi %02x", command);
128
129 if(leader && leader[0]) {
130 printf(" L=");
131 for(i = 0; leader[i]; i++)
132 printf("%02x", leader[i]);
133 }
134
135 for(i = 0; i < argcount; i++) {
136 char sep = i ? ',' : ' ';
137
138 if(args[i] == CSI_ARG_MISSING)
139 printf("%c*", sep);
140 else
141 printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
142 }
143
144 if(intermed && intermed[0]) {
145 printf(" I=");
146 for(i = 0; intermed[i]; i++)
147 printf("%02x", intermed[i]);
148 }
149
150 printf("\n");
151
152 return 1;
153}
154
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200155static int parser_osc(const char *command, size_t cmdlen, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200156{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200157
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200158 printf("osc ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200159 printhex(command, cmdlen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200160 printf("\n");
161
162 return 1;
163}
164
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200165static int parser_dcs(const char *command, size_t cmdlen, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200166{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200167
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200168 printf("dcs ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200169 printhex(command, cmdlen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200170 printf("\n");
171
172 return 1;
173}
174
175static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100176 parser_text, // text
177 parser_control, // control
178 parser_escape, // escape
179 parser_csi, // csi
180 parser_osc, // osc
181 parser_dcs, // dcs
182 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200183};
184
Bram Moolenaar707d2262019-12-04 22:16:54 +0100185// These callbacks are shared by State and Screen
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200186
187static int want_movecursor = 0;
188static VTermPos state_pos;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200189static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200190{
191 state_pos = pos;
192
193 if(want_movecursor)
194 printf("movecursor %d,%d\n", pos.row, pos.col);
195
196 return 1;
197}
198
199static int want_scrollrect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200200static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200201{
202 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;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200213static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200214{
215 if(!want_moverect)
216 return 0;
217
218 printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
219 src.start_row, src.end_row, src.start_col, src.end_col,
220 dest.start_row, dest.end_row, dest.start_col, dest.end_col);
221
222 return 1;
223}
224
225static int want_settermprop = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200226static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200227{
228 VTermValueType type;
229 if(!want_settermprop)
230 return 1;
231
232 type = vterm_get_prop_type(prop);
233 switch(type) {
234 case VTERM_VALUETYPE_BOOL:
235 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
236 return 1;
237 case VTERM_VALUETYPE_INT:
238 printf("settermprop %d %d\n", prop, val->number);
239 return 1;
240 case VTERM_VALUETYPE_STRING:
241 printf("settermprop %d \"%s\"\n", prop, val->string);
242 return 1;
243 case VTERM_VALUETYPE_COLOR:
244 printf("settermprop %d rgb(%d,%d,%d)\n", prop, val->color.red, val->color.green, val->color.blue);
245 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200246
247 case VTERM_N_VALUETYPES:
248 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200249 }
250
251 return 0;
252}
253
Bram Moolenaar707d2262019-12-04 22:16:54 +0100254// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200255
256static int want_state_putglyph = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200257static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200258{
259 int i;
260 if(!want_state_putglyph)
261 return 1;
262
263 printf("putglyph ");
264 for(i = 0; info->chars[i]; i++)
265 printf(i ? ",%x" : "%x", info->chars[i]);
266 printf(" %d %d,%d", info->width, pos.row, pos.col);
267 if(info->protected_cell)
268 printf(" prot");
269 if(info->dwl)
270 printf(" dwl");
271 if(info->dhl)
272 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
273 printf("\n");
274
275 return 1;
276}
277
278static int want_state_erase = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200279static int state_erase(VTermRect rect, int selective, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200280{
281 if(!want_state_erase)
282 return 1;
283
284 printf("erase %d..%d,%d..%d%s\n",
285 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
286 selective ? " selective" : "");
287
288 return 1;
289}
290
291static struct {
292 int bold;
293 int underline;
294 int italic;
295 int blink;
296 int reverse;
297 int strike;
298 int font;
299 VTermColor foreground;
300 VTermColor background;
301} state_pen;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200302static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200303{
304 switch(attr) {
305 case VTERM_ATTR_BOLD:
306 state_pen.bold = val->boolean;
307 break;
308 case VTERM_ATTR_UNDERLINE:
309 state_pen.underline = val->number;
310 break;
311 case VTERM_ATTR_ITALIC:
312 state_pen.italic = val->boolean;
313 break;
314 case VTERM_ATTR_BLINK:
315 state_pen.blink = val->boolean;
316 break;
317 case VTERM_ATTR_REVERSE:
318 state_pen.reverse = val->boolean;
319 break;
320 case VTERM_ATTR_STRIKE:
321 state_pen.strike = val->boolean;
322 break;
323 case VTERM_ATTR_FONT:
324 state_pen.font = val->number;
325 break;
326 case VTERM_ATTR_FOREGROUND:
327 state_pen.foreground = val->color;
328 break;
329 case VTERM_ATTR_BACKGROUND:
330 state_pen.background = val->color;
331 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200332
333 case VTERM_N_ATTRS:
334 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200335 }
336
337 return 1;
338}
339
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200340static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200341{
342 return 1;
343}
344
345VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100346 state_putglyph, // putglyph
347 movecursor, // movecursor
348 scrollrect, // scrollrect
349 moverect, // moverect
350 state_erase, // erase
351 NULL, // initpen
352 state_setpenattr, // setpenattr
353 settermprop, // settermprop
354 NULL, // bell
355 NULL, // resize
356 state_setlineinfo, // setlineinfo
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200357};
358
359static int want_screen_damage = 0;
360static int want_screen_damage_cells = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200361static int screen_damage(VTermRect rect, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200362{
363 if(!want_screen_damage)
364 return 1;
365
366 printf("damage %d..%d,%d..%d",
367 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
368
369 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200370 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200371 int row;
372 int col;
373
374 for(row = rect.start_row; row < rect.end_row; row++) {
375 int eol = rect.end_col;
376 while(eol > rect.start_col) {
377 VTermScreenCell cell;
378 VTermPos pos;
379 pos.row = row;
380 pos.col = eol-1;
381 vterm_screen_get_cell(screen, pos, &cell);
382 if(cell.chars[0])
383 break;
384
385 eol--;
386 }
387
388 if(eol == rect.start_col)
389 break;
390
391 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200392 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200393
394 printf(" %d<", row);
395 for(col = rect.start_col; col < eol; col++) {
396 VTermScreenCell cell;
397 VTermPos pos;
398 pos.row = row;
399 pos.col = col;
400 vterm_screen_get_cell(screen, pos, &cell);
401 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
402 }
403 printf(">");
404 }
405 }
406
407 printf("\n");
408
409 return 1;
410}
411
412static int want_screen_scrollback = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200413static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200414{
415 int eol;
416 int c;
417
418 if(!want_screen_scrollback)
419 return 1;
420
421 eol = cols;
422 while(eol && !cells[eol-1].chars[0])
423 eol--;
424
425 printf("sb_pushline %d =", cols);
426 for(c = 0; c < eol; c++)
427 printf(" %02X", cells[c].chars[0]);
428 printf("\n");
429
430 return 1;
431}
432
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200433static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200434{
435 int col;
436
437 if(!want_screen_scrollback)
438 return 0;
439
Bram Moolenaar707d2262019-12-04 22:16:54 +0100440 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200441 for(col = 0; col < cols; col++) {
442 if(col < 5)
443 cells[col].chars[0] = 'A' + col;
444 else
445 cells[col].chars[0] = 0;
446
447 cells[col].width = 1;
448 }
449
450 printf("sb_popline %d\n", cols);
451 return 1;
452}
453
454VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100455 screen_damage, // damage
456 moverect, // moverect
457 movecursor, // movecursor
458 settermprop, // settermprop
459 NULL, // bell
460 NULL, // resize
461 screen_sb_pushline, // sb_pushline
462 screen_sb_popline // sb_popline
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200463};
464
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200465int main(int argc UNUSED, char **argv UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200466{
467 char line[1024] = {0};
468 int flag;
469
470 int err;
471
472 setvbuf(stdout, NULL, _IONBF, 0);
473
474 while(fgets(line, sizeof line, stdin)) {
475 char *nl;
476 size_t outlen;
477 err = 0;
478
479 if((nl = strchr(line, '\n')))
480 *nl = '\0';
481
482 if(streq(line, "INIT")) {
483 if(!vt)
484 vt = vterm_new(25, 80);
485 }
486
487 else if(streq(line, "WANTPARSER")) {
488 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
489 }
490
491 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
492 int i = 9;
493 int sense = 1;
494 if(!state) {
495 state = vterm_obtain_state(vt);
496 vterm_state_set_callbacks(state, &state_cbs, NULL);
497 vterm_state_set_bold_highbright(state, 1);
498 vterm_state_reset(state, 1);
499 }
500
501 while(line[i] == ' ')
502 i++;
503 for( ; line[i]; i++)
504 switch(line[i]) {
505 case '+':
506 sense = 1;
507 break;
508 case '-':
509 sense = 0;
510 break;
511 case 'g':
512 want_state_putglyph = sense;
513 break;
514 case 's':
515 want_scrollrect = sense;
516 break;
517 case 'm':
518 want_moverect = sense;
519 break;
520 case 'e':
521 want_state_erase = sense;
522 break;
523 case 'p':
524 want_settermprop = sense;
525 break;
526 case 'f':
527 vterm_state_set_unrecognised_fallbacks(state, sense ? &parser_cbs : NULL, NULL);
528 break;
529 default:
530 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
531 }
532 }
533
534 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
535 int i = 10;
536 int sense = 1;
537 if(!screen)
538 screen = vterm_obtain_screen(vt);
539 vterm_screen_enable_altscreen(screen, 1);
540 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
541
542 while(line[i] == ' ')
543 i++;
544 for( ; line[i]; i++)
545 switch(line[i]) {
546 case '-':
547 sense = 0;
548 break;
549 case 'd':
550 want_screen_damage = sense;
551 break;
552 case 'D':
553 want_screen_damage = sense;
554 want_screen_damage_cells = sense;
555 break;
556 case 'm':
557 want_moverect = sense;
558 break;
559 case 'c':
560 want_movecursor = sense;
561 break;
562 case 'p':
563 want_settermprop = 1;
564 break;
565 case 'b':
566 want_screen_scrollback = sense;
567 break;
568 default:
569 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
570 }
571 }
572
573 else if(sscanf(line, "UTF8 %d", &flag)) {
574 vterm_set_utf8(vt, flag);
575 }
576
577 else if(streq(line, "RESET")) {
578 if(state) {
579 vterm_state_reset(state, 1);
580 vterm_state_get_cursorpos(state, &state_pos);
581 }
582 if(screen) {
583 vterm_screen_reset(screen, 1);
584 }
585 }
586
587 else if(strstartswith(line, "RESIZE ")) {
588 int rows, cols;
589 char *linep = line + 7;
590 while(linep[0] == ' ')
591 linep++;
592 sscanf(linep, "%d, %d", &rows, &cols);
593 vterm_set_size(vt, rows, cols);
594 }
595
596 else if(strstartswith(line, "PUSH ")) {
597 char *bytes = line + 5;
598 size_t len = inplace_hex2bytes(bytes);
599 size_t written = vterm_input_write(vt, bytes, len);
600 if(written < len)
601 fprintf(stderr, "! short write\n");
602 }
603
604 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100605 // This isn't really external API but it's hard to get this out any
606 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200607 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
608 if(encoding.enc->init)
609 (*encoding.enc->init)(encoding.enc, encoding.data);
610 }
611
612 else if(strstartswith(line, "ENCIN ")) {
613 char *bytes = line + 6;
614 size_t len = inplace_hex2bytes(bytes);
615
616 uint32_t cp[1024];
617 int cpi = 0;
618 size_t pos = 0;
619
620 (*encoding.enc->decode)(encoding.enc, encoding.data,
621 cp, &cpi, len, bytes, &pos, len);
622
623 if(cpi > 0) {
624 int i;
625 printf("encout ");
626 for(i = 0; i < cpi; i++) {
627 printf(i ? ",%x" : "%x", cp[i]);
628 }
629 printf("\n");
630 }
631 }
632
633 else if(strstartswith(line, "INCHAR ")) {
634 char *linep = line + 7;
635 unsigned int c = 0;
636 VTermModifier mod;
637 while(linep[0] == ' ')
638 linep++;
639 mod = strpe_modifiers(&linep);
640 sscanf(linep, " %x", &c);
641
642 vterm_keyboard_unichar(vt, c, mod);
643 }
644
645 else if(strstartswith(line, "INKEY ")) {
646 VTermModifier mod;
647 VTermKey key;
648 char *linep = line + 6;
649 while(linep[0] == ' ')
650 linep++;
651 mod = strpe_modifiers(&linep);
652 while(linep[0] == ' ')
653 linep++;
654 key = strp_key(linep);
655
656 vterm_keyboard_key(vt, key, mod);
657 }
658
659 else if(strstartswith(line, "PASTE ")) {
660 char *linep = line + 6;
661 if(streq(linep, "START"))
662 vterm_keyboard_start_paste(vt);
663 else if(streq(linep, "END"))
664 vterm_keyboard_end_paste(vt);
665 else
666 goto abort_line;
667 }
668
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200669 else if(strstartswith(line, "FOCUS ")) {
670 char *linep = line + 6;
671 if(streq(linep, "IN"))
672 vterm_state_focus_in(state);
673 else if(streq(linep, "OUT"))
674 vterm_state_focus_out(state);
675 else
676 goto abort_line;
677 }
678
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200679 else if(strstartswith(line, "MOUSEMOVE ")) {
680 char *linep = line + 10;
681 int row, col, len;
682 VTermModifier mod;
683 while(linep[0] == ' ')
684 linep++;
685 sscanf(linep, "%d,%d%n", &row, &col, &len);
686 linep += len;
687 while(linep[0] == ' ')
688 linep++;
689 mod = strpe_modifiers(&linep);
690 vterm_mouse_move(vt, row, col, mod);
691 }
692
693 else if(strstartswith(line, "MOUSEBTN ")) {
694 char *linep = line + 9;
695 char press;
696 int button, len;
697 VTermModifier mod;
698 while(linep[0] == ' ')
699 linep++;
700 sscanf(linep, "%c %d%n", &press, &button, &len);
701 linep += len;
702 while(linep[0] == ' ')
703 linep++;
704 mod = strpe_modifiers(&linep);
705 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
706 }
707
708 else if(strstartswith(line, "DAMAGEMERGE ")) {
709 char *linep = line + 12;
710 while(linep[0] == ' ')
711 linep++;
712 if(streq(linep, "CELL"))
713 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
714 else if(streq(linep, "ROW"))
715 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
716 else if(streq(linep, "SCREEN"))
717 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
718 else if(streq(linep, "SCROLL"))
719 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
720 }
721
722 else if(strstartswith(line, "DAMAGEFLUSH")) {
723 vterm_screen_flush_damage(screen);
724 }
725
726 else if(line[0] == '?') {
727 if(streq(line, "?cursor")) {
728 VTermPos pos;
729 vterm_state_get_cursorpos(state, &pos);
730 if(pos.row != state_pos.row)
731 printf("! row mismatch: state=%d,%d event=%d,%d\n",
732 pos.row, pos.col, state_pos.row, state_pos.col);
733 else if(pos.col != state_pos.col)
734 printf("! col mismatch: state=%d,%d event=%d,%d\n",
735 pos.row, pos.col, state_pos.row, state_pos.col);
736 else
737 printf("%d,%d\n", state_pos.row, state_pos.col);
738 }
739 else if(strstartswith(line, "?pen ")) {
740 VTermValue val;
741 char *linep = line + 5;
742 while(linep[0] == ' ')
743 linep++;
744
745#define BOOLSTR(v) ((v) ? "on" : "off")
746
747 if(streq(linep, "bold")) {
748 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
749 if(val.boolean != state_pen.bold)
750 printf("! pen bold mismatch; state=%s, event=%s\n",
751 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
752 else
753 printf("%s\n", BOOLSTR(state_pen.bold));
754 }
755 else if(streq(linep, "underline")) {
756 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
757 if(val.boolean != state_pen.underline)
758 printf("! pen underline mismatch; state=%d, event=%d\n",
759 val.boolean, state_pen.underline);
760 else
761 printf("%d\n", state_pen.underline);
762 }
763 else if(streq(linep, "italic")) {
764 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
765 if(val.boolean != state_pen.italic)
766 printf("! pen italic mismatch; state=%s, event=%s\n",
767 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
768 else
769 printf("%s\n", BOOLSTR(state_pen.italic));
770 }
771 else if(streq(linep, "blink")) {
772 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
773 if(val.boolean != state_pen.blink)
774 printf("! pen blink mismatch; state=%s, event=%s\n",
775 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
776 else
777 printf("%s\n", BOOLSTR(state_pen.blink));
778 }
779 else if(streq(linep, "reverse")) {
780 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
781 if(val.boolean != state_pen.reverse)
782 printf("! pen reverse mismatch; state=%s, event=%s\n",
783 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
784 else
785 printf("%s\n", BOOLSTR(state_pen.reverse));
786 }
787 else if(streq(linep, "font")) {
788 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
789 if(val.boolean != state_pen.font)
790 printf("! pen font mismatch; state=%d, event=%d\n",
791 val.boolean, state_pen.font);
792 else
793 printf("%d\n", state_pen.font);
794 }
795 else if(streq(linep, "foreground")) {
796 printf("rgb(%d,%d,%d)\n", state_pen.foreground.red, state_pen.foreground.green, state_pen.foreground.blue);
797 }
798 else if(streq(linep, "background")) {
799 printf("rgb(%d,%d,%d)\n", state_pen.background.red, state_pen.background.green, state_pen.background.blue);
800 }
801 else
802 printf("?\n");
803 }
804 else if(strstartswith(line, "?screen_chars ")) {
805 char *linep = line + 13;
806 VTermRect rect;
807 size_t len;
808 while(linep[0] == ' ')
809 linep++;
810 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
811 printf("! screen_chars unrecognised input\n");
812 goto abort_line;
813 }
814 len = vterm_screen_get_chars(screen, NULL, 0, rect);
815 if(len == (size_t)-1)
816 printf("! screen_chars error\n");
817 else if(len == 0)
818 printf("\n");
819 else {
820 uint32_t *chars = malloc(sizeof(uint32_t) * len);
821 size_t i;
822 vterm_screen_get_chars(screen, chars, len, rect);
823 for(i = 0; i < len; i++) {
824 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
825 }
826 free(chars);
827 }
828 }
829 else if(strstartswith(line, "?screen_text ")) {
830 char *linep = line + 12;
831 VTermRect rect;
832 size_t len;
833 while(linep[0] == ' ')
834 linep++;
835 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
836 printf("! screen_text unrecognised input\n");
837 goto abort_line;
838 }
839 len = vterm_screen_get_text(screen, NULL, 0, rect);
840 if(len == (size_t)-1)
841 printf("! screen_text error\n");
842 else if(len == 0)
843 printf("\n");
844 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100845 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200846 unsigned char *buffer = malloc(len + 4);
847 unsigned char *text = buffer + 2;
848 text[-2] = 0x55; text[-1] = 0xAA;
849 text[len] = 0x55; text[len+1] = 0xAA;
850
851 vterm_screen_get_text(screen, (char *)text, len, rect);
852
853 if(text[-2] != 0x55 || text[-1] != 0xAA)
854 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
855 else if(text[len] != 0x55 || text[len+1] != 0xAA)
856 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
857 else
858 {
859 size_t i;
860 for(i = 0; i < len; i++) {
861 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
862 }
863 }
864
865 free(buffer);
866 }
867 }
868 else if(strstartswith(line, "?screen_cell ")) {
869 char *linep = line + 12;
870 int i;
871 VTermPos pos;
872 VTermScreenCell cell;
873 while(linep[0] == ' ')
874 linep++;
875 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
876 printf("! screen_cell unrecognised input\n");
877 goto abort_line;
878 }
879 if(!vterm_screen_get_cell(screen, pos, &cell))
880 goto abort_line;
881 printf("{");
882 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
883 printf("%s0x%x", i ? "," : "", cell.chars[i]);
884 }
885 printf("} width=%d attrs={", cell.width);
886 if(cell.attrs.bold) printf("B");
887 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
888 if(cell.attrs.italic) printf("I");
889 if(cell.attrs.blink) printf("K");
890 if(cell.attrs.reverse) printf("R");
891 if(cell.attrs.font) printf("F%d", cell.attrs.font);
892 printf("} ");
893 if(cell.attrs.dwl) printf("dwl ");
894 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
895 printf("fg=rgb(%d,%d,%d) ", cell.fg.red, cell.fg.green, cell.fg.blue);
896 printf("bg=rgb(%d,%d,%d)\n", cell.bg.red, cell.bg.green, cell.bg.blue);
897 }
898 else if(strstartswith(line, "?screen_eol ")) {
899 VTermPos pos;
900 char *linep = line + 12;
901 while(linep[0] == ' ')
902 linep++;
903 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
904 printf("! screen_eol unrecognised input\n");
905 goto abort_line;
906 }
907 printf("%d\n", vterm_screen_is_eol(screen, pos));
908 }
909 else if(strstartswith(line, "?screen_attrs_extent ")) {
910 VTermPos pos;
911 VTermRect rect;
912 char *linep = line + 21;
913 while(linep[0] == ' ')
914 linep++;
915 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
916 printf("! screen_attrs_extent unrecognised input\n");
917 goto abort_line;
918 }
919 rect.start_col = 0;
920 rect.end_col = -1;
921 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
922 printf("! screen_attrs_extent failed\n");
923 goto abort_line;
924 }
925 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
926 }
927 else
928 printf("?\n");
929
930 memset(line, 0, sizeof line);
931 continue;
932 }
933
934 else
935 abort_line: err = 1;
936
937 outlen = vterm_output_get_buffer_current(vt);
938 if(outlen > 0) {
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200939 char outbuff[1024];
940 vterm_output_read(vt, outbuff, outlen);
941
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200942 term_output(outbuff, outlen, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200943 }
944
945 printf(err ? "?\n" : "DONE\n");
946 }
947
948 vterm_free(vt);
949
950 return 0;
951}