1 /* See LICENSE for license details. */ 
  14 #include <sys/ioctl.h> 
  15 #include <sys/select.h> 
  18 #include <sys/types.h> 
  24 #include <X11/Xatom.h> 
  26 #include <X11/Xutil.h> 
  27 #include <X11/cursorfont.h> 
  28 #include <X11/keysym.h> 
  29 #include <X11/Xft/Xft.h> 
  30 #include <X11/XKBlib.h> 
  31 #include <fontconfig/fontconfig.h> 
  43 #elif defined(__OpenBSD__) || defined(__NetBSD__) || defined(__APPLE__) 
  45 #elif defined(__FreeBSD__) || defined(__DragonFly__) 
  51 #define XEMBED_FOCUS_IN  4 
  52 #define XEMBED_FOCUS_OUT 5 
  55 #define UTF_INVALID   0xFFFD 
  57 #define ESC_BUF_SIZ   (128*UTF_SIZ) 
  58 #define ESC_ARG_SIZ   16 
  59 #define STR_BUF_SIZ   ESC_BUF_SIZ 
  60 #define STR_ARG_SIZ   ESC_ARG_SIZ 
  61 #define XK_ANY_MOD    UINT_MAX 
  63 #define XK_SWITCH_MOD (1<<13) 
  66 #define MIN(a, b)               ((a) < (b) ? (a) : (b)) 
  67 #define MAX(a, b)               ((a) < (b) ? (b) : (a)) 
  68 #define LEN(a)                  (sizeof(a) / sizeof(a)[0]) 
  69 #define NUMMAXLEN(x)            ((int)(sizeof(x) * 2.56 + 0.5) + 1) 
  70 #define DEFAULT(a, b)           (a) = (a) ? (a) : (b) 
  71 #define BETWEEN(x, a, b)        ((a) <= (x) && (x) <= (b)) 
  72 #define DIVCEIL(n, d)           (((n) + ((d) - 1)) / (d)) 
  73 #define ISCONTROLC0(c)          (BETWEEN(c, 0, 0x1f) || (c) == '\177') 
  74 #define ISCONTROLC1(c)          (BETWEEN(c, 0x80, 0x9f)) 
  75 #define ISCONTROL(c)            (ISCONTROLC0(c) || ISCONTROLC1(c)) 
  76 #define ISDELIM(u)              (utf8strchr(worddelimiters, u) != NULL) 
  77 #define LIMIT(x, a, b)          (x) = (x) < (a) ? (a) : (x) > (b) ? (b) : (x) 
  78 #define ATTRCMP(a, b)           ((a).mode != (b).mode || (a).fg != (b).fg || \ 
  80 #define IS_SET(flag)            ((term.mode & (flag)) != 0) 
  81 #define TIMEDIFF(t1, t2)        ((t1.tv_sec-t2.tv_sec)*1000 + \ 
  82                                 (t1.tv_nsec-t2.tv_nsec)/1E6) 
  83 #define MODBIT(x, set, bit)     ((set) ? ((x) |= (bit)) : ((x) &= ~(bit))) 
  85 #define TRUECOLOR(r,g,b)        (1 << 24 | (r) << 16 | (g) << 8 | (b)) 
  86 #define IS_TRUECOL(x)           (1 << 24 & (x)) 
  87 #define TRUERED(x)              (((x) & 0xff0000) >> 8) 
  88 #define TRUEGREEN(x)            (((x) & 0xff00)) 
  89 #define TRUEBLUE(x)             (((x) & 0xff) << 8) 
  92 #define ISO14755CMD             "dmenu -w %lu -p codepoint: </dev/null" 
  94 enum glyph_attribute 
{ 
  99         ATTR_UNDERLINE  
= 1 << 3, 
 101         ATTR_REVERSE    
= 1 << 5, 
 102         ATTR_INVISIBLE  
= 1 << 6, 
 103         ATTR_STRUCK     
= 1 << 7, 
 106         ATTR_WDUMMY     
= 1 << 10, 
 107         ATTR_BOLD_FAINT 
= ATTR_BOLD 
| ATTR_FAINT
, 
 110 enum cursor_movement 
{ 
 123         MODE_INSERT      
= 1 << 1, 
 124         MODE_APPKEYPAD   
= 1 << 2, 
 125         MODE_ALTSCREEN   
= 1 << 3, 
 127         MODE_MOUSEBTN    
= 1 << 5, 
 128         MODE_MOUSEMOTION 
= 1 << 6, 
 129         MODE_REVERSE     
= 1 << 7, 
 130         MODE_KBDLOCK     
= 1 << 8, 
 133         MODE_APPCURSOR   
= 1 << 11, 
 134         MODE_MOUSESGR    
= 1 << 12, 
 136         MODE_BLINK       
= 1 << 14, 
 137         MODE_FBLINK      
= 1 << 15, 
 138         MODE_FOCUS       
= 1 << 16, 
 139         MODE_MOUSEX10    
= 1 << 17, 
 140         MODE_MOUSEMANY   
= 1 << 18, 
 141         MODE_BRCKTPASTE  
= 1 << 19, 
 142         MODE_PRINT       
= 1 << 20, 
 144         MODE_SIXEL       
= 1 << 22, 
 145         MODE_MOUSE       
= MODE_MOUSEBTN
|MODE_MOUSEMOTION
|MODE_MOUSEX10\
 
 162         ESC_STR        
= 4,  /* OSC, PM, APC */ 
 164         ESC_STR_END    
= 16, /* a final string was encountered */ 
 165         ESC_TEST       
= 32, /* Enter in test mode */ 
 175 enum selection_mode 
{ 
 181 enum selection_type 
{ 
 186 enum selection_snap 
{ 
 191 typedef unsigned char uchar
; 
 192 typedef unsigned int uint
; 
 193 typedef unsigned long ulong
; 
 194 typedef unsigned short ushort
; 
 196 typedef uint_least32_t Rune
; 
 198 typedef XftDraw 
*Draw
; 
 199 typedef XftColor Color
; 
 202         Rune u
;           /* character code */ 
 203         ushort mode
;      /* attribute flags */ 
 204         uint32_t fg
;      /* foreground  */ 
 205         uint32_t bg
;      /* background  */ 
 211         Glyph attr
; /* current char attributes */ 
 217 /* CSI Escape sequence structs */ 
 218 /* ESC '[' [[ [<priv>] <arg> [;]] <mode> [<mode>]] */ 
 220         char buf
[ESC_BUF_SIZ
]; /* raw string */ 
 221         int len
;               /* raw string length */ 
 223         int arg
[ESC_ARG_SIZ
]; 
 224         int narg
;              /* nb of args */ 
 228 /* STR Escape sequence structs */ 
 229 /* ESC type [[ [<priv>] <arg> [;]] <mode>] ESC '\' */ 
 231         char type
;             /* ESC type ... */ 
 232         char buf
[STR_BUF_SIZ
]; /* raw string */ 
 233         int len
;               /* raw string length */ 
 234         char *args
[STR_ARG_SIZ
]; 
 235         int narg
;              /* nb of args */ 
 238 /* Internal representation of the screen */ 
 240         int row
;      /* nb row */ 
 241         int col
;      /* nb col */ 
 242         Line 
*line
;   /* screen */ 
 243         Line 
*alt
;    /* alternate screen */ 
 244         int *dirty
;  /* dirtyness of lines */ 
 245         XftGlyphFontSpec 
*specbuf
; /* font spec buffer used for rendering */ 
 246         TCursor c
;    /* cursor */ 
 247         int top
;      /* top    scroll limit */ 
 248         int bot
;      /* bottom scroll limit */ 
 249         int mode
;     /* terminal mode flags */ 
 250         int esc
;      /* escape state flags */ 
 251         char trantbl
[4]; /* charset table translation */ 
 252         int charset
;  /* current charset */ 
 253         int icharset
; /* selected charset for sequence */ 
 254         int numlock
; /* lock numbers in keyboard */ 
 258 /* Purely graphic info */ 
 264         Atom xembed
, wmdeletewin
, netwmname
, netwmpid
; 
 269         XSetWindowAttributes attrs
; 
 271         int isfixed
; /* is fixed geometry? */ 
 272         int l
, t
; /* left and top offset */ 
 273         int gm
; /* geometry mask */ 
 274         int tw
, th
; /* tty width and height */ 
 275         int w
, h
; /* window width and height */ 
 276         int ch
; /* char height */ 
 277         int cw
; /* char width  */ 
 278         char state
; /* focus, redraw, visible */ 
 279         int cursor
; /* cursor style */ 
 292         /* three valued logic variables: 0 indifferent, 1 on, -1 off */ 
 293         signed char appkey
;    /* application keypad */ 
 294         signed char appcursor
; /* application cursor */ 
 295         signed char crlf
;      /* crlf mode          */ 
 303          * Selection variables: 
 304          * nb – normalized coordinates of the beginning of the selection 
 305          * ne – normalized coordinates of the end of the selection 
 306          * ob – original coordinates of the beginning of the selection 
 307          * oe – original coordinates of the end of the selection 
 313         char *primary
, *clipboard
; 
 316         struct timespec tclick1
; 
 317         struct timespec tclick2
; 
 330         void (*func
)(const Arg 
*); 
 334 /* function definitions used in config.h */ 
 335 static void clipcopy(const Arg 
*); 
 336 static void clippaste(const Arg 
*); 
 337 static void numlock(const Arg 
*); 
 338 static void selpaste(const Arg 
*); 
 339 static void xzoom(const Arg 
*); 
 340 static void xzoomabs(const Arg 
*); 
 341 static void xzoomreset(const Arg 
*); 
 342 static void printsel(const Arg 
*); 
 343 static void printscreen(const Arg 
*) ; 
 344 static void iso14755(const Arg 
*); 
 345 static void toggleprinter(const Arg 
*); 
 346 static void sendbreak(const Arg 
*); 
 348 /* Config.h for applying patches and the configuration. */ 
 366 /* Drawing Context */ 
 368         Color col
[MAX(LEN(colorname
), 256)]; 
 369         Font font
, bfont
, ifont
, ibfont
; 
 373 static void die(const char *, ...); 
 374 static void draw(void); 
 375 static void redraw(void); 
 376 static void drawregion(int, int, int, int); 
 377 static void execsh(void); 
 378 static void stty(void); 
 379 static void sigchld(int); 
 380 static void run(void); 
 382 static void csidump(void); 
 383 static void csihandle(void); 
 384 static void csiparse(void); 
 385 static void csireset(void); 
 386 static int eschandle(uchar
); 
 387 static void strdump(void); 
 388 static void strhandle(void); 
 389 static void strparse(void); 
 390 static void strreset(void); 
 392 static int tattrset(int); 
 393 static void tprinter(char *, size_t); 
 394 static void tdumpsel(void); 
 395 static void tdumpline(int); 
 396 static void tdump(void); 
 397 static void tclearregion(int, int, int, int); 
 398 static void tcursor(int); 
 399 static void tdeletechar(int); 
 400 static void tdeleteline(int); 
 401 static void tinsertblank(int); 
 402 static void tinsertblankline(int); 
 403 static int tlinelen(int); 
 404 static void tmoveto(int, int); 
 405 static void tmoveato(int, int); 
 406 static void tnew(int, int); 
 407 static void tnewline(int); 
 408 static void tputtab(int); 
 409 static void tputc(Rune
); 
 410 static void treset(void); 
 411 static void tresize(int, int); 
 412 static void tscrollup(int, int); 
 413 static void tscrolldown(int, int); 
 414 static void tsetattr(int *, int); 
 415 static void tsetchar(Rune
, Glyph 
*, int, int); 
 416 static void tsetscroll(int, int); 
 417 static void tswapscreen(void); 
 418 static void tsetdirt(int, int); 
 419 static void tsetdirtattr(int); 
 420 static void tsetmode(int, int, int *, int); 
 421 static void tfulldirt(void); 
 422 static void techo(Rune
); 
 423 static void tcontrolcode(uchar 
); 
 424 static void tdectest(char ); 
 425 static void tdefutf8(char); 
 426 static int32_t tdefcolor(int *, int *, int); 
 427 static void tdeftran(char); 
 428 static inline int match(uint
, uint
); 
 429 static void ttynew(void); 
 430 static size_t ttyread(void); 
 431 static void ttyresize(void); 
 432 static void ttysend(char *, size_t); 
 433 static void ttywrite(const char *, size_t); 
 434 static void tstrsequence(uchar
); 
 436 static inline ushort 
sixd_to_16bit(int); 
 437 static int xmakeglyphfontspecs(XftGlyphFontSpec 
*, const Glyph 
*, int, int, int); 
 438 static void xdrawglyphfontspecs(const XftGlyphFontSpec 
*, Glyph
, int, int, int); 
 439 static void xdrawglyph(Glyph
, int, int); 
 440 static void xhints(void); 
 441 static void xclear(int, int, int, int); 
 442 static void xdrawcursor(void); 
 443 static void xinit(void); 
 444 static void xloadcols(void); 
 445 static int xsetcolorname(int, const char *); 
 446 static int xgeommasktogravity(int); 
 447 static int xloadfont(Font 
*, FcPattern 
*); 
 448 static void xloadfonts(char *, double); 
 449 static void xsettitle(char *); 
 450 static void xresettitle(void); 
 451 static void xsetpointermotion(int); 
 452 static void xseturgency(int); 
 453 static void xsetsel(char *, Time
); 
 454 static void xunloadfont(Font 
*); 
 455 static void xunloadfonts(void); 
 456 static void xresize(int, int); 
 458 static void expose(XEvent 
*); 
 459 static void visibility(XEvent 
*); 
 460 static void unmap(XEvent 
*); 
 461 static char *kmap(KeySym
, uint
); 
 462 static void kpress(XEvent 
*); 
 463 static void cmessage(XEvent 
*); 
 464 static void cresize(int, int); 
 465 static void resize(XEvent 
*); 
 466 static void focus(XEvent 
*); 
 467 static void brelease(XEvent 
*); 
 468 static void bpress(XEvent 
*); 
 469 static void bmotion(XEvent 
*); 
 470 static void propnotify(XEvent 
*); 
 471 static void selnotify(XEvent 
*); 
 472 static void selclear(XEvent 
*); 
 473 static void selrequest(XEvent 
*); 
 475 static void selinit(void); 
 476 static void selnormalize(void); 
 477 static inline int selected(int, int); 
 478 static char *getsel(void); 
 479 static void selcopy(Time
); 
 480 static void selscroll(int, int); 
 481 static void selsnap(int *, int *, int); 
 482 static int x2col(int); 
 483 static int y2row(int); 
 484 static void getbuttoninfo(XEvent 
*); 
 485 static void mousereport(XEvent 
*); 
 487 static size_t utf8decode(char *, Rune 
*, size_t); 
 488 static Rune 
utf8decodebyte(char, size_t *); 
 489 static size_t utf8encode(Rune
, char *); 
 490 static char utf8encodebyte(Rune
, size_t); 
 491 static char *utf8strchr(char *s
, Rune u
); 
 492 static size_t utf8validate(Rune 
*, size_t); 
 494 static ssize_t 
xwrite(int, const char *, size_t); 
 495 static void *xmalloc(size_t); 
 496 static void *xrealloc(void *, size_t); 
 497 static char *xstrdup(char *); 
 499 static void usage(void); 
 501 static void (*handler
[LASTEvent
])(XEvent 
*) = { 
 503         [ClientMessage
] = cmessage
, 
 504         [ConfigureNotify
] = resize
, 
 505         [VisibilityNotify
] = visibility
, 
 506         [UnmapNotify
] = unmap
, 
 510         [MotionNotify
] = bmotion
, 
 511         [ButtonPress
] = bpress
, 
 512         [ButtonRelease
] = brelease
, 
 514  * Uncomment if you want the selection to disappear when you select something 
 515  * different in another window. 
 517 /*      [SelectionClear] = selclear, */ 
 518         [SelectionNotify
] = selnotify
, 
 520  * PropertyNotify is only turned on when there is some INCR transfer happening 
 521  * for the selection retrieval. 
 523         [PropertyNotify
] = propnotify
, 
 524         [SelectionRequest
] = selrequest
, 
 531 static CSIEscape csiescseq
; 
 532 static STREscape strescseq
; 
 535 static Selection sel
; 
 537 static char **opt_cmd  
= NULL
; 
 538 static char *opt_class 
= NULL
; 
 539 static char *opt_embed 
= NULL
; 
 540 static char *opt_font  
= NULL
; 
 541 static char *opt_io    
= NULL
; 
 542 static char *opt_line  
= NULL
; 
 543 static char *opt_name  
= NULL
; 
 544 static char *opt_title 
= NULL
; 
 545 static int oldbutton   
= 3; /* button event on startup: 3 = release */ 
 547 static char *usedfont 
= NULL
; 
 548 static double usedfontsize 
= 0; 
 549 static double defaultfontsize 
= 0; 
 551 static uchar utfbyte
[UTF_SIZ 
+ 1] = {0x80,    0, 0xC0, 0xE0, 0xF0}; 
 552 static uchar utfmask
[UTF_SIZ 
+ 1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8}; 
 553 static Rune utfmin
[UTF_SIZ 
+ 1] = {       0,    0,  0x80,  0x800,  0x10000}; 
 554 static Rune utfmax
[UTF_SIZ 
+ 1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF}; 
 556 /* Font Ring Cache */ 
 570 /* Fontcache is an array now. A new font will be appended to the array. */ 
 571 static Fontcache frc
[16]; 
 572 static int frclen 
= 0; 
 575 xwrite(int fd
, const char *s
, size_t len
) 
 581                 r 
= write(fd
, s
, len
); 
 594         void *p 
= malloc(len
); 
 597                 die("Out of memory\n"); 
 603 xrealloc(void *p
, size_t len
) 
 605         if ((p 
= realloc(p
, len
)) == NULL
) 
 606                 die("Out of memory\n"); 
 614         if ((s 
= strdup(s
)) == NULL
) 
 615                 die("Out of memory\n"); 
 621 utf8decode(char *c
, Rune 
*u
, size_t clen
) 
 623         size_t i
, j
, len
, type
; 
 629         udecoded 
= utf8decodebyte(c
[0], &len
); 
 630         if (!BETWEEN(len
, 1, UTF_SIZ
)) 
 632         for (i 
= 1, j 
= 1; i 
< clen 
&& j 
< len
; ++i
, ++j
) { 
 633                 udecoded 
= (udecoded 
<< 6) | utf8decodebyte(c
[i
], &type
); 
 640         utf8validate(u
, len
); 
 646 utf8decodebyte(char c
, size_t *i
) 
 648         for (*i 
= 0; *i 
< LEN(utfmask
); ++(*i
)) 
 649                 if (((uchar
)c 
& utfmask
[*i
]) == utfbyte
[*i
]) 
 650                         return (uchar
)c 
& ~utfmask
[*i
]; 
 656 utf8encode(Rune u
, char *c
) 
 660         len 
= utf8validate(&u
, 0); 
 664         for (i 
= len 
- 1; i 
!= 0; --i
) { 
 665                 c
[i
] = utf8encodebyte(u
, 0); 
 668         c
[0] = utf8encodebyte(u
, len
); 
 674 utf8encodebyte(Rune u
, size_t i
) 
 676         return utfbyte
[i
] | (u 
& ~utfmask
[i
]); 
 680 utf8strchr(char *s
, Rune u
) 
 686         for (i 
= 0, j 
= 0; i 
< len
; i 
+= j
) { 
 687                 if (!(j 
= utf8decode(&s
[i
], &r
, len 
- i
))) 
 697 utf8validate(Rune 
*u
, size_t i
) 
 699         if (!BETWEEN(*u
, utfmin
[i
], utfmax
[i
]) || BETWEEN(*u
, 0xD800, 0xDFFF)) 
 701         for (i 
= 1; *u 
> utfmax
[i
]; ++i
) 
 710         clock_gettime(CLOCK_MONOTONIC
, &sel
.tclick1
); 
 711         clock_gettime(CLOCK_MONOTONIC
, &sel
.tclick2
); 
 716         sel
.clipboard 
= NULL
; 
 717         sel
.xtarget 
= XInternAtom(xw
.dpy
, "UTF8_STRING", 0); 
 718         if (sel
.xtarget 
== None
) 
 719                 sel
.xtarget 
= XA_STRING
; 
 728         return LIMIT(x
, 0, term
.col
-1); 
 737         return LIMIT(y
, 0, term
.row
-1); 
 745         if (term
.line
[y
][i 
- 1].mode 
& ATTR_WRAP
) 
 748         while (i 
> 0 && term
.line
[y
][i 
- 1].u 
== ' ') 
 759         if (sel
.type 
== SEL_REGULAR 
&& sel
.ob
.y 
!= sel
.oe
.y
) { 
 760                 sel
.nb
.x 
= sel
.ob
.y 
< sel
.oe
.y 
? sel
.ob
.x 
: sel
.oe
.x
; 
 761                 sel
.ne
.x 
= sel
.ob
.y 
< sel
.oe
.y 
? sel
.oe
.x 
: sel
.ob
.x
; 
 763                 sel
.nb
.x 
= MIN(sel
.ob
.x
, sel
.oe
.x
); 
 764                 sel
.ne
.x 
= MAX(sel
.ob
.x
, sel
.oe
.x
); 
 766         sel
.nb
.y 
= MIN(sel
.ob
.y
, sel
.oe
.y
); 
 767         sel
.ne
.y 
= MAX(sel
.ob
.y
, sel
.oe
.y
); 
 769         selsnap(&sel
.nb
.x
, &sel
.nb
.y
, -1); 
 770         selsnap(&sel
.ne
.x
, &sel
.ne
.y
, +1); 
 772         /* expand selection over line breaks */ 
 773         if (sel
.type 
== SEL_RECTANGULAR
) 
 775         i 
= tlinelen(sel
.nb
.y
); 
 778         if (tlinelen(sel
.ne
.y
) <= sel
.ne
.x
) 
 779                 sel
.ne
.x 
= term
.col 
- 1; 
 783 selected(int x
, int y
) 
 785         if (sel
.mode 
== SEL_EMPTY
) 
 788         if (sel
.type 
== SEL_RECTANGULAR
) 
 789                 return BETWEEN(y
, sel
.nb
.y
, sel
.ne
.y
) 
 790                     && BETWEEN(x
, sel
.nb
.x
, sel
.ne
.x
); 
 792         return BETWEEN(y
, sel
.nb
.y
, sel
.ne
.y
) 
 793             && (y 
!= sel
.nb
.y 
|| x 
>= sel
.nb
.x
) 
 794             && (y 
!= sel
.ne
.y 
|| x 
<= sel
.ne
.x
); 
 798 selsnap(int *x
, int *y
, int direction
) 
 800         int newx
, newy
, xt
, yt
; 
 801         int delim
, prevdelim
; 
 807                  * Snap around if the word wraps around at the end or 
 808                  * beginning of a line. 
 810                 prevgp 
= &term
.line
[*y
][*x
]; 
 811                 prevdelim 
= ISDELIM(prevgp
->u
); 
 813                         newx 
= *x 
+ direction
; 
 815                         if (!BETWEEN(newx
, 0, term
.col 
- 1)) { 
 817                                 newx 
= (newx 
+ term
.col
) % term
.col
; 
 818                                 if (!BETWEEN(newy
, 0, term
.row 
- 1)) 
 824                                         yt 
= newy
, xt 
= newx
; 
 825                                 if (!(term
.line
[yt
][xt
].mode 
& ATTR_WRAP
)) 
 829                         if (newx 
>= tlinelen(newy
)) 
 832                         gp 
= &term
.line
[newy
][newx
]; 
 833                         delim 
= ISDELIM(gp
->u
); 
 834                         if (!(gp
->mode 
& ATTR_WDUMMY
) && (delim 
!= prevdelim
 
 835                                         || (delim 
&& gp
->u 
!= prevgp
->u
))) 
 846                  * Snap around if the the previous line or the current one 
 847                  * has set ATTR_WRAP at its end. Then the whole next or 
 848                  * previous line will be selected. 
 850                 *x 
= (direction 
< 0) ? 0 : term
.col 
- 1; 
 852                         for (; *y 
> 0; *y 
+= direction
) { 
 853                                 if (!(term
.line
[*y
-1][term
.col
-1].mode
 
 858                 } else if (direction 
> 0) { 
 859                         for (; *y 
< term
.row
-1; *y 
+= direction
) { 
 860                                 if (!(term
.line
[*y
][term
.col
-1].mode
 
 871 getbuttoninfo(XEvent 
*e
) 
 874         uint state 
= e
->xbutton
.state 
& ~(Button1Mask 
| forceselmod
); 
 876         sel
.alt 
= IS_SET(MODE_ALTSCREEN
); 
 878         sel
.oe
.x 
= x2col(e
->xbutton
.x
); 
 879         sel
.oe
.y 
= y2row(e
->xbutton
.y
); 
 882         sel
.type 
= SEL_REGULAR
; 
 883         for (type 
= 1; type 
< LEN(selmasks
); ++type
) { 
 884                 if (match(selmasks
[type
], state
)) { 
 892 mousereport(XEvent 
*e
) 
 894         int x 
= x2col(e
->xbutton
.x
), y 
= y2row(e
->xbutton
.y
), 
 895             button 
= e
->xbutton
.button
, state 
= e
->xbutton
.state
, 
 901         if (e
->xbutton
.type 
== MotionNotify
) { 
 902                 if (x 
== ox 
&& y 
== oy
) 
 904                 if (!IS_SET(MODE_MOUSEMOTION
) && !IS_SET(MODE_MOUSEMANY
)) 
 906                 /* MOUSE_MOTION: no reporting if no button is pressed */ 
 907                 if (IS_SET(MODE_MOUSEMOTION
) && oldbutton 
== 3) 
 910                 button 
= oldbutton 
+ 32; 
 914                 if (!IS_SET(MODE_MOUSESGR
) && e
->xbutton
.type 
== ButtonRelease
) { 
 921                 if (e
->xbutton
.type 
== ButtonPress
) { 
 925                 } else if (e
->xbutton
.type 
== ButtonRelease
) { 
 927                         /* MODE_MOUSEX10: no button release reporting */ 
 928                         if (IS_SET(MODE_MOUSEX10
)) 
 930                         if (button 
== 64 || button 
== 65) 
 935         if (!IS_SET(MODE_MOUSEX10
)) { 
 936                 button 
+= ((state 
& ShiftMask  
) ? 4  : 0) 
 937                         + ((state 
& Mod4Mask   
) ? 8  : 0) 
 938                         + ((state 
& ControlMask
) ? 16 : 0); 
 941         if (IS_SET(MODE_MOUSESGR
)) { 
 942                 len 
= snprintf(buf
, sizeof(buf
), "\033[<%d;%d;%d%c", 
 944                                 e
->xbutton
.type 
== ButtonRelease 
? 'm' : 'M'); 
 945         } else if (x 
< 223 && y 
< 223) { 
 946                 len 
= snprintf(buf
, sizeof(buf
), "\033[M%c%c%c", 
 947                                 32+button
, 32+x
+1, 32+y
+1); 
 961         if (IS_SET(MODE_MOUSE
) && !(e
->xbutton
.state 
& forceselmod
)) { 
 966         for (ms 
= mshortcuts
; ms 
< mshortcuts 
+ LEN(mshortcuts
); ms
++) { 
 967                 if (e
->xbutton
.button 
== ms
->b
 
 968                                 && match(ms
->mask
, e
->xbutton
.state
)) { 
 969                         ttysend(ms
->s
, strlen(ms
->s
)); 
 974         if (e
->xbutton
.button 
== Button1
) { 
 975                 clock_gettime(CLOCK_MONOTONIC
, &now
); 
 977                 /* Clear previous selection, logically and visually. */ 
 979                 sel
.mode 
= SEL_EMPTY
; 
 980                 sel
.type 
= SEL_REGULAR
; 
 981                 sel
.oe
.x 
= sel
.ob
.x 
= x2col(e
->xbutton
.x
); 
 982                 sel
.oe
.y 
= sel
.ob
.y 
= y2row(e
->xbutton
.y
); 
 985                  * If the user clicks below predefined timeouts specific 
 986                  * snapping behaviour is exposed. 
 988                 if (TIMEDIFF(now
, sel
.tclick2
) <= tripleclicktimeout
) { 
 989                         sel
.snap 
= SNAP_LINE
; 
 990                 } else if (TIMEDIFF(now
, sel
.tclick1
) <= doubleclicktimeout
) { 
 991                         sel
.snap 
= SNAP_WORD
; 
 998                         sel
.mode 
= SEL_READY
; 
 999                 tsetdirt(sel
.nb
.y
, sel
.ne
.y
); 
1000                 sel
.tclick2 
= sel
.tclick1
; 
1009         int y
, bufsize
, lastx
, linelen
; 
1015         bufsize 
= (term
.col
+1) * (sel
.ne
.y
-sel
.nb
.y
+1) * UTF_SIZ
; 
1016         ptr 
= str 
= xmalloc(bufsize
); 
1018         /* append every set & selected glyph to the selection */ 
1019         for (y 
= sel
.nb
.y
; y 
<= sel
.ne
.y
; y
++) { 
1020                 if ((linelen 
= tlinelen(y
)) == 0) { 
1025                 if (sel
.type 
== SEL_RECTANGULAR
) { 
1026                         gp 
= &term
.line
[y
][sel
.nb
.x
]; 
1029                         gp 
= &term
.line
[y
][sel
.nb
.y 
== y 
? sel
.nb
.x 
: 0]; 
1030                         lastx 
= (sel
.ne
.y 
== y
) ? sel
.ne
.x 
: term
.col
-1; 
1032                 last 
= &term
.line
[y
][MIN(lastx
, linelen
-1)]; 
1033                 while (last 
>= gp 
&& last
->u 
== ' ') 
1036                 for ( ; gp 
<= last
; ++gp
) { 
1037                         if (gp
->mode 
& ATTR_WDUMMY
) 
1040                         ptr 
+= utf8encode(gp
->u
, ptr
); 
1044                  * Copy and pasting of line endings is inconsistent 
1045                  * in the inconsistent terminal and GUI world. 
1046                  * The best solution seems like to produce '\n' when 
1047                  * something is copied from st and convert '\n' to 
1048                  * '\r', when something to be pasted is received by 
1050                  * FIXME: Fix the computer world. 
1052                 if ((y 
< sel
.ne
.y 
|| lastx 
>= linelen
) && !(last
->mode 
& ATTR_WRAP
)) 
1062         xsetsel(getsel(), t
); 
1066 propnotify(XEvent 
*e
) 
1068         XPropertyEvent 
*xpev
; 
1069         Atom clipboard 
= XInternAtom(xw
.dpy
, "CLIPBOARD", 0); 
1071         xpev 
= &e
->xproperty
; 
1072         if (xpev
->state 
== PropertyNewValue 
&& 
1073                         (xpev
->atom 
== XA_PRIMARY 
|| 
1074                          xpev
->atom 
== clipboard
)) { 
1080 selnotify(XEvent 
*e
) 
1082         ulong nitems
, ofs
, rem
; 
1084         uchar 
*data
, *last
, *repl
; 
1085         Atom type
, incratom
, property
; 
1087         incratom 
= XInternAtom(xw
.dpy
, "INCR", 0); 
1090         if (e
->type 
== SelectionNotify
) { 
1091                 property 
= e
->xselection
.property
; 
1092         } else if(e
->type 
== PropertyNotify
) { 
1093                 property 
= e
->xproperty
.atom
; 
1097         if (property 
== None
) 
1101                 if (XGetWindowProperty(xw
.dpy
, xw
.win
, property
, ofs
, 
1102                                         BUFSIZ
/4, False
, AnyPropertyType
, 
1103                                         &type
, &format
, &nitems
, &rem
, 
1105                         fprintf(stderr
, "Clipboard allocation failed\n"); 
1109                 if (e
->type 
== PropertyNotify 
&& nitems 
== 0 && rem 
== 0) { 
1111                          * If there is some PropertyNotify with no data, then 
1112                          * this is the signal of the selection owner that all 
1113                          * data has been transferred. We won't need to receive 
1114                          * PropertyNotify events anymore. 
1116                         MODBIT(xw
.attrs
.event_mask
, 0, PropertyChangeMask
); 
1117                         XChangeWindowAttributes(xw
.dpy
, xw
.win
, CWEventMask
, 
1121                 if (type 
== incratom
) { 
1123                          * Activate the PropertyNotify events so we receive 
1124                          * when the selection owner does send us the next 
1127                         MODBIT(xw
.attrs
.event_mask
, 1, PropertyChangeMask
); 
1128                         XChangeWindowAttributes(xw
.dpy
, xw
.win
, CWEventMask
, 
1132                          * Deleting the property is the transfer start signal. 
1134                         XDeleteProperty(xw
.dpy
, xw
.win
, (int)property
); 
1139                  * As seen in getsel: 
1140                  * Line endings are inconsistent in the terminal and GUI world 
1141                  * copy and pasting. When receiving some selection data, 
1142                  * replace all '\n' with '\r'. 
1143                  * FIXME: Fix the computer world. 
1146                 last 
= data 
+ nitems 
* format 
/ 8; 
1147                 while ((repl 
= memchr(repl
, '\n', last 
- repl
))) { 
1151                 if (IS_SET(MODE_BRCKTPASTE
) && ofs 
== 0) 
1152                         ttywrite("\033[200~", 6); 
1153                 ttysend((char *)data
, nitems 
* format 
/ 8); 
1154                 if (IS_SET(MODE_BRCKTPASTE
) && rem 
== 0) 
1155                         ttywrite("\033[201~", 6); 
1157                 /* number of 32-bit chunks returned */ 
1158                 ofs 
+= nitems 
* format 
/ 32; 
1162          * Deleting the property again tells the selection owner to send the 
1163          * next data chunk in the property. 
1165         XDeleteProperty(xw
.dpy
, xw
.win
, (int)property
); 
1169 selpaste(const Arg 
*dummy
) 
1171         XConvertSelection(xw
.dpy
, XA_PRIMARY
, sel
.xtarget
, XA_PRIMARY
, 
1172                         xw
.win
, CurrentTime
); 
1176 clipcopy(const Arg 
*dummy
) 
1180         if (sel
.clipboard 
!= NULL
) 
1181                 free(sel
.clipboard
); 
1183         if (sel
.primary 
!= NULL
) { 
1184                 sel
.clipboard 
= xstrdup(sel
.primary
); 
1185                 clipboard 
= XInternAtom(xw
.dpy
, "CLIPBOARD", 0); 
1186                 XSetSelectionOwner(xw
.dpy
, clipboard
, xw
.win
, CurrentTime
); 
1191 clippaste(const Arg 
*dummy
) 
1195         clipboard 
= XInternAtom(xw
.dpy
, "CLIPBOARD", 0); 
1196         XConvertSelection(xw
.dpy
, clipboard
, sel
.xtarget
, clipboard
, 
1197                         xw
.win
, CurrentTime
); 
1205         sel
.mode 
= SEL_IDLE
; 
1207         tsetdirt(sel
.nb
.y
, sel
.ne
.y
); 
1211 selrequest(XEvent 
*e
) 
1213         XSelectionRequestEvent 
*xsre
; 
1214         XSelectionEvent xev
; 
1215         Atom xa_targets
, string
, clipboard
; 
1218         xsre 
= (XSelectionRequestEvent 
*) e
; 
1219         xev
.type 
= SelectionNotify
; 
1220         xev
.requestor 
= xsre
->requestor
; 
1221         xev
.selection 
= xsre
->selection
; 
1222         xev
.target 
= xsre
->target
; 
1223         xev
.time 
= xsre
->time
; 
1224         if (xsre
->property 
== None
) 
1225                 xsre
->property 
= xsre
->target
; 
1228         xev
.property 
= None
; 
1230         xa_targets 
= XInternAtom(xw
.dpy
, "TARGETS", 0); 
1231         if (xsre
->target 
== xa_targets
) { 
1232                 /* respond with the supported type */ 
1233                 string 
= sel
.xtarget
; 
1234                 XChangeProperty(xsre
->display
, xsre
->requestor
, xsre
->property
, 
1235                                 XA_ATOM
, 32, PropModeReplace
, 
1236                                 (uchar 
*) &string
, 1); 
1237                 xev
.property 
= xsre
->property
; 
1238         } else if (xsre
->target 
== sel
.xtarget 
|| xsre
->target 
== XA_STRING
) { 
1240                  * xith XA_STRING non ascii characters may be incorrect in the 
1241                  * requestor. It is not our problem, use utf8. 
1243                 clipboard 
= XInternAtom(xw
.dpy
, "CLIPBOARD", 0); 
1244                 if (xsre
->selection 
== XA_PRIMARY
) { 
1245                         seltext 
= sel
.primary
; 
1246                 } else if (xsre
->selection 
== clipboard
) { 
1247                         seltext 
= sel
.clipboard
; 
1250                                 "Unhandled clipboard selection 0x%lx\n", 
1254                 if (seltext 
!= NULL
) { 
1255                         XChangeProperty(xsre
->display
, xsre
->requestor
, 
1256                                         xsre
->property
, xsre
->target
, 
1258                                         (uchar 
*)seltext
, strlen(seltext
)); 
1259                         xev
.property 
= xsre
->property
; 
1263         /* all done, send a notification to the listener */ 
1264         if (!XSendEvent(xsre
->display
, xsre
->requestor
, 1, 0, (XEvent 
*) &xev
)) 
1265                 fprintf(stderr
, "Error sending SelectionNotify event\n"); 
1269 xsetsel(char *str
, Time t
) 
1274         XSetSelectionOwner(xw
.dpy
, XA_PRIMARY
, xw
.win
, t
); 
1275         if (XGetSelectionOwner(xw
.dpy
, XA_PRIMARY
) != xw
.win
) 
1282         if (IS_SET(MODE_MOUSE
) && !(e
->xbutton
.state 
& forceselmod
)) { 
1287         if (e
->xbutton
.button 
== Button2
) { 
1289         } else if (e
->xbutton
.button 
== Button1
) { 
1290                 if (sel
.mode 
== SEL_READY
) { 
1292                         selcopy(e
->xbutton
.time
); 
1295                 sel
.mode 
= SEL_IDLE
; 
1296                 tsetdirt(sel
.nb
.y
, sel
.ne
.y
); 
1303         int oldey
, oldex
, oldsby
, oldsey
; 
1305         if (IS_SET(MODE_MOUSE
) && !(e
->xbutton
.state 
& forceselmod
)) { 
1313         sel
.mode 
= SEL_READY
; 
1320         if (oldey 
!= sel
.oe
.y 
|| oldex 
!= sel
.oe
.x
) 
1321                 tsetdirt(MIN(sel
.nb
.y
, oldsby
), MAX(sel
.ne
.y
, oldsey
)); 
1325 die(const char *errstr
, ...) 
1329         va_start(ap
, errstr
); 
1330         vfprintf(stderr
, errstr
, ap
); 
1338         char **args
, *sh
, *prog
; 
1339         const struct passwd 
*pw
; 
1340         char buf
[sizeof(long) * 8 + 1]; 
1343         if ((pw 
= getpwuid(getuid())) == NULL
) { 
1345                         die("getpwuid:%s\n", strerror(errno
)); 
1347                         die("who are you?\n"); 
1350         if ((sh 
= getenv("SHELL")) == NULL
) 
1351                 sh 
= (pw
->pw_shell
[0]) ? pw
->pw_shell 
: shell
; 
1359         args 
= (opt_cmd
) ? opt_cmd 
: (char *[]) {prog
, NULL
}; 
1361         snprintf(buf
, sizeof(buf
), "%lu", xw
.win
); 
1363         unsetenv("COLUMNS"); 
1365         unsetenv("TERMCAP"); 
1366         setenv("LOGNAME", pw
->pw_name
, 1); 
1367         setenv("USER", pw
->pw_name
, 1); 
1368         setenv("SHELL", sh
, 1); 
1369         setenv("HOME", pw
->pw_dir
, 1); 
1370         setenv("TERM", termname
, 1); 
1371         setenv("WINDOWID", buf
, 1); 
1373         signal(SIGCHLD
, SIG_DFL
); 
1374         signal(SIGHUP
, SIG_DFL
); 
1375         signal(SIGINT
, SIG_DFL
); 
1376         signal(SIGQUIT
, SIG_DFL
); 
1377         signal(SIGTERM
, SIG_DFL
); 
1378         signal(SIGALRM
, SIG_DFL
); 
1390         if ((p 
= waitpid(pid
, &stat
, WNOHANG
)) < 0) 
1391                 die("Waiting for pid %hd failed: %s\n", pid
, strerror(errno
)); 
1396         if (!WIFEXITED(stat
) || WEXITSTATUS(stat
)) 
1397                 die("child finished with error '%d'\n", stat
); 
1405         char cmd
[_POSIX_ARG_MAX
], **p
, *q
, *s
; 
1408         if ((n 
= strlen(stty_args
)) > sizeof(cmd
)-1) 
1409                 die("incorrect stty parameters\n"); 
1410         memcpy(cmd
, stty_args
, n
); 
1412         siz 
= sizeof(cmd
) - n
; 
1413         for (p 
= opt_cmd
; p 
&& (s 
= *p
); ++p
) { 
1414                 if ((n 
= strlen(s
)) > siz
-1) 
1415                         die("stty parameter length too long\n"); 
1422         if (system(cmd
) != 0) 
1423             perror("Couldn't call stty"); 
1430         struct winsize w 
= {term
.row
, term
.col
, 0, 0}; 
1433                 term
.mode 
|= MODE_PRINT
; 
1434                 iofd 
= (!strcmp(opt_io
, "-")) ? 
1435                           1 : open(opt_io
, O_WRONLY 
| O_CREAT
, 0666); 
1437                         fprintf(stderr
, "Error opening %s:%s\n", 
1438                                 opt_io
, strerror(errno
)); 
1443                 if ((cmdfd 
= open(opt_line
, O_RDWR
)) < 0) 
1444                         die("open line failed: %s\n", strerror(errno
)); 
1450         /* seems to work fine on linux, openbsd and freebsd */ 
1451         if (openpty(&m
, &s
, NULL
, NULL
, &w
) < 0) 
1452                 die("openpty failed: %s\n", strerror(errno
)); 
1454         switch (pid 
= fork()) { 
1456                 die("fork failed\n"); 
1460                 setsid(); /* create a new process group */ 
1464                 if (ioctl(s
, TIOCSCTTY
, NULL
) < 0) 
1465                         die("ioctl TIOCSCTTY failed: %s\n", strerror(errno
)); 
1473                 signal(SIGCHLD
, sigchld
); 
1481         static char buf
[BUFSIZ
]; 
1482         static int buflen 
= 0; 
1484         int charsize
; /* size of utf8 char in bytes */ 
1488         /* append read bytes to unprocessed bytes */ 
1489         if ((ret 
= read(cmdfd
, buf
+buflen
, LEN(buf
)-buflen
)) < 0) 
1490                 die("Couldn't read from shell: %s\n", strerror(errno
)); 
1496                 if (IS_SET(MODE_UTF8
) && !IS_SET(MODE_SIXEL
)) { 
1497                         /* process a complete utf8 char */ 
1498                         charsize 
= utf8decode(ptr
, &unicodep
, buflen
); 
1508                         tputc(*ptr
++ & 0xFF); 
1512         /* keep any uncomplete utf8 char for the next call */ 
1514                 memmove(buf
, ptr
, buflen
); 
1520 ttywrite(const char *s
, size_t n
) 
1527          * Remember that we are using a pty, which might be a modem line. 
1528          * Writing too much will clog the line. That's why we are doing this 
1530          * FIXME: Migrate the world to Plan 9. 
1535                 FD_SET(cmdfd
, &wfd
); 
1536                 FD_SET(cmdfd
, &rfd
); 
1538                 /* Check if we can write. */ 
1539                 if (pselect(cmdfd
+1, &rfd
, &wfd
, NULL
, NULL
, NULL
) < 0) { 
1542                         die("select failed: %s\n", strerror(errno
)); 
1544                 if (FD_ISSET(cmdfd
, &wfd
)) { 
1546                          * Only write the bytes written by ttywrite() or the 
1547                          * default of 256. This seems to be a reasonable value 
1548                          * for a serial line. Bigger values might clog the I/O. 
1550                         if ((r 
= write(cmdfd
, s
, (n 
< lim
)? n 
: lim
)) < 0) 
1554                                  * We weren't able to write out everything. 
1555                                  * This means the buffer is getting full 
1563                                 /* All bytes have been written. */ 
1567                 if (FD_ISSET(cmdfd
, &rfd
)) 
1573         die("write error on tty: %s\n", strerror(errno
)); 
1577 ttysend(char *s
, size_t n
) 
1584         if (!IS_SET(MODE_ECHO
)) 
1588         for (t 
= s
; t 
< lim
; t 
+= len
) { 
1589                 if (IS_SET(MODE_UTF8
) && !IS_SET(MODE_SIXEL
)) { 
1590                         len 
= utf8decode(t
, &u
, n
); 
1607         w
.ws_row 
= term
.row
; 
1608         w
.ws_col 
= term
.col
; 
1609         w
.ws_xpixel 
= xw
.tw
; 
1610         w
.ws_ypixel 
= xw
.th
; 
1611         if (ioctl(cmdfd
, TIOCSWINSZ
, &w
) < 0) 
1612                 fprintf(stderr
, "Couldn't set window size: %s\n", strerror(errno
)); 
1620         for (i 
= 0; i 
< term
.row
-1; i
++) { 
1621                 for (j 
= 0; j 
< term
.col
-1; j
++) { 
1622                         if (term
.line
[i
][j
].mode 
& attr
) 
1631 tsetdirt(int top
, int bot
) 
1635         LIMIT(top
, 0, term
.row
-1); 
1636         LIMIT(bot
, 0, term
.row
-1); 
1638         for (i 
= top
; i 
<= bot
; i
++) 
1643 tsetdirtattr(int attr
) 
1647         for (i 
= 0; i 
< term
.row
-1; i
++) { 
1648                 for (j 
= 0; j 
< term
.col
-1; j
++) { 
1649                         if (term
.line
[i
][j
].mode 
& attr
) { 
1660         tsetdirt(0, term
.row
-1); 
1666         static TCursor c
[2]; 
1667         int alt 
= IS_SET(MODE_ALTSCREEN
); 
1669         if (mode 
== CURSOR_SAVE
) { 
1671         } else if (mode 
== CURSOR_LOAD
) { 
1673                 tmoveto(c
[alt
].x
, c
[alt
].y
); 
1682         term
.c 
= (TCursor
){{ 
1686         }, .x 
= 0, .y 
= 0, .state 
= CURSOR_DEFAULT
}; 
1688         memset(term
.tabs
, 0, term
.col 
* sizeof(*term
.tabs
)); 
1689         for (i 
= tabspaces
; i 
< term
.col
; i 
+= tabspaces
) 
1692         term
.bot 
= term
.row 
- 1; 
1693         term
.mode 
= MODE_WRAP
|MODE_UTF8
; 
1694         memset(term
.trantbl
, CS_USA
, sizeof(term
.trantbl
)); 
1697         for (i 
= 0; i 
< 2; i
++) { 
1699                 tcursor(CURSOR_SAVE
); 
1700                 tclearregion(0, 0, term
.col
-1, term
.row
-1); 
1706 tnew(int col
, int row
) 
1708         term 
= (Term
){ .c 
= { .attr 
= { .fg 
= defaultfg
, .bg 
= defaultbg 
} } }; 
1718         Line 
*tmp 
= term
.line
; 
1720         term
.line 
= term
.alt
; 
1722         term
.mode 
^= MODE_ALTSCREEN
; 
1727 tscrolldown(int orig
, int n
) 
1732         LIMIT(n
, 0, term
.bot
-orig
+1); 
1734         tsetdirt(orig
, term
.bot
-n
); 
1735         tclearregion(0, term
.bot
-n
+1, term
.col
-1, term
.bot
); 
1737         for (i 
= term
.bot
; i 
>= orig
+n
; i
--) { 
1738                 temp 
= term
.line
[i
]; 
1739                 term
.line
[i
] = term
.line
[i
-n
]; 
1740                 term
.line
[i
-n
] = temp
; 
1747 tscrollup(int orig
, int n
) 
1752         LIMIT(n
, 0, term
.bot
-orig
+1); 
1754         tclearregion(0, orig
, term
.col
-1, orig
+n
-1); 
1755         tsetdirt(orig
+n
, term
.bot
); 
1757         for (i 
= orig
; i 
<= term
.bot
-n
; i
++) { 
1758                 temp 
= term
.line
[i
]; 
1759                 term
.line
[i
] = term
.line
[i
+n
]; 
1760                 term
.line
[i
+n
] = temp
; 
1763         selscroll(orig
, -n
); 
1767 selscroll(int orig
, int n
) 
1772         if (BETWEEN(sel
.ob
.y
, orig
, term
.bot
) || BETWEEN(sel
.oe
.y
, orig
, term
.bot
)) { 
1773                 if ((sel
.ob
.y 
+= n
) > term
.bot 
|| (sel
.oe
.y 
+= n
) < term
.top
) { 
1777                 if (sel
.type 
== SEL_RECTANGULAR
) { 
1778                         if (sel
.ob
.y 
< term
.top
) 
1779                                 sel
.ob
.y 
= term
.top
; 
1780                         if (sel
.oe
.y 
> term
.bot
) 
1781                                 sel
.oe
.y 
= term
.bot
; 
1783                         if (sel
.ob
.y 
< term
.top
) { 
1784                                 sel
.ob
.y 
= term
.top
; 
1787                         if (sel
.oe
.y 
> term
.bot
) { 
1788                                 sel
.oe
.y 
= term
.bot
; 
1789                                 sel
.oe
.x 
= term
.col
; 
1797 tnewline(int first_col
) 
1801         if (y 
== term
.bot
) { 
1802                 tscrollup(term
.top
, 1); 
1806         tmoveto(first_col 
? 0 : term
.c
.x
, y
); 
1812         char *p 
= csiescseq
.buf
, *np
; 
1821         csiescseq
.buf
[csiescseq
.len
] = '\0'; 
1822         while (p 
< csiescseq
.buf
+csiescseq
.len
) { 
1824                 v 
= strtol(p
, &np
, 10); 
1827                 if (v 
== LONG_MAX 
|| v 
== LONG_MIN
) 
1829                 csiescseq
.arg
[csiescseq
.narg
++] = v
; 
1831                 if (*p 
!= ';' || csiescseq
.narg 
== ESC_ARG_SIZ
) 
1835         csiescseq
.mode
[0] = *p
++; 
1836         csiescseq
.mode
[1] = (p 
< csiescseq
.buf
+csiescseq
.len
) ? *p 
: '\0'; 
1839 /* for absolute user moves, when decom is set */ 
1841 tmoveato(int x
, int y
) 
1843         tmoveto(x
, y 
+ ((term
.c
.state 
& CURSOR_ORIGIN
) ? term
.top
: 0)); 
1847 tmoveto(int x
, int y
) 
1851         if (term
.c
.state 
& CURSOR_ORIGIN
) { 
1856                 maxy 
= term
.row 
- 1; 
1858         term
.c
.state 
&= ~CURSOR_WRAPNEXT
; 
1859         term
.c
.x 
= LIMIT(x
, 0, term
.col
-1); 
1860         term
.c
.y 
= LIMIT(y
, miny
, maxy
); 
1864 tsetchar(Rune u
, Glyph 
*attr
, int x
, int y
) 
1866         static char *vt100_0
[62] = { /* 0x41 - 0x7e */ 
1867                 "↑", "↓", "→", "←", "█", "▚", "☃", /* A - G */ 
1868                 0, 0, 0, 0, 0, 0, 0, 0, /* H - O */ 
1869                 0, 0, 0, 0, 0, 0, 0, 0, /* P - W */ 
1870                 0, 0, 0, 0, 0, 0, 0, " ", /* X - _ */ 
1871                 "◆", "▒", "␉", "␌", "␍", "␊", "°", "±", /* ` - g */ 
1872                 "", "␋", "┘", "┐", "┌", "└", "┼", "⎺", /* h - o */ 
1873                 "⎻", "─", "⎼", "⎽", "├", "┤", "┴", "┬", /* p - w */ 
1874                 "│", "≤", "≥", "π", "≠", "£", "·", /* x - ~ */ 
1878          * The table is proudly stolen from rxvt. 
1880         if (term
.trantbl
[term
.charset
] == CS_GRAPHIC0 
&& 
1881            BETWEEN(u
, 0x41, 0x7e) && vt100_0
[u 
- 0x41]) 
1882                 utf8decode(vt100_0
[u 
- 0x41], &u
, UTF_SIZ
); 
1884         if (term
.line
[y
][x
].mode 
& ATTR_WIDE
) { 
1885                 if (x
+1 < term
.col
) { 
1886                         term
.line
[y
][x
+1].u 
= ' '; 
1887                         term
.line
[y
][x
+1].mode 
&= ~ATTR_WDUMMY
; 
1889         } else if (term
.line
[y
][x
].mode 
& ATTR_WDUMMY
) { 
1890                 term
.line
[y
][x
-1].u 
= ' '; 
1891                 term
.line
[y
][x
-1].mode 
&= ~ATTR_WIDE
; 
1895         term
.line
[y
][x
] = *attr
; 
1896         term
.line
[y
][x
].u 
= u
; 
1900 tclearregion(int x1
, int y1
, int x2
, int y2
) 
1906                 temp 
= x1
, x1 
= x2
, x2 
= temp
; 
1908                 temp 
= y1
, y1 
= y2
, y2 
= temp
; 
1910         LIMIT(x1
, 0, term
.col
-1); 
1911         LIMIT(x2
, 0, term
.col
-1); 
1912         LIMIT(y1
, 0, term
.row
-1); 
1913         LIMIT(y2
, 0, term
.row
-1); 
1915         for (y 
= y1
; y 
<= y2
; y
++) { 
1917                 for (x 
= x1
; x 
<= x2
; x
++) { 
1918                         gp 
= &term
.line
[y
][x
]; 
1921                         gp
->fg 
= term
.c
.attr
.fg
; 
1922                         gp
->bg 
= term
.c
.attr
.bg
; 
1935         LIMIT(n
, 0, term
.col 
- term
.c
.x
); 
1939         size 
= term
.col 
- src
; 
1940         line 
= term
.line
[term
.c
.y
]; 
1942         memmove(&line
[dst
], &line
[src
], size 
* sizeof(Glyph
)); 
1943         tclearregion(term
.col
-n
, term
.c
.y
, term
.col
-1, term
.c
.y
); 
1952         LIMIT(n
, 0, term
.col 
- term
.c
.x
); 
1956         size 
= term
.col 
- dst
; 
1957         line 
= term
.line
[term
.c
.y
]; 
1959         memmove(&line
[dst
], &line
[src
], size 
* sizeof(Glyph
)); 
1960         tclearregion(src
, term
.c
.y
, dst 
- 1, term
.c
.y
); 
1964 tinsertblankline(int n
) 
1966         if (BETWEEN(term
.c
.y
, term
.top
, term
.bot
)) 
1967                 tscrolldown(term
.c
.y
, n
); 
1973         if (BETWEEN(term
.c
.y
, term
.top
, term
.bot
)) 
1974                 tscrollup(term
.c
.y
, n
); 
1978 tdefcolor(int *attr
, int *npar
, int l
) 
1983         switch (attr
[*npar 
+ 1]) { 
1984         case 2: /* direct color in RGB space */ 
1985                 if (*npar 
+ 4 >= l
) { 
1987                                 "erresc(38): Incorrect number of parameters (%d)\n", 
1991                 r 
= attr
[*npar 
+ 2]; 
1992                 g 
= attr
[*npar 
+ 3]; 
1993                 b 
= attr
[*npar 
+ 4]; 
1995                 if (!BETWEEN(r
, 0, 255) || !BETWEEN(g
, 0, 255) || !BETWEEN(b
, 0, 255)) 
1996                         fprintf(stderr
, "erresc: bad rgb color (%u,%u,%u)\n", 
1999                         idx 
= TRUECOLOR(r
, g
, b
); 
2001         case 5: /* indexed color */ 
2002                 if (*npar 
+ 2 >= l
) { 
2004                                 "erresc(38): Incorrect number of parameters (%d)\n", 
2009                 if (!BETWEEN(attr
[*npar
], 0, 255)) 
2010                         fprintf(stderr
, "erresc: bad fgcolor %d\n", attr
[*npar
]); 
2014         case 0: /* implemented defined (only foreground) */ 
2015         case 1: /* transparent */ 
2016         case 3: /* direct color in CMY space */ 
2017         case 4: /* direct color in CMYK space */ 
2020                         "erresc(38): gfx attr %d unknown\n", attr
[*npar
]); 
2028 tsetattr(int *attr
, int l
) 
2033         for (i 
= 0; i 
< l
; i
++) { 
2036                         term
.c
.attr
.mode 
&= ~( 
2045                         term
.c
.attr
.fg 
= defaultfg
; 
2046                         term
.c
.attr
.bg 
= defaultbg
; 
2049                         term
.c
.attr
.mode 
|= ATTR_BOLD
; 
2052                         term
.c
.attr
.mode 
|= ATTR_FAINT
; 
2055                         term
.c
.attr
.mode 
|= ATTR_ITALIC
; 
2058                         term
.c
.attr
.mode 
|= ATTR_UNDERLINE
; 
2060                 case 5: /* slow blink */ 
2062                 case 6: /* rapid blink */ 
2063                         term
.c
.attr
.mode 
|= ATTR_BLINK
; 
2066                         term
.c
.attr
.mode 
|= ATTR_REVERSE
; 
2069                         term
.c
.attr
.mode 
|= ATTR_INVISIBLE
; 
2072                         term
.c
.attr
.mode 
|= ATTR_STRUCK
; 
2075                         term
.c
.attr
.mode 
&= ~(ATTR_BOLD 
| ATTR_FAINT
); 
2078                         term
.c
.attr
.mode 
&= ~ATTR_ITALIC
; 
2081                         term
.c
.attr
.mode 
&= ~ATTR_UNDERLINE
; 
2084                         term
.c
.attr
.mode 
&= ~ATTR_BLINK
; 
2087                         term
.c
.attr
.mode 
&= ~ATTR_REVERSE
; 
2090                         term
.c
.attr
.mode 
&= ~ATTR_INVISIBLE
; 
2093                         term
.c
.attr
.mode 
&= ~ATTR_STRUCK
; 
2096                         if ((idx 
= tdefcolor(attr
, &i
, l
)) >= 0) 
2097                                 term
.c
.attr
.fg 
= idx
; 
2100                         term
.c
.attr
.fg 
= defaultfg
; 
2103                         if ((idx 
= tdefcolor(attr
, &i
, l
)) >= 0) 
2104                                 term
.c
.attr
.bg 
= idx
; 
2107                         term
.c
.attr
.bg 
= defaultbg
; 
2110                         if (BETWEEN(attr
[i
], 30, 37)) { 
2111                                 term
.c
.attr
.fg 
= attr
[i
] - 30; 
2112                         } else if (BETWEEN(attr
[i
], 40, 47)) { 
2113                                 term
.c
.attr
.bg 
= attr
[i
] - 40; 
2114                         } else if (BETWEEN(attr
[i
], 90, 97)) { 
2115                                 term
.c
.attr
.fg 
= attr
[i
] - 90 + 8; 
2116                         } else if (BETWEEN(attr
[i
], 100, 107)) { 
2117                                 term
.c
.attr
.bg 
= attr
[i
] - 100 + 8; 
2120                                         "erresc(default): gfx attr %d unknown\n", 
2121                                         attr
[i
]), csidump(); 
2129 tsetscroll(int t
, int b
) 
2133         LIMIT(t
, 0, term
.row
-1); 
2134         LIMIT(b
, 0, term
.row
-1); 
2145 tsetmode(int priv
, int set
, int *args
, int narg
) 
2150         for (lim 
= args 
+ narg
; args 
< lim
; ++args
) { 
2153                         case 1: /* DECCKM -- Cursor key */ 
2154                                 MODBIT(term
.mode
, set
, MODE_APPCURSOR
); 
2156                         case 5: /* DECSCNM -- Reverse video */ 
2158                                 MODBIT(term
.mode
, set
, MODE_REVERSE
); 
2159                                 if (mode 
!= term
.mode
) 
2162                         case 6: /* DECOM -- Origin */ 
2163                                 MODBIT(term
.c
.state
, set
, CURSOR_ORIGIN
); 
2166                         case 7: /* DECAWM -- Auto wrap */ 
2167                                 MODBIT(term
.mode
, set
, MODE_WRAP
); 
2169                         case 0:  /* Error (IGNORED) */ 
2170                         case 2:  /* DECANM -- ANSI/VT52 (IGNORED) */ 
2171                         case 3:  /* DECCOLM -- Column  (IGNORED) */ 
2172                         case 4:  /* DECSCLM -- Scroll (IGNORED) */ 
2173                         case 8:  /* DECARM -- Auto repeat (IGNORED) */ 
2174                         case 18: /* DECPFF -- Printer feed (IGNORED) */ 
2175                         case 19: /* DECPEX -- Printer extent (IGNORED) */ 
2176                         case 42: /* DECNRCM -- National characters (IGNORED) */ 
2177                         case 12: /* att610 -- Start blinking cursor (IGNORED) */ 
2179                         case 25: /* DECTCEM -- Text Cursor Enable Mode */ 
2180                                 MODBIT(term
.mode
, !set
, MODE_HIDE
); 
2182                         case 9:    /* X10 mouse compatibility mode */ 
2183                                 xsetpointermotion(0); 
2184                                 MODBIT(term
.mode
, 0, MODE_MOUSE
); 
2185                                 MODBIT(term
.mode
, set
, MODE_MOUSEX10
); 
2187                         case 1000: /* 1000: report button press */ 
2188                                 xsetpointermotion(0); 
2189                                 MODBIT(term
.mode
, 0, MODE_MOUSE
); 
2190                                 MODBIT(term
.mode
, set
, MODE_MOUSEBTN
); 
2192                         case 1002: /* 1002: report motion on button press */ 
2193                                 xsetpointermotion(0); 
2194                                 MODBIT(term
.mode
, 0, MODE_MOUSE
); 
2195                                 MODBIT(term
.mode
, set
, MODE_MOUSEMOTION
); 
2197                         case 1003: /* 1003: enable all mouse motions */ 
2198                                 xsetpointermotion(set
); 
2199                                 MODBIT(term
.mode
, 0, MODE_MOUSE
); 
2200                                 MODBIT(term
.mode
, set
, MODE_MOUSEMANY
); 
2202                         case 1004: /* 1004: send focus events to tty */ 
2203                                 MODBIT(term
.mode
, set
, MODE_FOCUS
); 
2205                         case 1006: /* 1006: extended reporting mode */ 
2206                                 MODBIT(term
.mode
, set
, MODE_MOUSESGR
); 
2209                                 MODBIT(term
.mode
, set
, MODE_8BIT
); 
2211                         case 1049: /* swap screen & set/restore cursor as xterm */ 
2212                                 if (!allowaltscreen
) 
2214                                 tcursor((set
) ? CURSOR_SAVE 
: CURSOR_LOAD
); 
2216                         case 47: /* swap screen */ 
2218                                 if (!allowaltscreen
) 
2220                                 alt 
= IS_SET(MODE_ALTSCREEN
); 
2222                                         tclearregion(0, 0, term
.col
-1, 
2225                                 if (set 
^ alt
) /* set is always 1 or 0 */ 
2231                                 tcursor((set
) ? CURSOR_SAVE 
: CURSOR_LOAD
); 
2233                         case 2004: /* 2004: bracketed paste mode */ 
2234                                 MODBIT(term
.mode
, set
, MODE_BRCKTPASTE
); 
2236                         /* Not implemented mouse modes. See comments there. */ 
2237                         case 1001: /* mouse highlight mode; can hang the 
2238                                       terminal by design when implemented. */ 
2239                         case 1005: /* UTF-8 mouse mode; will confuse 
2240                                       applications not supporting UTF-8 
2242                         case 1015: /* urxvt mangled mouse mode; incompatible 
2243                                       and can be mistaken for other control 
2247                                         "erresc: unknown private set/reset mode %d\n", 
2253                         case 0:  /* Error (IGNORED) */ 
2255                         case 2:  /* KAM -- keyboard action */ 
2256                                 MODBIT(term
.mode
, set
, MODE_KBDLOCK
); 
2258                         case 4:  /* IRM -- Insertion-replacement */ 
2259                                 MODBIT(term
.mode
, set
, MODE_INSERT
); 
2261                         case 12: /* SRM -- Send/Receive */ 
2262                                 MODBIT(term
.mode
, !set
, MODE_ECHO
); 
2264                         case 20: /* LNM -- Linefeed/new line */ 
2265                                 MODBIT(term
.mode
, set
, MODE_CRLF
); 
2269                                         "erresc: unknown set/reset mode %d\n", 
2283         switch (csiescseq
.mode
[0]) { 
2286                 fprintf(stderr
, "erresc: unknown csi "); 
2290         case '@': /* ICH -- Insert <n> blank char */ 
2291                 DEFAULT(csiescseq
.arg
[0], 1); 
2292                 tinsertblank(csiescseq
.arg
[0]); 
2294         case 'A': /* CUU -- Cursor <n> Up */ 
2295                 DEFAULT(csiescseq
.arg
[0], 1); 
2296                 tmoveto(term
.c
.x
, term
.c
.y
-csiescseq
.arg
[0]); 
2298         case 'B': /* CUD -- Cursor <n> Down */ 
2299         case 'e': /* VPR --Cursor <n> Down */ 
2300                 DEFAULT(csiescseq
.arg
[0], 1); 
2301                 tmoveto(term
.c
.x
, term
.c
.y
+csiescseq
.arg
[0]); 
2303         case 'i': /* MC -- Media Copy */ 
2304                 switch (csiescseq
.arg
[0]) { 
2309                         tdumpline(term
.c
.y
); 
2315                         term
.mode 
&= ~MODE_PRINT
; 
2318                         term
.mode 
|= MODE_PRINT
; 
2322         case 'c': /* DA -- Device Attributes */ 
2323                 if (csiescseq
.arg
[0] == 0) 
2324                         ttywrite(vtiden
, sizeof(vtiden
) - 1); 
2326         case 'C': /* CUF -- Cursor <n> Forward */ 
2327         case 'a': /* HPR -- Cursor <n> Forward */ 
2328                 DEFAULT(csiescseq
.arg
[0], 1); 
2329                 tmoveto(term
.c
.x
+csiescseq
.arg
[0], term
.c
.y
); 
2331         case 'D': /* CUB -- Cursor <n> Backward */ 
2332                 DEFAULT(csiescseq
.arg
[0], 1); 
2333                 tmoveto(term
.c
.x
-csiescseq
.arg
[0], term
.c
.y
); 
2335         case 'E': /* CNL -- Cursor <n> Down and first col */ 
2336                 DEFAULT(csiescseq
.arg
[0], 1); 
2337                 tmoveto(0, term
.c
.y
+csiescseq
.arg
[0]); 
2339         case 'F': /* CPL -- Cursor <n> Up and first col */ 
2340                 DEFAULT(csiescseq
.arg
[0], 1); 
2341                 tmoveto(0, term
.c
.y
-csiescseq
.arg
[0]); 
2343         case 'g': /* TBC -- Tabulation clear */ 
2344                 switch (csiescseq
.arg
[0]) { 
2345                 case 0: /* clear current tab stop */ 
2346                         term
.tabs
[term
.c
.x
] = 0; 
2348                 case 3: /* clear all the tabs */ 
2349                         memset(term
.tabs
, 0, term
.col 
* sizeof(*term
.tabs
)); 
2355         case 'G': /* CHA -- Move to <col> */ 
2357                 DEFAULT(csiescseq
.arg
[0], 1); 
2358                 tmoveto(csiescseq
.arg
[0]-1, term
.c
.y
); 
2360         case 'H': /* CUP -- Move to <row> <col> */ 
2362                 DEFAULT(csiescseq
.arg
[0], 1); 
2363                 DEFAULT(csiescseq
.arg
[1], 1); 
2364                 tmoveato(csiescseq
.arg
[1]-1, csiescseq
.arg
[0]-1); 
2366         case 'I': /* CHT -- Cursor Forward Tabulation <n> tab stops */ 
2367                 DEFAULT(csiescseq
.arg
[0], 1); 
2368                 tputtab(csiescseq
.arg
[0]); 
2370         case 'J': /* ED -- Clear screen */ 
2372                 switch (csiescseq
.arg
[0]) { 
2374                         tclearregion(term
.c
.x
, term
.c
.y
, term
.col
-1, term
.c
.y
); 
2375                         if (term
.c
.y 
< term
.row
-1) { 
2376                                 tclearregion(0, term
.c
.y
+1, term
.col
-1, 
2382                                 tclearregion(0, 0, term
.col
-1, term
.c
.y
-1); 
2383                         tclearregion(0, term
.c
.y
, term
.c
.x
, term
.c
.y
); 
2386                         tclearregion(0, 0, term
.col
-1, term
.row
-1); 
2392         case 'K': /* EL -- Clear line */ 
2393                 switch (csiescseq
.arg
[0]) { 
2395                         tclearregion(term
.c
.x
, term
.c
.y
, term
.col
-1, 
2399                         tclearregion(0, term
.c
.y
, term
.c
.x
, term
.c
.y
); 
2402                         tclearregion(0, term
.c
.y
, term
.col
-1, term
.c
.y
); 
2406         case 'S': /* SU -- Scroll <n> line up */ 
2407                 DEFAULT(csiescseq
.arg
[0], 1); 
2408                 tscrollup(term
.top
, csiescseq
.arg
[0]); 
2410         case 'T': /* SD -- Scroll <n> line down */ 
2411                 DEFAULT(csiescseq
.arg
[0], 1); 
2412                 tscrolldown(term
.top
, csiescseq
.arg
[0]); 
2414         case 'L': /* IL -- Insert <n> blank lines */ 
2415                 DEFAULT(csiescseq
.arg
[0], 1); 
2416                 tinsertblankline(csiescseq
.arg
[0]); 
2418         case 'l': /* RM -- Reset Mode */ 
2419                 tsetmode(csiescseq
.priv
, 0, csiescseq
.arg
, csiescseq
.narg
); 
2421         case 'M': /* DL -- Delete <n> lines */ 
2422                 DEFAULT(csiescseq
.arg
[0], 1); 
2423                 tdeleteline(csiescseq
.arg
[0]); 
2425         case 'X': /* ECH -- Erase <n> char */ 
2426                 DEFAULT(csiescseq
.arg
[0], 1); 
2427                 tclearregion(term
.c
.x
, term
.c
.y
, 
2428                                 term
.c
.x 
+ csiescseq
.arg
[0] - 1, term
.c
.y
); 
2430         case 'P': /* DCH -- Delete <n> char */ 
2431                 DEFAULT(csiescseq
.arg
[0], 1); 
2432                 tdeletechar(csiescseq
.arg
[0]); 
2434         case 'Z': /* CBT -- Cursor Backward Tabulation <n> tab stops */ 
2435                 DEFAULT(csiescseq
.arg
[0], 1); 
2436                 tputtab(-csiescseq
.arg
[0]); 
2438         case 'd': /* VPA -- Move to <row> */ 
2439                 DEFAULT(csiescseq
.arg
[0], 1); 
2440                 tmoveato(term
.c
.x
, csiescseq
.arg
[0]-1); 
2442         case 'h': /* SM -- Set terminal mode */ 
2443                 tsetmode(csiescseq
.priv
, 1, csiescseq
.arg
, csiescseq
.narg
); 
2445         case 'm': /* SGR -- Terminal attribute (color) */ 
2446                 tsetattr(csiescseq
.arg
, csiescseq
.narg
); 
2448         case 'n': /* DSR – Device Status Report (cursor position) */ 
2449                 if (csiescseq
.arg
[0] == 6) { 
2450                         len 
= snprintf(buf
, sizeof(buf
),"\033[%i;%iR", 
2451                                         term
.c
.y
+1, term
.c
.x
+1); 
2455         case 'r': /* DECSTBM -- Set Scrolling Region */ 
2456                 if (csiescseq
.priv
) { 
2459                         DEFAULT(csiescseq
.arg
[0], 1); 
2460                         DEFAULT(csiescseq
.arg
[1], term
.row
); 
2461                         tsetscroll(csiescseq
.arg
[0]-1, csiescseq
.arg
[1]-1); 
2465         case 's': /* DECSC -- Save cursor position (ANSI.SYS) */ 
2466                 tcursor(CURSOR_SAVE
); 
2468         case 'u': /* DECRC -- Restore cursor position (ANSI.SYS) */ 
2469                 tcursor(CURSOR_LOAD
); 
2472                 switch (csiescseq
.mode
[1]) { 
2473                 case 'q': /* DECSCUSR -- Set Cursor Style */ 
2474                         DEFAULT(csiescseq
.arg
[0], 1); 
2475                         if (!BETWEEN(csiescseq
.arg
[0], 0, 6)) { 
2478                         xw
.cursor 
= csiescseq
.arg
[0]; 
2494         for (i 
= 0; i 
< csiescseq
.len
; i
++) { 
2495                 c 
= csiescseq
.buf
[i
] & 0xff; 
2498                 } else if (c 
== '\n') { 
2500                 } else if (c 
== '\r') { 
2502                 } else if (c 
== 0x1b) { 
2505                         printf("(%02x)", c
); 
2514         memset(&csiescseq
, 0, sizeof(csiescseq
)); 
2523         term
.esc 
&= ~(ESC_STR_END
|ESC_STR
); 
2525         par 
= (narg 
= strescseq
.narg
) ? atoi(strescseq
.args
[0]) : 0; 
2527         switch (strescseq
.type
) { 
2528         case ']': /* OSC -- Operating System Command */ 
2534                                 xsettitle(strescseq
.args
[1]); 
2536                 case 4: /* color set */ 
2539                         p 
= strescseq
.args
[2]; 
2541                 case 104: /* color reset, here p = NULL */ 
2542                         j 
= (narg 
> 1) ? atoi(strescseq
.args
[1]) : -1; 
2543                         if (xsetcolorname(j
, p
)) { 
2544                                 fprintf(stderr
, "erresc: invalid color %s\n", p
); 
2547                                  * TODO if defaultbg color is changed, borders 
2555         case 'k': /* old title set compatibility */ 
2556                 xsettitle(strescseq
.args
[0]); 
2558         case 'P': /* DCS -- Device Control String */ 
2559                 term
.mode 
|= ESC_DCS
; 
2560         case '_': /* APC -- Application Program Command */ 
2561         case '^': /* PM -- Privacy Message */ 
2565         fprintf(stderr
, "erresc: unknown str "); 
2573         char *p 
= strescseq
.buf
; 
2576         strescseq
.buf
[strescseq
.len
] = '\0'; 
2581         while (strescseq
.narg 
< STR_ARG_SIZ
) { 
2582                 strescseq
.args
[strescseq
.narg
++] = p
; 
2583                 while ((c 
= *p
) != ';' && c 
!= '\0') 
2597         printf("ESC%c", strescseq
.type
); 
2598         for (i 
= 0; i 
< strescseq
.len
; i
++) { 
2599                 c 
= strescseq
.buf
[i
] & 0xff; 
2602                 } else if (isprint(c
)) { 
2604                 } else if (c 
== '\n') { 
2606                 } else if (c 
== '\r') { 
2608                 } else if (c 
== 0x1b) { 
2611                         printf("(%02x)", c
); 
2620         memset(&strescseq
, 0, sizeof(strescseq
)); 
2624 sendbreak(const Arg 
*arg
) 
2626         if (tcsendbreak(cmdfd
, 0)) 
2627                 perror("Error sending break"); 
2631 tprinter(char *s
, size_t len
) 
2633         if (iofd 
!= -1 && xwrite(iofd
, s
, len
) < 0) { 
2634                 fprintf(stderr
, "Error writing in %s:%s\n", 
2635                         opt_io
, strerror(errno
)); 
2642 iso14755(const Arg 
*arg
) 
2644         char cmd
[sizeof(ISO14755CMD
) + NUMMAXLEN(xw
.win
)]; 
2646         char *us
, *e
, codepoint
[9], uc
[UTF_SIZ
]; 
2647         unsigned long utf32
; 
2649         snprintf(cmd
, sizeof(cmd
), ISO14755CMD
, xw
.win
); 
2650         if (!(p 
= popen(cmd
, "r"))) 
2653         us 
= fgets(codepoint
, sizeof(codepoint
), p
); 
2656         if (!us 
|| *us 
== '\0' || *us 
== '-' || strlen(us
) > 7) 
2658         if ((utf32 
= strtoul(us
, &e
, 16)) == ULONG_MAX 
|| 
2659             (*e 
!= '\n' && *e 
!= '\0')) 
2662         ttysend(uc
, utf8encode(utf32
, uc
)); 
2666 toggleprinter(const Arg 
*arg
) 
2668         term
.mode 
^= MODE_PRINT
; 
2672 printscreen(const Arg 
*arg
) 
2678 printsel(const Arg 
*arg
) 
2688         if ((ptr 
= getsel())) { 
2689                 tprinter(ptr
, strlen(ptr
)); 
2700         bp 
= &term
.line
[n
][0]; 
2701         end 
= &bp
[MIN(tlinelen(n
), term
.col
) - 1]; 
2702         if (bp 
!= end 
|| bp
->u 
!= ' ') { 
2703                 for ( ;bp 
<= end
; ++bp
) 
2704                         tprinter(buf
, utf8encode(bp
->u
, buf
)); 
2714         for (i 
= 0; i 
< term
.row
; ++i
) 
2724                 while (x 
< term
.col 
&& n
--) 
2725                         for (++x
; x 
< term
.col 
&& !term
.tabs
[x
]; ++x
) 
2728                 while (x 
> 0 && n
++) 
2729                         for (--x
; x 
> 0 && !term
.tabs
[x
]; --x
) 
2732         term
.c
.x 
= LIMIT(x
, 0, term
.col
-1); 
2738         if (ISCONTROL(u
)) { /* control code */ 
2743                 } else if (u 
!= '\n' && u 
!= '\r' && u 
!= '\t') { 
2752 tdefutf8(char ascii
) 
2755                 term
.mode 
|= MODE_UTF8
; 
2756         else if (ascii 
== '@') 
2757                 term
.mode 
&= ~MODE_UTF8
; 
2761 tdeftran(char ascii
) 
2763         static char cs
[] = "0B"; 
2764         static int vcs
[] = {CS_GRAPHIC0
, CS_USA
}; 
2767         if ((p 
= strchr(cs
, ascii
)) == NULL
) { 
2768                 fprintf(stderr
, "esc unhandled charset: ESC ( %c\n", ascii
); 
2770                 term
.trantbl
[term
.icharset
] = vcs
[p 
- cs
]; 
2779         if (c 
== '8') { /* DEC screen alignment test. */ 
2780                 for (x 
= 0; x 
< term
.col
; ++x
) { 
2781                         for (y 
= 0; y 
< term
.row
; ++y
) 
2782                                 tsetchar('E', &term
.c
.attr
, x
, y
); 
2788 tstrsequence(uchar c
) 
2793         case 0x90:   /* DCS -- Device Control String */ 
2795                 term
.esc 
|= ESC_DCS
; 
2797         case 0x9f:   /* APC -- Application Program Command */ 
2800         case 0x9e:   /* PM -- Privacy Message */ 
2803         case 0x9d:   /* OSC -- Operating System Command */ 
2808         term
.esc 
|= ESC_STR
; 
2812 tcontrolcode(uchar ascii
) 
2819                 tmoveto(term
.c
.x
-1, term
.c
.y
); 
2822                 tmoveto(0, term
.c
.y
); 
2827                 /* go to first col if the mode is set */ 
2828                 tnewline(IS_SET(MODE_CRLF
)); 
2830         case '\a':   /* BEL */ 
2831                 if (term
.esc 
& ESC_STR_END
) { 
2832                         /* backwards compatibility to xterm */ 
2835                         if (!(xw
.state 
& WIN_FOCUSED
)) 
2838                                 XkbBell(xw
.dpy
, xw
.win
, bellvolume
, (Atom
)NULL
); 
2841         case '\033': /* ESC */ 
2843                 term
.esc 
&= ~(ESC_CSI
|ESC_ALTCHARSET
|ESC_TEST
); 
2844                 term
.esc 
|= ESC_START
; 
2846         case '\016': /* SO (LS1 -- Locking shift 1) */ 
2847         case '\017': /* SI (LS0 -- Locking shift 0) */ 
2848                 term
.charset 
= 1 - (ascii 
- '\016'); 
2850         case '\032': /* SUB */ 
2851                 tsetchar('?', &term
.c
.attr
, term
.c
.x
, term
.c
.y
); 
2852         case '\030': /* CAN */ 
2855         case '\005': /* ENQ (IGNORED) */ 
2856         case '\000': /* NUL (IGNORED) */ 
2857         case '\021': /* XON (IGNORED) */ 
2858         case '\023': /* XOFF (IGNORED) */ 
2859         case 0177:   /* DEL (IGNORED) */ 
2861         case 0x80:   /* TODO: PAD */ 
2862         case 0x81:   /* TODO: HOP */ 
2863         case 0x82:   /* TODO: BPH */ 
2864         case 0x83:   /* TODO: NBH */ 
2865         case 0x84:   /* TODO: IND */ 
2867         case 0x85:   /* NEL -- Next line */ 
2868                 tnewline(1); /* always go to first col */ 
2870         case 0x86:   /* TODO: SSA */ 
2871         case 0x87:   /* TODO: ESA */ 
2873         case 0x88:   /* HTS -- Horizontal tab stop */ 
2874                 term
.tabs
[term
.c
.x
] = 1; 
2876         case 0x89:   /* TODO: HTJ */ 
2877         case 0x8a:   /* TODO: VTS */ 
2878         case 0x8b:   /* TODO: PLD */ 
2879         case 0x8c:   /* TODO: PLU */ 
2880         case 0x8d:   /* TODO: RI */ 
2881         case 0x8e:   /* TODO: SS2 */ 
2882         case 0x8f:   /* TODO: SS3 */ 
2883         case 0x91:   /* TODO: PU1 */ 
2884         case 0x92:   /* TODO: PU2 */ 
2885         case 0x93:   /* TODO: STS */ 
2886         case 0x94:   /* TODO: CCH */ 
2887         case 0x95:   /* TODO: MW */ 
2888         case 0x96:   /* TODO: SPA */ 
2889         case 0x97:   /* TODO: EPA */ 
2890         case 0x98:   /* TODO: SOS */ 
2891         case 0x99:   /* TODO: SGCI */ 
2893         case 0x9a:   /* DECID -- Identify Terminal */ 
2894                 ttywrite(vtiden
, sizeof(vtiden
) - 1); 
2896         case 0x9b:   /* TODO: CSI */ 
2897         case 0x9c:   /* TODO: ST */ 
2899         case 0x90:   /* DCS -- Device Control String */ 
2900         case 0x9d:   /* OSC -- Operating System Command */ 
2901         case 0x9e:   /* PM -- Privacy Message */ 
2902         case 0x9f:   /* APC -- Application Program Command */ 
2903                 tstrsequence(ascii
); 
2906         /* only CAN, SUB, \a and C1 chars interrupt a sequence */ 
2907         term
.esc 
&= ~(ESC_STR_END
|ESC_STR
); 
2911  * returns 1 when the sequence is finished and it hasn't to read 
2912  * more characters for this sequence, otherwise 0 
2915 eschandle(uchar ascii
) 
2919                 term
.esc 
|= ESC_CSI
; 
2922                 term
.esc 
|= ESC_TEST
; 
2925                 term
.esc 
|= ESC_UTF8
; 
2927         case 'P': /* DCS -- Device Control String */ 
2928         case '_': /* APC -- Application Program Command */ 
2929         case '^': /* PM -- Privacy Message */ 
2930         case ']': /* OSC -- Operating System Command */ 
2931         case 'k': /* old title set compatibility */ 
2932                 tstrsequence(ascii
); 
2934         case 'n': /* LS2 -- Locking shift 2 */ 
2935         case 'o': /* LS3 -- Locking shift 3 */ 
2936                 term
.charset 
= 2 + (ascii 
- 'n'); 
2938         case '(': /* GZD4 -- set primary charset G0 */ 
2939         case ')': /* G1D4 -- set secondary charset G1 */ 
2940         case '*': /* G2D4 -- set tertiary charset G2 */ 
2941         case '+': /* G3D4 -- set quaternary charset G3 */ 
2942                 term
.icharset 
= ascii 
- '('; 
2943                 term
.esc 
|= ESC_ALTCHARSET
; 
2945         case 'D': /* IND -- Linefeed */ 
2946                 if (term
.c
.y 
== term
.bot
) { 
2947                         tscrollup(term
.top
, 1); 
2949                         tmoveto(term
.c
.x
, term
.c
.y
+1); 
2952         case 'E': /* NEL -- Next line */ 
2953                 tnewline(1); /* always go to first col */ 
2955         case 'H': /* HTS -- Horizontal tab stop */ 
2956                 term
.tabs
[term
.c
.x
] = 1; 
2958         case 'M': /* RI -- Reverse index */ 
2959                 if (term
.c
.y 
== term
.top
) { 
2960                         tscrolldown(term
.top
, 1); 
2962                         tmoveto(term
.c
.x
, term
.c
.y
-1); 
2965         case 'Z': /* DECID -- Identify Terminal */ 
2966                 ttywrite(vtiden
, sizeof(vtiden
) - 1); 
2968         case 'c': /* RIS -- Reset to inital state */ 
2973         case '=': /* DECPAM -- Application keypad */ 
2974                 term
.mode 
|= MODE_APPKEYPAD
; 
2976         case '>': /* DECPNM -- Normal keypad */ 
2977                 term
.mode 
&= ~MODE_APPKEYPAD
; 
2979         case '7': /* DECSC -- Save Cursor */ 
2980                 tcursor(CURSOR_SAVE
); 
2982         case '8': /* DECRC -- Restore Cursor */ 
2983                 tcursor(CURSOR_LOAD
); 
2985         case '\\': /* ST -- String Terminator */ 
2986                 if (term
.esc 
& ESC_STR_END
) 
2990                 fprintf(stderr
, "erresc: unknown sequence ESC 0x%02X '%c'\n", 
2991                         (uchar
) ascii
, isprint(ascii
)? ascii
:'.'); 
3005         control 
= ISCONTROL(u
); 
3006         if (!IS_SET(MODE_UTF8
) && !IS_SET(MODE_SIXEL
)) { 
3010                 len 
= utf8encode(u
, c
); 
3011                 if (!control 
&& (width 
= wcwidth(u
)) == -1) { 
3012                         memcpy(c
, "\357\277\275", 4); /* UTF_INVALID */ 
3017         if (IS_SET(MODE_PRINT
)) 
3021          * STR sequence must be checked before anything else 
3022          * because it uses all following characters until it 
3023          * receives a ESC, a SUB, a ST or any other C1 control 
3026         if (term
.esc 
& ESC_STR
) { 
3027                 if (u 
== '\a' || u 
== 030 || u 
== 032 || u 
== 033 || 
3029                         term
.esc 
&= ~(ESC_START
|ESC_STR
|ESC_DCS
); 
3030                         if (IS_SET(MODE_SIXEL
)) { 
3031                                 /* TODO: render sixel */; 
3032                                 term
.mode 
&= ~MODE_SIXEL
; 
3035                         term
.esc 
|= ESC_STR_END
; 
3036                         goto check_control_code
; 
3040                 if (IS_SET(MODE_SIXEL
)) { 
3041                         /* TODO: implement sixel mode */ 
3044                 if (term
.esc
&ESC_DCS 
&& strescseq
.len 
== 0 && u 
== 'q') 
3045                         term
.mode 
|= MODE_SIXEL
; 
3047                 if (strescseq
.len
+len 
>= sizeof(strescseq
.buf
)-1) { 
3049                          * Here is a bug in terminals. If the user never sends 
3050                          * some code to stop the str or esc command, then st 
3051                          * will stop responding. But this is better than 
3052                          * silently failing with unknown characters. At least 
3053                          * then users will report back. 
3055                          * In the case users ever get fixed, here is the code: 
3064                 memmove(&strescseq
.buf
[strescseq
.len
], c
, len
); 
3065                 strescseq
.len 
+= len
; 
3071          * Actions of control codes must be performed as soon they arrive 
3072          * because they can be embedded inside a control sequence, and 
3073          * they must not cause conflicts with sequences. 
3078                  * control codes are not shown ever 
3081         } else if (term
.esc 
& ESC_START
) { 
3082                 if (term
.esc 
& ESC_CSI
) { 
3083                         csiescseq
.buf
[csiescseq
.len
++] = u
; 
3084                         if (BETWEEN(u
, 0x40, 0x7E) 
3085                                         || csiescseq
.len 
>= \
 
3086                                         sizeof(csiescseq
.buf
)-1) { 
3092                 } else if (term
.esc 
& ESC_UTF8
) { 
3094                 } else if (term
.esc 
& ESC_ALTCHARSET
) { 
3096                 } else if (term
.esc 
& ESC_TEST
) { 
3101                         /* sequence already finished */ 
3105                  * All characters which form part of a sequence are not 
3110         if (sel
.ob
.x 
!= -1 && BETWEEN(term
.c
.y
, sel
.ob
.y
, sel
.oe
.y
)) 
3113         gp 
= &term
.line
[term
.c
.y
][term
.c
.x
]; 
3114         if (IS_SET(MODE_WRAP
) && (term
.c
.state 
& CURSOR_WRAPNEXT
)) { 
3115                 gp
->mode 
|= ATTR_WRAP
; 
3117                 gp 
= &term
.line
[term
.c
.y
][term
.c
.x
]; 
3120         if (IS_SET(MODE_INSERT
) && term
.c
.x
+width 
< term
.col
) 
3121                 memmove(gp
+width
, gp
, (term
.col 
- term
.c
.x 
- width
) * sizeof(Glyph
)); 
3123         if (term
.c
.x
+width 
> term
.col
) { 
3125                 gp 
= &term
.line
[term
.c
.y
][term
.c
.x
]; 
3128         tsetchar(u
, &term
.c
.attr
, term
.c
.x
, term
.c
.y
); 
3131                 gp
->mode 
|= ATTR_WIDE
; 
3132                 if (term
.c
.x
+1 < term
.col
) { 
3134                         gp
[1].mode 
= ATTR_WDUMMY
; 
3137         if (term
.c
.x
+width 
< term
.col
) { 
3138                 tmoveto(term
.c
.x
+width
, term
.c
.y
); 
3140                 term
.c
.state 
|= CURSOR_WRAPNEXT
; 
3145 tresize(int col
, int row
) 
3148         int minrow 
= MIN(row
, term
.row
); 
3149         int mincol 
= MIN(col
, term
.col
); 
3153         if (col 
< 1 || row 
< 1) { 
3155                         "tresize: error resizing to %dx%d\n", col
, row
); 
3160          * slide screen to keep cursor where we expect it - 
3161          * tscrollup would work here, but we can optimize to 
3162          * memmove because we're freeing the earlier lines 
3164         for (i 
= 0; i 
<= term
.c
.y 
- row
; i
++) { 
3168         /* ensure that both src and dst are not NULL */ 
3170                 memmove(term
.line
, term
.line 
+ i
, row 
* sizeof(Line
)); 
3171                 memmove(term
.alt
, term
.alt 
+ i
, row 
* sizeof(Line
)); 
3173         for (i 
+= row
; i 
< term
.row
; i
++) { 
3178         /* resize to new width */ 
3179         term
.specbuf 
= xrealloc(term
.specbuf
, col 
* sizeof(XftGlyphFontSpec
)); 
3181         /* resize to new height */ 
3182         term
.line 
= xrealloc(term
.line
, row 
* sizeof(Line
)); 
3183         term
.alt  
= xrealloc(term
.alt
,  row 
* sizeof(Line
)); 
3184         term
.dirty 
= xrealloc(term
.dirty
, row 
* sizeof(*term
.dirty
)); 
3185         term
.tabs 
= xrealloc(term
.tabs
, col 
* sizeof(*term
.tabs
)); 
3187         /* resize each row to new width, zero-pad if needed */ 
3188         for (i 
= 0; i 
< minrow
; i
++) { 
3189                 term
.line
[i
] = xrealloc(term
.line
[i
], col 
* sizeof(Glyph
)); 
3190                 term
.alt
[i
]  = xrealloc(term
.alt
[i
],  col 
* sizeof(Glyph
)); 
3193         /* allocate any new rows */ 
3194         for (/* i == minrow */; i 
< row
; i
++) { 
3195                 term
.line
[i
] = xmalloc(col 
* sizeof(Glyph
)); 
3196                 term
.alt
[i
] = xmalloc(col 
* sizeof(Glyph
)); 
3198         if (col 
> term
.col
) { 
3199                 bp 
= term
.tabs 
+ term
.col
; 
3201                 memset(bp
, 0, sizeof(*term
.tabs
) * (col 
- term
.col
)); 
3202                 while (--bp 
> term
.tabs 
&& !*bp
) 
3204                 for (bp 
+= tabspaces
; bp 
< term
.tabs 
+ col
; bp 
+= tabspaces
) 
3207         /* update terminal size */ 
3210         /* reset scrolling region */ 
3211         tsetscroll(0, row
-1); 
3212         /* make use of the LIMIT in tmoveto */ 
3213         tmoveto(term
.c
.x
, term
.c
.y
); 
3214         /* Clearing both screens (it makes dirty all lines) */ 
3216         for (i 
= 0; i 
< 2; i
++) { 
3217                 if (mincol 
< col 
&& 0 < minrow
) { 
3218                         tclearregion(mincol
, 0, col 
- 1, minrow 
- 1); 
3220                 if (0 < col 
&& minrow 
< row
) { 
3221                         tclearregion(0, minrow
, col 
- 1, row 
- 1); 
3224                 tcursor(CURSOR_LOAD
); 
3230 xresize(int col
, int row
) 
3232         xw
.tw 
= MAX(1, col 
* xw
.cw
); 
3233         xw
.th 
= MAX(1, row 
* xw
.ch
); 
3235         XFreePixmap(xw
.dpy
, xw
.buf
); 
3236         xw
.buf 
= XCreatePixmap(xw
.dpy
, xw
.win
, xw
.w
, xw
.h
, 
3237                         DefaultDepth(xw
.dpy
, xw
.scr
)); 
3238         XftDrawChange(xw
.draw
, xw
.buf
); 
3239         xclear(0, 0, xw
.w
, xw
.h
); 
3243 sixd_to_16bit(int x
) 
3245         return x 
== 0 ? 0 : 0x3737 + 0x2828 * x
; 
3249 xloadcolor(int i
, const char *name
, Color 
*ncolor
) 
3251         XRenderColor color 
= { .alpha 
= 0xffff }; 
3254                 if (BETWEEN(i
, 16, 255)) { /* 256 color */ 
3255                         if (i 
< 6*6*6+16) { /* same colors as xterm */ 
3256                                 color
.red   
= sixd_to_16bit( ((i
-16)/36)%6 ); 
3257                                 color
.green 
= sixd_to_16bit( ((i
-16)/6) %6 ); 
3258                                 color
.blue  
= sixd_to_16bit( ((i
-16)/1) %6 ); 
3259                         } else { /* greyscale */ 
3260                                 color
.red 
= 0x0808 + 0x0a0a * (i 
- (6*6*6+16)); 
3261                                 color
.green 
= color
.blue 
= color
.red
; 
3263                         return XftColorAllocValue(xw
.dpy
, xw
.vis
, 
3264                                                   xw
.cmap
, &color
, ncolor
); 
3266                         name 
= colorname
[i
]; 
3269         return XftColorAllocName(xw
.dpy
, xw
.vis
, xw
.cmap
, name
, ncolor
); 
3280                 for (cp 
= dc
.col
; cp 
< &dc
.col
[LEN(dc
.col
)]; ++cp
) 
3281                         XftColorFree(xw
.dpy
, xw
.vis
, xw
.cmap
, cp
); 
3284         for (i 
= 0; i 
< LEN(dc
.col
); i
++) 
3285                 if (!xloadcolor(i
, NULL
, &dc
.col
[i
])) { 
3287                                 die("Could not allocate color '%s'\n", colorname
[i
]); 
3289                                 die("Could not allocate color %d\n", i
); 
3295 xsetcolorname(int x
, const char *name
) 
3299         if (!BETWEEN(x
, 0, LEN(dc
.col
))) 
3303         if (!xloadcolor(x
, name
, &ncolor
)) 
3306         XftColorFree(xw
.dpy
, xw
.vis
, xw
.cmap
, &dc
.col
[x
]); 
3313  * Absolute coordinates. 
3316 xclear(int x1
, int y1
, int x2
, int y2
) 
3318         XftDrawRect(xw
.draw
, 
3319                         &dc
.col
[IS_SET(MODE_REVERSE
)? defaultfg 
: defaultbg
], 
3320                         x1
, y1
, x2
-x1
, y2
-y1
); 
3326         XClassHint 
class = {opt_name 
? opt_name 
: termname
, 
3327                             opt_class 
? opt_class 
: termname
}; 
3328         XWMHints wm 
= {.flags 
= InputHint
, .input 
= 1}; 
3329         XSizeHints 
*sizeh 
= NULL
; 
3331         sizeh 
= XAllocSizeHints(); 
3333         sizeh
->flags 
= PSize 
| PResizeInc 
| PBaseSize
; 
3334         sizeh
->height 
= xw
.h
; 
3335         sizeh
->width 
= xw
.w
; 
3336         sizeh
->height_inc 
= xw
.ch
; 
3337         sizeh
->width_inc 
= xw
.cw
; 
3338         sizeh
->base_height 
= 2 * borderpx
; 
3339         sizeh
->base_width 
= 2 * borderpx
; 
3341                 sizeh
->flags 
|= PMaxSize 
| PMinSize
; 
3342                 sizeh
->min_width 
= sizeh
->max_width 
= xw
.w
; 
3343                 sizeh
->min_height 
= sizeh
->max_height 
= xw
.h
; 
3345         if (xw
.gm 
& (XValue
|YValue
)) { 
3346                 sizeh
->flags 
|= USPosition 
| PWinGravity
; 
3349                 sizeh
->win_gravity 
= xgeommasktogravity(xw
.gm
); 
3352         XSetWMProperties(xw
.dpy
, xw
.win
, NULL
, NULL
, NULL
, 0, sizeh
, &wm
, 
3358 xgeommasktogravity(int mask
) 
3360         switch (mask 
& (XNegative
|YNegative
)) { 
3362                 return NorthWestGravity
; 
3364                 return NorthEastGravity
; 
3366                 return SouthWestGravity
; 
3369         return SouthEastGravity
; 
3373 xloadfont(Font 
*f
, FcPattern 
*pattern
) 
3378         int wantattr
, haveattr
; 
3380         match 
= XftFontMatch(xw
.dpy
, xw
.scr
, pattern
, &result
); 
3384         if (!(f
->match 
= XftFontOpenPattern(xw
.dpy
, match
))) { 
3385                 FcPatternDestroy(match
); 
3389         if ((XftPatternGetInteger(pattern
, "slant", 0, &wantattr
) == 
3392                  * Check if xft was unable to find a font with the appropriate 
3393                  * slant but gave us one anyway. Try to mitigate. 
3395                 if ((XftPatternGetInteger(f
->match
->pattern
, "slant", 0, 
3396                     &haveattr
) != XftResultMatch
) || haveattr 
< wantattr
) { 
3398                         fputs("st: font slant does not match\n", stderr
); 
3402         if ((XftPatternGetInteger(pattern
, "weight", 0, &wantattr
) == 
3404                 if ((XftPatternGetInteger(f
->match
->pattern
, "weight", 0, 
3405                     &haveattr
) != XftResultMatch
) || haveattr 
!= wantattr
) { 
3407                         fputs("st: font weight does not match\n", stderr
); 
3411         XftTextExtentsUtf8(xw
.dpy
, f
->match
, 
3412                 (const FcChar8 
*) ascii_printable
, 
3413                 strlen(ascii_printable
), &extents
); 
3416         f
->pattern 
= FcPatternDuplicate(pattern
); 
3418         f
->ascent 
= f
->match
->ascent
; 
3419         f
->descent 
= f
->match
->descent
; 
3421         f
->rbearing 
= f
->match
->max_advance_width
; 
3423         f
->height 
= f
->ascent 
+ f
->descent
; 
3424         f
->width 
= DIVCEIL(extents
.xOff
, strlen(ascii_printable
)); 
3430 xloadfonts(char *fontstr
, double fontsize
) 
3436         if (fontstr
[0] == '-') { 
3437                 pattern 
= XftXlfdParse(fontstr
, False
, False
); 
3439                 pattern 
= FcNameParse((FcChar8 
*)fontstr
); 
3443                 die("st: can't open font %s\n", fontstr
); 
3446                 FcPatternDel(pattern
, FC_PIXEL_SIZE
); 
3447                 FcPatternDel(pattern
, FC_SIZE
); 
3448                 FcPatternAddDouble(pattern
, FC_PIXEL_SIZE
, (double)fontsize
); 
3449                 usedfontsize 
= fontsize
; 
3451                 if (FcPatternGetDouble(pattern
, FC_PIXEL_SIZE
, 0, &fontval
) == 
3453                         usedfontsize 
= fontval
; 
3454                 } else if (FcPatternGetDouble(pattern
, FC_SIZE
, 0, &fontval
) == 
3459                          * Default font size is 12, if none given. This is to 
3460                          * have a known usedfontsize value. 
3462                         FcPatternAddDouble(pattern
, FC_PIXEL_SIZE
, 12); 
3465                 defaultfontsize 
= usedfontsize
; 
3468         if (xloadfont(&dc
.font
, pattern
)) 
3469                 die("st: can't open font %s\n", fontstr
); 
3471         if (usedfontsize 
< 0) { 
3472                 FcPatternGetDouble(dc
.font
.match
->pattern
, 
3473                                    FC_PIXEL_SIZE
, 0, &fontval
); 
3474                 usedfontsize 
= fontval
; 
3476                         defaultfontsize 
= fontval
; 
3479         /* Setting character width and height. */ 
3480         xw
.cw 
= ceilf(dc
.font
.width 
* cwscale
); 
3481         xw
.ch 
= ceilf(dc
.font
.height 
* chscale
); 
3483         FcPatternDel(pattern
, FC_SLANT
); 
3484         FcPatternAddInteger(pattern
, FC_SLANT
, FC_SLANT_ITALIC
); 
3485         if (xloadfont(&dc
.ifont
, pattern
)) 
3486                 die("st: can't open font %s\n", fontstr
); 
3488         FcPatternDel(pattern
, FC_WEIGHT
); 
3489         FcPatternAddInteger(pattern
, FC_WEIGHT
, FC_WEIGHT_BOLD
); 
3490         if (xloadfont(&dc
.ibfont
, pattern
)) 
3491                 die("st: can't open font %s\n", fontstr
); 
3493         FcPatternDel(pattern
, FC_SLANT
); 
3494         FcPatternAddInteger(pattern
, FC_SLANT
, FC_SLANT_ROMAN
); 
3495         if (xloadfont(&dc
.bfont
, pattern
)) 
3496                 die("st: can't open font %s\n", fontstr
); 
3498         FcPatternDestroy(pattern
); 
3502 xunloadfont(Font 
*f
) 
3504         XftFontClose(xw
.dpy
, f
->match
); 
3505         FcPatternDestroy(f
->pattern
); 
3507                 FcFontSetDestroy(f
->set
); 
3513         /* Free the loaded fonts in the font cache.  */ 
3515                 XftFontClose(xw
.dpy
, frc
[--frclen
].font
); 
3517         xunloadfont(&dc
.font
); 
3518         xunloadfont(&dc
.bfont
); 
3519         xunloadfont(&dc
.ifont
); 
3520         xunloadfont(&dc
.ibfont
); 
3524 xzoom(const Arg 
*arg
) 
3528         larg
.f 
= usedfontsize 
+ arg
->f
; 
3533 xzoomabs(const Arg 
*arg
) 
3536         xloadfonts(usedfont
, arg
->f
); 
3544 xzoomreset(const Arg 
*arg
) 
3548         if (defaultfontsize 
> 0) { 
3549                 larg
.f 
= defaultfontsize
; 
3560         pid_t thispid 
= getpid(); 
3561         XColor xmousefg
, xmousebg
; 
3563         if (!(xw
.dpy 
= XOpenDisplay(NULL
))) 
3564                 die("Can't open display\n"); 
3565         xw
.scr 
= XDefaultScreen(xw
.dpy
); 
3566         xw
.vis 
= XDefaultVisual(xw
.dpy
, xw
.scr
); 
3570                 die("Could not init fontconfig.\n"); 
3572         usedfont 
= (opt_font 
== NULL
)? font 
: opt_font
; 
3573         xloadfonts(usedfont
, 0); 
3576         xw
.cmap 
= XDefaultColormap(xw
.dpy
, xw
.scr
); 
3579         /* adjust fixed window geometry */ 
3580         xw
.w 
= 2 * borderpx 
+ term
.col 
* xw
.cw
; 
3581         xw
.h 
= 2 * borderpx 
+ term
.row 
* xw
.ch
; 
3582         if (xw
.gm 
& XNegative
) 
3583                 xw
.l 
+= DisplayWidth(xw
.dpy
, xw
.scr
) - xw
.w 
- 2; 
3584         if (xw
.gm 
& YNegative
) 
3585                 xw
.t 
+= DisplayHeight(xw
.dpy
, xw
.scr
) - xw
.h 
- 2; 
3588         xw
.attrs
.background_pixel 
= dc
.col
[defaultbg
].pixel
; 
3589         xw
.attrs
.border_pixel 
= dc
.col
[defaultbg
].pixel
; 
3590         xw
.attrs
.bit_gravity 
= NorthWestGravity
; 
3591         xw
.attrs
.event_mask 
= FocusChangeMask 
| KeyPressMask
 
3592                 | ExposureMask 
| VisibilityChangeMask 
| StructureNotifyMask
 
3593                 | ButtonMotionMask 
| ButtonPressMask 
| ButtonReleaseMask
; 
3594         xw
.attrs
.colormap 
= xw
.cmap
; 
3596         if (!(opt_embed 
&& (parent 
= strtol(opt_embed
, NULL
, 0)))) 
3597                 parent 
= XRootWindow(xw
.dpy
, xw
.scr
); 
3598         xw
.win 
= XCreateWindow(xw
.dpy
, parent
, xw
.l
, xw
.t
, 
3599                         xw
.w
, xw
.h
, 0, XDefaultDepth(xw
.dpy
, xw
.scr
), InputOutput
, 
3600                         xw
.vis
, CWBackPixel 
| CWBorderPixel 
| CWBitGravity
 
3601                         | CWEventMask 
| CWColormap
, &xw
.attrs
); 
3603         memset(&gcvalues
, 0, sizeof(gcvalues
)); 
3604         gcvalues
.graphics_exposures 
= False
; 
3605         dc
.gc 
= XCreateGC(xw
.dpy
, parent
, GCGraphicsExposures
, 
3607         xw
.buf 
= XCreatePixmap(xw
.dpy
, xw
.win
, xw
.w
, xw
.h
, 
3608                         DefaultDepth(xw
.dpy
, xw
.scr
)); 
3609         XSetForeground(xw
.dpy
, dc
.gc
, dc
.col
[defaultbg
].pixel
); 
3610         XFillRectangle(xw
.dpy
, xw
.buf
, dc
.gc
, 0, 0, xw
.w
, xw
.h
); 
3612         /* Xft rendering context */ 
3613         xw
.draw 
= XftDrawCreate(xw
.dpy
, xw
.buf
, xw
.vis
, xw
.cmap
); 
3616         if ((xw
.xim 
= XOpenIM(xw
.dpy
, NULL
, NULL
, NULL
)) == NULL
) { 
3617                 XSetLocaleModifiers("@im=local"); 
3618                 if ((xw
.xim 
=  XOpenIM(xw
.dpy
, NULL
, NULL
, NULL
)) == NULL
) { 
3619                         XSetLocaleModifiers("@im="); 
3620                         if ((xw
.xim 
= XOpenIM(xw
.dpy
, 
3621                                         NULL
, NULL
, NULL
)) == NULL
) { 
3622                                 die("XOpenIM failed. Could not open input" 
3627         xw
.xic 
= XCreateIC(xw
.xim
, XNInputStyle
, XIMPreeditNothing
 
3628                                            | XIMStatusNothing
, XNClientWindow
, xw
.win
, 
3629                                            XNFocusWindow
, xw
.win
, NULL
); 
3631                 die("XCreateIC failed. Could not obtain input method.\n"); 
3633         /* white cursor, black outline */ 
3634         cursor 
= XCreateFontCursor(xw
.dpy
, mouseshape
); 
3635         XDefineCursor(xw
.dpy
, xw
.win
, cursor
); 
3637         if (XParseColor(xw
.dpy
, xw
.cmap
, colorname
[mousefg
], &xmousefg
) == 0) { 
3638                 xmousefg
.red   
= 0xffff; 
3639                 xmousefg
.green 
= 0xffff; 
3640                 xmousefg
.blue  
= 0xffff; 
3643         if (XParseColor(xw
.dpy
, xw
.cmap
, colorname
[mousebg
], &xmousebg
) == 0) { 
3644                 xmousebg
.red   
= 0x0000; 
3645                 xmousebg
.green 
= 0x0000; 
3646                 xmousebg
.blue  
= 0x0000; 
3649         XRecolorCursor(xw
.dpy
, cursor
, &xmousefg
, &xmousebg
); 
3651         xw
.xembed 
= XInternAtom(xw
.dpy
, "_XEMBED", False
); 
3652         xw
.wmdeletewin 
= XInternAtom(xw
.dpy
, "WM_DELETE_WINDOW", False
); 
3653         xw
.netwmname 
= XInternAtom(xw
.dpy
, "_NET_WM_NAME", False
); 
3654         XSetWMProtocols(xw
.dpy
, xw
.win
, &xw
.wmdeletewin
, 1); 
3656         xw
.netwmpid 
= XInternAtom(xw
.dpy
, "_NET_WM_PID", False
); 
3657         XChangeProperty(xw
.dpy
, xw
.win
, xw
.netwmpid
, XA_CARDINAL
, 32, 
3658                         PropModeReplace
, (uchar 
*)&thispid
, 1); 
3661         XMapWindow(xw
.dpy
, xw
.win
); 
3663         XSync(xw
.dpy
, False
); 
3667 xmakeglyphfontspecs(XftGlyphFontSpec 
*specs
, const Glyph 
*glyphs
, int len
, int x
, int y
) 
3669         float winx 
= borderpx 
+ x 
* xw
.cw
, winy 
= borderpx 
+ y 
* xw
.ch
, xp
, yp
; 
3670         ushort mode
, prevmode 
= USHRT_MAX
; 
3671         Font 
*font 
= &dc
.font
; 
3672         int frcflags 
= FRC_NORMAL
; 
3673         float runewidth 
= xw
.cw
; 
3677         FcPattern 
*fcpattern
, *fontpattern
; 
3678         FcFontSet 
*fcsets
[] = { NULL 
}; 
3679         FcCharSet 
*fccharset
; 
3680         int i
, f
, numspecs 
= 0; 
3682         for (i 
= 0, xp 
= winx
, yp 
= winy 
+ font
->ascent
; i 
< len
; ++i
) { 
3683                 /* Fetch rune and mode for current glyph. */ 
3685                 mode 
= glyphs
[i
].mode
; 
3687                 /* Skip dummy wide-character spacing. */ 
3688                 if (mode 
== ATTR_WDUMMY
) 
3691                 /* Determine font for glyph if different from previous glyph. */ 
3692                 if (prevmode 
!= mode
) { 
3695                         frcflags 
= FRC_NORMAL
; 
3696                         runewidth 
= xw
.cw 
* ((mode 
& ATTR_WIDE
) ? 2.0f 
: 1.0f
); 
3697                         if ((mode 
& ATTR_ITALIC
) && (mode 
& ATTR_BOLD
)) { 
3699                                 frcflags 
= FRC_ITALICBOLD
; 
3700                         } else if (mode 
& ATTR_ITALIC
) { 
3702                                 frcflags 
= FRC_ITALIC
; 
3703                         } else if (mode 
& ATTR_BOLD
) { 
3705                                 frcflags 
= FRC_BOLD
; 
3707                         yp 
= winy 
+ font
->ascent
; 
3710                 /* Lookup character index with default font. */ 
3711                 glyphidx 
= XftCharIndex(xw
.dpy
, font
->match
, rune
); 
3713                         specs
[numspecs
].font 
= font
->match
; 
3714                         specs
[numspecs
].glyph 
= glyphidx
; 
3715                         specs
[numspecs
].x 
= (short)xp
; 
3716                         specs
[numspecs
].y 
= (short)yp
; 
3722                 /* Fallback on font cache, search the font cache for match. */ 
3723                 for (f 
= 0; f 
< frclen
; f
++) { 
3724                         glyphidx 
= XftCharIndex(xw
.dpy
, frc
[f
].font
, rune
); 
3725                         /* Everything correct. */ 
3726                         if (glyphidx 
&& frc
[f
].flags 
== frcflags
) 
3728                         /* We got a default font for a not found glyph. */ 
3729                         if (!glyphidx 
&& frc
[f
].flags 
== frcflags
 
3730                                         && frc
[f
].unicodep 
== rune
) { 
3735                 /* Nothing was found. Use fontconfig to find matching font. */ 
3738                                 font
->set 
= FcFontSort(0, font
->pattern
, 
3740                         fcsets
[0] = font
->set
; 
3743                          * Nothing was found in the cache. Now use 
3744                          * some dozen of Fontconfig calls to get the 
3745                          * font for one single character. 
3747                          * Xft and fontconfig are design failures. 
3749                         fcpattern 
= FcPatternDuplicate(font
->pattern
); 
3750                         fccharset 
= FcCharSetCreate(); 
3752                         FcCharSetAddChar(fccharset
, rune
); 
3753                         FcPatternAddCharSet(fcpattern
, FC_CHARSET
, 
3755                         FcPatternAddBool(fcpattern
, FC_SCALABLE
, 1); 
3757                         FcConfigSubstitute(0, fcpattern
, 
3759                         FcDefaultSubstitute(fcpattern
); 
3761                         fontpattern 
= FcFontSetMatch(0, fcsets
, 1, 
3765                          * Overwrite or create the new cache entry. 
3767                         if (frclen 
>= LEN(frc
)) { 
3768                                 frclen 
= LEN(frc
) - 1; 
3769                                 XftFontClose(xw
.dpy
, frc
[frclen
].font
); 
3770                                 frc
[frclen
].unicodep 
= 0; 
3773                         frc
[frclen
].font 
= XftFontOpenPattern(xw
.dpy
, 
3775                         frc
[frclen
].flags 
= frcflags
; 
3776                         frc
[frclen
].unicodep 
= rune
; 
3778                         glyphidx 
= XftCharIndex(xw
.dpy
, frc
[frclen
].font
, rune
); 
3783                         FcPatternDestroy(fcpattern
); 
3784                         FcCharSetDestroy(fccharset
); 
3787                 specs
[numspecs
].font 
= frc
[f
].font
; 
3788                 specs
[numspecs
].glyph 
= glyphidx
; 
3789                 specs
[numspecs
].x 
= (short)xp
; 
3790                 specs
[numspecs
].y 
= (short)yp
; 
3799 xdrawglyphfontspecs(const XftGlyphFontSpec 
*specs
, Glyph base
, int len
, int x
, int y
) 
3801         int charlen 
= len 
* ((base
.mode 
& ATTR_WIDE
) ? 2 : 1); 
3802         int winx 
= borderpx 
+ x 
* xw
.cw
, winy 
= borderpx 
+ y 
* xw
.ch
, 
3803             width 
= charlen 
* xw
.cw
; 
3804         Color 
*fg
, *bg
, *temp
, revfg
, revbg
, truefg
, truebg
; 
3805         XRenderColor colfg
, colbg
; 
3808         /* Fallback on color display for attributes not supported by the font */ 
3809         if (base
.mode 
& ATTR_ITALIC 
&& base
.mode 
& ATTR_BOLD
) { 
3810                 if (dc
.ibfont
.badslant 
|| dc
.ibfont
.badweight
) 
3811                         base
.fg 
= defaultattr
; 
3812         } else if ((base
.mode 
& ATTR_ITALIC 
&& dc
.ifont
.badslant
) || 
3813             (base
.mode 
& ATTR_BOLD 
&& dc
.bfont
.badweight
)) { 
3814                 base
.fg 
= defaultattr
; 
3817         if (IS_TRUECOL(base
.fg
)) { 
3818                 colfg
.alpha 
= 0xffff; 
3819                 colfg
.red 
= TRUERED(base
.fg
); 
3820                 colfg
.green 
= TRUEGREEN(base
.fg
); 
3821                 colfg
.blue 
= TRUEBLUE(base
.fg
); 
3822                 XftColorAllocValue(xw
.dpy
, xw
.vis
, xw
.cmap
, &colfg
, &truefg
); 
3825                 fg 
= &dc
.col
[base
.fg
]; 
3828         if (IS_TRUECOL(base
.bg
)) { 
3829                 colbg
.alpha 
= 0xffff; 
3830                 colbg
.green 
= TRUEGREEN(base
.bg
); 
3831                 colbg
.red 
= TRUERED(base
.bg
); 
3832                 colbg
.blue 
= TRUEBLUE(base
.bg
); 
3833                 XftColorAllocValue(xw
.dpy
, xw
.vis
, xw
.cmap
, &colbg
, &truebg
); 
3836                 bg 
= &dc
.col
[base
.bg
]; 
3839         /* Change basic system colors [0-7] to bright system colors [8-15] */ 
3840         if ((base
.mode 
& ATTR_BOLD_FAINT
) == ATTR_BOLD 
&& BETWEEN(base
.fg
, 0, 7)) 
3841                 fg 
= &dc
.col
[base
.fg 
+ 8]; 
3843         if (IS_SET(MODE_REVERSE
)) { 
3844                 if (fg 
== &dc
.col
[defaultfg
]) { 
3845                         fg 
= &dc
.col
[defaultbg
]; 
3847                         colfg
.red 
= ~fg
->color
.red
; 
3848                         colfg
.green 
= ~fg
->color
.green
; 
3849                         colfg
.blue 
= ~fg
->color
.blue
; 
3850                         colfg
.alpha 
= fg
->color
.alpha
; 
3851                         XftColorAllocValue(xw
.dpy
, xw
.vis
, xw
.cmap
, &colfg
, 
3856                 if (bg 
== &dc
.col
[defaultbg
]) { 
3857                         bg 
= &dc
.col
[defaultfg
]; 
3859                         colbg
.red 
= ~bg
->color
.red
; 
3860                         colbg
.green 
= ~bg
->color
.green
; 
3861                         colbg
.blue 
= ~bg
->color
.blue
; 
3862                         colbg
.alpha 
= bg
->color
.alpha
; 
3863                         XftColorAllocValue(xw
.dpy
, xw
.vis
, xw
.cmap
, &colbg
, 
3869         if (base
.mode 
& ATTR_REVERSE
) { 
3875         if ((base
.mode 
& ATTR_BOLD_FAINT
) == ATTR_FAINT
) { 
3876                 colfg
.red 
= fg
->color
.red 
/ 2; 
3877                 colfg
.green 
= fg
->color
.green 
/ 2; 
3878                 colfg
.blue 
= fg
->color
.blue 
/ 2; 
3879                 XftColorAllocValue(xw
.dpy
, xw
.vis
, xw
.cmap
, &colfg
, &revfg
); 
3883         if (base
.mode 
& ATTR_BLINK 
&& term
.mode 
& MODE_BLINK
) 
3886         if (base
.mode 
& ATTR_INVISIBLE
) 
3889         /* Intelligent cleaning up of the borders. */ 
3891                 xclear(0, (y 
== 0)? 0 : winy
, borderpx
, 
3892                         winy 
+ xw
.ch 
+ ((y 
>= term
.row
-1)? xw
.h 
: 0)); 
3894         if (x 
+ charlen 
>= term
.col
) { 
3895                 xclear(winx 
+ width
, (y 
== 0)? 0 : winy
, xw
.w
, 
3896                         ((y 
>= term
.row
-1)? xw
.h 
: (winy 
+ xw
.ch
))); 
3899                 xclear(winx
, 0, winx 
+ width
, borderpx
); 
3900         if (y 
== term
.row
-1) 
3901                 xclear(winx
, winy 
+ xw
.ch
, winx 
+ width
, xw
.h
); 
3903         /* Clean up the region we want to draw to. */ 
3904         XftDrawRect(xw
.draw
, bg
, winx
, winy
, width
, xw
.ch
); 
3906         /* Set the clip region because Xft is sometimes dirty. */ 
3911         XftDrawSetClipRectangles(xw
.draw
, winx
, winy
, &r
, 1); 
3913         /* Render the glyphs. */ 
3914         XftDrawGlyphFontSpec(xw
.draw
, fg
, specs
, len
); 
3916         /* Render underline and strikethrough. */ 
3917         if (base
.mode 
& ATTR_UNDERLINE
) { 
3918                 XftDrawRect(xw
.draw
, fg
, winx
, winy 
+ dc
.font
.ascent 
+ 1, 
3922         if (base
.mode 
& ATTR_STRUCK
) { 
3923                 XftDrawRect(xw
.draw
, fg
, winx
, winy 
+ 2 * dc
.font
.ascent 
/ 3, 
3927         /* Reset clip to none. */ 
3928         XftDrawSetClip(xw
.draw
, 0); 
3932 xdrawglyph(Glyph g
, int x
, int y
) 
3935         XftGlyphFontSpec spec
; 
3937         numspecs 
= xmakeglyphfontspecs(&spec
, &g
, 1, x
, y
); 
3938         xdrawglyphfontspecs(&spec
, g
, numspecs
, x
, y
); 
3944         static int oldx 
= 0, oldy 
= 0; 
3946         Glyph g 
= {' ', ATTR_NULL
, defaultbg
, defaultcs
}, og
; 
3947         int ena_sel 
= sel
.ob
.x 
!= -1 && sel
.alt 
== IS_SET(MODE_ALTSCREEN
); 
3950         LIMIT(oldx
, 0, term
.col
-1); 
3951         LIMIT(oldy
, 0, term
.row
-1); 
3955         /* adjust position if in dummy */ 
3956         if (term
.line
[oldy
][oldx
].mode 
& ATTR_WDUMMY
) 
3958         if (term
.line
[term
.c
.y
][curx
].mode 
& ATTR_WDUMMY
) 
3961         /* remove the old cursor */ 
3962         og 
= term
.line
[oldy
][oldx
]; 
3963         if (ena_sel 
&& selected(oldx
, oldy
)) 
3964                 og
.mode 
^= ATTR_REVERSE
; 
3965         xdrawglyph(og
, oldx
, oldy
); 
3967         g
.u 
= term
.line
[term
.c
.y
][term
.c
.x
].u
; 
3970          * Select the right color for the right mode. 
3972         if (IS_SET(MODE_REVERSE
)) { 
3973                 g
.mode 
|= ATTR_REVERSE
; 
3975                 if (ena_sel 
&& selected(term
.c
.x
, term
.c
.y
)) { 
3976                         drawcol 
= dc
.col
[defaultcs
]; 
3979                         drawcol 
= dc
.col
[defaultrcs
]; 
3983                 if (ena_sel 
&& selected(term
.c
.x
, term
.c
.y
)) { 
3984                         drawcol 
= dc
.col
[defaultrcs
]; 
3988                         drawcol 
= dc
.col
[defaultcs
]; 
3992         if (IS_SET(MODE_HIDE
)) 
3995         /* draw the new one */ 
3996         if (xw
.state 
& WIN_FOCUSED
) { 
3997                 switch (xw
.cursor
) { 
3998                 case 7: /* st extension: snowman */ 
3999                         utf8decode("☃", &g
.u
, UTF_SIZ
); 
4000                 case 0: /* Blinking Block */ 
4001                 case 1: /* Blinking Block (Default) */ 
4002                 case 2: /* Steady Block */ 
4003                         g
.mode 
|= term
.line
[term
.c
.y
][curx
].mode 
& ATTR_WIDE
; 
4004                         xdrawglyph(g
, term
.c
.x
, term
.c
.y
); 
4006                 case 3: /* Blinking Underline */ 
4007                 case 4: /* Steady Underline */ 
4008                         XftDrawRect(xw
.draw
, &drawcol
, 
4009                                         borderpx 
+ curx 
* xw
.cw
, 
4010                                         borderpx 
+ (term
.c
.y 
+ 1) * xw
.ch 
- \
 
4012                                         xw
.cw
, cursorthickness
); 
4014                 case 5: /* Blinking bar */ 
4015                 case 6: /* Steady bar */ 
4016                         XftDrawRect(xw
.draw
, &drawcol
, 
4017                                         borderpx 
+ curx 
* xw
.cw
, 
4018                                         borderpx 
+ term
.c
.y 
* xw
.ch
, 
4019                                         cursorthickness
, xw
.ch
); 
4023                 XftDrawRect(xw
.draw
, &drawcol
, 
4024                                 borderpx 
+ curx 
* xw
.cw
, 
4025                                 borderpx 
+ term
.c
.y 
* xw
.ch
, 
4027                 XftDrawRect(xw
.draw
, &drawcol
, 
4028                                 borderpx 
+ curx 
* xw
.cw
, 
4029                                 borderpx 
+ term
.c
.y 
* xw
.ch
, 
4031                 XftDrawRect(xw
.draw
, &drawcol
, 
4032                                 borderpx 
+ (curx 
+ 1) * xw
.cw 
- 1, 
4033                                 borderpx 
+ term
.c
.y 
* xw
.ch
, 
4035                 XftDrawRect(xw
.draw
, &drawcol
, 
4036                                 borderpx 
+ curx 
* xw
.cw
, 
4037                                 borderpx 
+ (term
.c
.y 
+ 1) * xw
.ch 
- 1, 
4040         oldx 
= curx
, oldy 
= term
.c
.y
; 
4049         Xutf8TextListToTextProperty(xw
.dpy
, &p
, 1, XUTF8StringStyle
, 
4051         XSetWMName(xw
.dpy
, xw
.win
, &prop
); 
4052         XSetTextProperty(xw
.dpy
, xw
.win
, &prop
, xw
.netwmname
); 
4059         xsettitle(opt_title 
? opt_title 
: "st"); 
4072         drawregion(0, 0, term
.col
, term
.row
); 
4073         XCopyArea(xw
.dpy
, xw
.buf
, xw
.win
, dc
.gc
, 0, 0, xw
.w
, 
4075         XSetForeground(xw
.dpy
, dc
.gc
, 
4076                         dc
.col
[IS_SET(MODE_REVERSE
)? 
4077                                 defaultfg 
: defaultbg
].pixel
); 
4081 drawregion(int x1
, int y1
, int x2
, int y2
) 
4083         int i
, x
, y
, ox
, numspecs
; 
4085         XftGlyphFontSpec 
*specs
; 
4086         int ena_sel 
= sel
.ob
.x 
!= -1 && sel
.alt 
== IS_SET(MODE_ALTSCREEN
); 
4088         if (!(xw
.state 
& WIN_VISIBLE
)) 
4091         for (y 
= y1
; y 
< y2
; y
++) { 
4097                 specs 
= term
.specbuf
; 
4098                 numspecs 
= xmakeglyphfontspecs(specs
, &term
.line
[y
][x1
], x2 
- x1
, x1
, y
); 
4101                 for (x 
= x1
; x 
< x2 
&& i 
< numspecs
; x
++) { 
4102                         new = term
.line
[y
][x
]; 
4103                         if (new.mode 
== ATTR_WDUMMY
) 
4105                         if (ena_sel 
&& selected(x
, y
)) 
4106                                 new.mode 
^= ATTR_REVERSE
; 
4107                         if (i 
> 0 && ATTRCMP(base
, new)) { 
4108                                 xdrawglyphfontspecs(specs
, base
, i
, ox
, y
); 
4120                         xdrawglyphfontspecs(specs
, base
, i
, ox
, y
); 
4132 visibility(XEvent 
*ev
) 
4134         XVisibilityEvent 
*e 
= &ev
->xvisibility
; 
4136         MODBIT(xw
.state
, e
->state 
!= VisibilityFullyObscured
, WIN_VISIBLE
); 
4142         xw
.state 
&= ~WIN_VISIBLE
; 
4146 xsetpointermotion(int set
) 
4148         MODBIT(xw
.attrs
.event_mask
, set
, PointerMotionMask
); 
4149         XChangeWindowAttributes(xw
.dpy
, xw
.win
, CWEventMask
, &xw
.attrs
); 
4153 xseturgency(int add
) 
4155         XWMHints 
*h 
= XGetWMHints(xw
.dpy
, xw
.win
); 
4157         MODBIT(h
->flags
, add
, XUrgencyHint
); 
4158         XSetWMHints(xw
.dpy
, xw
.win
, h
); 
4165         XFocusChangeEvent 
*e 
= &ev
->xfocus
; 
4167         if (e
->mode 
== NotifyGrab
) 
4170         if (ev
->type 
== FocusIn
) { 
4171                 XSetICFocus(xw
.xic
); 
4172                 xw
.state 
|= WIN_FOCUSED
; 
4174                 if (IS_SET(MODE_FOCUS
)) 
4175                         ttywrite("\033[I", 3); 
4177                 XUnsetICFocus(xw
.xic
); 
4178                 xw
.state 
&= ~WIN_FOCUSED
; 
4179                 if (IS_SET(MODE_FOCUS
)) 
4180                         ttywrite("\033[O", 3); 
4185 match(uint mask
, uint state
) 
4187         return mask 
== XK_ANY_MOD 
|| mask 
== (state 
& ~ignoremod
); 
4191 numlock(const Arg 
*dummy
) 
4197 kmap(KeySym k
, uint state
) 
4202         /* Check for mapped keys out of X11 function keys. */ 
4203         for (i 
= 0; i 
< LEN(mappedkeys
); i
++) { 
4204                 if (mappedkeys
[i
] == k
) 
4207         if (i 
== LEN(mappedkeys
)) { 
4208                 if ((k 
& 0xFFFF) < 0xFD00) 
4212         for (kp 
= key
; kp 
< key 
+ LEN(key
); kp
++) { 
4216                 if (!match(kp
->mask
, state
)) 
4219                 if (IS_SET(MODE_APPKEYPAD
) ? kp
->appkey 
< 0 : kp
->appkey 
> 0) 
4221                 if (term
.numlock 
&& kp
->appkey 
== 2) 
4224                 if (IS_SET(MODE_APPCURSOR
) ? kp
->appcursor 
< 0 : kp
->appcursor 
> 0) 
4227                 if (IS_SET(MODE_CRLF
) ? kp
->crlf 
< 0 : kp
->crlf 
> 0) 
4239         XKeyEvent 
*e 
= &ev
->xkey
; 
4241         char buf
[32], *customkey
; 
4247         if (IS_SET(MODE_KBDLOCK
)) 
4250         len 
= XmbLookupString(xw
.xic
, e
, buf
, sizeof buf
, &ksym
, &status
); 
4252         for (bp 
= shortcuts
; bp 
< shortcuts 
+ LEN(shortcuts
); bp
++) { 
4253                 if (ksym 
== bp
->keysym 
&& match(bp
->mod
, e
->state
)) { 
4254                         bp
->func(&(bp
->arg
)); 
4259         /* 2. custom keys from config.h */ 
4260         if ((customkey 
= kmap(ksym
, e
->state
))) { 
4261                 ttysend(customkey
, strlen(customkey
)); 
4265         /* 3. composed string from input method */ 
4268         if (len 
== 1 && e
->state 
& Mod1Mask
) { 
4269                 if (IS_SET(MODE_8BIT
)) { 
4272                                 len 
= utf8encode(c
, buf
); 
4289          *  http://standards.freedesktop.org/xembed-spec/xembed-spec-latest.html 
4291         if (e
->xclient
.message_type 
== xw
.xembed 
&& e
->xclient
.format 
== 32) { 
4292                 if (e
->xclient
.data
.l
[1] == XEMBED_FOCUS_IN
) { 
4293                         xw
.state 
|= WIN_FOCUSED
; 
4295                 } else if (e
->xclient
.data
.l
[1] == XEMBED_FOCUS_OUT
) { 
4296                         xw
.state 
&= ~WIN_FOCUSED
; 
4298         } else if (e
->xclient
.data
.l
[0] == xw
.wmdeletewin
) { 
4299                 /* Send SIGHUP to shell */ 
4306 cresize(int width
, int height
) 
4315         col 
= (xw
.w 
- 2 * borderpx
) / xw
.cw
; 
4316         row 
= (xw
.h 
- 2 * borderpx
) / xw
.ch
; 
4325         if (e
->xconfigure
.width 
== xw
.w 
&& e
->xconfigure
.height 
== xw
.h
) 
4328         cresize(e
->xconfigure
.width
, e
->xconfigure
.height
); 
4336         int w 
= xw
.w
, h 
= xw
.h
; 
4338         int xfd 
= XConnectionNumber(xw
.dpy
), xev
, blinkset 
= 0, dodraw 
= 0; 
4339         struct timespec drawtimeout
, *tv 
= NULL
, now
, last
, lastblink
; 
4342         /* Waiting for window mapping */ 
4344                 XNextEvent(xw
.dpy
, &ev
); 
4346                  * This XFilterEvent call is required because of XOpenIM. It 
4347                  * does filter out the key event and some client message for 
4348                  * the input method too. 
4350                 if (XFilterEvent(&ev
, None
)) 
4352                 if (ev
.type 
== ConfigureNotify
) { 
4353                         w 
= ev
.xconfigure
.width
; 
4354                         h 
= ev
.xconfigure
.height
; 
4356         } while (ev
.type 
!= MapNotify
); 
4362         clock_gettime(CLOCK_MONOTONIC
, &last
); 
4365         for (xev 
= actionfps
;;) { 
4367                 FD_SET(cmdfd
, &rfd
); 
4370                 if (pselect(MAX(xfd
, cmdfd
)+1, &rfd
, NULL
, NULL
, tv
, NULL
) < 0) { 
4373                         die("select failed: %s\n", strerror(errno
)); 
4375                 if (FD_ISSET(cmdfd
, &rfd
)) { 
4378                                 blinkset 
= tattrset(ATTR_BLINK
); 
4380                                         MODBIT(term
.mode
, 0, MODE_BLINK
); 
4384                 if (FD_ISSET(xfd
, &rfd
)) 
4387                 clock_gettime(CLOCK_MONOTONIC
, &now
); 
4388                 drawtimeout
.tv_sec 
= 0; 
4389                 drawtimeout
.tv_nsec 
=  (1000 * 1E6
)/ xfps
; 
4393                 if (blinktimeout 
&& TIMEDIFF(now
, lastblink
) > blinktimeout
) { 
4394                         tsetdirtattr(ATTR_BLINK
); 
4395                         term
.mode 
^= MODE_BLINK
; 
4399                 deltatime 
= TIMEDIFF(now
, last
); 
4400                 if (deltatime 
> 1000 / (xev 
? xfps 
: actionfps
)) { 
4406                         while (XPending(xw
.dpy
)) { 
4407                                 XNextEvent(xw
.dpy
, &ev
); 
4408                                 if (XFilterEvent(&ev
, None
)) 
4410                                 if (handler
[ev
.type
]) 
4411                                         (handler
[ev
.type
])(&ev
); 
4417                         if (xev 
&& !FD_ISSET(xfd
, &rfd
)) 
4419                         if (!FD_ISSET(cmdfd
, &rfd
) && !FD_ISSET(xfd
, &rfd
)) { 
4421                                         if (TIMEDIFF(now
, lastblink
) \
 
4423                                                 drawtimeout
.tv_nsec 
= 1000; 
4425                                                 drawtimeout
.tv_nsec 
= (1E6 
* \
 
4430                                         drawtimeout
.tv_sec 
= \
 
4431                                             drawtimeout
.tv_nsec 
/ 1E9
; 
4432                                         drawtimeout
.tv_nsec 
%= (long)1E9
; 
4444         die("usage: %s [-aiv] [-c class] [-f font] [-g geometry]" 
4445             " [-n name] [-o file]\n" 
4446             "          [-T title] [-t title] [-w windowid]" 
4447             " [[-e] command [args ...]]\n" 
4448             "       %s [-aiv] [-c class] [-f font] [-g geometry]" 
4449             " [-n name] [-o file]\n" 
4450             "          [-T title] [-t title] [-w windowid] -l line" 
4451             " [stty_args ...]\n", argv0
, argv0
); 
4455 main(int argc
, char *argv
[]) 
4457         uint cols 
= 80, rows 
= 24; 
4461         xw
.cursor 
= cursorshape
; 
4468                 opt_class 
= EARGF(usage()); 
4475                 opt_font 
= EARGF(usage()); 
4478                 xw
.gm 
= XParseGeometry(EARGF(usage()), 
4479                                 &xw
.l
, &xw
.t
, &cols
, &rows
); 
4485                 opt_io 
= EARGF(usage()); 
4488                 opt_line 
= EARGF(usage()); 
4491                 opt_name 
= EARGF(usage()); 
4495                 opt_title 
= EARGF(usage()); 
4498                 opt_embed 
= EARGF(usage()); 
4501                 die("%s " VERSION 
" (c) 2010-2016 st engineers\n", argv0
); 
4509                 /* eat all remaining arguments */ 
4511                 if (!opt_title 
&& !opt_line
) 
4512                         opt_title 
= basename(xstrdup(argv
[0])); 
4514         setlocale(LC_CTYPE
, ""); 
4515         XSetLocaleModifiers(""); 
4516         tnew(MAX(cols
, 1), MAX(rows
, 1));