blob: 289d82928198f620f818f2df955fc1a5393a5a3a [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 Moolenaard19a8f92020-04-11 21:42:48 +0200156static int parser_osc(const char *command, size_t cmdlen, 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 Moolenaar88c1ee82020-04-12 13:38:57 +0200160 printhex(command, cmdlen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200161 printf("\n");
162
163 return 1;
164}
165
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200166static int parser_dcs(const char *command, size_t cmdlen, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200167{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200168
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200169 printf("dcs ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200170 printhex(command, cmdlen);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200171 printf("\n");
172
173 return 1;
174}
175
176static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100177 parser_text, // text
178 parser_control, // control
179 parser_escape, // escape
180 parser_csi, // csi
181 parser_osc, // osc
182 parser_dcs, // dcs
183 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200184};
185
Bram Moolenaar707d2262019-12-04 22:16:54 +0100186// These callbacks are shared by State and Screen
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200187
188static int want_movecursor = 0;
189static VTermPos state_pos;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200190static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200191{
192 state_pos = pos;
193
194 if(want_movecursor)
195 printf("movecursor %d,%d\n", pos.row, pos.col);
196
197 return 1;
198}
199
200static int want_scrollrect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200201static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200202{
203 if(!want_scrollrect)
204 return 0;
205
206 printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
207 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
208 downward, rightward);
209
210 return 1;
211}
212
213static int want_moverect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200214static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200215{
216 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;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200227static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200228{
229 VTermValueType type;
230 if(!want_settermprop)
231 return 1;
232
233 type = vterm_get_prop_type(prop);
234 switch(type) {
235 case VTERM_VALUETYPE_BOOL:
236 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
237 return 1;
238 case VTERM_VALUETYPE_INT:
239 printf("settermprop %d %d\n", prop, val->number);
240 return 1;
241 case VTERM_VALUETYPE_STRING:
242 printf("settermprop %d \"%s\"\n", prop, val->string);
243 return 1;
244 case VTERM_VALUETYPE_COLOR:
245 printf("settermprop %d rgb(%d,%d,%d)\n", prop, val->color.red, val->color.green, val->color.blue);
246 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200247
248 case VTERM_N_VALUETYPES:
249 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200250 }
251
252 return 0;
253}
254
Bram Moolenaar707d2262019-12-04 22:16:54 +0100255// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200256
257static int want_state_putglyph = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200258static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200259{
260 int i;
261 if(!want_state_putglyph)
262 return 1;
263
264 printf("putglyph ");
265 for(i = 0; info->chars[i]; i++)
266 printf(i ? ",%x" : "%x", info->chars[i]);
267 printf(" %d %d,%d", info->width, pos.row, pos.col);
268 if(info->protected_cell)
269 printf(" prot");
270 if(info->dwl)
271 printf(" dwl");
272 if(info->dhl)
273 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
274 printf("\n");
275
276 return 1;
277}
278
279static int want_state_erase = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200280static int state_erase(VTermRect rect, int selective, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200281{
282 if(!want_state_erase)
283 return 1;
284
285 printf("erase %d..%d,%d..%d%s\n",
286 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
287 selective ? " selective" : "");
288
289 return 1;
290}
291
292static struct {
293 int bold;
294 int underline;
295 int italic;
296 int blink;
297 int reverse;
298 int strike;
299 int font;
300 VTermColor foreground;
301 VTermColor background;
302} state_pen;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200303static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200304{
305 switch(attr) {
306 case VTERM_ATTR_BOLD:
307 state_pen.bold = val->boolean;
308 break;
309 case VTERM_ATTR_UNDERLINE:
310 state_pen.underline = val->number;
311 break;
312 case VTERM_ATTR_ITALIC:
313 state_pen.italic = val->boolean;
314 break;
315 case VTERM_ATTR_BLINK:
316 state_pen.blink = val->boolean;
317 break;
318 case VTERM_ATTR_REVERSE:
319 state_pen.reverse = val->boolean;
320 break;
321 case VTERM_ATTR_STRIKE:
322 state_pen.strike = val->boolean;
323 break;
324 case VTERM_ATTR_FONT:
325 state_pen.font = val->number;
326 break;
327 case VTERM_ATTR_FOREGROUND:
328 state_pen.foreground = val->color;
329 break;
330 case VTERM_ATTR_BACKGROUND:
331 state_pen.background = val->color;
332 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200333
334 case VTERM_N_ATTRS:
335 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200336 }
337
338 return 1;
339}
340
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200341static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200342{
343 return 1;
344}
345
346VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100347 state_putglyph, // putglyph
348 movecursor, // movecursor
349 scrollrect, // scrollrect
350 moverect, // moverect
351 state_erase, // erase
352 NULL, // initpen
353 state_setpenattr, // setpenattr
354 settermprop, // settermprop
355 NULL, // bell
356 NULL, // resize
357 state_setlineinfo, // setlineinfo
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200358};
359
360static int want_screen_damage = 0;
361static int want_screen_damage_cells = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200362static int screen_damage(VTermRect rect, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200363{
364 if(!want_screen_damage)
365 return 1;
366
367 printf("damage %d..%d,%d..%d",
368 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
369
370 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200371 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200372 int row;
373 int col;
374
375 for(row = rect.start_row; row < rect.end_row; row++) {
376 int eol = rect.end_col;
377 while(eol > rect.start_col) {
378 VTermScreenCell cell;
379 VTermPos pos;
380 pos.row = row;
381 pos.col = eol-1;
382 vterm_screen_get_cell(screen, pos, &cell);
383 if(cell.chars[0])
384 break;
385
386 eol--;
387 }
388
389 if(eol == rect.start_col)
390 break;
391
392 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200393 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200394
395 printf(" %d<", row);
396 for(col = rect.start_col; col < eol; col++) {
397 VTermScreenCell cell;
398 VTermPos pos;
399 pos.row = row;
400 pos.col = col;
401 vterm_screen_get_cell(screen, pos, &cell);
402 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
403 }
404 printf(">");
405 }
406 }
407
408 printf("\n");
409
410 return 1;
411}
412
413static int want_screen_scrollback = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200414static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200415{
416 int eol;
417 int c;
418
419 if(!want_screen_scrollback)
420 return 1;
421
422 eol = cols;
423 while(eol && !cells[eol-1].chars[0])
424 eol--;
425
426 printf("sb_pushline %d =", cols);
427 for(c = 0; c < eol; c++)
428 printf(" %02X", cells[c].chars[0]);
429 printf("\n");
430
431 return 1;
432}
433
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200434static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200435{
436 int col;
437
438 if(!want_screen_scrollback)
439 return 0;
440
Bram Moolenaar707d2262019-12-04 22:16:54 +0100441 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200442 for(col = 0; col < cols; col++) {
443 if(col < 5)
444 cells[col].chars[0] = 'A' + col;
445 else
446 cells[col].chars[0] = 0;
447
448 cells[col].width = 1;
449 }
450
451 printf("sb_popline %d\n", cols);
452 return 1;
453}
454
455VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100456 screen_damage, // damage
457 moverect, // moverect
458 movecursor, // movecursor
459 settermprop, // settermprop
460 NULL, // bell
461 NULL, // resize
462 screen_sb_pushline, // sb_pushline
463 screen_sb_popline // sb_popline
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200464};
465
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200466int main(int argc UNUSED, char **argv UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200467{
468 char line[1024] = {0};
469 int flag;
470
471 int err;
472
473 setvbuf(stdout, NULL, _IONBF, 0);
474
475 while(fgets(line, sizeof line, stdin)) {
476 char *nl;
477 size_t outlen;
478 err = 0;
479
480 if((nl = strchr(line, '\n')))
481 *nl = '\0';
482
483 if(streq(line, "INIT")) {
484 if(!vt)
485 vt = vterm_new(25, 80);
Bram Moolenaar94d729c2020-05-17 21:50:16 +0200486
487 // Somehow this makes tests fail
488 // vterm_output_set_callback(vt, term_output, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200489 }
490
491 else if(streq(line, "WANTPARSER")) {
492 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
493 }
494
495 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
496 int i = 9;
497 int sense = 1;
498 if(!state) {
499 state = vterm_obtain_state(vt);
500 vterm_state_set_callbacks(state, &state_cbs, NULL);
501 vterm_state_set_bold_highbright(state, 1);
502 vterm_state_reset(state, 1);
503 }
504
505 while(line[i] == ' ')
506 i++;
507 for( ; line[i]; i++)
508 switch(line[i]) {
509 case '+':
510 sense = 1;
511 break;
512 case '-':
513 sense = 0;
514 break;
515 case 'g':
516 want_state_putglyph = sense;
517 break;
518 case 's':
519 want_scrollrect = sense;
520 break;
521 case 'm':
522 want_moverect = sense;
523 break;
524 case 'e':
525 want_state_erase = sense;
526 break;
527 case 'p':
528 want_settermprop = sense;
529 break;
530 case 'f':
531 vterm_state_set_unrecognised_fallbacks(state, sense ? &parser_cbs : NULL, NULL);
532 break;
533 default:
534 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
535 }
536 }
537
538 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
539 int i = 10;
540 int sense = 1;
541 if(!screen)
542 screen = vterm_obtain_screen(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200543 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
544
545 while(line[i] == ' ')
546 i++;
547 for( ; line[i]; i++)
548 switch(line[i]) {
549 case '-':
550 sense = 0;
551 break;
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200552 case 'a':
553 vterm_screen_enable_altscreen(screen, 1);
554 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200555 case 'd':
556 want_screen_damage = sense;
557 break;
558 case 'D':
559 want_screen_damage = sense;
560 want_screen_damage_cells = sense;
561 break;
562 case 'm':
563 want_moverect = sense;
564 break;
565 case 'c':
566 want_movecursor = sense;
567 break;
568 case 'p':
569 want_settermprop = 1;
570 break;
571 case 'b':
572 want_screen_scrollback = sense;
573 break;
574 default:
575 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
576 }
577 }
578
579 else if(sscanf(line, "UTF8 %d", &flag)) {
580 vterm_set_utf8(vt, flag);
581 }
582
583 else if(streq(line, "RESET")) {
584 if(state) {
585 vterm_state_reset(state, 1);
586 vterm_state_get_cursorpos(state, &state_pos);
587 }
588 if(screen) {
589 vterm_screen_reset(screen, 1);
590 }
591 }
592
593 else if(strstartswith(line, "RESIZE ")) {
594 int rows, cols;
595 char *linep = line + 7;
596 while(linep[0] == ' ')
597 linep++;
598 sscanf(linep, "%d, %d", &rows, &cols);
599 vterm_set_size(vt, rows, cols);
600 }
601
602 else if(strstartswith(line, "PUSH ")) {
603 char *bytes = line + 5;
604 size_t len = inplace_hex2bytes(bytes);
605 size_t written = vterm_input_write(vt, bytes, len);
606 if(written < len)
607 fprintf(stderr, "! short write\n");
608 }
609
610 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100611 // This isn't really external API but it's hard to get this out any
612 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200613 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
614 if(encoding.enc->init)
615 (*encoding.enc->init)(encoding.enc, encoding.data);
616 }
617
618 else if(strstartswith(line, "ENCIN ")) {
619 char *bytes = line + 6;
620 size_t len = inplace_hex2bytes(bytes);
621
622 uint32_t cp[1024];
623 int cpi = 0;
624 size_t pos = 0;
625
626 (*encoding.enc->decode)(encoding.enc, encoding.data,
627 cp, &cpi, len, bytes, &pos, len);
628
629 if(cpi > 0) {
630 int i;
631 printf("encout ");
632 for(i = 0; i < cpi; i++) {
633 printf(i ? ",%x" : "%x", cp[i]);
634 }
635 printf("\n");
636 }
637 }
638
639 else if(strstartswith(line, "INCHAR ")) {
640 char *linep = line + 7;
641 unsigned int c = 0;
642 VTermModifier mod;
643 while(linep[0] == ' ')
644 linep++;
645 mod = strpe_modifiers(&linep);
646 sscanf(linep, " %x", &c);
647
648 vterm_keyboard_unichar(vt, c, mod);
649 }
650
651 else if(strstartswith(line, "INKEY ")) {
652 VTermModifier mod;
653 VTermKey key;
654 char *linep = line + 6;
655 while(linep[0] == ' ')
656 linep++;
657 mod = strpe_modifiers(&linep);
658 while(linep[0] == ' ')
659 linep++;
660 key = strp_key(linep);
661
662 vterm_keyboard_key(vt, key, mod);
663 }
664
665 else if(strstartswith(line, "PASTE ")) {
666 char *linep = line + 6;
667 if(streq(linep, "START"))
668 vterm_keyboard_start_paste(vt);
669 else if(streq(linep, "END"))
670 vterm_keyboard_end_paste(vt);
671 else
672 goto abort_line;
673 }
674
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200675 else if(strstartswith(line, "FOCUS ")) {
676 char *linep = line + 6;
677 if(streq(linep, "IN"))
678 vterm_state_focus_in(state);
679 else if(streq(linep, "OUT"))
680 vterm_state_focus_out(state);
681 else
682 goto abort_line;
683 }
684
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200685 else if(strstartswith(line, "MOUSEMOVE ")) {
686 char *linep = line + 10;
687 int row, col, len;
688 VTermModifier mod;
689 while(linep[0] == ' ')
690 linep++;
691 sscanf(linep, "%d,%d%n", &row, &col, &len);
692 linep += len;
693 while(linep[0] == ' ')
694 linep++;
695 mod = strpe_modifiers(&linep);
696 vterm_mouse_move(vt, row, col, mod);
697 }
698
699 else if(strstartswith(line, "MOUSEBTN ")) {
700 char *linep = line + 9;
701 char press;
702 int button, len;
703 VTermModifier mod;
704 while(linep[0] == ' ')
705 linep++;
706 sscanf(linep, "%c %d%n", &press, &button, &len);
707 linep += len;
708 while(linep[0] == ' ')
709 linep++;
710 mod = strpe_modifiers(&linep);
711 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
712 }
713
714 else if(strstartswith(line, "DAMAGEMERGE ")) {
715 char *linep = line + 12;
716 while(linep[0] == ' ')
717 linep++;
718 if(streq(linep, "CELL"))
719 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
720 else if(streq(linep, "ROW"))
721 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
722 else if(streq(linep, "SCREEN"))
723 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
724 else if(streq(linep, "SCROLL"))
725 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
726 }
727
728 else if(strstartswith(line, "DAMAGEFLUSH")) {
729 vterm_screen_flush_damage(screen);
730 }
731
732 else if(line[0] == '?') {
733 if(streq(line, "?cursor")) {
734 VTermPos pos;
735 vterm_state_get_cursorpos(state, &pos);
736 if(pos.row != state_pos.row)
737 printf("! row mismatch: state=%d,%d event=%d,%d\n",
738 pos.row, pos.col, state_pos.row, state_pos.col);
739 else if(pos.col != state_pos.col)
740 printf("! col mismatch: state=%d,%d event=%d,%d\n",
741 pos.row, pos.col, state_pos.row, state_pos.col);
742 else
743 printf("%d,%d\n", state_pos.row, state_pos.col);
744 }
745 else if(strstartswith(line, "?pen ")) {
746 VTermValue val;
747 char *linep = line + 5;
748 while(linep[0] == ' ')
749 linep++;
750
751#define BOOLSTR(v) ((v) ? "on" : "off")
752
753 if(streq(linep, "bold")) {
754 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
755 if(val.boolean != state_pen.bold)
756 printf("! pen bold mismatch; state=%s, event=%s\n",
757 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
758 else
759 printf("%s\n", BOOLSTR(state_pen.bold));
760 }
761 else if(streq(linep, "underline")) {
762 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
763 if(val.boolean != state_pen.underline)
764 printf("! pen underline mismatch; state=%d, event=%d\n",
765 val.boolean, state_pen.underline);
766 else
767 printf("%d\n", state_pen.underline);
768 }
769 else if(streq(linep, "italic")) {
770 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
771 if(val.boolean != state_pen.italic)
772 printf("! pen italic mismatch; state=%s, event=%s\n",
773 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
774 else
775 printf("%s\n", BOOLSTR(state_pen.italic));
776 }
777 else if(streq(linep, "blink")) {
778 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
779 if(val.boolean != state_pen.blink)
780 printf("! pen blink mismatch; state=%s, event=%s\n",
781 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
782 else
783 printf("%s\n", BOOLSTR(state_pen.blink));
784 }
785 else if(streq(linep, "reverse")) {
786 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
787 if(val.boolean != state_pen.reverse)
788 printf("! pen reverse mismatch; state=%s, event=%s\n",
789 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
790 else
791 printf("%s\n", BOOLSTR(state_pen.reverse));
792 }
793 else if(streq(linep, "font")) {
794 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
795 if(val.boolean != state_pen.font)
796 printf("! pen font mismatch; state=%d, event=%d\n",
797 val.boolean, state_pen.font);
798 else
799 printf("%d\n", state_pen.font);
800 }
801 else if(streq(linep, "foreground")) {
802 printf("rgb(%d,%d,%d)\n", state_pen.foreground.red, state_pen.foreground.green, state_pen.foreground.blue);
803 }
804 else if(streq(linep, "background")) {
805 printf("rgb(%d,%d,%d)\n", state_pen.background.red, state_pen.background.green, state_pen.background.blue);
806 }
807 else
808 printf("?\n");
809 }
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200810 else if(strstartswith(line, "?lineinfo ")) {
811 char *linep = line + 10;
812 int row;
813 const VTermLineInfo *info;
814 while(linep[0] == ' ')
815 linep++;
816 if(sscanf(linep, "%d", &row) < 1) {
817 printf("! lineinfo unrecognised input\n");
818 goto abort_line;
819 }
820 info = vterm_state_get_lineinfo(state, row);
821 if(info->doublewidth)
822 printf("dwl ");
823 if(info->doubleheight)
824 printf("dhl ");
825 if(info->continuation)
826 printf("cont ");
827 printf("\n");
828 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200829 else if(strstartswith(line, "?screen_chars ")) {
830 char *linep = line + 13;
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_chars unrecognised input\n");
837 goto abort_line;
838 }
839 len = vterm_screen_get_chars(screen, NULL, 0, rect);
840 if(len == (size_t)-1)
841 printf("! screen_chars error\n");
842 else if(len == 0)
843 printf("\n");
844 else {
845 uint32_t *chars = malloc(sizeof(uint32_t) * len);
846 size_t i;
847 vterm_screen_get_chars(screen, chars, len, rect);
848 for(i = 0; i < len; i++) {
849 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
850 }
851 free(chars);
852 }
853 }
854 else if(strstartswith(line, "?screen_text ")) {
855 char *linep = line + 12;
856 VTermRect rect;
857 size_t len;
858 while(linep[0] == ' ')
859 linep++;
860 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
861 printf("! screen_text unrecognised input\n");
862 goto abort_line;
863 }
864 len = vterm_screen_get_text(screen, NULL, 0, rect);
865 if(len == (size_t)-1)
866 printf("! screen_text error\n");
867 else if(len == 0)
868 printf("\n");
869 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100870 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200871 unsigned char *buffer = malloc(len + 4);
872 unsigned char *text = buffer + 2;
873 text[-2] = 0x55; text[-1] = 0xAA;
874 text[len] = 0x55; text[len+1] = 0xAA;
875
876 vterm_screen_get_text(screen, (char *)text, len, rect);
877
878 if(text[-2] != 0x55 || text[-1] != 0xAA)
879 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
880 else if(text[len] != 0x55 || text[len+1] != 0xAA)
881 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
882 else
883 {
884 size_t i;
885 for(i = 0; i < len; i++) {
886 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
887 }
888 }
889
890 free(buffer);
891 }
892 }
893 else if(strstartswith(line, "?screen_cell ")) {
894 char *linep = line + 12;
895 int i;
896 VTermPos pos;
897 VTermScreenCell cell;
898 while(linep[0] == ' ')
899 linep++;
900 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
901 printf("! screen_cell unrecognised input\n");
902 goto abort_line;
903 }
904 if(!vterm_screen_get_cell(screen, pos, &cell))
905 goto abort_line;
906 printf("{");
907 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
908 printf("%s0x%x", i ? "," : "", cell.chars[i]);
909 }
910 printf("} width=%d attrs={", cell.width);
911 if(cell.attrs.bold) printf("B");
912 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
913 if(cell.attrs.italic) printf("I");
914 if(cell.attrs.blink) printf("K");
915 if(cell.attrs.reverse) printf("R");
916 if(cell.attrs.font) printf("F%d", cell.attrs.font);
917 printf("} ");
918 if(cell.attrs.dwl) printf("dwl ");
919 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
920 printf("fg=rgb(%d,%d,%d) ", cell.fg.red, cell.fg.green, cell.fg.blue);
921 printf("bg=rgb(%d,%d,%d)\n", cell.bg.red, cell.bg.green, cell.bg.blue);
922 }
923 else if(strstartswith(line, "?screen_eol ")) {
924 VTermPos pos;
925 char *linep = line + 12;
926 while(linep[0] == ' ')
927 linep++;
928 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
929 printf("! screen_eol unrecognised input\n");
930 goto abort_line;
931 }
932 printf("%d\n", vterm_screen_is_eol(screen, pos));
933 }
934 else if(strstartswith(line, "?screen_attrs_extent ")) {
935 VTermPos pos;
936 VTermRect rect;
937 char *linep = line + 21;
938 while(linep[0] == ' ')
939 linep++;
940 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
941 printf("! screen_attrs_extent unrecognised input\n");
942 goto abort_line;
943 }
944 rect.start_col = 0;
945 rect.end_col = -1;
946 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
947 printf("! screen_attrs_extent failed\n");
948 goto abort_line;
949 }
950 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
951 }
952 else
953 printf("?\n");
954
955 memset(line, 0, sizeof line);
956 continue;
957 }
958
959 else
960 abort_line: err = 1;
961
962 outlen = vterm_output_get_buffer_current(vt);
963 if(outlen > 0) {
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200964 char outbuff[1024];
965 vterm_output_read(vt, outbuff, outlen);
966
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200967 term_output(outbuff, outlen, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200968 }
969
970 printf(err ? "?\n" : "DONE\n");
971 }
972
973 vterm_free(vt);
974
975 return 0;
976}