1 #include "vterm_internal.h"
2
3 #include <stdio.h>
4 #include <string.h>
5
6 #include "rect.h"
7 #include "utf8.h"
8
9 #define UNICODE_SPACE 0x20
10 #define UNICODE_LINEFEED 0x0a
11
12 /* State of the pen at some moment in time, also used in a cell */
13 typedef struct
14 {
15 /* After the bitfield */
16 VTermColor fg, bg;
17
18 unsigned int bold : 1;
19 unsigned int underline : 2;
20 unsigned int italic : 1;
21 unsigned int blink : 1;
22 unsigned int reverse : 1;
23 unsigned int strike : 1;
24 unsigned int font : 4; /* 0 to 9 */
25
26 /* Extra state storage that isn't strictly pen-related */
27 unsigned int protected_cell : 1;
28 unsigned int dwl : 1; /* on a DECDWL or DECDHL line */
29 unsigned int dhl : 2; /* on a DECDHL line (1=top 2=bottom) */
30 } ScreenPen;
31
32 /* Internal representation of a screen cell */
33 typedef struct
34 {
35 uint32_t chars[VTERM_MAX_CHARS_PER_CELL];
36 ScreenPen pen;
37 } ScreenCell;
38
39 static int vterm_screen_set_cell(VTermScreen *screen, VTermPos pos, const VTermScreenCell *cell);
40
41 struct VTermScreen
42 {
43 VTerm *vt;
44 VTermState *state;
45
46 const VTermScreenCallbacks *callbacks;
47 void *cbdata;
48
49 VTermDamageSize damage_merge;
50 /* start_row == -1 => no damage */
51 VTermRect damaged;
52 VTermRect pending_scrollrect;
53 int pending_scroll_downward, pending_scroll_rightward;
54
55 int rows;
56 int cols;
57 int global_reverse;
58
59 /* Primary and Altscreen. buffers[1] is lazily allocated as needed */
60 ScreenCell *buffers[2];
61
62 /* buffer will == buffers[0] or buffers[1], depending on altscreen */
63 ScreenCell *buffer;
64
65 /* buffer for a single screen row used in scrollback storage callbacks */
66 VTermScreenCell *sb_buffer;
67
68 ScreenPen pen;
69 };
70
getcell(const VTermScreen * screen,int row,int col)71 static inline ScreenCell *getcell(const VTermScreen *screen, int row, int col)
72 {
73 if(row < 0 || row >= screen->rows)
74 return NULL;
75 if(col < 0 || col >= screen->cols)
76 return NULL;
77 return screen->buffer + (screen->cols * row) + col;
78 }
79
realloc_buffer(VTermScreen * screen,ScreenCell * buffer,int new_rows,int new_cols)80 static ScreenCell *realloc_buffer(VTermScreen *screen, ScreenCell *buffer, int new_rows, int new_cols)
81 {
82 ScreenCell *new_buffer = vterm_allocator_malloc(screen->vt, sizeof(ScreenCell) * new_rows * new_cols);
83
84 for(int row = 0; row < new_rows; row++) {
85 for(int col = 0; col < new_cols; col++) {
86 ScreenCell *new_cell = new_buffer + row*new_cols + col;
87
88 if(buffer && row < screen->rows && col < screen->cols)
89 *new_cell = buffer[row * screen->cols + col];
90 else {
91 new_cell->chars[0] = 0;
92 new_cell->pen = screen->pen;
93 }
94 }
95 }
96
97 if(buffer)
98 vterm_allocator_free(screen->vt, buffer);
99
100 return new_buffer;
101 }
102
damagerect(VTermScreen * screen,VTermRect rect)103 static void damagerect(VTermScreen *screen, VTermRect rect)
104 {
105 VTermRect emit;
106
107 switch(screen->damage_merge) {
108 case VTERM_DAMAGE_CELL:
109 /* Always emit damage event */
110 emit = rect;
111 break;
112
113 case VTERM_DAMAGE_ROW:
114 /* Emit damage longer than one row. Try to merge with existing damage in
115 * the same row */
116 if(rect.end_row > rect.start_row + 1) {
117 // Bigger than 1 line - flush existing, emit this
118 vterm_screen_flush_damage(screen);
119 emit = rect;
120 }
121 else if(screen->damaged.start_row == -1) {
122 // None stored yet
123 screen->damaged = rect;
124 return;
125 }
126 else if(rect.start_row == screen->damaged.start_row) {
127 // Merge with the stored line
128 if(screen->damaged.start_col > rect.start_col)
129 screen->damaged.start_col = rect.start_col;
130 if(screen->damaged.end_col < rect.end_col)
131 screen->damaged.end_col = rect.end_col;
132 return;
133 }
134 else {
135 // Emit the currently stored line, store a new one
136 emit = screen->damaged;
137 screen->damaged = rect;
138 }
139 break;
140
141 case VTERM_DAMAGE_SCREEN:
142 case VTERM_DAMAGE_SCROLL:
143 /* Never emit damage event */
144 if(screen->damaged.start_row == -1)
145 screen->damaged = rect;
146 else {
147 rect_expand(&screen->damaged, &rect);
148 }
149 return;
150
151 default:
152 DEBUG_LOG("TODO: Maybe merge damage for level %d\n", screen->damage_merge);
153 return;
154 }
155
156 if(screen->callbacks && screen->callbacks->damage)
157 (*screen->callbacks->damage)(emit, screen->cbdata);
158 }
159
damagescreen(VTermScreen * screen)160 static void damagescreen(VTermScreen *screen)
161 {
162 VTermRect rect = {
163 .start_row = 0,
164 .end_row = screen->rows,
165 .start_col = 0,
166 .end_col = screen->cols,
167 };
168
169 damagerect(screen, rect);
170 }
171
putglyph(VTermGlyphInfo * info,VTermPos pos,void * user)172 static int putglyph(VTermGlyphInfo *info, VTermPos pos, void *user)
173 {
174 VTermScreen *screen = user;
175 ScreenCell *cell = getcell(screen, pos.row, pos.col);
176
177 if(!cell)
178 return 0;
179
180 int i;
181 for(i = 0; i < VTERM_MAX_CHARS_PER_CELL && info->chars[i]; i++) {
182 cell->chars[i] = info->chars[i];
183 cell->pen = screen->pen;
184 }
185 if(i < VTERM_MAX_CHARS_PER_CELL)
186 cell->chars[i] = 0;
187
188 for(int col = 1; col < info->width; col++)
189 getcell(screen, pos.row, pos.col + col)->chars[0] = (uint32_t)-1;
190
191 VTermRect rect = {
192 .start_row = pos.row,
193 .end_row = pos.row+1,
194 .start_col = pos.col,
195 .end_col = pos.col+info->width,
196 };
197
198 cell->pen.protected_cell = info->protected_cell;
199 cell->pen.dwl = info->dwl;
200 cell->pen.dhl = info->dhl;
201
202 damagerect(screen, rect);
203
204 return 1;
205 }
206
moverect_internal(VTermRect dest,VTermRect src,void * user)207 static int moverect_internal(VTermRect dest, VTermRect src, void *user)
208 {
209 VTermScreen *screen = user;
210
211 if(screen->callbacks && screen->callbacks->sb_pushline &&
212 dest.start_row == 0 && dest.start_col == 0 && // starts top-left corner
213 dest.end_col == screen->cols && // full width
214 screen->buffer == screen->buffers[0]) { // not altscreen
215 VTermPos pos;
216 for(pos.row = 0; pos.row < src.start_row; pos.row++) {
217 for(pos.col = 0; pos.col < screen->cols; pos.col++)
218 vterm_screen_get_cell(screen, pos, screen->sb_buffer + pos.col);
219
220 (screen->callbacks->sb_pushline)(screen->cols, screen->sb_buffer, screen->cbdata);
221 }
222 }
223
224 int cols = src.end_col - src.start_col;
225 int downward = src.start_row - dest.start_row;
226
227 int init_row, test_row, inc_row;
228 if(downward < 0) {
229 init_row = dest.end_row - 1;
230 test_row = dest.start_row - 1;
231 inc_row = -1;
232 }
233 else {
234 init_row = dest.start_row;
235 test_row = dest.end_row;
236 inc_row = +1;
237 }
238
239 for(int row = init_row; row != test_row; row += inc_row)
240 memmove(getcell(screen, row, dest.start_col),
241 getcell(screen, row + downward, src.start_col),
242 cols * sizeof(ScreenCell));
243
244 return 1;
245 }
246
moverect_user(VTermRect dest,VTermRect src,void * user)247 static int moverect_user(VTermRect dest, VTermRect src, void *user)
248 {
249 VTermScreen *screen = user;
250
251 if(screen->callbacks && screen->callbacks->moverect) {
252 if(screen->damage_merge != VTERM_DAMAGE_SCROLL)
253 // Avoid an infinite loop
254 vterm_screen_flush_damage(screen);
255
256 if((*screen->callbacks->moverect)(dest, src, screen->cbdata))
257 return 1;
258 }
259
260 damagerect(screen, dest);
261
262 return 1;
263 }
264
erase_internal(VTermRect rect,int selective,void * user)265 static int erase_internal(VTermRect rect, int selective, void *user)
266 {
267 VTermScreen *screen = user;
268
269 for(int row = rect.start_row; row < screen->state->rows && row < rect.end_row; row++) {
270 const VTermLineInfo *info = vterm_state_get_lineinfo(screen->state, row);
271
272 for(int col = rect.start_col; col < rect.end_col; col++) {
273 ScreenCell *cell = getcell(screen, row, col);
274
275 if(selective && cell->pen.protected_cell)
276 continue;
277
278 cell->chars[0] = 0;
279 cell->pen = screen->pen;
280 cell->pen.dwl = info->doublewidth;
281 cell->pen.dhl = info->doubleheight;
282 }
283 }
284
285 return 1;
286 }
287
erase_user(VTermRect rect,int selective,void * user)288 static int erase_user(VTermRect rect, int selective, void *user)
289 {
290 VTermScreen *screen = user;
291
292 damagerect(screen, rect);
293
294 return 1;
295 }
296
erase(VTermRect rect,int selective,void * user)297 static int erase(VTermRect rect, int selective, void *user)
298 {
299 erase_internal(rect, selective, user);
300 return erase_user(rect, 0, user);
301 }
302
scrollrect(VTermRect rect,int downward,int rightward,void * user)303 static int scrollrect(VTermRect rect, int downward, int rightward, void *user)
304 {
305 VTermScreen *screen = user;
306
307 if(screen->damage_merge != VTERM_DAMAGE_SCROLL) {
308 vterm_scroll_rect(rect, downward, rightward,
309 moverect_internal, erase_internal, screen);
310
311 vterm_screen_flush_damage(screen);
312
313 vterm_scroll_rect(rect, downward, rightward,
314 moverect_user, erase_user, screen);
315
316 return 1;
317 }
318
319 if(screen->damaged.start_row != -1 &&
320 !rect_intersects(&rect, &screen->damaged)) {
321 vterm_screen_flush_damage(screen);
322 }
323
324 if(screen->pending_scrollrect.start_row == -1) {
325 screen->pending_scrollrect = rect;
326 screen->pending_scroll_downward = downward;
327 screen->pending_scroll_rightward = rightward;
328 }
329 else if(rect_equal(&screen->pending_scrollrect, &rect) &&
330 ((screen->pending_scroll_downward == 0 && downward == 0) ||
331 (screen->pending_scroll_rightward == 0 && rightward == 0))) {
332 screen->pending_scroll_downward += downward;
333 screen->pending_scroll_rightward += rightward;
334 }
335 else {
336 vterm_screen_flush_damage(screen);
337
338 screen->pending_scrollrect = rect;
339 screen->pending_scroll_downward = downward;
340 screen->pending_scroll_rightward = rightward;
341 }
342
343 vterm_scroll_rect(rect, downward, rightward,
344 moverect_internal, erase_internal, screen);
345
346 if(screen->damaged.start_row == -1)
347 return 1;
348
349 if(rect_contains(&rect, &screen->damaged)) {
350 /* Scroll region entirely contains the damage; just move it */
351 vterm_rect_move(&screen->damaged, -downward, -rightward);
352 rect_clip(&screen->damaged, &rect);
353 }
354 /* There are a number of possible cases here, but lets restrict this to only
355 * the common case where we might actually gain some performance by
356 * optimising it. Namely, a vertical scroll that neatly cuts the damage
357 * region in half.
358 */
359 else if(rect.start_col <= screen->damaged.start_col &&
360 rect.end_col >= screen->damaged.end_col &&
361 rightward == 0) {
362 if(screen->damaged.start_row >= rect.start_row &&
363 screen->damaged.start_row < rect.end_row) {
364 screen->damaged.start_row -= downward;
365 if(screen->damaged.start_row < rect.start_row)
366 screen->damaged.start_row = rect.start_row;
367 if(screen->damaged.start_row > rect.end_row)
368 screen->damaged.start_row = rect.end_row;
369 }
370 if(screen->damaged.end_row >= rect.start_row &&
371 screen->damaged.end_row < rect.end_row) {
372 screen->damaged.end_row -= downward;
373 if(screen->damaged.end_row < rect.start_row)
374 screen->damaged.end_row = rect.start_row;
375 if(screen->damaged.end_row > rect.end_row)
376 screen->damaged.end_row = rect.end_row;
377 }
378 }
379 else {
380 DEBUG_LOG("TODO: Just flush and redo damaged=" STRFrect " rect=" STRFrect "\n",
381 ARGSrect(screen->damaged), ARGSrect(rect));
382 }
383
384 return 1;
385 }
386
movecursor(VTermPos pos,VTermPos oldpos,int visible,void * user)387 static int movecursor(VTermPos pos, VTermPos oldpos, int visible, void *user)
388 {
389 VTermScreen *screen = user;
390
391 if(screen->callbacks && screen->callbacks->movecursor)
392 return (*screen->callbacks->movecursor)(pos, oldpos, visible, screen->cbdata);
393
394 return 0;
395 }
396
setpenattr(VTermAttr attr,VTermValue * val,void * user)397 static int setpenattr(VTermAttr attr, VTermValue *val, void *user)
398 {
399 VTermScreen *screen = user;
400
401 switch(attr) {
402 case VTERM_ATTR_BOLD:
403 screen->pen.bold = val->boolean;
404 return 1;
405 case VTERM_ATTR_UNDERLINE:
406 screen->pen.underline = val->number;
407 return 1;
408 case VTERM_ATTR_ITALIC:
409 screen->pen.italic = val->boolean;
410 return 1;
411 case VTERM_ATTR_BLINK:
412 screen->pen.blink = val->boolean;
413 return 1;
414 case VTERM_ATTR_REVERSE:
415 screen->pen.reverse = val->boolean;
416 return 1;
417 case VTERM_ATTR_STRIKE:
418 screen->pen.strike = val->boolean;
419 return 1;
420 case VTERM_ATTR_FONT:
421 screen->pen.font = val->number;
422 return 1;
423 case VTERM_ATTR_FOREGROUND:
424 screen->pen.fg = val->color;
425 return 1;
426 case VTERM_ATTR_BACKGROUND:
427 screen->pen.bg = val->color;
428 return 1;
429
430 case VTERM_N_ATTRS:
431 return 0;
432 }
433
434 return 0;
435 }
436
settermprop(VTermProp prop,VTermValue * val,void * user)437 static int settermprop(VTermProp prop, VTermValue *val, void *user)
438 {
439 VTermScreen *screen = user;
440
441 switch(prop) {
442 case VTERM_PROP_ALTSCREEN:
443 if(val->boolean && !screen->buffers[1])
444 return 0;
445
446 screen->buffer = val->boolean ? screen->buffers[1] : screen->buffers[0];
447 /* only send a damage event on disable; because during enable there's an
448 * erase that sends a damage anyway
449 */
450 if(!val->boolean)
451 damagescreen(screen);
452 break;
453 case VTERM_PROP_REVERSE:
454 screen->global_reverse = val->boolean;
455 damagescreen(screen);
456 break;
457 default:
458 ; /* ignore */
459 }
460
461 if(screen->callbacks && screen->callbacks->settermprop)
462 return (*screen->callbacks->settermprop)(prop, val, screen->cbdata);
463
464 return 1;
465 }
466
bell(void * user)467 static int bell(void *user)
468 {
469 VTermScreen *screen = user;
470
471 if(screen->callbacks && screen->callbacks->bell)
472 return (*screen->callbacks->bell)(screen->cbdata);
473
474 return 0;
475 }
476
resize(int new_rows,int new_cols,VTermPos * delta,void * user)477 static int resize(int new_rows, int new_cols, VTermPos *delta, void *user)
478 {
479 VTermScreen *screen = user;
480
481 int is_altscreen = (screen->buffers[1] && screen->buffer == screen->buffers[1]);
482
483 int old_rows = screen->rows;
484 int old_cols = screen->cols;
485
486 if(!is_altscreen && new_rows < old_rows) {
487 // Fewer rows - determine if we're going to scroll at all, and if so, push
488 // those lines to scrollback
489 VTermPos pos = { 0, 0 };
490 VTermPos cursor = screen->state->pos;
491 // Find the first blank row after the cursor.
492 for(pos.row = old_rows - 1; pos.row >= new_rows; pos.row--)
493 if(!vterm_screen_is_eol(screen, pos) || cursor.row == pos.row)
494 break;
495
496 int first_blank_row = pos.row + 1;
497 if(first_blank_row > new_rows) {
498 VTermRect rect = {
499 .start_row = 0,
500 .end_row = old_rows,
501 .start_col = 0,
502 .end_col = old_cols,
503 };
504 scrollrect(rect, first_blank_row - new_rows, 0, user);
505 vterm_screen_flush_damage(screen);
506
507 delta->row -= first_blank_row - new_rows;
508 }
509 }
510
511 screen->buffers[0] = realloc_buffer(screen, screen->buffers[0], new_rows, new_cols);
512 if(screen->buffers[1])
513 screen->buffers[1] = realloc_buffer(screen, screen->buffers[1], new_rows, new_cols);
514
515 screen->buffer = is_altscreen ? screen->buffers[1] : screen->buffers[0];
516
517 screen->rows = new_rows;
518 screen->cols = new_cols;
519
520 if(screen->sb_buffer)
521 vterm_allocator_free(screen->vt, screen->sb_buffer);
522
523 screen->sb_buffer = vterm_allocator_malloc(screen->vt, sizeof(VTermScreenCell) * new_cols);
524
525 if(new_cols > old_cols) {
526 VTermRect rect = {
527 .start_row = 0,
528 .end_row = old_rows,
529 .start_col = old_cols,
530 .end_col = new_cols,
531 };
532 damagerect(screen, rect);
533 }
534
535 if(new_rows > old_rows) {
536 if(!is_altscreen && screen->callbacks && screen->callbacks->sb_popline) {
537 int rows = new_rows - old_rows;
538 while(rows) {
539 if(!(screen->callbacks->sb_popline(screen->cols, screen->sb_buffer, screen->cbdata)))
540 break;
541
542 VTermRect rect = {
543 .start_row = 0,
544 .end_row = screen->rows,
545 .start_col = 0,
546 .end_col = screen->cols,
547 };
548 scrollrect(rect, -1, 0, user);
549
550 VTermPos pos = { 0, 0 };
551 for(pos.col = 0; pos.col < screen->cols; pos.col += screen->sb_buffer[pos.col].width)
552 vterm_screen_set_cell(screen, pos, screen->sb_buffer + pos.col);
553
554 rect.end_row = 1;
555 damagerect(screen, rect);
556
557 vterm_screen_flush_damage(screen);
558
559 rows--;
560 delta->row++;
561 }
562 }
563
564 VTermRect rect = {
565 .start_row = old_rows,
566 .end_row = new_rows,
567 .start_col = 0,
568 .end_col = new_cols,
569 };
570 damagerect(screen, rect);
571 }
572
573 if(screen->callbacks && screen->callbacks->resize)
574 return (*screen->callbacks->resize)(new_rows, new_cols, screen->cbdata);
575
576 return 1;
577 }
578
setlineinfo(int row,const VTermLineInfo * newinfo,const VTermLineInfo * oldinfo,void * user)579 static int setlineinfo(int row, const VTermLineInfo *newinfo, const VTermLineInfo *oldinfo, void *user)
580 {
581 VTermScreen *screen = user;
582
583 if(newinfo->doublewidth != oldinfo->doublewidth ||
584 newinfo->doubleheight != oldinfo->doubleheight) {
585 for(int col = 0; col < screen->cols; col++) {
586 ScreenCell *cell = getcell(screen, row, col);
587 cell->pen.dwl = newinfo->doublewidth;
588 cell->pen.dhl = newinfo->doubleheight;
589 }
590
591 VTermRect rect = {
592 .start_row = row,
593 .end_row = row + 1,
594 .start_col = 0,
595 .end_col = newinfo->doublewidth ? screen->cols / 2 : screen->cols,
596 };
597 damagerect(screen, rect);
598
599 if(newinfo->doublewidth) {
600 rect.start_col = screen->cols / 2;
601 rect.end_col = screen->cols;
602
603 erase_internal(rect, 0, user);
604 }
605 }
606
607 return 1;
608 }
609
610 static VTermStateCallbacks state_cbs = {
611 .putglyph = &putglyph,
612 .movecursor = &movecursor,
613 .scrollrect = &scrollrect,
614 .erase = &erase,
615 .setpenattr = &setpenattr,
616 .settermprop = &settermprop,
617 .bell = &bell,
618 .resize = &resize,
619 .setlineinfo = &setlineinfo,
620 };
621
screen_new(VTerm * vt)622 static VTermScreen *screen_new(VTerm *vt)
623 {
624 VTermState *state = vterm_obtain_state(vt);
625 if(!state)
626 return NULL;
627
628 VTermScreen *screen = vterm_allocator_malloc(vt, sizeof(VTermScreen));
629 int rows, cols;
630
631 vterm_get_size(vt, &rows, &cols);
632
633 screen->vt = vt;
634 screen->state = state;
635
636 screen->damage_merge = VTERM_DAMAGE_CELL;
637 screen->damaged.start_row = -1;
638 screen->pending_scrollrect.start_row = -1;
639
640 screen->rows = rows;
641 screen->cols = cols;
642
643 screen->callbacks = NULL;
644 screen->cbdata = NULL;
645
646 screen->buffers[0] = realloc_buffer(screen, NULL, rows, cols);
647
648 screen->buffer = screen->buffers[0];
649
650 screen->sb_buffer = vterm_allocator_malloc(screen->vt, sizeof(VTermScreenCell) * cols);
651
652 vterm_state_set_callbacks(screen->state, &state_cbs, screen);
653
654 return screen;
655 }
656
vterm_screen_free(VTermScreen * screen)657 INTERNAL void vterm_screen_free(VTermScreen *screen)
658 {
659 vterm_allocator_free(screen->vt, screen->buffers[0]);
660 if(screen->buffers[1])
661 vterm_allocator_free(screen->vt, screen->buffers[1]);
662
663 vterm_allocator_free(screen->vt, screen->sb_buffer);
664
665 vterm_allocator_free(screen->vt, screen);
666 }
667
vterm_screen_reset(VTermScreen * screen,int hard)668 void vterm_screen_reset(VTermScreen *screen, int hard)
669 {
670 screen->damaged.start_row = -1;
671 screen->pending_scrollrect.start_row = -1;
672 vterm_state_reset(screen->state, hard);
673 vterm_screen_flush_damage(screen);
674 }
675
_get_chars(const VTermScreen * screen,const int utf8,void * buffer,size_t len,const VTermRect rect)676 static size_t _get_chars(const VTermScreen *screen, const int utf8, void *buffer, size_t len, const VTermRect rect)
677 {
678 size_t outpos = 0;
679 int padding = 0;
680
681 #define PUT(c) \
682 if(utf8) { \
683 size_t thislen = utf8_seqlen(c); \
684 if(buffer && outpos + thislen <= len) \
685 outpos += fill_utf8((c), (char *)buffer + outpos); \
686 else \
687 outpos += thislen; \
688 } \
689 else { \
690 if(buffer && outpos + 1 <= len) \
691 ((uint32_t*)buffer)[outpos++] = (c); \
692 else \
693 outpos++; \
694 }
695
696 for(int row = rect.start_row; row < rect.end_row; row++) {
697 for(int col = rect.start_col; col < rect.end_col; col++) {
698 ScreenCell *cell = getcell(screen, row, col);
699
700 if(cell->chars[0] == 0)
701 // Erased cell, might need a space
702 padding++;
703 else if(cell->chars[0] == (uint32_t)-1)
704 // Gap behind a double-width char, do nothing
705 ;
706 else {
707 while(padding) {
708 PUT(UNICODE_SPACE);
709 padding--;
710 }
711 for(int i = 0; i < VTERM_MAX_CHARS_PER_CELL && cell->chars[i]; i++) {
712 PUT(cell->chars[i]);
713 }
714 }
715 }
716
717 if(row < rect.end_row - 1) {
718 PUT(UNICODE_LINEFEED);
719 padding = 0;
720 }
721 }
722
723 return outpos;
724 }
725
vterm_screen_get_chars(const VTermScreen * screen,uint32_t * chars,size_t len,const VTermRect rect)726 size_t vterm_screen_get_chars(const VTermScreen *screen, uint32_t *chars, size_t len, const VTermRect rect)
727 {
728 return _get_chars(screen, 0, chars, len, rect);
729 }
730
vterm_screen_get_text(const VTermScreen * screen,char * str,size_t len,const VTermRect rect)731 size_t vterm_screen_get_text(const VTermScreen *screen, char *str, size_t len, const VTermRect rect)
732 {
733 return _get_chars(screen, 1, str, len, rect);
734 }
735
736 /* Copy internal to external representation of a screen cell */
vterm_screen_get_cell(const VTermScreen * screen,VTermPos pos,VTermScreenCell * cell)737 int vterm_screen_get_cell(const VTermScreen *screen, VTermPos pos, VTermScreenCell *cell)
738 {
739 ScreenCell *intcell = getcell(screen, pos.row, pos.col);
740 if(!intcell)
741 return 0;
742
743 for(int i = 0; ; i++) {
744 cell->chars[i] = intcell->chars[i];
745 if(!intcell->chars[i])
746 break;
747 }
748
749 cell->attrs.bold = intcell->pen.bold;
750 cell->attrs.underline = intcell->pen.underline;
751 cell->attrs.italic = intcell->pen.italic;
752 cell->attrs.blink = intcell->pen.blink;
753 cell->attrs.reverse = intcell->pen.reverse ^ screen->global_reverse;
754 cell->attrs.strike = intcell->pen.strike;
755 cell->attrs.font = intcell->pen.font;
756
757 cell->attrs.dwl = intcell->pen.dwl;
758 cell->attrs.dhl = intcell->pen.dhl;
759
760 cell->fg = intcell->pen.fg;
761 cell->bg = intcell->pen.bg;
762
763 if(pos.col < (screen->cols - 1) &&
764 getcell(screen, pos.row, pos.col + 1)->chars[0] == (uint32_t)-1)
765 cell->width = 2;
766 else
767 cell->width = 1;
768
769 return 1;
770 }
771
772 /* Copy external to internal representation of a screen cell */
773 /* static because it's only used internally for sb_popline during resize */
vterm_screen_set_cell(VTermScreen * screen,VTermPos pos,const VTermScreenCell * cell)774 static int vterm_screen_set_cell(VTermScreen *screen, VTermPos pos, const VTermScreenCell *cell)
775 {
776 ScreenCell *intcell = getcell(screen, pos.row, pos.col);
777 if(!intcell)
778 return 0;
779
780 for(int i = 0; ; i++) {
781 intcell->chars[i] = cell->chars[i];
782 if(!cell->chars[i])
783 break;
784 }
785
786 intcell->pen.bold = cell->attrs.bold;
787 intcell->pen.underline = cell->attrs.underline;
788 intcell->pen.italic = cell->attrs.italic;
789 intcell->pen.blink = cell->attrs.blink;
790 intcell->pen.reverse = cell->attrs.reverse ^ screen->global_reverse;
791 intcell->pen.strike = cell->attrs.strike;
792 intcell->pen.font = cell->attrs.font;
793
794 intcell->pen.fg = cell->fg;
795 intcell->pen.bg = cell->bg;
796
797 if(cell->width == 2)
798 getcell(screen, pos.row, pos.col + 1)->chars[0] = (uint32_t)-1;
799
800 return 1;
801 }
802
vterm_screen_is_eol(const VTermScreen * screen,VTermPos pos)803 int vterm_screen_is_eol(const VTermScreen *screen, VTermPos pos)
804 {
805 /* This cell is EOL if this and every cell to the right is black */
806 for(; pos.col < screen->cols; pos.col++) {
807 ScreenCell *cell = getcell(screen, pos.row, pos.col);
808 if(cell->chars[0] != 0)
809 return 0;
810 }
811
812 return 1;
813 }
814
vterm_obtain_screen(VTerm * vt)815 VTermScreen *vterm_obtain_screen(VTerm *vt)
816 {
817 if(vt->screen)
818 return vt->screen;
819
820 VTermScreen *screen = screen_new(vt);
821 vt->screen = screen;
822
823 return screen;
824 }
825
vterm_screen_enable_altscreen(VTermScreen * screen,int altscreen)826 void vterm_screen_enable_altscreen(VTermScreen *screen, int altscreen)
827 {
828
829 if(!screen->buffers[1] && altscreen) {
830 int rows, cols;
831 vterm_get_size(screen->vt, &rows, &cols);
832
833 screen->buffers[1] = realloc_buffer(screen, NULL, rows, cols);
834 }
835 }
836
vterm_screen_set_callbacks(VTermScreen * screen,const VTermScreenCallbacks * callbacks,void * user)837 void vterm_screen_set_callbacks(VTermScreen *screen, const VTermScreenCallbacks *callbacks, void *user)
838 {
839 screen->callbacks = callbacks;
840 screen->cbdata = user;
841 }
842
vterm_screen_get_cbdata(VTermScreen * screen)843 void *vterm_screen_get_cbdata(VTermScreen *screen)
844 {
845 return screen->cbdata;
846 }
847
vterm_screen_set_unrecognised_fallbacks(VTermScreen * screen,const VTermParserCallbacks * fallbacks,void * user)848 void vterm_screen_set_unrecognised_fallbacks(VTermScreen *screen, const VTermParserCallbacks *fallbacks, void *user)
849 {
850 vterm_state_set_unrecognised_fallbacks(screen->state, fallbacks, user);
851 }
852
vterm_screen_get_unrecognised_fbdata(VTermScreen * screen)853 void *vterm_screen_get_unrecognised_fbdata(VTermScreen *screen)
854 {
855 return vterm_state_get_unrecognised_fbdata(screen->state);
856 }
857
vterm_screen_flush_damage(VTermScreen * screen)858 void vterm_screen_flush_damage(VTermScreen *screen)
859 {
860 if(screen->pending_scrollrect.start_row != -1) {
861 vterm_scroll_rect(screen->pending_scrollrect, screen->pending_scroll_downward, screen->pending_scroll_rightward,
862 moverect_user, erase_user, screen);
863
864 screen->pending_scrollrect.start_row = -1;
865 }
866
867 if(screen->damaged.start_row != -1) {
868 if(screen->callbacks && screen->callbacks->damage)
869 (*screen->callbacks->damage)(screen->damaged, screen->cbdata);
870
871 screen->damaged.start_row = -1;
872 }
873 }
874
vterm_screen_set_damage_merge(VTermScreen * screen,VTermDamageSize size)875 void vterm_screen_set_damage_merge(VTermScreen *screen, VTermDamageSize size)
876 {
877 vterm_screen_flush_damage(screen);
878 screen->damage_merge = size;
879 }
880
attrs_differ(VTermAttrMask attrs,ScreenCell * a,ScreenCell * b)881 static int attrs_differ(VTermAttrMask attrs, ScreenCell *a, ScreenCell *b)
882 {
883 if((attrs & VTERM_ATTR_BOLD_MASK) && (a->pen.bold != b->pen.bold))
884 return 1;
885 if((attrs & VTERM_ATTR_UNDERLINE_MASK) && (a->pen.underline != b->pen.underline))
886 return 1;
887 if((attrs & VTERM_ATTR_ITALIC_MASK) && (a->pen.italic != b->pen.italic))
888 return 1;
889 if((attrs & VTERM_ATTR_BLINK_MASK) && (a->pen.blink != b->pen.blink))
890 return 1;
891 if((attrs & VTERM_ATTR_REVERSE_MASK) && (a->pen.reverse != b->pen.reverse))
892 return 1;
893 if((attrs & VTERM_ATTR_STRIKE_MASK) && (a->pen.strike != b->pen.strike))
894 return 1;
895 if((attrs & VTERM_ATTR_FONT_MASK) && (a->pen.font != b->pen.font))
896 return 1;
897 if((attrs & VTERM_ATTR_FOREGROUND_MASK) && !vterm_color_is_equal(&a->pen.fg, &b->pen.fg))
898 return 1;
899 if((attrs & VTERM_ATTR_BACKGROUND_MASK) && !vterm_color_is_equal(&a->pen.bg, &b->pen.bg))
900 return 1;
901
902 return 0;
903 }
904
vterm_screen_get_attrs_extent(const VTermScreen * screen,VTermRect * extent,VTermPos pos,VTermAttrMask attrs)905 int vterm_screen_get_attrs_extent(const VTermScreen *screen, VTermRect *extent, VTermPos pos, VTermAttrMask attrs)
906 {
907 ScreenCell *target = getcell(screen, pos.row, pos.col);
908
909 // TODO: bounds check
910 extent->start_row = pos.row;
911 extent->end_row = pos.row + 1;
912
913 if(extent->start_col < 0)
914 extent->start_col = 0;
915 if(extent->end_col < 0)
916 extent->end_col = screen->cols;
917
918 int col;
919
920 for(col = pos.col - 1; col >= extent->start_col; col--)
921 if(attrs_differ(attrs, target, getcell(screen, pos.row, col)))
922 break;
923 extent->start_col = col + 1;
924
925 for(col = pos.col + 1; col < extent->end_col; col++)
926 if(attrs_differ(attrs, target, getcell(screen, pos.row, col)))
927 break;
928 extent->end_col = col - 1;
929
930 return 1;
931 }
932
vterm_screen_convert_color_to_rgb(const VTermScreen * screen,VTermColor * col)933 void vterm_screen_convert_color_to_rgb(const VTermScreen *screen, VTermColor *col)
934 {
935 vterm_state_convert_color_to_rgb(screen->state, col);
936 }
937