blob: 4e24198cdee2e9128aeb811d6c971d26c95e98a9 [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;
Bram Moolenaar7da34152021-11-24 19:30:55 +000016 if(sscanf(inpos, "%2x", &ch) < 1)
17 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +020018 *outpos = ch;
19 outpos += 1; inpos += 2;
20 }
21
22 return outpos - s;
23}
24
25static VTermModifier strpe_modifiers(char **strp)
26{
27 VTermModifier state = 0;
28
29 while((*strp)[0]) {
30 switch(((*strp)++)[0]) {
31 case 'S': state |= VTERM_MOD_SHIFT; break;
32 case 'C': state |= VTERM_MOD_CTRL; break;
33 case 'A': state |= VTERM_MOD_ALT; break;
34 default: return state;
35 }
36 }
37
38 return state;
39}
40
41static VTermKey strp_key(char *str)
42{
43 static struct {
44 char *name;
45 VTermKey key;
46 } keys[] = {
47 { "Up", VTERM_KEY_UP },
48 { "Tab", VTERM_KEY_TAB },
49 { "Enter", VTERM_KEY_ENTER },
50 { "KP0", VTERM_KEY_KP_0 },
Bram Moolenaarc4c9f7e2020-05-17 20:52:45 +020051 { "F1", VTERM_KEY_FUNCTION(1) },
Bram Moolenaare4f25e42017-07-07 11:54:15 +020052 { NULL, VTERM_KEY_NONE },
53 };
54 int i;
55
56 for(i = 0; keys[i].name; i++) {
57 if(streq(str, keys[i].name))
58 return keys[i].key;
59 }
60
61 return VTERM_KEY_NONE;
62}
63
Bram Moolenaare5886cc2020-05-21 20:10:04 +020064static void print_color(const VTermColor *col)
65{
66 if (VTERM_COLOR_IS_RGB(col)) {
67 printf("rgb(%d,%d,%d", col->red, col->green, col->blue);
68 }
69 else if (VTERM_COLOR_IS_INDEXED(col)) {
70 printf("idx(%d", col->index);
71 }
72 else {
73 printf("invalid(%d", col->type);
74 }
75 if (VTERM_COLOR_IS_DEFAULT_FG(col)) {
76 printf(",is_default_fg");
77 }
78 if (VTERM_COLOR_IS_DEFAULT_BG(col)) {
79 printf(",is_default_bg");
80 }
81 printf(")");
82}
83
Bram Moolenaare4f25e42017-07-07 11:54:15 +020084static VTerm *vt;
85static VTermState *state;
86static VTermScreen *screen;
87
88static VTermEncodingInstance encoding;
89
Bram Moolenaarcd630be2020-04-12 14:50:26 +020090static void term_output(const char *s, size_t len, void *user UNUSED)
Bram Moolenaar88c1ee82020-04-12 13:38:57 +020091{
92 size_t i;
93
94 printf("output ");
95 for(i = 0; i < len; i++)
96 printf("%x%s", (unsigned char)s[i], i < len-1 ? "," : "\n");
97}
98
99static void printhex(const char *s, size_t len)
100{
101 while(len--)
Bram Moolenaar7da34152021-11-24 19:30:55 +0000102 printf("%02x", (uint8_t)(s++)[0]);
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200103}
104
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200105static int parser_text(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200106{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200107 size_t i;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200108
109 printf("text ");
110 for(i = 0; i < len; i++) {
111 unsigned char b = bytes[i];
112 if(b < 0x20 || b == 0x7f || (b >= 0x80 && b < 0xa0))
113 break;
114 printf(i ? ",%x" : "%x", b);
115 }
116 printf("\n");
117
118 return i;
119}
120
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200121static int parser_control(unsigned char control, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200122{
123 printf("control %02x\n", control);
124
125 return 1;
126}
127
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200128static int parser_escape(const char bytes[], size_t len, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200129{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200130 if(bytes[0] >= 0x20 && bytes[0] < 0x30) {
131 if(len < 2)
132 return -1;
133 len = 2;
134 }
135 else {
136 len = 1;
137 }
138
139 printf("escape ");
Bram Moolenaar88c1ee82020-04-12 13:38:57 +0200140 printhex(bytes, len);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200141 printf("\n");
142
143 return len;
144}
145
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200146static 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 +0200147{
148 int i;
149 printf("csi %02x", command);
150
151 if(leader && leader[0]) {
152 printf(" L=");
153 for(i = 0; leader[i]; i++)
154 printf("%02x", leader[i]);
155 }
156
157 for(i = 0; i < argcount; i++) {
158 char sep = i ? ',' : ' ';
159
160 if(args[i] == CSI_ARG_MISSING)
161 printf("%c*", sep);
162 else
163 printf("%c%ld%s", sep, CSI_ARG(args[i]), CSI_ARG_HAS_MORE(args[i]) ? "+" : "");
164 }
165
166 if(intermed && intermed[0]) {
167 printf(" I=");
168 for(i = 0; intermed[i]; i++)
169 printf("%02x", intermed[i]);
170 }
171
172 printf("\n");
173
174 return 1;
175}
176
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200177static int parser_osc(int command, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200178{
Bram Moolenaar81ea1df2020-04-11 18:01:41 +0200179
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200180 printf("osc ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200181
182 if(frag.initial) {
183 if(command == -1)
184 printf("[");
185 else
186 printf("[%d;", command);
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
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200199static int parser_dcs(const char *command, size_t commandlen, VTermStringFragment frag, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200200{
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200201 printf("dcs ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200202
203 if(frag.initial) {
204 size_t i;
205 printf("[");
206 for(i = 0; i < commandlen; i++)
207 printf("%02x", command[i]);
208 }
209
210 printhex(frag.str, frag.len);
211
212 if(frag.final)
213 printf("]");
214
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200215 printf("\n");
216
217 return 1;
218}
219
Bram Moolenaar7da34152021-11-24 19:30:55 +0000220static int parser_apc(VTermStringFragment frag, void *user UNUSED)
221{
222 printf("apc ");
223
224 if(frag.initial)
225 printf("[");
226
227 printhex(frag.str, frag.len);
228
229 if(frag.final)
230 printf("]");
231
232 printf("\n");
233
234 return 1;
235}
236
237static int parser_pm(VTermStringFragment frag, void *user UNUSED)
238{
239 printf("pm ");
240
241 if(frag.initial)
242 printf("[");
243
244 printhex(frag.str, frag.len);
245
246 if(frag.final)
247 printf("]");
248
249 printf("\n");
250
251 return 1;
252}
253
254static int parser_sos(VTermStringFragment frag, void *user UNUSED)
255{
256 printf("sos ");
257
258 if(frag.initial)
259 printf("[");
260
261 printhex(frag.str, frag.len);
262
263 if(frag.final)
264 printf("]");
265
266 printf("\n");
267
268 return 1;
269}
270
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200271static VTermParserCallbacks parser_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100272 parser_text, // text
273 parser_control, // control
274 parser_escape, // escape
275 parser_csi, // csi
276 parser_osc, // osc
277 parser_dcs, // dcs
Bram Moolenaar7da34152021-11-24 19:30:55 +0000278 parser_apc, // apc
279 parser_pm, // pm
280 parser_sos, // sos
Bram Moolenaar707d2262019-12-04 22:16:54 +0100281 NULL // resize
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200282};
283
Bram Moolenaard8637282020-05-20 18:41:41 +0200284static VTermStateFallbacks fallbacks = {
285 parser_control, // control
286 parser_csi, // csi
287 parser_osc, // osc
Bram Moolenaar7da34152021-11-24 19:30:55 +0000288 parser_dcs, // dcs
289 parser_apc, // dcs
290 parser_pm, // pm
291 parser_sos // sos
Bram Moolenaard8637282020-05-20 18:41:41 +0200292};
293
294/* These callbacks are shared by State and Screen */
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200295
296static int want_movecursor = 0;
297static VTermPos state_pos;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200298static int movecursor(VTermPos pos, VTermPos oldpos UNUSED, int visible UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200299{
300 state_pos = pos;
301
302 if(want_movecursor)
303 printf("movecursor %d,%d\n", pos.row, pos.col);
304
305 return 1;
306}
307
308static int want_scrollrect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200309static int scrollrect(VTermRect rect, int downward, int rightward, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200310{
311 if(!want_scrollrect)
312 return 0;
313
314 printf("scrollrect %d..%d,%d..%d => %+d,%+d\n",
315 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
316 downward, rightward);
317
318 return 1;
319}
320
321static int want_moverect = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200322static int moverect(VTermRect dest, VTermRect src, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200323{
324 if(!want_moverect)
325 return 0;
326
327 printf("moverect %d..%d,%d..%d -> %d..%d,%d..%d\n",
328 src.start_row, src.end_row, src.start_col, src.end_col,
329 dest.start_row, dest.end_row, dest.start_col, dest.end_col);
330
331 return 1;
332}
333
334static int want_settermprop = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200335static int settermprop(VTermProp prop, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200336{
337 VTermValueType type;
338 if(!want_settermprop)
339 return 1;
340
341 type = vterm_get_prop_type(prop);
342 switch(type) {
343 case VTERM_VALUETYPE_BOOL:
344 printf("settermprop %d %s\n", prop, val->boolean ? "true" : "false");
345 return 1;
346 case VTERM_VALUETYPE_INT:
347 printf("settermprop %d %d\n", prop, val->number);
348 return 1;
349 case VTERM_VALUETYPE_STRING:
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200350 printf("settermprop %d %s\"%.*s\"%s\n", prop,
351 val->string.initial ? "[" : "", val->string.len, val->string.str, val->string.final ? "]" : "");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200352 return 1;
353 case VTERM_VALUETYPE_COLOR:
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200354 printf("settermprop %d ", prop);
355 print_color(&val->color);
356 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200357 return 1;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200358
359 case VTERM_N_VALUETYPES:
360 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200361 }
362
363 return 0;
364}
365
Bram Moolenaar707d2262019-12-04 22:16:54 +0100366// These callbacks are for State
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200367
368static int want_state_putglyph = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200369static int state_putglyph(VTermGlyphInfo *info, VTermPos pos, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200370{
371 int i;
372 if(!want_state_putglyph)
373 return 1;
374
375 printf("putglyph ");
Bram Moolenaarbe593bf2020-05-19 21:20:04 +0200376 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && info->chars[i]; i++)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200377 printf(i ? ",%x" : "%x", info->chars[i]);
378 printf(" %d %d,%d", info->width, pos.row, pos.col);
379 if(info->protected_cell)
380 printf(" prot");
381 if(info->dwl)
382 printf(" dwl");
383 if(info->dhl)
384 printf(" dhl-%s", info->dhl == 1 ? "top" : info->dhl == 2 ? "bottom" : "?" );
385 printf("\n");
386
387 return 1;
388}
389
390static int want_state_erase = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200391static int state_erase(VTermRect rect, int selective, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200392{
393 if(!want_state_erase)
394 return 1;
395
396 printf("erase %d..%d,%d..%d%s\n",
397 rect.start_row, rect.end_row, rect.start_col, rect.end_col,
398 selective ? " selective" : "");
399
400 return 1;
401}
402
403static struct {
404 int bold;
405 int underline;
406 int italic;
407 int blink;
408 int reverse;
Bram Moolenaard8637282020-05-20 18:41:41 +0200409 int conceal;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200410 int strike;
411 int font;
412 VTermColor foreground;
413 VTermColor background;
414} state_pen;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200415static int state_setpenattr(VTermAttr attr, VTermValue *val, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200416{
417 switch(attr) {
418 case VTERM_ATTR_BOLD:
419 state_pen.bold = val->boolean;
420 break;
421 case VTERM_ATTR_UNDERLINE:
422 state_pen.underline = val->number;
423 break;
424 case VTERM_ATTR_ITALIC:
425 state_pen.italic = val->boolean;
426 break;
427 case VTERM_ATTR_BLINK:
428 state_pen.blink = val->boolean;
429 break;
430 case VTERM_ATTR_REVERSE:
431 state_pen.reverse = val->boolean;
432 break;
Bram Moolenaard8637282020-05-20 18:41:41 +0200433 case VTERM_ATTR_CONCEAL:
434 state_pen.conceal = val->boolean;
435 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200436 case VTERM_ATTR_STRIKE:
437 state_pen.strike = val->boolean;
438 break;
439 case VTERM_ATTR_FONT:
440 state_pen.font = val->number;
441 break;
442 case VTERM_ATTR_FOREGROUND:
443 state_pen.foreground = val->color;
444 break;
445 case VTERM_ATTR_BACKGROUND:
446 state_pen.background = val->color;
447 break;
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200448
449 case VTERM_N_ATTRS:
450 return 0;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200451 }
452
453 return 1;
454}
455
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200456static int state_setlineinfo(int row UNUSED, const VTermLineInfo *newinfo UNUSED, const VTermLineInfo *oldinfo UNUSED, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200457{
458 return 1;
459}
460
461VTermStateCallbacks state_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100462 state_putglyph, // putglyph
463 movecursor, // movecursor
464 scrollrect, // scrollrect
465 moverect, // moverect
466 state_erase, // erase
467 NULL, // initpen
468 state_setpenattr, // setpenattr
469 settermprop, // settermprop
470 NULL, // bell
471 NULL, // resize
472 state_setlineinfo, // setlineinfo
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200473};
474
Bram Moolenaar7da34152021-11-24 19:30:55 +0000475static int selection_set(VTermSelectionMask mask, VTermStringFragment frag, void *user UNUSED)
476{
477 printf("selection-set mask=%04X ", mask);
478 if(frag.initial)
479 printf("[");
480 printhex(frag.str, frag.len);
481 if(frag.final)
482 printf("]");
483 printf("\n");
484
485 return 1;
486}
487
488static int selection_query(VTermSelectionMask mask, void *user UNUSED)
489{
490 printf("selection-query mask=%04X\n", mask);
491
492 return 1;
493}
494
495VTermSelectionCallbacks selection_cbs = {
496 .set = selection_set,
497 .query = selection_query,
498};
499
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200500static int want_screen_damage = 0;
501static int want_screen_damage_cells = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200502static int screen_damage(VTermRect rect, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200503{
504 if(!want_screen_damage)
505 return 1;
506
507 printf("damage %d..%d,%d..%d",
508 rect.start_row, rect.end_row, rect.start_col, rect.end_col);
509
510 if(want_screen_damage_cells) {
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200511 int equals = FALSE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200512 int row;
513 int col;
514
515 for(row = rect.start_row; row < rect.end_row; row++) {
516 int eol = rect.end_col;
517 while(eol > rect.start_col) {
518 VTermScreenCell cell;
519 VTermPos pos;
520 pos.row = row;
521 pos.col = eol-1;
522 vterm_screen_get_cell(screen, pos, &cell);
523 if(cell.chars[0])
524 break;
525
526 eol--;
527 }
528
529 if(eol == rect.start_col)
530 break;
531
532 if(!equals)
Bram Moolenaarb2a76ec2017-07-25 21:34:46 +0200533 printf(" ="), equals = TRUE;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200534
535 printf(" %d<", row);
536 for(col = rect.start_col; col < eol; col++) {
537 VTermScreenCell cell;
538 VTermPos pos;
539 pos.row = row;
540 pos.col = col;
541 vterm_screen_get_cell(screen, pos, &cell);
542 printf(col == rect.start_col ? "%02X" : " %02X", cell.chars[0]);
543 }
544 printf(">");
545 }
546 }
547
548 printf("\n");
549
550 return 1;
551}
552
553static int want_screen_scrollback = 0;
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200554static int screen_sb_pushline(int cols, const VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200555{
556 int eol;
557 int c;
558
559 if(!want_screen_scrollback)
560 return 1;
561
562 eol = cols;
563 while(eol && !cells[eol-1].chars[0])
564 eol--;
565
566 printf("sb_pushline %d =", cols);
567 for(c = 0; c < eol; c++)
568 printf(" %02X", cells[c].chars[0]);
569 printf("\n");
570
571 return 1;
572}
573
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200574static int screen_sb_popline(int cols, VTermScreenCell *cells, void *user UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200575{
576 int col;
577
578 if(!want_screen_scrollback)
579 return 0;
580
Bram Moolenaar707d2262019-12-04 22:16:54 +0100581 // All lines of scrollback contain "ABCDE"
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200582 for(col = 0; col < cols; col++) {
583 if(col < 5)
584 cells[col].chars[0] = 'A' + col;
585 else
586 cells[col].chars[0] = 0;
587
588 cells[col].width = 1;
589 }
590
591 printf("sb_popline %d\n", cols);
592 return 1;
593}
594
595VTermScreenCallbacks screen_cbs = {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100596 screen_damage, // damage
597 moverect, // moverect
598 movecursor, // movecursor
599 settermprop, // settermprop
600 NULL, // bell
601 NULL, // resize
602 screen_sb_pushline, // sb_pushline
603 screen_sb_popline // sb_popline
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200604};
605
Bram Moolenaard19a8f92020-04-11 21:42:48 +0200606int main(int argc UNUSED, char **argv UNUSED)
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200607{
608 char line[1024] = {0};
609 int flag;
610
611 int err;
612
613 setvbuf(stdout, NULL, _IONBF, 0);
614
615 while(fgets(line, sizeof line, stdin)) {
616 char *nl;
617 size_t outlen;
618 err = 0;
619
620 if((nl = strchr(line, '\n')))
621 *nl = '\0';
622
623 if(streq(line, "INIT")) {
624 if(!vt)
625 vt = vterm_new(25, 80);
Bram Moolenaar94d729c2020-05-17 21:50:16 +0200626
627 // Somehow this makes tests fail
628 // vterm_output_set_callback(vt, term_output, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200629 }
630
631 else if(streq(line, "WANTPARSER")) {
632 vterm_parser_set_callbacks(vt, &parser_cbs, NULL);
633 }
634
635 else if(strstartswith(line, "WANTSTATE") && (line[9] == '\0' || line[9] == ' ')) {
636 int i = 9;
637 int sense = 1;
638 if(!state) {
639 state = vterm_obtain_state(vt);
640 vterm_state_set_callbacks(state, &state_cbs, NULL);
Bram Moolenaar7da34152021-11-24 19:30:55 +0000641 vterm_state_set_selection_callbacks(state, &selection_cbs, NULL, NULL, 1024);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200642 vterm_state_set_bold_highbright(state, 1);
643 vterm_state_reset(state, 1);
644 }
645
646 while(line[i] == ' ')
647 i++;
648 for( ; line[i]; i++)
649 switch(line[i]) {
650 case '+':
651 sense = 1;
652 break;
653 case '-':
654 sense = 0;
655 break;
656 case 'g':
657 want_state_putglyph = sense;
658 break;
659 case 's':
660 want_scrollrect = sense;
661 break;
662 case 'm':
663 want_moverect = sense;
664 break;
665 case 'e':
666 want_state_erase = sense;
667 break;
668 case 'p':
669 want_settermprop = sense;
670 break;
671 case 'f':
Bram Moolenaard8637282020-05-20 18:41:41 +0200672 vterm_state_set_unrecognised_fallbacks(state, sense ? &fallbacks : NULL, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200673 break;
674 default:
675 fprintf(stderr, "Unrecognised WANTSTATE flag '%c'\n", line[i]);
676 }
677 }
678
679 else if(strstartswith(line, "WANTSCREEN") && (line[10] == '\0' || line[10] == ' ')) {
680 int i = 10;
681 int sense = 1;
682 if(!screen)
683 screen = vterm_obtain_screen(vt);
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200684 vterm_screen_set_callbacks(screen, &screen_cbs, NULL);
685
686 while(line[i] == ' ')
687 i++;
688 for( ; line[i]; i++)
689 switch(line[i]) {
690 case '-':
691 sense = 0;
692 break;
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200693 case 'a':
694 vterm_screen_enable_altscreen(screen, 1);
695 break;
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200696 case 'd':
697 want_screen_damage = sense;
698 break;
699 case 'D':
700 want_screen_damage = sense;
701 want_screen_damage_cells = sense;
702 break;
703 case 'm':
704 want_moverect = sense;
705 break;
706 case 'c':
707 want_movecursor = sense;
708 break;
709 case 'p':
710 want_settermprop = 1;
711 break;
712 case 'b':
713 want_screen_scrollback = sense;
714 break;
715 default:
716 fprintf(stderr, "Unrecognised WANTSCREEN flag '%c'\n", line[i]);
717 }
718 }
719
720 else if(sscanf(line, "UTF8 %d", &flag)) {
721 vterm_set_utf8(vt, flag);
722 }
723
724 else if(streq(line, "RESET")) {
725 if(state) {
726 vterm_state_reset(state, 1);
727 vterm_state_get_cursorpos(state, &state_pos);
728 }
729 if(screen) {
730 vterm_screen_reset(screen, 1);
731 }
732 }
733
734 else if(strstartswith(line, "RESIZE ")) {
735 int rows, cols;
736 char *linep = line + 7;
737 while(linep[0] == ' ')
738 linep++;
739 sscanf(linep, "%d, %d", &rows, &cols);
740 vterm_set_size(vt, rows, cols);
741 }
742
743 else if(strstartswith(line, "PUSH ")) {
744 char *bytes = line + 5;
745 size_t len = inplace_hex2bytes(bytes);
746 size_t written = vterm_input_write(vt, bytes, len);
747 if(written < len)
748 fprintf(stderr, "! short write\n");
749 }
750
751 else if(streq(line, "WANTENCODING")) {
Bram Moolenaar707d2262019-12-04 22:16:54 +0100752 // This isn't really external API but it's hard to get this out any
753 // other way
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200754 encoding.enc = vterm_lookup_encoding(ENC_UTF8, 'u');
755 if(encoding.enc->init)
756 (*encoding.enc->init)(encoding.enc, encoding.data);
757 }
758
759 else if(strstartswith(line, "ENCIN ")) {
760 char *bytes = line + 6;
761 size_t len = inplace_hex2bytes(bytes);
762
763 uint32_t cp[1024];
764 int cpi = 0;
765 size_t pos = 0;
766
767 (*encoding.enc->decode)(encoding.enc, encoding.data,
768 cp, &cpi, len, bytes, &pos, len);
769
770 if(cpi > 0) {
771 int i;
772 printf("encout ");
773 for(i = 0; i < cpi; i++) {
774 printf(i ? ",%x" : "%x", cp[i]);
775 }
776 printf("\n");
777 }
778 }
779
780 else if(strstartswith(line, "INCHAR ")) {
781 char *linep = line + 7;
782 unsigned int c = 0;
783 VTermModifier mod;
784 while(linep[0] == ' ')
785 linep++;
786 mod = strpe_modifiers(&linep);
787 sscanf(linep, " %x", &c);
788
789 vterm_keyboard_unichar(vt, c, mod);
790 }
791
792 else if(strstartswith(line, "INKEY ")) {
793 VTermModifier mod;
794 VTermKey key;
795 char *linep = line + 6;
796 while(linep[0] == ' ')
797 linep++;
798 mod = strpe_modifiers(&linep);
799 while(linep[0] == ' ')
800 linep++;
801 key = strp_key(linep);
802
803 vterm_keyboard_key(vt, key, mod);
804 }
805
806 else if(strstartswith(line, "PASTE ")) {
807 char *linep = line + 6;
808 if(streq(linep, "START"))
809 vterm_keyboard_start_paste(vt);
810 else if(streq(linep, "END"))
811 vterm_keyboard_end_paste(vt);
812 else
813 goto abort_line;
814 }
815
Bram Moolenaarb5b49a32018-03-25 16:20:37 +0200816 else if(strstartswith(line, "FOCUS ")) {
817 char *linep = line + 6;
818 if(streq(linep, "IN"))
819 vterm_state_focus_in(state);
820 else if(streq(linep, "OUT"))
821 vterm_state_focus_out(state);
822 else
823 goto abort_line;
824 }
825
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200826 else if(strstartswith(line, "MOUSEMOVE ")) {
827 char *linep = line + 10;
828 int row, col, len;
829 VTermModifier mod;
830 while(linep[0] == ' ')
831 linep++;
832 sscanf(linep, "%d,%d%n", &row, &col, &len);
833 linep += len;
834 while(linep[0] == ' ')
835 linep++;
836 mod = strpe_modifiers(&linep);
837 vterm_mouse_move(vt, row, col, mod);
838 }
839
840 else if(strstartswith(line, "MOUSEBTN ")) {
841 char *linep = line + 9;
842 char press;
843 int button, len;
844 VTermModifier mod;
845 while(linep[0] == ' ')
846 linep++;
847 sscanf(linep, "%c %d%n", &press, &button, &len);
848 linep += len;
849 while(linep[0] == ' ')
850 linep++;
851 mod = strpe_modifiers(&linep);
852 vterm_mouse_button(vt, button, (press == 'd' || press == 'D'), mod);
853 }
854
Bram Moolenaar7da34152021-11-24 19:30:55 +0000855 else if(strstartswith(line, "SELECTION ")) {
856 char *linep = line + 10;
857 unsigned int mask;
858 int len;
859 VTermStringFragment frag = { 0 };
860 sscanf(linep, "%x%n", &mask, &len);
861 linep += len;
862 while(linep[0] == ' ')
863 linep++;
864 if(linep[0] == '[') {
865 frag.initial = TRUE;
866 linep++;
867 while(linep[0] == ' ')
868 linep++;
869 }
870 frag.len = inplace_hex2bytes(linep);
871 frag.str = linep;
872 linep += frag.len * 2;
873 while(linep[0] == ' ')
874 linep++;
875 if(linep[0] == ']') {
876 frag.final = TRUE;
877 }
878 vterm_state_send_selection(state, mask, frag);
879 }
880
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200881 else if(strstartswith(line, "DAMAGEMERGE ")) {
882 char *linep = line + 12;
883 while(linep[0] == ' ')
884 linep++;
885 if(streq(linep, "CELL"))
886 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_CELL);
887 else if(streq(linep, "ROW"))
888 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_ROW);
889 else if(streq(linep, "SCREEN"))
890 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCREEN);
891 else if(streq(linep, "SCROLL"))
892 vterm_screen_set_damage_merge(screen, VTERM_DAMAGE_SCROLL);
893 }
894
895 else if(strstartswith(line, "DAMAGEFLUSH")) {
896 vterm_screen_flush_damage(screen);
897 }
898
899 else if(line[0] == '?') {
900 if(streq(line, "?cursor")) {
901 VTermPos pos;
902 vterm_state_get_cursorpos(state, &pos);
903 if(pos.row != state_pos.row)
904 printf("! row mismatch: state=%d,%d event=%d,%d\n",
905 pos.row, pos.col, state_pos.row, state_pos.col);
906 else if(pos.col != state_pos.col)
907 printf("! col mismatch: state=%d,%d event=%d,%d\n",
908 pos.row, pos.col, state_pos.row, state_pos.col);
909 else
910 printf("%d,%d\n", state_pos.row, state_pos.col);
911 }
912 else if(strstartswith(line, "?pen ")) {
913 VTermValue val;
914 char *linep = line + 5;
915 while(linep[0] == ' ')
916 linep++;
917
918#define BOOLSTR(v) ((v) ? "on" : "off")
919
920 if(streq(linep, "bold")) {
921 vterm_state_get_penattr(state, VTERM_ATTR_BOLD, &val);
922 if(val.boolean != state_pen.bold)
923 printf("! pen bold mismatch; state=%s, event=%s\n",
924 BOOLSTR(val.boolean), BOOLSTR(state_pen.bold));
925 else
926 printf("%s\n", BOOLSTR(state_pen.bold));
927 }
928 else if(streq(linep, "underline")) {
929 vterm_state_get_penattr(state, VTERM_ATTR_UNDERLINE, &val);
930 if(val.boolean != state_pen.underline)
931 printf("! pen underline mismatch; state=%d, event=%d\n",
932 val.boolean, state_pen.underline);
933 else
934 printf("%d\n", state_pen.underline);
935 }
936 else if(streq(linep, "italic")) {
937 vterm_state_get_penattr(state, VTERM_ATTR_ITALIC, &val);
938 if(val.boolean != state_pen.italic)
939 printf("! pen italic mismatch; state=%s, event=%s\n",
940 BOOLSTR(val.boolean), BOOLSTR(state_pen.italic));
941 else
942 printf("%s\n", BOOLSTR(state_pen.italic));
943 }
944 else if(streq(linep, "blink")) {
945 vterm_state_get_penattr(state, VTERM_ATTR_BLINK, &val);
946 if(val.boolean != state_pen.blink)
947 printf("! pen blink mismatch; state=%s, event=%s\n",
948 BOOLSTR(val.boolean), BOOLSTR(state_pen.blink));
949 else
950 printf("%s\n", BOOLSTR(state_pen.blink));
951 }
952 else if(streq(linep, "reverse")) {
953 vterm_state_get_penattr(state, VTERM_ATTR_REVERSE, &val);
954 if(val.boolean != state_pen.reverse)
955 printf("! pen reverse mismatch; state=%s, event=%s\n",
956 BOOLSTR(val.boolean), BOOLSTR(state_pen.reverse));
957 else
958 printf("%s\n", BOOLSTR(state_pen.reverse));
959 }
960 else if(streq(linep, "font")) {
961 vterm_state_get_penattr(state, VTERM_ATTR_FONT, &val);
962 if(val.boolean != state_pen.font)
963 printf("! pen font mismatch; state=%d, event=%d\n",
964 val.boolean, state_pen.font);
965 else
966 printf("%d\n", state_pen.font);
967 }
968 else if(streq(linep, "foreground")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200969 print_color(&state_pen.foreground);
970 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200971 }
972 else if(streq(linep, "background")) {
Bram Moolenaare5886cc2020-05-21 20:10:04 +0200973 print_color(&state_pen.background);
974 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200975 }
976 else
977 printf("?\n");
978 }
Bram Moolenaar88d68de2020-05-18 21:51:01 +0200979 else if(strstartswith(line, "?lineinfo ")) {
980 char *linep = line + 10;
981 int row;
982 const VTermLineInfo *info;
983 while(linep[0] == ' ')
984 linep++;
985 if(sscanf(linep, "%d", &row) < 1) {
986 printf("! lineinfo unrecognised input\n");
987 goto abort_line;
988 }
989 info = vterm_state_get_lineinfo(state, row);
990 if(info->doublewidth)
991 printf("dwl ");
992 if(info->doubleheight)
993 printf("dhl ");
994 if(info->continuation)
995 printf("cont ");
996 printf("\n");
997 }
Bram Moolenaare4f25e42017-07-07 11:54:15 +0200998 else if(strstartswith(line, "?screen_chars ")) {
999 char *linep = line + 13;
1000 VTermRect rect;
1001 size_t len;
1002 while(linep[0] == ' ')
1003 linep++;
Bram Moolenaar501e7772022-10-16 14:35:46 +01001004 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) == 4)
1005 ; // fine
1006 else if(sscanf(linep, "%d", &rect.start_row) == 1) {
1007 rect.end_row = rect.start_row + 1;
1008 rect.start_col = 0;
1009 vterm_get_size(vt, NULL, &rect.end_col);
1010 }
1011 else {
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001012 printf("! screen_chars unrecognised input\n");
1013 goto abort_line;
1014 }
1015 len = vterm_screen_get_chars(screen, NULL, 0, rect);
1016 if(len == (size_t)-1)
1017 printf("! screen_chars error\n");
1018 else if(len == 0)
1019 printf("\n");
1020 else {
1021 uint32_t *chars = malloc(sizeof(uint32_t) * len);
1022 size_t i;
1023 vterm_screen_get_chars(screen, chars, len, rect);
1024 for(i = 0; i < len; i++) {
1025 printf("0x%02x%s", chars[i], i < len-1 ? "," : "\n");
1026 }
1027 free(chars);
1028 }
1029 }
1030 else if(strstartswith(line, "?screen_text ")) {
1031 char *linep = line + 12;
1032 VTermRect rect;
1033 size_t len;
1034 while(linep[0] == ' ')
1035 linep++;
1036 if(sscanf(linep, "%d,%d,%d,%d", &rect.start_row, &rect.start_col, &rect.end_row, &rect.end_col) < 4) {
1037 printf("! screen_text unrecognised input\n");
1038 goto abort_line;
1039 }
1040 len = vterm_screen_get_text(screen, NULL, 0, rect);
1041 if(len == (size_t)-1)
1042 printf("! screen_text error\n");
1043 else if(len == 0)
1044 printf("\n");
1045 else {
Bram Moolenaar707d2262019-12-04 22:16:54 +01001046 // Put an overwrite guard at both ends of the buffer
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001047 unsigned char *buffer = malloc(len + 4);
1048 unsigned char *text = buffer + 2;
1049 text[-2] = 0x55; text[-1] = 0xAA;
1050 text[len] = 0x55; text[len+1] = 0xAA;
1051
1052 vterm_screen_get_text(screen, (char *)text, len, rect);
1053
1054 if(text[-2] != 0x55 || text[-1] != 0xAA)
1055 printf("! screen_get_text buffer overrun left [%02x,%02x]\n", text[-2], text[-1]);
1056 else if(text[len] != 0x55 || text[len+1] != 0xAA)
1057 printf("! screen_get_text buffer overrun right [%02x,%02x]\n", text[len], text[len+1]);
1058 else
1059 {
1060 size_t i;
1061 for(i = 0; i < len; i++) {
1062 printf("0x%02x%s", text[i], i < len-1 ? "," : "\n");
1063 }
1064 }
1065
1066 free(buffer);
1067 }
1068 }
1069 else if(strstartswith(line, "?screen_cell ")) {
1070 char *linep = line + 12;
1071 int i;
1072 VTermPos pos;
1073 VTermScreenCell cell;
1074 while(linep[0] == ' ')
1075 linep++;
1076 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1077 printf("! screen_cell unrecognised input\n");
1078 goto abort_line;
1079 }
1080 if(!vterm_screen_get_cell(screen, pos, &cell))
1081 goto abort_line;
1082 printf("{");
1083 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell.chars[i]; i++) {
1084 printf("%s0x%x", i ? "," : "", cell.chars[i]);
1085 }
1086 printf("} width=%d attrs={", cell.width);
1087 if(cell.attrs.bold) printf("B");
1088 if(cell.attrs.underline) printf("U%d", cell.attrs.underline);
1089 if(cell.attrs.italic) printf("I");
1090 if(cell.attrs.blink) printf("K");
1091 if(cell.attrs.reverse) printf("R");
1092 if(cell.attrs.font) printf("F%d", cell.attrs.font);
1093 printf("} ");
1094 if(cell.attrs.dwl) printf("dwl ");
1095 if(cell.attrs.dhl) printf("dhl-%s ", cell.attrs.dhl == 2 ? "bottom" : "top");
Bram Moolenaare5886cc2020-05-21 20:10:04 +02001096 printf("fg=");
1097 vterm_screen_convert_color_to_rgb(screen, &cell.fg);
1098 print_color(&cell.fg);
1099 printf(" bg=");
1100 vterm_screen_convert_color_to_rgb(screen, &cell.bg);
1101 print_color(&cell.bg);
1102 printf("\n");
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001103 }
1104 else if(strstartswith(line, "?screen_eol ")) {
1105 VTermPos pos;
1106 char *linep = line + 12;
1107 while(linep[0] == ' ')
1108 linep++;
1109 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1110 printf("! screen_eol unrecognised input\n");
1111 goto abort_line;
1112 }
1113 printf("%d\n", vterm_screen_is_eol(screen, pos));
1114 }
1115 else if(strstartswith(line, "?screen_attrs_extent ")) {
1116 VTermPos pos;
1117 VTermRect rect;
1118 char *linep = line + 21;
1119 while(linep[0] == ' ')
1120 linep++;
1121 if(sscanf(linep, "%d,%d\n", &pos.row, &pos.col) < 2) {
1122 printf("! screen_attrs_extent unrecognised input\n");
1123 goto abort_line;
1124 }
1125 rect.start_col = 0;
1126 rect.end_col = -1;
1127 if(!vterm_screen_get_attrs_extent(screen, &rect, pos, ~0)) {
1128 printf("! screen_attrs_extent failed\n");
1129 goto abort_line;
1130 }
1131 printf("%d,%d-%d,%d\n", rect.start_row, rect.start_col, rect.end_row, rect.end_col);
1132 }
1133 else
1134 printf("?\n");
1135
1136 memset(line, 0, sizeof line);
1137 continue;
1138 }
1139
1140 else
1141 abort_line: err = 1;
1142
1143 outlen = vterm_output_get_buffer_current(vt);
1144 if(outlen > 0) {
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001145 char outbuff[1024];
1146 vterm_output_read(vt, outbuff, outlen);
1147
Bram Moolenaar88c1ee82020-04-12 13:38:57 +02001148 term_output(outbuff, outlen, NULL);
Bram Moolenaare4f25e42017-07-07 11:54:15 +02001149 }
1150
1151 printf(err ? "?\n" : "DONE\n");
1152 }
1153
1154 vterm_free(vt);
1155
1156 return 0;
1157}