From 58ceb6f09e32d9a2a84507471593e228dafaf69c Mon Sep 17 00:00:00 2001 From: Hektor Misplon Date: Thu, 7 May 2020 18:13:56 +0000 Subject: [PATCH] Upgrade to st-8.3 --- .suckless/st/.gitignore | 1 - .suckless/st/FAQ | 195 -- .suckless/st/LEGACY | 17 - .suckless/st/LICENSE | 34 - .suckless/st/Makefile | 58 - .suckless/st/README | 34 - .suckless/st/TODO | 28 - .suckless/st/arg.h | 50 - .suckless/st/config.def.h | 459 --- .suckless/st/config.h | 460 --- .suckless/st/config.mk | 37 - .suckless/st/hb.c | 136 - .suckless/st/hb.h | 7 - .suckless/st/hb.o | Bin 4552 -> 0 bytes .suckless/st/patches/st-anysize-0.8.1.diff | 152 - .../patches/st-ligatures-20200430-0.8.3.diff | 306 -- .suckless/st/patches/st-nordtheme-0.8.2.diff | 100 - .suckless/st/st | Bin 102984 -> 0 bytes .suckless/st/st.1 | 176 -- .suckless/st/st.c | 2600 ----------------- .suckless/st/st.c.orig | 2599 ---------------- .suckless/st/st.c.rej | 12 - .suckless/st/st.h | 125 - .suckless/st/st.info | 222 -- .suckless/st/st.o | Bin 73248 -> 0 bytes .suckless/st/win.h | 39 - .suckless/st/x.c | 2055 ------------- .suckless/st/x.c.orig | 2045 ------------- .suckless/st/x.o | Bin 73808 -> 0 bytes 29 files changed, 11947 deletions(-) delete mode 100644 .suckless/st/.gitignore delete mode 100644 .suckless/st/FAQ delete mode 100644 .suckless/st/LEGACY delete mode 100644 .suckless/st/LICENSE delete mode 100644 .suckless/st/Makefile delete mode 100644 .suckless/st/README delete mode 100644 .suckless/st/TODO delete mode 100644 .suckless/st/arg.h delete mode 100644 .suckless/st/config.def.h delete mode 100644 .suckless/st/config.h delete mode 100644 .suckless/st/config.mk delete mode 100644 .suckless/st/hb.c delete mode 100644 .suckless/st/hb.h delete mode 100644 .suckless/st/hb.o delete mode 100644 .suckless/st/patches/st-anysize-0.8.1.diff delete mode 100644 .suckless/st/patches/st-ligatures-20200430-0.8.3.diff delete mode 100644 .suckless/st/patches/st-nordtheme-0.8.2.diff delete mode 100755 .suckless/st/st delete mode 100644 .suckless/st/st.1 delete mode 100644 .suckless/st/st.c delete mode 100644 .suckless/st/st.c.orig delete mode 100644 .suckless/st/st.c.rej delete mode 100644 .suckless/st/st.h delete mode 100644 .suckless/st/st.info delete mode 100644 .suckless/st/st.o delete mode 100644 .suckless/st/win.h delete mode 100644 .suckless/st/x.c delete mode 100644 .suckless/st/x.c.orig delete mode 100644 .suckless/st/x.o diff --git a/.suckless/st/.gitignore b/.suckless/st/.gitignore deleted file mode 100644 index 645e6a2..0000000 --- a/.suckless/st/.gitignore +++ /dev/null @@ -1 +0,0 @@ -*.old diff --git a/.suckless/st/FAQ b/.suckless/st/FAQ deleted file mode 100644 index 78c769a..0000000 --- a/.suckless/st/FAQ +++ /dev/null @@ -1,195 +0,0 @@ -## Why does st not handle utmp entries? - -Use the excellent tool of [utmp](https://git.suckless.org/utmp/) for this task. - -## Some _random program_ complains that st is unknown/not recognised/unsupported/whatever! - -It means that st doesn’t have any terminfo entry on your system. Chances are -you did not `make install`. If you just want to test it without installing it, -you can manually run `tic -sx st.info`. - -## Nothing works, and nothing is said about an unknown terminal! - -* Some programs just assume they’re running in xterm i.e. they don’t rely on - terminfo. What you see is the current state of the “xterm compliance”. -* Some programs don’t complain about the lacking st description and default to - another terminal. In that case see the question about terminfo. - -## How do I scroll back up? - -Using a terminal multiplexer. - -* `st -e tmux` using C-b [ -* `st -e screen` using C-a ESC - -## Why doesn't the Del key work in some programs? - -Taken from the terminfo manpage: - - If the terminal has a keypad that transmits codes when the keys - are pressed, this information can be given. Note that it is not - possible to handle terminals where the keypad only works in - local (this applies, for example, to the unshifted HP 2621 keys). - If the keypad can be set to transmit or not transmit, give these - codes as smkx and rmkx. Otherwise the keypad is assumed to - always transmit. - -In the st case smkx=E[?1hE= and rmkx=E[?1lE>, so it is mandatory that -applications which want to test against keypad keys send these -sequences. - -But buggy applications (like bash and irssi, for example) don't do this. A fast -solution for them is to use the following command: - - $ printf '\033[?1h\033=' >/dev/tty - -or - $ tput smkx - -In the case of bash, readline is used. Readline has a different note in its -manpage about this issue: - - enable-keypad (Off) - When set to On, readline will try to enable the - application keypad when it is called. Some systems - need this to enable arrow keys. - -Adding this option to your .inputrc will fix the keypad problem for all -applications using readline. - -If you are using zsh, then read the zsh FAQ -: - - It should be noted that the O / [ confusion can occur with other keys - such as Home and End. Some systems let you query the key sequences - sent by these keys from the system's terminal database, terminfo. - Unfortunately, the key sequences given there typically apply to the - mode that is not the one zsh uses by default (it's the "application" - mode rather than the "raw" mode). Explaining the use of terminfo is - outside of the scope of this FAQ, but if you wish to use the key - sequences given there you can tell the line editor to turn on - "application" mode when it starts and turn it off when it stops: - - function zle-line-init () { echoti smkx } - function zle-line-finish () { echoti rmkx } - zle -N zle-line-init - zle -N zle-line-finish - -Putting these lines into your .zshrc will fix the problems. - -## How can I use meta in 8bit mode? - -St supports meta in 8bit mode, but the default terminfo entry doesn't -use this capability. If you want it, you have to use the 'st-meta' value -in TERM. - -## I cannot compile st in OpenBSD - -OpenBSD lacks librt, despite it being mandatory in POSIX -. -If you want to compile st for OpenBSD you have to remove -lrt from config.mk, and -st will compile without any loss of functionality, because all the functions are -included in libc on this platform. - -## The Backspace Case - -St is emulating the Linux way of handling backspace being delete and delete being -backspace. - -This is an issue that was discussed in suckless mailing list -. Here is why some old grumpy -terminal users wants its backspace to be how he feels it: - - Well, I am going to comment why I want to change the behaviour - of this key. When ASCII was defined in 1968, communication - with computers was done using punched cards, or hardcopy - terminals (basically a typewriter machine connected with the - computer using a serial port). ASCII defines DELETE as 7F, - because, in punched-card terms, it means all the holes of the - card punched; it is thus a kind of 'physical delete'. In the - same way, the BACKSPACE key was a non-destructive backspace, - as on a typewriter. So, if you wanted to delete a character, - you had to BACKSPACE and then DELETE. Another use of BACKSPACE - was to type accented characters, for example 'a BACKSPACE `'. - The VT100 had no BACKSPACE key; it was generated using the - CONTROL key as another control character (CONTROL key sets to - 0 b7 b6 b5, so it converts H (code 0x48) into BACKSPACE (code - 0x08)), but it had a DELETE key in a similar position where - the BACKSPACE key is located today on common PC keyboards. - All the terminal emulators emulated the difference between - these keys correctly: the backspace key generated a BACKSPACE - (^H) and delete key generated a DELETE (^?). - - But a problem arose when Linus Torvalds wrote Linux. Unlike - earlier terminals, the Linux virtual terminal (the terminal - emulator integrated in the kernel) returned a DELETE when - backspace was pressed, due to the VT100 having a DELETE key in - the same position. This created a lot of problems (see [1] - and [2]). Since Linux has become the king, a lot of terminal - emulators today generate a DELETE when the backspace key is - pressed in order to avoid problems with Linux. The result is - that the only way of generating a BACKSPACE on these systems - is by using CONTROL + H. (I also think that emacs had an - important point here because the CONTROL + H prefix is used - in emacs in some commands (help commands).) - - From point of view of the kernel, you can change the key - for deleting a previous character with stty erase. When you - connect a real terminal into a machine you describe the type - of terminal, so getty configures the correct value of stty - erase for this terminal. In the case of terminal emulators, - however, you don't have any getty that can set the correct - value of stty erase, so you always get the default value. - For this reason, it is necessary to add 'stty erase ^H' to your - profile if you have changed the value of the backspace key. - Of course, another solution is for st itself to modify the - value of stty erase. I usually have the inverse problem: - when I connect to non-Unix machines, I have to press CONTROL + - h to get a BACKSPACE. The inverse problem occurs when a user - connects to my Unix machines from a different system with a - correct backspace key. - - [1] http://www.ibb.net/~anne/keyboard.html - [2] http://www.tldp.org/HOWTO/Keyboard-and-Console-HOWTO-5.html - -## But I really want the old grumpy behaviour of my terminal - -Apply [1]. - -[1] https://st.suckless.org/patches/delkey - -## Why do images not work in st (in programs such as w3m)? - -This is a terrible hack that overdraws an image on top of the terminal emulator -window. It also relies on a very specific way the terminal draws it's contents. - -A more proper (but limited way) would be using sixels. Which st doesn't -support. - -## BadLength X error in Xft when trying to render emoji - -Xft makes st crash when rendering color emojis with the following error: - -"X Error of failed request: BadLength (poly request too large or internal Xlib length error)" - Major opcode of failed request: 139 (RENDER) - Minor opcode of failed request: 20 (RenderAddGlyphs) - Serial number of failed request: 1595 - Current serial number in output stream: 1818" - -This is a known bug in Xft (not st) which happens on some platforms and -combination of particular fonts and fontconfig settings. - -See also: -https://gitlab.freedesktop.org/xorg/lib/libxft/issues/6 -https://bugs.freedesktop.org/show_bug.cgi?id=107534 -https://bugzilla.redhat.com/show_bug.cgi?id=1498269 - -The solution is to remove color emoji fonts or disable this in the fontconfig -XML configuration. As an ugly workaround (which may work only on newer -fontconfig versions (FC_COLOR)), the following code can be used to mask color -fonts: - - FcPatternAddBool(fcpattern, FC_COLOR, FcFalse); - -Please don't bother reporting this bug to st, but notify the upstream Xft -developers about fixing this bug. diff --git a/.suckless/st/LEGACY b/.suckless/st/LEGACY deleted file mode 100644 index bf28b1e..0000000 --- a/.suckless/st/LEGACY +++ /dev/null @@ -1,17 +0,0 @@ -A STATEMENT ON LEGACY SUPPORT - -In the terminal world there is much cruft that comes from old and unsup‐ -ported terminals that inherit incompatible modes and escape sequences -which noone is able to know, except when he/she comes from that time and -developed a graphical vt100 emulator at that time. - -One goal of st is to only support what is really needed. When you en‐ -counter a sequence which you really need, implement it. But while you -are at it, do not add the other cruft you might encounter while sneek‐ -ing at other terminal emulators. History has bloated them and there is -no real evidence that most of the sequences are used today. - - -Christoph Lohmann <20h@r-36.net> -2012-09-13T07:00:36.081271045+02:00 - diff --git a/.suckless/st/LICENSE b/.suckless/st/LICENSE deleted file mode 100644 index c356c39..0000000 --- a/.suckless/st/LICENSE +++ /dev/null @@ -1,34 +0,0 @@ -MIT/X Consortium License - -© 2014-2018 Hiltjo Posthuma -© 2018 Devin J. Pohly -© 2014-2017 Quentin Rameau -© 2009-2012 Aurélien APTEL -© 2008-2017 Anselm R Garbe -© 2012-2017 Roberto E. Vargas Caballero -© 2012-2016 Christoph Lohmann <20h at r-36 dot net> -© 2013 Eon S. Jeon -© 2013 Alexander Sedov -© 2013 Mark Edgar -© 2013-2014 Eric Pruitt -© 2013 Michael Forney -© 2013-2014 Markus Teich -© 2014-2015 Laslo Hunhold - -Permission is hereby granted, free of charge, to any person obtaining a -copy of this software and associated documentation files (the "Software"), -to deal in the Software without restriction, including without limitation -the rights to use, copy, modify, merge, publish, distribute, sublicense, -and/or sell copies of the Software, and to permit persons to whom the -Software is furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL -THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING -FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER -DEALINGS IN THE SOFTWARE. diff --git a/.suckless/st/Makefile b/.suckless/st/Makefile deleted file mode 100644 index 38240da..0000000 --- a/.suckless/st/Makefile +++ /dev/null @@ -1,58 +0,0 @@ -# st - simple terminal -# See LICENSE file for copyright and license details. -.POSIX: - -include config.mk - -SRC = st.c x.c hb.c -OBJ = $(SRC:.c=.o) - -all: options st - -options: - @echo st build options: - @echo "CFLAGS = $(STCFLAGS)" - @echo "LDFLAGS = $(STLDFLAGS)" - @echo "CC = $(CC)" - -config.h: - cp config.def.h config.h - -.c.o: - $(CC) $(STCFLAGS) -c $< - -st.o: config.h st.h win.h -x.o: arg.h config.h st.h win.h hb.h -hb.o: st.h - -$(OBJ): config.h config.mk - -st: $(OBJ) - $(CC) -o $@ $(OBJ) $(STLDFLAGS) - -clean: - rm -f st $(OBJ) st-$(VERSION).tar.gz - -dist: clean - mkdir -p st-$(VERSION) - cp -R FAQ LEGACY TODO LICENSE Makefile README config.mk\ - config.def.h st.info st.1 arg.h st.h win.h $(SRC)\ - st-$(VERSION) - tar -cf - st-$(VERSION) | gzip > st-$(VERSION).tar.gz - rm -rf st-$(VERSION) - -install: st - mkdir -p $(DESTDIR)$(PREFIX)/bin - cp -f st $(DESTDIR)$(PREFIX)/bin - chmod 755 $(DESTDIR)$(PREFIX)/bin/st - mkdir -p $(DESTDIR)$(MANPREFIX)/man1 - sed "s/VERSION/$(VERSION)/g" < st.1 > $(DESTDIR)$(MANPREFIX)/man1/st.1 - chmod 644 $(DESTDIR)$(MANPREFIX)/man1/st.1 - tic -sx st.info - @echo Please see the README file regarding the terminfo entry of st. - -uninstall: - rm -f $(DESTDIR)$(PREFIX)/bin/st - rm -f $(DESTDIR)$(MANPREFIX)/man1/st.1 - -.PHONY: all options clean dist install uninstall diff --git a/.suckless/st/README b/.suckless/st/README deleted file mode 100644 index 6a846ed..0000000 --- a/.suckless/st/README +++ /dev/null @@ -1,34 +0,0 @@ -st - simple terminal --------------------- -st is a simple terminal emulator for X which sucks less. - - -Requirements ------------- -In order to build st you need the Xlib header files. - - -Installation ------------- -Edit config.mk to match your local setup (st is installed into -the /usr/local namespace by default). - -Afterwards enter the following command to build and install st (if -necessary as root): - - make clean install - - -Running st ----------- -If you did not install st with make clean install, you must compile -the st terminfo entry with the following command: - - tic -sx st.info - -See the man page for additional details. - -Credits -------- -Based on Aurélien APTEL bt source code. - diff --git a/.suckless/st/TODO b/.suckless/st/TODO deleted file mode 100644 index 5f74cd5..0000000 --- a/.suckless/st/TODO +++ /dev/null @@ -1,28 +0,0 @@ -vt emulation ------------- - -* double-height support - -code & interface ----------------- - -* add a simple way to do multiplexing - -drawing -------- -* add diacritics support to xdraws() - * switch to a suckless font drawing library -* make the font cache simpler -* add better support for brightening of the upper colors - -bugs ----- - -* fix shift up/down (shift selection in emacs) -* remove DEC test sequence when appropriate - -misc ----- - - $ grep -nE 'XXX|TODO' st.c - diff --git a/.suckless/st/arg.h b/.suckless/st/arg.h deleted file mode 100644 index a22e019..0000000 --- a/.suckless/st/arg.h +++ /dev/null @@ -1,50 +0,0 @@ -/* - * Copy me if you can. - * by 20h - */ - -#ifndef ARG_H__ -#define ARG_H__ - -extern char *argv0; - -/* use main(int argc, char *argv[]) */ -#define ARGBEGIN for (argv0 = *argv, argv++, argc--;\ - argv[0] && argv[0][0] == '-'\ - && argv[0][1];\ - argc--, argv++) {\ - char argc_;\ - char **argv_;\ - int brk_;\ - if (argv[0][1] == '-' && argv[0][2] == '\0') {\ - argv++;\ - argc--;\ - break;\ - }\ - int i_;\ - for (i_ = 1, brk_ = 0, argv_ = argv;\ - argv[0][i_] && !brk_;\ - i_++) {\ - if (argv_ != argv)\ - break;\ - argc_ = argv[0][i_];\ - switch (argc_) - -#define ARGEND }\ - } - -#define ARGC() argc_ - -#define EARGF(x) ((argv[0][i_+1] == '\0' && argv[1] == NULL)?\ - ((x), abort(), (char *)0) :\ - (brk_ = 1, (argv[0][i_+1] != '\0')?\ - (&argv[0][i_+1]) :\ - (argc--, argv++, argv[0]))) - -#define ARGF() ((argv[0][i_+1] == '\0' && argv[1] == NULL)?\ - (char *)0 :\ - (brk_ = 1, (argv[0][i_+1] != '\0')?\ - (&argv[0][i_+1]) :\ - (argc--, argv++, argv[0]))) - -#endif diff --git a/.suckless/st/config.def.h b/.suckless/st/config.def.h deleted file mode 100644 index 013b9a6..0000000 --- a/.suckless/st/config.def.h +++ /dev/null @@ -1,459 +0,0 @@ -/* See LICENSE file for copyright and license details. */ - -/* - * appearance - * - * font: see http://freedesktop.org/software/fontconfig/fontconfig-user.html - */ -static char *font = "Liberation Mono:pixelsize=12:antialias=true:autohint=true"; -static int borderpx = 2; - -/* - * What program is execed by st depends of these precedence rules: - * 1: program passed with -e - * 2: utmp option - * 3: SHELL environment variable - * 4: value of shell in /etc/passwd - * 5: value of shell in config.h - */ -static char *shell = "/bin/sh"; -char *utmp = NULL; -char *stty_args = "stty raw pass8 nl -echo -iexten -cstopb 38400"; - -/* identification sequence returned in DA and DECID */ -char *vtiden = "\033[?6c"; - -/* Kerning / character bounding-box multipliers */ -static float cwscale = 1.0; -static float chscale = 1.0; - -/* - * word delimiter string - * - * More advanced example: L" `'\"()[]{}" - */ -wchar_t *worddelimiters = L" "; - -/* selection timeouts (in milliseconds) */ -static unsigned int doubleclicktimeout = 300; -static unsigned int tripleclicktimeout = 600; - -/* alt screens */ -int allowaltscreen = 1; - -/* frames per second st should at maximum draw to the screen */ -static unsigned int xfps = 120; -static unsigned int actionfps = 30; - -/* - * blinking timeout (set to 0 to disable blinking) for the terminal blinking - * attribute. - */ -static unsigned int blinktimeout = 800; - -/* - * thickness of underline and bar cursors - */ -static unsigned int cursorthickness = 2; - -/* - * bell volume. It must be a value between -100 and 100. Use 0 for disabling - * it - */ -static int bellvolume = 0; - -/* default TERM value */ -char *termname = "st-256color"; - -/* - * spaces per tab - * - * When you are changing this value, don't forget to adapt the »it« value in - * the st.info and appropriately install the st.info in the environment where - * you use this st version. - * - * it#$tabspaces, - * - * Secondly make sure your kernel is not expanding tabs. When running `stty - * -a` »tab0« should appear. You can tell the terminal to not expand tabs by - * running following command: - * - * stty tabs - */ -unsigned int tabspaces = 8; - -/* Terminal colors (16 first used in escape sequence) */ -static const char *colorname[] = { - /* 8 normal colors */ - "#3b4252", /* black */ - "#bf616a", /* red */ - "#a3be8c", /* green */ - "#ebcb8b", /* yellow */ - "#81a1c1", /* blue */ - "#b48ead", /* magenta */ - "#88c0d0", /* cyan */ - "#e5e9f0", /* white */ - - /* 8 bright colors */ - "#4c566a", /* black */ - "#bf616a", /* red */ - "#a3be8c", /* green */ - "#ebcb8b", /* yellow */ - "#81a1c1", /* blue */ - "#b48ead", /* magenta */ - "#8fbcbb", /* cyan */ - "#eceff4", /* white */ - - [255] = 0, - - /* more colors can be added after 255 to use with DefaultXX */ - "#2e3440", /* background */ - "#d8dee9", /* foreground */ -}; - - -/* - * Default colors (colorname index) - * foreground, background, cursor, reverse cursor - */ -unsigned int defaultfg = 257; -unsigned int defaultbg = 256; -static unsigned int defaultcs = 257; -static unsigned int defaultrcs = 256; - -/* - * Default shape of cursor - * 2: Block ("█") - * 4: Underline ("_") - * 6: Bar ("|") - * 7: Snowman ("☃") - */ -static unsigned int cursorshape = 2; - -/* - * Default columns and rows numbers - */ - -static unsigned int cols = 80; -static unsigned int rows = 24; - -/* - * Default colour and shape of the mouse cursor - */ -static unsigned int mouseshape = XC_xterm; -static unsigned int mousefg = 7; -static unsigned int mousebg = 0; - -/* - * Color used to display font attributes when fontconfig selected a font which - * doesn't match the ones requested. - */ -static unsigned int defaultattr = 11; - -/* - * Force mouse select/shortcuts while mask is active (when MODE_MOUSE is set). - * Note that if you want to use ShiftMask with selmasks, set this to an other - * modifier, set to 0 to not use it. - */ -static uint forcemousemod = ShiftMask; - -/* - * Internal mouse shortcuts. - * Beware that overloading Button1 will disable the selection. - */ -static MouseShortcut mshortcuts[] = { - /* mask button function argument release */ - { XK_ANY_MOD, Button2, selpaste, {.i = 0}, 1 }, - { XK_ANY_MOD, Button4, ttysend, {.s = "\031"} }, - { XK_ANY_MOD, Button5, ttysend, {.s = "\005"} }, -}; - -/* Internal keyboard shortcuts. */ -#define MODKEY Mod1Mask -#define TERMMOD (ControlMask|ShiftMask) - -static Shortcut shortcuts[] = { - /* mask keysym function argument */ - { XK_ANY_MOD, XK_Break, sendbreak, {.i = 0} }, - { ControlMask, XK_Print, toggleprinter, {.i = 0} }, - { ShiftMask, XK_Print, printscreen, {.i = 0} }, - { XK_ANY_MOD, XK_Print, printsel, {.i = 0} }, - { TERMMOD, XK_Prior, zoom, {.f = +1} }, - { TERMMOD, XK_Next, zoom, {.f = -1} }, - { TERMMOD, XK_Home, zoomreset, {.f = 0} }, - { TERMMOD, XK_C, clipcopy, {.i = 0} }, - { TERMMOD, XK_V, clippaste, {.i = 0} }, - { TERMMOD, XK_Y, selpaste, {.i = 0} }, - { ShiftMask, XK_Insert, selpaste, {.i = 0} }, - { TERMMOD, XK_Num_Lock, numlock, {.i = 0} }, -}; - -/* - * Special keys (change & recompile st.info accordingly) - * - * Mask value: - * * Use XK_ANY_MOD to match the key no matter modifiers state - * * Use XK_NO_MOD to match the key alone (no modifiers) - * appkey value: - * * 0: no value - * * > 0: keypad application mode enabled - * * = 2: term.numlock = 1 - * * < 0: keypad application mode disabled - * appcursor value: - * * 0: no value - * * > 0: cursor application mode enabled - * * < 0: cursor application mode disabled - * - * Be careful with the order of the definitions because st searches in - * this table sequentially, so any XK_ANY_MOD must be in the last - * position for a key. - */ - -/* - * If you want keys other than the X11 function keys (0xFD00 - 0xFFFF) - * to be mapped below, add them to this array. - */ -static KeySym mappedkeys[] = { -1 }; - -/* - * State bits to ignore when matching key or button events. By default, - * numlock (Mod2Mask) and keyboard layout (XK_SWITCH_MOD) are ignored. - */ -static uint ignoremod = Mod2Mask|XK_SWITCH_MOD; - -/* - * This is the huge key array which defines all compatibility to the Linux - * world. Please decide about changes wisely. - */ -static Key key[] = { - /* keysym mask string appkey appcursor */ - { XK_KP_Home, ShiftMask, "\033[2J", 0, -1}, - { XK_KP_Home, ShiftMask, "\033[1;2H", 0, +1}, - { XK_KP_Home, XK_ANY_MOD, "\033[H", 0, -1}, - { XK_KP_Home, XK_ANY_MOD, "\033[1~", 0, +1}, - { XK_KP_Up, XK_ANY_MOD, "\033Ox", +1, 0}, - { XK_KP_Up, XK_ANY_MOD, "\033[A", 0, -1}, - { XK_KP_Up, XK_ANY_MOD, "\033OA", 0, +1}, - { XK_KP_Down, XK_ANY_MOD, "\033Or", +1, 0}, - { XK_KP_Down, XK_ANY_MOD, "\033[B", 0, -1}, - { XK_KP_Down, XK_ANY_MOD, "\033OB", 0, +1}, - { XK_KP_Left, XK_ANY_MOD, "\033Ot", +1, 0}, - { XK_KP_Left, XK_ANY_MOD, "\033[D", 0, -1}, - { XK_KP_Left, XK_ANY_MOD, "\033OD", 0, +1}, - { XK_KP_Right, XK_ANY_MOD, "\033Ov", +1, 0}, - { XK_KP_Right, XK_ANY_MOD, "\033[C", 0, -1}, - { XK_KP_Right, XK_ANY_MOD, "\033OC", 0, +1}, - { XK_KP_Prior, ShiftMask, "\033[5;2~", 0, 0}, - { XK_KP_Prior, XK_ANY_MOD, "\033[5~", 0, 0}, - { XK_KP_Begin, XK_ANY_MOD, "\033[E", 0, 0}, - { XK_KP_End, ControlMask, "\033[J", -1, 0}, - { XK_KP_End, ControlMask, "\033[1;5F", +1, 0}, - { XK_KP_End, ShiftMask, "\033[K", -1, 0}, - { XK_KP_End, ShiftMask, "\033[1;2F", +1, 0}, - { XK_KP_End, XK_ANY_MOD, "\033[4~", 0, 0}, - { XK_KP_Next, ShiftMask, "\033[6;2~", 0, 0}, - { XK_KP_Next, XK_ANY_MOD, "\033[6~", 0, 0}, - { XK_KP_Insert, ShiftMask, "\033[2;2~", +1, 0}, - { XK_KP_Insert, ShiftMask, "\033[4l", -1, 0}, - { XK_KP_Insert, ControlMask, "\033[L", -1, 0}, - { XK_KP_Insert, ControlMask, "\033[2;5~", +1, 0}, - { XK_KP_Insert, XK_ANY_MOD, "\033[4h", -1, 0}, - { XK_KP_Insert, XK_ANY_MOD, "\033[2~", +1, 0}, - { XK_KP_Delete, ControlMask, "\033[M", -1, 0}, - { XK_KP_Delete, ControlMask, "\033[3;5~", +1, 0}, - { XK_KP_Delete, ShiftMask, "\033[2K", -1, 0}, - { XK_KP_Delete, ShiftMask, "\033[3;2~", +1, 0}, - { XK_KP_Delete, XK_ANY_MOD, "\033[P", -1, 0}, - { XK_KP_Delete, XK_ANY_MOD, "\033[3~", +1, 0}, - { XK_KP_Multiply, XK_ANY_MOD, "\033Oj", +2, 0}, - { XK_KP_Add, XK_ANY_MOD, "\033Ok", +2, 0}, - { XK_KP_Enter, XK_ANY_MOD, "\033OM", +2, 0}, - { XK_KP_Enter, XK_ANY_MOD, "\r", -1, 0}, - { XK_KP_Subtract, XK_ANY_MOD, "\033Om", +2, 0}, - { XK_KP_Decimal, XK_ANY_MOD, "\033On", +2, 0}, - { XK_KP_Divide, XK_ANY_MOD, "\033Oo", +2, 0}, - { XK_KP_0, XK_ANY_MOD, "\033Op", +2, 0}, - { XK_KP_1, XK_ANY_MOD, "\033Oq", +2, 0}, - { XK_KP_2, XK_ANY_MOD, "\033Or", +2, 0}, - { XK_KP_3, XK_ANY_MOD, "\033Os", +2, 0}, - { XK_KP_4, XK_ANY_MOD, "\033Ot", +2, 0}, - { XK_KP_5, XK_ANY_MOD, "\033Ou", +2, 0}, - { XK_KP_6, XK_ANY_MOD, "\033Ov", +2, 0}, - { XK_KP_7, XK_ANY_MOD, "\033Ow", +2, 0}, - { XK_KP_8, XK_ANY_MOD, "\033Ox", +2, 0}, - { XK_KP_9, XK_ANY_MOD, "\033Oy", +2, 0}, - { XK_Up, ShiftMask, "\033[1;2A", 0, 0}, - { XK_Up, Mod1Mask, "\033[1;3A", 0, 0}, - { XK_Up, ShiftMask|Mod1Mask,"\033[1;4A", 0, 0}, - { XK_Up, ControlMask, "\033[1;5A", 0, 0}, - { XK_Up, ShiftMask|ControlMask,"\033[1;6A", 0, 0}, - { XK_Up, ControlMask|Mod1Mask,"\033[1;7A", 0, 0}, - { XK_Up,ShiftMask|ControlMask|Mod1Mask,"\033[1;8A", 0, 0}, - { XK_Up, XK_ANY_MOD, "\033[A", 0, -1}, - { XK_Up, XK_ANY_MOD, "\033OA", 0, +1}, - { XK_Down, ShiftMask, "\033[1;2B", 0, 0}, - { XK_Down, Mod1Mask, "\033[1;3B", 0, 0}, - { XK_Down, ShiftMask|Mod1Mask,"\033[1;4B", 0, 0}, - { XK_Down, ControlMask, "\033[1;5B", 0, 0}, - { XK_Down, ShiftMask|ControlMask,"\033[1;6B", 0, 0}, - { XK_Down, ControlMask|Mod1Mask,"\033[1;7B", 0, 0}, - { XK_Down,ShiftMask|ControlMask|Mod1Mask,"\033[1;8B",0, 0}, - { XK_Down, XK_ANY_MOD, "\033[B", 0, -1}, - { XK_Down, XK_ANY_MOD, "\033OB", 0, +1}, - { XK_Left, ShiftMask, "\033[1;2D", 0, 0}, - { XK_Left, Mod1Mask, "\033[1;3D", 0, 0}, - { XK_Left, ShiftMask|Mod1Mask,"\033[1;4D", 0, 0}, - { XK_Left, ControlMask, "\033[1;5D", 0, 0}, - { XK_Left, ShiftMask|ControlMask,"\033[1;6D", 0, 0}, - { XK_Left, ControlMask|Mod1Mask,"\033[1;7D", 0, 0}, - { XK_Left,ShiftMask|ControlMask|Mod1Mask,"\033[1;8D",0, 0}, - { XK_Left, XK_ANY_MOD, "\033[D", 0, -1}, - { XK_Left, XK_ANY_MOD, "\033OD", 0, +1}, - { XK_Right, ShiftMask, "\033[1;2C", 0, 0}, - { XK_Right, Mod1Mask, "\033[1;3C", 0, 0}, - { XK_Right, ShiftMask|Mod1Mask,"\033[1;4C", 0, 0}, - { XK_Right, ControlMask, "\033[1;5C", 0, 0}, - { XK_Right, ShiftMask|ControlMask,"\033[1;6C", 0, 0}, - { XK_Right, ControlMask|Mod1Mask,"\033[1;7C", 0, 0}, - { XK_Right,ShiftMask|ControlMask|Mod1Mask,"\033[1;8C",0, 0}, - { XK_Right, XK_ANY_MOD, "\033[C", 0, -1}, - { XK_Right, XK_ANY_MOD, "\033OC", 0, +1}, - { XK_ISO_Left_Tab, ShiftMask, "\033[Z", 0, 0}, - { XK_Return, Mod1Mask, "\033\r", 0, 0}, - { XK_Return, XK_ANY_MOD, "\r", 0, 0}, - { XK_Insert, ShiftMask, "\033[4l", -1, 0}, - { XK_Insert, ShiftMask, "\033[2;2~", +1, 0}, - { XK_Insert, ControlMask, "\033[L", -1, 0}, - { XK_Insert, ControlMask, "\033[2;5~", +1, 0}, - { XK_Insert, XK_ANY_MOD, "\033[4h", -1, 0}, - { XK_Insert, XK_ANY_MOD, "\033[2~", +1, 0}, - { XK_Delete, ControlMask, "\033[M", -1, 0}, - { XK_Delete, ControlMask, "\033[3;5~", +1, 0}, - { XK_Delete, ShiftMask, "\033[2K", -1, 0}, - { XK_Delete, ShiftMask, "\033[3;2~", +1, 0}, - { XK_Delete, XK_ANY_MOD, "\033[P", -1, 0}, - { XK_Delete, XK_ANY_MOD, "\033[3~", +1, 0}, - { XK_BackSpace, XK_NO_MOD, "\177", 0, 0}, - { XK_BackSpace, Mod1Mask, "\033\177", 0, 0}, - { XK_Home, ShiftMask, "\033[2J", 0, -1}, - { XK_Home, ShiftMask, "\033[1;2H", 0, +1}, - { XK_Home, XK_ANY_MOD, "\033[H", 0, -1}, - { XK_Home, XK_ANY_MOD, "\033[1~", 0, +1}, - { XK_End, ControlMask, "\033[J", -1, 0}, - { XK_End, ControlMask, "\033[1;5F", +1, 0}, - { XK_End, ShiftMask, "\033[K", -1, 0}, - { XK_End, ShiftMask, "\033[1;2F", +1, 0}, - { XK_End, XK_ANY_MOD, "\033[4~", 0, 0}, - { XK_Prior, ControlMask, "\033[5;5~", 0, 0}, - { XK_Prior, ShiftMask, "\033[5;2~", 0, 0}, - { XK_Prior, XK_ANY_MOD, "\033[5~", 0, 0}, - { XK_Next, ControlMask, "\033[6;5~", 0, 0}, - { XK_Next, ShiftMask, "\033[6;2~", 0, 0}, - { XK_Next, XK_ANY_MOD, "\033[6~", 0, 0}, - { XK_F1, XK_NO_MOD, "\033OP" , 0, 0}, - { XK_F1, /* F13 */ ShiftMask, "\033[1;2P", 0, 0}, - { XK_F1, /* F25 */ ControlMask, "\033[1;5P", 0, 0}, - { XK_F1, /* F37 */ Mod4Mask, "\033[1;6P", 0, 0}, - { XK_F1, /* F49 */ Mod1Mask, "\033[1;3P", 0, 0}, - { XK_F1, /* F61 */ Mod3Mask, "\033[1;4P", 0, 0}, - { XK_F2, XK_NO_MOD, "\033OQ" , 0, 0}, - { XK_F2, /* F14 */ ShiftMask, "\033[1;2Q", 0, 0}, - { XK_F2, /* F26 */ ControlMask, "\033[1;5Q", 0, 0}, - { XK_F2, /* F38 */ Mod4Mask, "\033[1;6Q", 0, 0}, - { XK_F2, /* F50 */ Mod1Mask, "\033[1;3Q", 0, 0}, - { XK_F2, /* F62 */ Mod3Mask, "\033[1;4Q", 0, 0}, - { XK_F3, XK_NO_MOD, "\033OR" , 0, 0}, - { XK_F3, /* F15 */ ShiftMask, "\033[1;2R", 0, 0}, - { XK_F3, /* F27 */ ControlMask, "\033[1;5R", 0, 0}, - { XK_F3, /* F39 */ Mod4Mask, "\033[1;6R", 0, 0}, - { XK_F3, /* F51 */ Mod1Mask, "\033[1;3R", 0, 0}, - { XK_F3, /* F63 */ Mod3Mask, "\033[1;4R", 0, 0}, - { XK_F4, XK_NO_MOD, "\033OS" , 0, 0}, - { XK_F4, /* F16 */ ShiftMask, "\033[1;2S", 0, 0}, - { XK_F4, /* F28 */ ControlMask, "\033[1;5S", 0, 0}, - { XK_F4, /* F40 */ Mod4Mask, "\033[1;6S", 0, 0}, - { XK_F4, /* F52 */ Mod1Mask, "\033[1;3S", 0, 0}, - { XK_F5, XK_NO_MOD, "\033[15~", 0, 0}, - { XK_F5, /* F17 */ ShiftMask, "\033[15;2~", 0, 0}, - { XK_F5, /* F29 */ ControlMask, "\033[15;5~", 0, 0}, - { XK_F5, /* F41 */ Mod4Mask, "\033[15;6~", 0, 0}, - { XK_F5, /* F53 */ Mod1Mask, "\033[15;3~", 0, 0}, - { XK_F6, XK_NO_MOD, "\033[17~", 0, 0}, - { XK_F6, /* F18 */ ShiftMask, "\033[17;2~", 0, 0}, - { XK_F6, /* F30 */ ControlMask, "\033[17;5~", 0, 0}, - { XK_F6, /* F42 */ Mod4Mask, "\033[17;6~", 0, 0}, - { XK_F6, /* F54 */ Mod1Mask, "\033[17;3~", 0, 0}, - { XK_F7, XK_NO_MOD, "\033[18~", 0, 0}, - { XK_F7, /* F19 */ ShiftMask, "\033[18;2~", 0, 0}, - { XK_F7, /* F31 */ ControlMask, "\033[18;5~", 0, 0}, - { XK_F7, /* F43 */ Mod4Mask, "\033[18;6~", 0, 0}, - { XK_F7, /* F55 */ Mod1Mask, "\033[18;3~", 0, 0}, - { XK_F8, XK_NO_MOD, "\033[19~", 0, 0}, - { XK_F8, /* F20 */ ShiftMask, "\033[19;2~", 0, 0}, - { XK_F8, /* F32 */ ControlMask, "\033[19;5~", 0, 0}, - { XK_F8, /* F44 */ Mod4Mask, "\033[19;6~", 0, 0}, - { XK_F8, /* F56 */ Mod1Mask, "\033[19;3~", 0, 0}, - { XK_F9, XK_NO_MOD, "\033[20~", 0, 0}, - { XK_F9, /* F21 */ ShiftMask, "\033[20;2~", 0, 0}, - { XK_F9, /* F33 */ ControlMask, "\033[20;5~", 0, 0}, - { XK_F9, /* F45 */ Mod4Mask, "\033[20;6~", 0, 0}, - { XK_F9, /* F57 */ Mod1Mask, "\033[20;3~", 0, 0}, - { XK_F10, XK_NO_MOD, "\033[21~", 0, 0}, - { XK_F10, /* F22 */ ShiftMask, "\033[21;2~", 0, 0}, - { XK_F10, /* F34 */ ControlMask, "\033[21;5~", 0, 0}, - { XK_F10, /* F46 */ Mod4Mask, "\033[21;6~", 0, 0}, - { XK_F10, /* F58 */ Mod1Mask, "\033[21;3~", 0, 0}, - { XK_F11, XK_NO_MOD, "\033[23~", 0, 0}, - { XK_F11, /* F23 */ ShiftMask, "\033[23;2~", 0, 0}, - { XK_F11, /* F35 */ ControlMask, "\033[23;5~", 0, 0}, - { XK_F11, /* F47 */ Mod4Mask, "\033[23;6~", 0, 0}, - { XK_F11, /* F59 */ Mod1Mask, "\033[23;3~", 0, 0}, - { XK_F12, XK_NO_MOD, "\033[24~", 0, 0}, - { XK_F12, /* F24 */ ShiftMask, "\033[24;2~", 0, 0}, - { XK_F12, /* F36 */ ControlMask, "\033[24;5~", 0, 0}, - { XK_F12, /* F48 */ Mod4Mask, "\033[24;6~", 0, 0}, - { XK_F12, /* F60 */ Mod1Mask, "\033[24;3~", 0, 0}, - { XK_F13, XK_NO_MOD, "\033[1;2P", 0, 0}, - { XK_F14, XK_NO_MOD, "\033[1;2Q", 0, 0}, - { XK_F15, XK_NO_MOD, "\033[1;2R", 0, 0}, - { XK_F16, XK_NO_MOD, "\033[1;2S", 0, 0}, - { XK_F17, XK_NO_MOD, "\033[15;2~", 0, 0}, - { XK_F18, XK_NO_MOD, "\033[17;2~", 0, 0}, - { XK_F19, XK_NO_MOD, "\033[18;2~", 0, 0}, - { XK_F20, XK_NO_MOD, "\033[19;2~", 0, 0}, - { XK_F21, XK_NO_MOD, "\033[20;2~", 0, 0}, - { XK_F22, XK_NO_MOD, "\033[21;2~", 0, 0}, - { XK_F23, XK_NO_MOD, "\033[23;2~", 0, 0}, - { XK_F24, XK_NO_MOD, "\033[24;2~", 0, 0}, - { XK_F25, XK_NO_MOD, "\033[1;5P", 0, 0}, - { XK_F26, XK_NO_MOD, "\033[1;5Q", 0, 0}, - { XK_F27, XK_NO_MOD, "\033[1;5R", 0, 0}, - { XK_F28, XK_NO_MOD, "\033[1;5S", 0, 0}, - { XK_F29, XK_NO_MOD, "\033[15;5~", 0, 0}, - { XK_F30, XK_NO_MOD, "\033[17;5~", 0, 0}, - { XK_F31, XK_NO_MOD, "\033[18;5~", 0, 0}, - { XK_F32, XK_NO_MOD, "\033[19;5~", 0, 0}, - { XK_F33, XK_NO_MOD, "\033[20;5~", 0, 0}, - { XK_F34, XK_NO_MOD, "\033[21;5~", 0, 0}, - { XK_F35, XK_NO_MOD, "\033[23;5~", 0, 0}, -}; - -/* - * Selection types' masks. - * Use the same masks as usual. - * Button1Mask is always unset, to make masks match between ButtonPress. - * ButtonRelease and MotionNotify. - * If no match is found, regular selection is used. - */ -static uint selmasks[] = { - [SEL_RECTANGULAR] = Mod1Mask, -}; - -/* - * Printable characters in ASCII, used to estimate the advance width - * of single wide characters. - */ -static char ascii_printable[] = - " !\"#$%&'()*+,-./0123456789:;<=>?" - "@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_" - "`abcdefghijklmnopqrstuvwxyz{|}~"; diff --git a/.suckless/st/config.h b/.suckless/st/config.h deleted file mode 100644 index 053be8b..0000000 --- a/.suckless/st/config.h +++ /dev/null @@ -1,460 +0,0 @@ -/* See LICENSE file for copyright and license details. */ - -/* - * appearance - * - * font: see http://freedesktop.org/software/fontconfig/fontconfig-user.html - */ -static char *font = "Iosevka:pixelsize=20:antialias=true:autohint:true"; -static int borderpx = 8; - -/* - * What program is execed by st depends of these precedence rules: - * 1: program passed with -e - * 2: utmp option - * 3: SHELL environment variable - * 4: value of shell in /etc/passwd - * 5: value of shell in config.h - */ -static char *shell = "/bin/sh"; -char *utmp = NULL; -char *stty_args = "stty raw pass8 nl -echo -iexten -cstopb 38400"; - -/* identification sequence returned in DA and DECID */ -char *vtiden = "\033[?6c"; - -/* Kerning / character bounding-box multipliers */ -static float cwscale = 1.0; -static float chscale = 1.0; - -/* - * word delimiter string - * - * More advanced example: L" `'\"()[]{}" - */ -wchar_t *worddelimiters = L" "; - -/* selection timeouts (in milliseconds) */ -static unsigned int doubleclicktimeout = 300; -static unsigned int tripleclicktimeout = 600; - -/* alt screens */ -int allowaltscreen = 1; - -/* frames per second st should at maximum draw to the screen */ -static unsigned int xfps = 120; -static unsigned int actionfps = 30; - -/* - * blinking timeout (set to 0 to disable blinking) for the terminal blinking - * attribute. - */ -static unsigned int blinktimeout = 800; - -/* - * thickness of underline and bar cursors - */ -static unsigned int cursorthickness = 2; - -/* - * bell volume. It must be a value between -100 and 100. Use 0 for disabling - * it - */ -static int bellvolume = 0; - -/* default TERM value */ -char *termname = "st-256color"; - -/* - * spaces per tab - * - * When you are changing this value, don't forget to adapt the »it« value in - * the st.info and appropriately install the st.info in the environment where - * you use this st version. - * - * it#$tabspaces, - * - * Secondly make sure your kernel is not expanding tabs. When running `stty - * -a` »tab0« should appear. You can tell the terminal to not expand tabs by - * running following command: - * - * stty tabs - */ -unsigned int tabspaces = 8; - -/* Terminal colors (16 first used in escape sequence) */ -static const char *colorname[] = { - /* 8 normal colors */ - "#000000", /* black */ - "#bf616a", /* red */ - "#a3be8c", /* green */ - "#ebcb8b", /* yellow */ - "#81a1c1", /* blue */ - "#b48ead", /* magenta */ - "#88c0d0", /* cyan */ - "#ffffff", /* white */ - - /* 8 bright colors */ - "#111111", /* black */ - "#bf616a", /* red */ - "#a3be8c", /* green */ - "#ebcb8b", /* yellow */ - "#81a1c1", /* blue */ - "#b48ead", /* magenta */ - "#8fbcbb", /* cyan */ - "#eeeeee", /* white */ - - [255] = 0, - - /* more colors can be added after 255 to use with DefaultXX */ - "#000000", /* background */ - "#d8dee9", /* foreground */ -}; - - -/* - * Default colors (colorname index) - * foreground, background, cursor, reverse cursor - */ -unsigned int defaultfg = 257; -unsigned int defaultbg = 256; -static unsigned int defaultcs = 257; -static unsigned int defaultrcs = 256; - - -/* - * Default shape of cursor - * 2: Block ("█") - * 4: Underline ("_") - * 6: Bar ("|") - * 7: Snowman ("☃") - */ -static unsigned int cursorshape = 2; - -/* - * Default columns and rows numbers - */ - -static unsigned int cols = 80; -static unsigned int rows = 24; - -/* - * Default colour and shape of the mouse cursor - */ -static unsigned int mouseshape = XC_xterm; -static unsigned int mousefg = 7; -static unsigned int mousebg = 0; - -/* - * Color used to display font attributes when fontconfig selected a font which - * doesn't match the ones requested. - */ -static unsigned int defaultattr = 11; - -/* - * Force mouse select/shortcuts while mask is active (when MODE_MOUSE is set). - * Note that if you want to use ShiftMask with selmasks, set this to an other - * modifier, set to 0 to not use it. - */ -static uint forcemousemod = ShiftMask; - -/* - * Internal mouse shortcuts. - * Beware that overloading Button1 will disable the selection. - */ -static MouseShortcut mshortcuts[] = { - /* mask button function argument release */ - { XK_ANY_MOD, Button2, selpaste, {.i = 0}, 1 }, - { XK_ANY_MOD, Button4, ttysend, {.s = "\031"} }, - { XK_ANY_MOD, Button5, ttysend, {.s = "\005"} }, -}; - -/* Internal keyboard shortcuts. */ -#define MODKEY Mod1Mask -#define TERMMOD (ControlMask|ShiftMask) - -static Shortcut shortcuts[] = { - /* mask keysym function argument */ - { XK_ANY_MOD, XK_Break, sendbreak, {.i = 0} }, - { ControlMask, XK_Print, toggleprinter, {.i = 0} }, - { ShiftMask, XK_Print, printscreen, {.i = 0} }, - { XK_ANY_MOD, XK_Print, printsel, {.i = 0} }, - { TERMMOD, XK_Prior, zoom, {.f = +1} }, - { TERMMOD, XK_Next, zoom, {.f = -1} }, - { TERMMOD, XK_Home, zoomreset, {.f = 0} }, - { TERMMOD, XK_C, clipcopy, {.i = 0} }, - { TERMMOD, XK_V, clippaste, {.i = 0} }, - { TERMMOD, XK_Y, selpaste, {.i = 0} }, - { ShiftMask, XK_Insert, selpaste, {.i = 0} }, - { TERMMOD, XK_Num_Lock, numlock, {.i = 0} }, -}; - -/* - * Special keys (change & recompile st.info accordingly) - * - * Mask value: - * * Use XK_ANY_MOD to match the key no matter modifiers state - * * Use XK_NO_MOD to match the key alone (no modifiers) - * appkey value: - * * 0: no value - * * > 0: keypad application mode enabled - * * = 2: term.numlock = 1 - * * < 0: keypad application mode disabled - * appcursor value: - * * 0: no value - * * > 0: cursor application mode enabled - * * < 0: cursor application mode disabled - * - * Be careful with the order of the definitions because st searches in - * this table sequentially, so any XK_ANY_MOD must be in the last - * position for a key. - */ - -/* - * If you want keys other than the X11 function keys (0xFD00 - 0xFFFF) - * to be mapped below, add them to this array. - */ -static KeySym mappedkeys[] = { -1 }; - -/* - * State bits to ignore when matching key or button events. By default, - * numlock (Mod2Mask) and keyboard layout (XK_SWITCH_MOD) are ignored. - */ -static uint ignoremod = Mod2Mask|XK_SWITCH_MOD; - -/* - * This is the huge key array which defines all compatibility to the Linux - * world. Please decide about changes wisely. - */ -static Key key[] = { - /* keysym mask string appkey appcursor */ - { XK_KP_Home, ShiftMask, "\033[2J", 0, -1}, - { XK_KP_Home, ShiftMask, "\033[1;2H", 0, +1}, - { XK_KP_Home, XK_ANY_MOD, "\033[H", 0, -1}, - { XK_KP_Home, XK_ANY_MOD, "\033[1~", 0, +1}, - { XK_KP_Up, XK_ANY_MOD, "\033Ox", +1, 0}, - { XK_KP_Up, XK_ANY_MOD, "\033[A", 0, -1}, - { XK_KP_Up, XK_ANY_MOD, "\033OA", 0, +1}, - { XK_KP_Down, XK_ANY_MOD, "\033Or", +1, 0}, - { XK_KP_Down, XK_ANY_MOD, "\033[B", 0, -1}, - { XK_KP_Down, XK_ANY_MOD, "\033OB", 0, +1}, - { XK_KP_Left, XK_ANY_MOD, "\033Ot", +1, 0}, - { XK_KP_Left, XK_ANY_MOD, "\033[D", 0, -1}, - { XK_KP_Left, XK_ANY_MOD, "\033OD", 0, +1}, - { XK_KP_Right, XK_ANY_MOD, "\033Ov", +1, 0}, - { XK_KP_Right, XK_ANY_MOD, "\033[C", 0, -1}, - { XK_KP_Right, XK_ANY_MOD, "\033OC", 0, +1}, - { XK_KP_Prior, ShiftMask, "\033[5;2~", 0, 0}, - { XK_KP_Prior, XK_ANY_MOD, "\033[5~", 0, 0}, - { XK_KP_Begin, XK_ANY_MOD, "\033[E", 0, 0}, - { XK_KP_End, ControlMask, "\033[J", -1, 0}, - { XK_KP_End, ControlMask, "\033[1;5F", +1, 0}, - { XK_KP_End, ShiftMask, "\033[K", -1, 0}, - { XK_KP_End, ShiftMask, "\033[1;2F", +1, 0}, - { XK_KP_End, XK_ANY_MOD, "\033[4~", 0, 0}, - { XK_KP_Next, ShiftMask, "\033[6;2~", 0, 0}, - { XK_KP_Next, XK_ANY_MOD, "\033[6~", 0, 0}, - { XK_KP_Insert, ShiftMask, "\033[2;2~", +1, 0}, - { XK_KP_Insert, ShiftMask, "\033[4l", -1, 0}, - { XK_KP_Insert, ControlMask, "\033[L", -1, 0}, - { XK_KP_Insert, ControlMask, "\033[2;5~", +1, 0}, - { XK_KP_Insert, XK_ANY_MOD, "\033[4h", -1, 0}, - { XK_KP_Insert, XK_ANY_MOD, "\033[2~", +1, 0}, - { XK_KP_Delete, ControlMask, "\033[M", -1, 0}, - { XK_KP_Delete, ControlMask, "\033[3;5~", +1, 0}, - { XK_KP_Delete, ShiftMask, "\033[2K", -1, 0}, - { XK_KP_Delete, ShiftMask, "\033[3;2~", +1, 0}, - { XK_KP_Delete, XK_ANY_MOD, "\033[P", -1, 0}, - { XK_KP_Delete, XK_ANY_MOD, "\033[3~", +1, 0}, - { XK_KP_Multiply, XK_ANY_MOD, "\033Oj", +2, 0}, - { XK_KP_Add, XK_ANY_MOD, "\033Ok", +2, 0}, - { XK_KP_Enter, XK_ANY_MOD, "\033OM", +2, 0}, - { XK_KP_Enter, XK_ANY_MOD, "\r", -1, 0}, - { XK_KP_Subtract, XK_ANY_MOD, "\033Om", +2, 0}, - { XK_KP_Decimal, XK_ANY_MOD, "\033On", +2, 0}, - { XK_KP_Divide, XK_ANY_MOD, "\033Oo", +2, 0}, - { XK_KP_0, XK_ANY_MOD, "\033Op", +2, 0}, - { XK_KP_1, XK_ANY_MOD, "\033Oq", +2, 0}, - { XK_KP_2, XK_ANY_MOD, "\033Or", +2, 0}, - { XK_KP_3, XK_ANY_MOD, "\033Os", +2, 0}, - { XK_KP_4, XK_ANY_MOD, "\033Ot", +2, 0}, - { XK_KP_5, XK_ANY_MOD, "\033Ou", +2, 0}, - { XK_KP_6, XK_ANY_MOD, "\033Ov", +2, 0}, - { XK_KP_7, XK_ANY_MOD, "\033Ow", +2, 0}, - { XK_KP_8, XK_ANY_MOD, "\033Ox", +2, 0}, - { XK_KP_9, XK_ANY_MOD, "\033Oy", +2, 0}, - { XK_Up, ShiftMask, "\033[1;2A", 0, 0}, - { XK_Up, Mod1Mask, "\033[1;3A", 0, 0}, - { XK_Up, ShiftMask|Mod1Mask,"\033[1;4A", 0, 0}, - { XK_Up, ControlMask, "\033[1;5A", 0, 0}, - { XK_Up, ShiftMask|ControlMask,"\033[1;6A", 0, 0}, - { XK_Up, ControlMask|Mod1Mask,"\033[1;7A", 0, 0}, - { XK_Up,ShiftMask|ControlMask|Mod1Mask,"\033[1;8A", 0, 0}, - { XK_Up, XK_ANY_MOD, "\033[A", 0, -1}, - { XK_Up, XK_ANY_MOD, "\033OA", 0, +1}, - { XK_Down, ShiftMask, "\033[1;2B", 0, 0}, - { XK_Down, Mod1Mask, "\033[1;3B", 0, 0}, - { XK_Down, ShiftMask|Mod1Mask,"\033[1;4B", 0, 0}, - { XK_Down, ControlMask, "\033[1;5B", 0, 0}, - { XK_Down, ShiftMask|ControlMask,"\033[1;6B", 0, 0}, - { XK_Down, ControlMask|Mod1Mask,"\033[1;7B", 0, 0}, - { XK_Down,ShiftMask|ControlMask|Mod1Mask,"\033[1;8B",0, 0}, - { XK_Down, XK_ANY_MOD, "\033[B", 0, -1}, - { XK_Down, XK_ANY_MOD, "\033OB", 0, +1}, - { XK_Left, ShiftMask, "\033[1;2D", 0, 0}, - { XK_Left, Mod1Mask, "\033[1;3D", 0, 0}, - { XK_Left, ShiftMask|Mod1Mask,"\033[1;4D", 0, 0}, - { XK_Left, ControlMask, "\033[1;5D", 0, 0}, - { XK_Left, ShiftMask|ControlMask,"\033[1;6D", 0, 0}, - { XK_Left, ControlMask|Mod1Mask,"\033[1;7D", 0, 0}, - { XK_Left,ShiftMask|ControlMask|Mod1Mask,"\033[1;8D",0, 0}, - { XK_Left, XK_ANY_MOD, "\033[D", 0, -1}, - { XK_Left, XK_ANY_MOD, "\033OD", 0, +1}, - { XK_Right, ShiftMask, "\033[1;2C", 0, 0}, - { XK_Right, Mod1Mask, "\033[1;3C", 0, 0}, - { XK_Right, ShiftMask|Mod1Mask,"\033[1;4C", 0, 0}, - { XK_Right, ControlMask, "\033[1;5C", 0, 0}, - { XK_Right, ShiftMask|ControlMask,"\033[1;6C", 0, 0}, - { XK_Right, ControlMask|Mod1Mask,"\033[1;7C", 0, 0}, - { XK_Right,ShiftMask|ControlMask|Mod1Mask,"\033[1;8C",0, 0}, - { XK_Right, XK_ANY_MOD, "\033[C", 0, -1}, - { XK_Right, XK_ANY_MOD, "\033OC", 0, +1}, - { XK_ISO_Left_Tab, ShiftMask, "\033[Z", 0, 0}, - { XK_Return, Mod1Mask, "\033\r", 0, 0}, - { XK_Return, XK_ANY_MOD, "\r", 0, 0}, - { XK_Insert, ShiftMask, "\033[4l", -1, 0}, - { XK_Insert, ShiftMask, "\033[2;2~", +1, 0}, - { XK_Insert, ControlMask, "\033[L", -1, 0}, - { XK_Insert, ControlMask, "\033[2;5~", +1, 0}, - { XK_Insert, XK_ANY_MOD, "\033[4h", -1, 0}, - { XK_Insert, XK_ANY_MOD, "\033[2~", +1, 0}, - { XK_Delete, ControlMask, "\033[M", -1, 0}, - { XK_Delete, ControlMask, "\033[3;5~", +1, 0}, - { XK_Delete, ShiftMask, "\033[2K", -1, 0}, - { XK_Delete, ShiftMask, "\033[3;2~", +1, 0}, - { XK_Delete, XK_ANY_MOD, "\033[P", -1, 0}, - { XK_Delete, XK_ANY_MOD, "\033[3~", +1, 0}, - { XK_BackSpace, XK_NO_MOD, "\177", 0, 0}, - { XK_BackSpace, Mod1Mask, "\033\177", 0, 0}, - { XK_Home, ShiftMask, "\033[2J", 0, -1}, - { XK_Home, ShiftMask, "\033[1;2H", 0, +1}, - { XK_Home, XK_ANY_MOD, "\033[H", 0, -1}, - { XK_Home, XK_ANY_MOD, "\033[1~", 0, +1}, - { XK_End, ControlMask, "\033[J", -1, 0}, - { XK_End, ControlMask, "\033[1;5F", +1, 0}, - { XK_End, ShiftMask, "\033[K", -1, 0}, - { XK_End, ShiftMask, "\033[1;2F", +1, 0}, - { XK_End, XK_ANY_MOD, "\033[4~", 0, 0}, - { XK_Prior, ControlMask, "\033[5;5~", 0, 0}, - { XK_Prior, ShiftMask, "\033[5;2~", 0, 0}, - { XK_Prior, XK_ANY_MOD, "\033[5~", 0, 0}, - { XK_Next, ControlMask, "\033[6;5~", 0, 0}, - { XK_Next, ShiftMask, "\033[6;2~", 0, 0}, - { XK_Next, XK_ANY_MOD, "\033[6~", 0, 0}, - { XK_F1, XK_NO_MOD, "\033OP" , 0, 0}, - { XK_F1, /* F13 */ ShiftMask, "\033[1;2P", 0, 0}, - { XK_F1, /* F25 */ ControlMask, "\033[1;5P", 0, 0}, - { XK_F1, /* F37 */ Mod4Mask, "\033[1;6P", 0, 0}, - { XK_F1, /* F49 */ Mod1Mask, "\033[1;3P", 0, 0}, - { XK_F1, /* F61 */ Mod3Mask, "\033[1;4P", 0, 0}, - { XK_F2, XK_NO_MOD, "\033OQ" , 0, 0}, - { XK_F2, /* F14 */ ShiftMask, "\033[1;2Q", 0, 0}, - { XK_F2, /* F26 */ ControlMask, "\033[1;5Q", 0, 0}, - { XK_F2, /* F38 */ Mod4Mask, "\033[1;6Q", 0, 0}, - { XK_F2, /* F50 */ Mod1Mask, "\033[1;3Q", 0, 0}, - { XK_F2, /* F62 */ Mod3Mask, "\033[1;4Q", 0, 0}, - { XK_F3, XK_NO_MOD, "\033OR" , 0, 0}, - { XK_F3, /* F15 */ ShiftMask, "\033[1;2R", 0, 0}, - { XK_F3, /* F27 */ ControlMask, "\033[1;5R", 0, 0}, - { XK_F3, /* F39 */ Mod4Mask, "\033[1;6R", 0, 0}, - { XK_F3, /* F51 */ Mod1Mask, "\033[1;3R", 0, 0}, - { XK_F3, /* F63 */ Mod3Mask, "\033[1;4R", 0, 0}, - { XK_F4, XK_NO_MOD, "\033OS" , 0, 0}, - { XK_F4, /* F16 */ ShiftMask, "\033[1;2S", 0, 0}, - { XK_F4, /* F28 */ ControlMask, "\033[1;5S", 0, 0}, - { XK_F4, /* F40 */ Mod4Mask, "\033[1;6S", 0, 0}, - { XK_F4, /* F52 */ Mod1Mask, "\033[1;3S", 0, 0}, - { XK_F5, XK_NO_MOD, "\033[15~", 0, 0}, - { XK_F5, /* F17 */ ShiftMask, "\033[15;2~", 0, 0}, - { XK_F5, /* F29 */ ControlMask, "\033[15;5~", 0, 0}, - { XK_F5, /* F41 */ Mod4Mask, "\033[15;6~", 0, 0}, - { XK_F5, /* F53 */ Mod1Mask, "\033[15;3~", 0, 0}, - { XK_F6, XK_NO_MOD, "\033[17~", 0, 0}, - { XK_F6, /* F18 */ ShiftMask, "\033[17;2~", 0, 0}, - { XK_F6, /* F30 */ ControlMask, "\033[17;5~", 0, 0}, - { XK_F6, /* F42 */ Mod4Mask, "\033[17;6~", 0, 0}, - { XK_F6, /* F54 */ Mod1Mask, "\033[17;3~", 0, 0}, - { XK_F7, XK_NO_MOD, "\033[18~", 0, 0}, - { XK_F7, /* F19 */ ShiftMask, "\033[18;2~", 0, 0}, - { XK_F7, /* F31 */ ControlMask, "\033[18;5~", 0, 0}, - { XK_F7, /* F43 */ Mod4Mask, "\033[18;6~", 0, 0}, - { XK_F7, /* F55 */ Mod1Mask, "\033[18;3~", 0, 0}, - { XK_F8, XK_NO_MOD, "\033[19~", 0, 0}, - { XK_F8, /* F20 */ ShiftMask, "\033[19;2~", 0, 0}, - { XK_F8, /* F32 */ ControlMask, "\033[19;5~", 0, 0}, - { XK_F8, /* F44 */ Mod4Mask, "\033[19;6~", 0, 0}, - { XK_F8, /* F56 */ Mod1Mask, "\033[19;3~", 0, 0}, - { XK_F9, XK_NO_MOD, "\033[20~", 0, 0}, - { XK_F9, /* F21 */ ShiftMask, "\033[20;2~", 0, 0}, - { XK_F9, /* F33 */ ControlMask, "\033[20;5~", 0, 0}, - { XK_F9, /* F45 */ Mod4Mask, "\033[20;6~", 0, 0}, - { XK_F9, /* F57 */ Mod1Mask, "\033[20;3~", 0, 0}, - { XK_F10, XK_NO_MOD, "\033[21~", 0, 0}, - { XK_F10, /* F22 */ ShiftMask, "\033[21;2~", 0, 0}, - { XK_F10, /* F34 */ ControlMask, "\033[21;5~", 0, 0}, - { XK_F10, /* F46 */ Mod4Mask, "\033[21;6~", 0, 0}, - { XK_F10, /* F58 */ Mod1Mask, "\033[21;3~", 0, 0}, - { XK_F11, XK_NO_MOD, "\033[23~", 0, 0}, - { XK_F11, /* F23 */ ShiftMask, "\033[23;2~", 0, 0}, - { XK_F11, /* F35 */ ControlMask, "\033[23;5~", 0, 0}, - { XK_F11, /* F47 */ Mod4Mask, "\033[23;6~", 0, 0}, - { XK_F11, /* F59 */ Mod1Mask, "\033[23;3~", 0, 0}, - { XK_F12, XK_NO_MOD, "\033[24~", 0, 0}, - { XK_F12, /* F24 */ ShiftMask, "\033[24;2~", 0, 0}, - { XK_F12, /* F36 */ ControlMask, "\033[24;5~", 0, 0}, - { XK_F12, /* F48 */ Mod4Mask, "\033[24;6~", 0, 0}, - { XK_F12, /* F60 */ Mod1Mask, "\033[24;3~", 0, 0}, - { XK_F13, XK_NO_MOD, "\033[1;2P", 0, 0}, - { XK_F14, XK_NO_MOD, "\033[1;2Q", 0, 0}, - { XK_F15, XK_NO_MOD, "\033[1;2R", 0, 0}, - { XK_F16, XK_NO_MOD, "\033[1;2S", 0, 0}, - { XK_F17, XK_NO_MOD, "\033[15;2~", 0, 0}, - { XK_F18, XK_NO_MOD, "\033[17;2~", 0, 0}, - { XK_F19, XK_NO_MOD, "\033[18;2~", 0, 0}, - { XK_F20, XK_NO_MOD, "\033[19;2~", 0, 0}, - { XK_F21, XK_NO_MOD, "\033[20;2~", 0, 0}, - { XK_F22, XK_NO_MOD, "\033[21;2~", 0, 0}, - { XK_F23, XK_NO_MOD, "\033[23;2~", 0, 0}, - { XK_F24, XK_NO_MOD, "\033[24;2~", 0, 0}, - { XK_F25, XK_NO_MOD, "\033[1;5P", 0, 0}, - { XK_F26, XK_NO_MOD, "\033[1;5Q", 0, 0}, - { XK_F27, XK_NO_MOD, "\033[1;5R", 0, 0}, - { XK_F28, XK_NO_MOD, "\033[1;5S", 0, 0}, - { XK_F29, XK_NO_MOD, "\033[15;5~", 0, 0}, - { XK_F30, XK_NO_MOD, "\033[17;5~", 0, 0}, - { XK_F31, XK_NO_MOD, "\033[18;5~", 0, 0}, - { XK_F32, XK_NO_MOD, "\033[19;5~", 0, 0}, - { XK_F33, XK_NO_MOD, "\033[20;5~", 0, 0}, - { XK_F34, XK_NO_MOD, "\033[21;5~", 0, 0}, - { XK_F35, XK_NO_MOD, "\033[23;5~", 0, 0}, -}; - -/* - * Selection types' masks. - * Use the same masks as usual. - * Button1Mask is always unset, to make masks match between ButtonPress. - * ButtonRelease and MotionNotify. - * If no match is found, regular selection is used. - */ -static uint selmasks[] = { - [SEL_RECTANGULAR] = Mod1Mask, -}; - -/* - * Printable characters in ASCII, used to estimate the advance width - * of single wide characters. - */ -static char ascii_printable[] = - " !\"#$%&'()*+,-./0123456789:;<=>?" - "@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_" - "`abcdefghijklmnopqrstuvwxyz{|}~"; diff --git a/.suckless/st/config.mk b/.suckless/st/config.mk deleted file mode 100644 index a021b2c..0000000 --- a/.suckless/st/config.mk +++ /dev/null @@ -1,37 +0,0 @@ -# st version -VERSION = 0.8.2 - -# Customize below to fit your system - -# paths -PREFIX = /usr/local -MANPREFIX = $(PREFIX)/share/man - -X11INC = /usr/X11R6/include -X11LIB = /usr/X11R6/lib - -PKG_CONFIG = pkg-config - -# includes and libs -INCS = -I$(X11INC) \ - `$(PKG_CONFIG) --cflags fontconfig` \ - `$(PKG_CONFIG) --cflags freetype2` \ - `$(PKG_CONFIG) --cflags harfbuzz` -LIBS = -L$(X11LIB) -lm -lrt -lX11 -lutil -lXft \ - `$(PKG_CONFIG) --libs fontconfig` \ - `$(PKG_CONFIG) --libs freetype2` \ - `$(PKG_CONFIG) --libs harfbuzz` - -# flags -STCPPFLAGS = -DVERSION=\"$(VERSION)\" -D_XOPEN_SOURCE=600 -STCFLAGS = $(INCS) $(STCPPFLAGS) $(CPPFLAGS) $(CFLAGS) -STLDFLAGS = $(LIBS) $(LDFLAGS) - -# OpenBSD: -#CPPFLAGS = -DVERSION=\"$(VERSION)\" -D_XOPEN_SOURCE=600 -D_BSD_SOURCE -#LIBS = -L$(X11LIB) -lm -lX11 -lutil -lXft \ -# `pkg-config --libs fontconfig` \ -# `pkg-config --libs freetype2` - -# compiler and linker -# CC = c99 diff --git a/.suckless/st/hb.c b/.suckless/st/hb.c deleted file mode 100644 index 7df2828..0000000 --- a/.suckless/st/hb.c +++ /dev/null @@ -1,136 +0,0 @@ -#include -#include -#include -#include -#include -#include - -#include "st.h" - -void hbtransformsegment(XftFont *xfont, const Glyph *string, hb_codepoint_t *codepoints, int start, int length); -hb_font_t *hbfindfont(XftFont *match); - -typedef struct { - XftFont *match; - hb_font_t *font; -} HbFontMatch; - -static int hbfontslen = 0; -static HbFontMatch *hbfontcache = NULL; - -void -hbunloadfonts() -{ - for (int i = 0; i < hbfontslen; i++) { - hb_font_destroy(hbfontcache[i].font); - XftUnlockFace(hbfontcache[i].match); - } - - if (hbfontcache != NULL) { - free(hbfontcache); - hbfontcache = NULL; - } - hbfontslen = 0; -} - -hb_font_t * -hbfindfont(XftFont *match) -{ - for (int i = 0; i < hbfontslen; i++) { - if (hbfontcache[i].match == match) - return hbfontcache[i].font; - } - - /* Font not found in cache, caching it now. */ - hbfontcache = realloc(hbfontcache, sizeof(HbFontMatch) * (hbfontslen + 1)); - FT_Face face = XftLockFace(match); - hb_font_t *font = hb_ft_font_create(face, NULL); - if (font == NULL) - die("Failed to load Harfbuzz font."); - - hbfontcache[hbfontslen].match = match; - hbfontcache[hbfontslen].font = font; - hbfontslen += 1; - - return font; -} - -void -hbtransform(XftGlyphFontSpec *specs, const Glyph *glyphs, size_t len, int x, int y) -{ - int start = 0, length = 1, gstart = 0; - hb_codepoint_t *codepoints = calloc(len, sizeof(hb_codepoint_t)); - - for (int idx = 1, specidx = 1; idx < len; idx++) { - if (glyphs[idx].mode & ATTR_WDUMMY) { - length += 1; - continue; - } - - if (specs[specidx].font != specs[start].font || ATTRCMP(glyphs[gstart], glyphs[idx]) || selected(x + idx, y) != selected(x + gstart, y)) { - hbtransformsegment(specs[start].font, glyphs, codepoints, gstart, length); - - /* Reset the sequence. */ - length = 1; - start = specidx; - gstart = idx; - } else { - length += 1; - } - - specidx++; - } - - /* EOL. */ - hbtransformsegment(specs[start].font, glyphs, codepoints, gstart, length); - - /* Apply the transformation to glyph specs. */ - for (int i = 0, specidx = 0; i < len; i++) { - if (glyphs[i].mode & ATTR_WDUMMY) - continue; - - if (codepoints[i] != specs[specidx].glyph) - ((Glyph *)glyphs)[i].mode |= ATTR_LIGA; - - specs[specidx++].glyph = codepoints[i]; - } - - free(codepoints); -} - -void -hbtransformsegment(XftFont *xfont, const Glyph *string, hb_codepoint_t *codepoints, int start, int length) -{ - hb_font_t *font = hbfindfont(xfont); - if (font == NULL) - return; - - Rune rune; - ushort mode = USHRT_MAX; - hb_buffer_t *buffer = hb_buffer_create(); - hb_buffer_set_direction(buffer, HB_DIRECTION_LTR); - - /* Fill buffer with codepoints. */ - for (int i = start; i < (start+length); i++) { - rune = string[i].u; - mode = string[i].mode; - if (mode & ATTR_WDUMMY) - rune = 0x0020; - hb_buffer_add_codepoints(buffer, &rune, 1, 0, 1); - } - - /* Shape the segment. */ - hb_shape(font, buffer, NULL, 0); - - /* Get new glyph info. */ - hb_glyph_info_t *info = hb_buffer_get_glyph_infos(buffer, NULL); - - /* Write new codepoints. */ - for (int i = 0; i < length; i++) { - hb_codepoint_t gid = info[i].codepoint; - codepoints[start+i] = gid; - } - - /* Cleanup. */ - hb_buffer_destroy(buffer); -} diff --git a/.suckless/st/hb.h b/.suckless/st/hb.h deleted file mode 100644 index b3e02d0..0000000 --- a/.suckless/st/hb.h +++ /dev/null @@ -1,7 +0,0 @@ -#include -#include -#include - -void hbunloadfonts(); -void hbtransform(XftGlyphFontSpec *, const Glyph *, size_t, int, int); - diff --git a/.suckless/st/hb.o b/.suckless/st/hb.o deleted file mode 100644 index 4d2292d9efc0707c752526f9a9e535bfe0af08f5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 4552 zcmbW5e{36P8OPs?ukF!fwpX%ZwgKLBOy0DaWCbaPFy$O4c|ESH#FCb=b)IuR*H#@n z*k_bhrJLlr+HrUc@gINi7t(}~koW_(4mRaSyG$sIN`N{|;`fF+1<|rXghYjTp7-82 zd3)rfiMP7E@B4Ya&-1*$?meqdPnpvLib6zD$U|hOCnzD$MR(d0tT{n$Awy*DQQG;d z*7-a`XEQ&=n!3NGEAOh*+rxU^nUr?LZ*ef5^81}NCG#<~JiS3Z7J_#BPTV)HGcj6s zTC{UPd;IvCpOm`)_CAC5?$+76c(UHnK3(~fN_%EJPBRJTE3~r=jW^owcV1VhwMAF{ z0^?$-{4B0NB=`5cyZ#zR zz6k~~qSsWWT4(*qo;q%1E`04+-ug9le??dRteVQE(VOX0L%Ws-LE_7B@53+Z((kQI4G&il+Sl;~)b+6~$%dV;GIhaCdxplh|j*&6y73%|kgD%aCX4aq7+Lu&ckzi;03^7~;lrZcR^$AV%k+LgC40 zuEw5?L0B*jQ};c(@<%oHQf^S)x}dhdXIdpanO}KFJ*@mAZ~X?(B!twwP2MI?Y6Fad7d@(72F7R zl1y0p#O4pM4`sSot=Xld(@54D#ZrC?$_qwb^$nVFpYY%*mutxD3Y*4ssOE`?MEemvrEFa$6f~BN{(9U_=w0NG_Tjs6`V1 zjHs3ZQ;zF?{J*+y?-xR2kWjEMa_tBnuxvE`&4Fw*@q8p3P4?7mH1&-=Ms(!dUL!iz z*_Vqx6*-{nkB$K{qA8Go6kM`V%@{Pn{bxMZ)ejPDF;WSO#l!4cVQ7{7y}z3=vg^%H zNCf`vRw587SoQ)?fNR_!6t{9N^umt)2-&{sg_+e6vhSw8=tQ=o3qlC{jD*I)QQchb zhVB7rzU)i@8-a&i8aK5%3LNu_eC&(?b3*iILilqb{5v81yTCE;eRu$% zu=5J_kKoyDzN;bp?GS!Fgx?6^2Voq$#n7LH6$KRPvAm^HHIQ~@*vG#5&C%V{=F zkuQ~@nj!jBcbaI--daK-MF0||u z3wGVXRP*g}*=`oNZU?k%7(umZTTZos6A{8tw;IoGg^#P6p@gp%KfUPs^@BZX1vK_8W=YgYUe!F+Pq) zD1zUBFA|oNkC_&KS_=Lt&V%|fNgtE+*??Z$H(v;F5r0mKgXd51Cj$C^`NQm&1Dtos zGXXC0dI2ut|0IO7AIxAw{{!#{{c4E*=bWSd!;=28#PJ&_;=IebSWhudd|$B$F7y$2 zZ)7g?w@Z9K2!vke9Cf!z{Fub??ic#1#6KeOQxZ=|{F1~!Dsl1tW8T{(enZmVA@K)s zqEJwOCp=<2MJ%pc^xE$vViOc761Or3CO4cV4 zLcwA2p|L(?e|Na}-r#ozi_nYj!7`&eT->uH-8!}46z8FKntojot)^WorX739A?Z@l zDU$Sjt3}ey26j+BnjR%-tFgfDv2?xR*y))^<{n@d3+T|>t`y46;)2by|Nm7ITyg7z zU-}7PSn_=R{GP8BZLIOEpkx9bxllapP?q?Ea<-!FJpvD^a>axO!}iW|r)IBMl` zKXfo=(EkPQKO~E}Rz{=H!I%%q5Zc0D+{3~w@{4)`4A>qK0$%@+=f5k!g`cSNU>9v! z`pXYE|0lV>e18$ diff --git a/.suckless/st/patches/st-anysize-0.8.1.diff b/.suckless/st/patches/st-anysize-0.8.1.diff deleted file mode 100644 index 1eaf2ba..0000000 --- a/.suckless/st/patches/st-anysize-0.8.1.diff +++ /dev/null @@ -1,152 +0,0 @@ -diff --git a/x.c b/x.c -index 00cb6b1..8f87c0f 100644 ---- a/x.c -+++ b/x.c -@@ -78,6 +78,7 @@ typedef XftGlyphFontSpec GlyphFontSpec; - typedef struct { - int tw, th; /* tty width and height */ - int w, h; /* window width and height */ -+ int hborderpx, vborderpx; - int ch; /* char height */ - int cw; /* char width */ - int mode; /* window state/mode flags */ -@@ -311,7 +312,7 @@ zoomreset(const Arg *arg) - int - evcol(XEvent *e) - { -- int x = e->xbutton.x - borderpx; -+ int x = e->xbutton.x - win.hborderpx; - LIMIT(x, 0, win.tw - 1); - return x / win.cw; - } -@@ -319,7 +320,7 @@ evcol(XEvent *e) - int - evrow(XEvent *e) - { -- int y = e->xbutton.y - borderpx; -+ int y = e->xbutton.y - win.vborderpx; - LIMIT(y, 0, win.th - 1); - return y / win.ch; - } -@@ -675,6 +676,9 @@ cresize(int width, int height) - col = MAX(1, col); - row = MAX(1, row); - -+ win.hborderpx = (win.w - col * win.cw) / 2; -+ win.vborderpx = (win.h - row * win.ch) / 2; -+ - tresize(col, row); - xresize(col, row); - ttyresize(win.tw, win.th); -@@ -793,8 +797,8 @@ xhints(void) - sizeh->flags = PSize | PResizeInc | PBaseSize | PMinSize; - sizeh->height = win.h; - sizeh->width = win.w; -- sizeh->height_inc = win.ch; -- sizeh->width_inc = win.cw; -+ sizeh->height_inc = 1; -+ sizeh->width_inc = 1; - sizeh->base_height = 2 * borderpx; - sizeh->base_width = 2 * borderpx; - sizeh->min_height = win.ch + 2 * borderpx; -@@ -1022,8 +1026,8 @@ xinit(int cols, int rows) - xloadcols(); - - /* adjust fixed window geometry */ -- win.w = 2 * borderpx + cols * win.cw; -- win.h = 2 * borderpx + rows * win.ch; -+ win.w = 2 * win.hborderpx + cols * win.cw; -+ win.h = 2 * win.vborderpx + rows * win.ch; - if (xw.gm & XNegative) - xw.l += DisplayWidth(xw.dpy, xw.scr) - win.w - 2; - if (xw.gm & YNegative) -@@ -1123,7 +1127,7 @@ xinit(int cols, int rows) - int - xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x, int y) - { -- float winx = borderpx + x * win.cw, winy = borderpx + y * win.ch, xp, yp; -+ float winx = win.hborderpx + x * win.cw, winy = win.vborderpx + y * win.ch, xp, yp; - ushort mode, prevmode = USHRT_MAX; - Font *font = &dc.font; - int frcflags = FRC_NORMAL; -@@ -1259,7 +1263,7 @@ void - xdrawglyphfontspecs(const XftGlyphFontSpec *specs, Glyph base, int len, int x, int y) - { - int charlen = len * ((base.mode & ATTR_WIDE) ? 2 : 1); -- int winx = borderpx + x * win.cw, winy = borderpx + y * win.ch, -+ int winx = win.hborderpx + x * win.cw, winy = win.vborderpx + y * win.ch, - width = charlen * win.cw; - Color *fg, *bg, *temp, revfg, revbg, truefg, truebg; - XRenderColor colfg, colbg; -@@ -1349,17 +1353,17 @@ xdrawglyphfontspecs(const XftGlyphFontSpec *specs, Glyph base, int len, int x, i - - /* Intelligent cleaning up of the borders. */ - if (x == 0) { -- xclear(0, (y == 0)? 0 : winy, borderpx, -+ xclear(0, (y == 0)? 0 : winy, win.vborderpx, - winy + win.ch + -- ((winy + win.ch >= borderpx + win.th)? win.h : 0)); -+ ((winy + win.ch >= win.vborderpx + win.th)? win.h : 0)); - } -- if (winx + width >= borderpx + win.tw) { -+ if (winx + width >= win.hborderpx + win.tw) { - xclear(winx + width, (y == 0)? 0 : winy, win.w, -- ((winy + win.ch >= borderpx + win.th)? win.h : (winy + win.ch))); -+ ((winy + win.ch >= win.vborderpx + win.th)? win.h : (winy + win.ch))); - } - if (y == 0) -- xclear(winx, 0, winx + width, borderpx); -- if (winy + win.ch >= borderpx + win.th) -+ xclear(winx, 0, winx + width, win.hborderpx); -+ if (winy + win.ch >= win.vborderpx + win.th) - xclear(winx, winy + win.ch, winx + width, win.h); - - /* Clean up the region we want to draw to. */ -@@ -1452,35 +1456,35 @@ xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og) - case 3: /* Blinking Underline */ - case 4: /* Steady Underline */ - XftDrawRect(xw.draw, &drawcol, -- borderpx + cx * win.cw, -- borderpx + (cy + 1) * win.ch - \ -+ win.hborderpx + cx * win.cw, -+ win.vborderpx + (cy + 1) * win.ch - \ - cursorthickness, - win.cw, cursorthickness); - break; - case 5: /* Blinking bar */ - case 6: /* Steady bar */ - XftDrawRect(xw.draw, &drawcol, -- borderpx + cx * win.cw, -- borderpx + cy * win.ch, -+ win.hborderpx + cx * win.cw, -+ win.vborderpx + cy * win.ch, - cursorthickness, win.ch); - break; - } - } else { - XftDrawRect(xw.draw, &drawcol, -- borderpx + cx * win.cw, -- borderpx + cy * win.ch, -+ win.hborderpx + cx * win.cw, -+ win.vborderpx + cy * win.ch, - win.cw - 1, 1); - XftDrawRect(xw.draw, &drawcol, -- borderpx + cx * win.cw, -- borderpx + cy * win.ch, -+ win.hborderpx + cx * win.cw, -+ win.vborderpx + cy * win.ch, - 1, win.ch - 1); - XftDrawRect(xw.draw, &drawcol, -- borderpx + (cx + 1) * win.cw - 1, -- borderpx + cy * win.ch, -+ win.hborderpx + (cx + 1) * win.cw - 1, -+ win.vborderpx + cy * win.ch, - 1, win.ch - 1); - XftDrawRect(xw.draw, &drawcol, -- borderpx + cx * win.cw, -- borderpx + (cy + 1) * win.ch - 1, -+ win.hborderpx + cx * win.cw, -+ win.vborderpx + (cy + 1) * win.ch - 1, - win.cw, 1); - } - } diff --git a/.suckless/st/patches/st-ligatures-20200430-0.8.3.diff b/.suckless/st/patches/st-ligatures-20200430-0.8.3.diff deleted file mode 100644 index 3f2131e..0000000 --- a/.suckless/st/patches/st-ligatures-20200430-0.8.3.diff +++ /dev/null @@ -1,306 +0,0 @@ -diff --git a/Makefile b/Makefile -index 470ac86..38240da 100644 ---- a/Makefile -+++ b/Makefile -@@ -4,7 +4,7 @@ - - include config.mk - --SRC = st.c x.c -+SRC = st.c x.c hb.c - OBJ = $(SRC:.c=.o) - - all: options st -@@ -22,7 +22,8 @@ config.h: - $(CC) $(STCFLAGS) -c $< - - st.o: config.h st.h win.h --x.o: arg.h config.h st.h win.h -+x.o: arg.h config.h st.h win.h hb.h -+hb.o: st.h - - $(OBJ): config.h config.mk - -diff --git a/config.mk b/config.mk -index beafc35..3df5c83 100644 ---- a/config.mk -+++ b/config.mk -@@ -15,10 +15,12 @@ PKG_CONFIG = pkg-config - # includes and libs - INCS = -I$(X11INC) \ - `$(PKG_CONFIG) --cflags fontconfig` \ -- `$(PKG_CONFIG) --cflags freetype2` -+ `$(PKG_CONFIG) --cflags freetype2` \ -+ `$(PKG_CONFIG) --cflags harfbuzz` - LIBS = -L$(X11LIB) -lm -lrt -lX11 -lutil -lXft \ - `$(PKG_CONFIG) --libs fontconfig` \ -- `$(PKG_CONFIG) --libs freetype2` -+ `$(PKG_CONFIG) --libs freetype2` \ -+ `$(PKG_CONFIG) --libs harfbuzz` - - # flags - STCPPFLAGS = -DVERSION=\"$(VERSION)\" -D_XOPEN_SOURCE=600 -diff --git a/hb.c b/hb.c -new file mode 100644 -index 0000000..7df2828 ---- /dev/null -+++ b/hb.c -@@ -0,0 +1,136 @@ -+#include -+#include -+#include -+#include -+#include -+#include -+ -+#include "st.h" -+ -+void hbtransformsegment(XftFont *xfont, const Glyph *string, hb_codepoint_t *codepoints, int start, int length); -+hb_font_t *hbfindfont(XftFont *match); -+ -+typedef struct { -+ XftFont *match; -+ hb_font_t *font; -+} HbFontMatch; -+ -+static int hbfontslen = 0; -+static HbFontMatch *hbfontcache = NULL; -+ -+void -+hbunloadfonts() -+{ -+ for (int i = 0; i < hbfontslen; i++) { -+ hb_font_destroy(hbfontcache[i].font); -+ XftUnlockFace(hbfontcache[i].match); -+ } -+ -+ if (hbfontcache != NULL) { -+ free(hbfontcache); -+ hbfontcache = NULL; -+ } -+ hbfontslen = 0; -+} -+ -+hb_font_t * -+hbfindfont(XftFont *match) -+{ -+ for (int i = 0; i < hbfontslen; i++) { -+ if (hbfontcache[i].match == match) -+ return hbfontcache[i].font; -+ } -+ -+ /* Font not found in cache, caching it now. */ -+ hbfontcache = realloc(hbfontcache, sizeof(HbFontMatch) * (hbfontslen + 1)); -+ FT_Face face = XftLockFace(match); -+ hb_font_t *font = hb_ft_font_create(face, NULL); -+ if (font == NULL) -+ die("Failed to load Harfbuzz font."); -+ -+ hbfontcache[hbfontslen].match = match; -+ hbfontcache[hbfontslen].font = font; -+ hbfontslen += 1; -+ -+ return font; -+} -+ -+void -+hbtransform(XftGlyphFontSpec *specs, const Glyph *glyphs, size_t len, int x, int y) -+{ -+ int start = 0, length = 1, gstart = 0; -+ hb_codepoint_t *codepoints = calloc(len, sizeof(hb_codepoint_t)); -+ -+ for (int idx = 1, specidx = 1; idx < len; idx++) { -+ if (glyphs[idx].mode & ATTR_WDUMMY) { -+ length += 1; -+ continue; -+ } -+ -+ if (specs[specidx].font != specs[start].font || ATTRCMP(glyphs[gstart], glyphs[idx]) || selected(x + idx, y) != selected(x + gstart, y)) { -+ hbtransformsegment(specs[start].font, glyphs, codepoints, gstart, length); -+ -+ /* Reset the sequence. */ -+ length = 1; -+ start = specidx; -+ gstart = idx; -+ } else { -+ length += 1; -+ } -+ -+ specidx++; -+ } -+ -+ /* EOL. */ -+ hbtransformsegment(specs[start].font, glyphs, codepoints, gstart, length); -+ -+ /* Apply the transformation to glyph specs. */ -+ for (int i = 0, specidx = 0; i < len; i++) { -+ if (glyphs[i].mode & ATTR_WDUMMY) -+ continue; -+ -+ if (codepoints[i] != specs[specidx].glyph) -+ ((Glyph *)glyphs)[i].mode |= ATTR_LIGA; -+ -+ specs[specidx++].glyph = codepoints[i]; -+ } -+ -+ free(codepoints); -+} -+ -+void -+hbtransformsegment(XftFont *xfont, const Glyph *string, hb_codepoint_t *codepoints, int start, int length) -+{ -+ hb_font_t *font = hbfindfont(xfont); -+ if (font == NULL) -+ return; -+ -+ Rune rune; -+ ushort mode = USHRT_MAX; -+ hb_buffer_t *buffer = hb_buffer_create(); -+ hb_buffer_set_direction(buffer, HB_DIRECTION_LTR); -+ -+ /* Fill buffer with codepoints. */ -+ for (int i = start; i < (start+length); i++) { -+ rune = string[i].u; -+ mode = string[i].mode; -+ if (mode & ATTR_WDUMMY) -+ rune = 0x0020; -+ hb_buffer_add_codepoints(buffer, &rune, 1, 0, 1); -+ } -+ -+ /* Shape the segment. */ -+ hb_shape(font, buffer, NULL, 0); -+ -+ /* Get new glyph info. */ -+ hb_glyph_info_t *info = hb_buffer_get_glyph_infos(buffer, NULL); -+ -+ /* Write new codepoints. */ -+ for (int i = 0; i < length; i++) { -+ hb_codepoint_t gid = info[i].codepoint; -+ codepoints[start+i] = gid; -+ } -+ -+ /* Cleanup. */ -+ hb_buffer_destroy(buffer); -+} -diff --git a/hb.h b/hb.h -new file mode 100644 -index 0000000..b3e02d0 ---- /dev/null -+++ b/hb.h -@@ -0,0 +1,7 @@ -+#include -+#include -+#include -+ -+void hbunloadfonts(); -+void hbtransform(XftGlyphFontSpec *, const Glyph *, size_t, int, int); -+ -diff --git a/st.c b/st.c -index 2bf133f..747f7b4 100644 ---- a/st.c -+++ b/st.c -@@ -2599,7 +2599,8 @@ draw(void) - drawregion(0, 0, term.col, term.row); - xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], -- term.ocx, term.ocy, term.line[term.ocy][term.ocx]); -+ term.ocx, term.ocy, term.line[term.ocy][term.ocx], -+ term.line[term.ocy], term.col); - term.ocx = cx; - term.ocy = term.c.y; - xfinishdraw(); -diff --git a/st.h b/st.h -index d978458..c9b279b 100644 ---- a/st.h -+++ b/st.h -@@ -11,7 +11,8 @@ - #define DIVCEIL(n, d) (((n) + ((d) - 1)) / (d)) - #define DEFAULT(a, b) (a) = (a) ? (a) : (b) - #define LIMIT(x, a, b) (x) = (x) < (a) ? (a) : (x) > (b) ? (b) : (x) --#define ATTRCMP(a, b) ((a).mode != (b).mode || (a).fg != (b).fg || \ -+#define ATTRCMP(a, b) (((a).mode & (~ATTR_WRAP) & (~ATTR_LIGA)) != ((b).mode & (~ATTR_WRAP) & (~ATTR_LIGA)) || \ -+ (a).fg != (b).fg || \ - (a).bg != (b).bg) - #define TIMEDIFF(t1, t2) ((t1.tv_sec-t2.tv_sec)*1000 + \ - (t1.tv_nsec-t2.tv_nsec)/1E6) -@@ -33,6 +34,7 @@ enum glyph_attribute { - ATTR_WRAP = 1 << 8, - ATTR_WIDE = 1 << 9, - ATTR_WDUMMY = 1 << 10, -+ ATTR_LIGA = 1 << 11, - ATTR_BOLD_FAINT = ATTR_BOLD | ATTR_FAINT, - }; - -diff --git a/win.h b/win.h -index a6ef1b9..bc0d180 100644 ---- a/win.h -+++ b/win.h -@@ -25,7 +25,7 @@ enum win_mode { - - void xbell(void); - void xclipcopy(void); --void xdrawcursor(int, int, Glyph, int, int, Glyph); -+void xdrawcursor(int, int, Glyph, int, int, Glyph, Line, int); - void xdrawline(Line, int, int, int); - void xfinishdraw(void); - void xloadcols(void); -diff --git a/x.c b/x.c -index e5f1737..3334a83 100644 ---- a/x.c -+++ b/x.c -@@ -19,6 +19,7 @@ char *argv0; - #include "arg.h" - #include "st.h" - #include "win.h" -+#include "hb.h" - - /* types used in config.h */ - typedef struct { -@@ -1031,6 +1032,9 @@ xunloadfont(Font *f) - void - xunloadfonts(void) - { -+ /* Clear Harfbuzz font cache. */ -+ hbunloadfonts(); -+ - /* Free the loaded fonts in the font cache. */ - while (frclen > 0) - XftFontClose(xw.dpy, frc[--frclen].font); -@@ -1229,7 +1233,7 @@ xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x - mode = glyphs[i].mode; - - /* Skip dummy wide-character spacing. */ -- if (mode == ATTR_WDUMMY) -+ if (mode & ATTR_WDUMMY) - continue; - - /* Determine font for glyph if different from previous glyph. */ -@@ -1336,6 +1340,9 @@ xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x - numspecs++; - } - -+ /* Harfbuzz transformation for ligatures. */ -+ hbtransform(specs, glyphs, len, x, y); -+ - return numspecs; - } - -@@ -1485,14 +1492,17 @@ xdrawglyph(Glyph g, int x, int y) - } - - void --xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og) -+xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og, Line line, int len) - { - Color drawcol; - - /* remove the old cursor */ - if (selected(ox, oy)) - og.mode ^= ATTR_REVERSE; -- xdrawglyph(og, ox, oy); -+ -+ /* Redraw the line where cursor was previously. -+ * It will restore the ligatures broken by the cursor. */ -+ xdrawline(line, 0, oy, len); - - if (IS_SET(MODE_HIDE)) - return; diff --git a/.suckless/st/patches/st-nordtheme-0.8.2.diff b/.suckless/st/patches/st-nordtheme-0.8.2.diff deleted file mode 100644 index 01e9231..0000000 --- a/.suckless/st/patches/st-nordtheme-0.8.2.diff +++ /dev/null @@ -1,100 +0,0 @@ -From 3e8f83d40c6fdd89653ef547aa16cffe1f3c1793 Mon Sep 17 00:00:00 2001 -From: aleks -Date: Sun, 4 Aug 2019 13:23:10 +0200 -Subject: [PATCH] Apply nord-theme - -*"Inspired by the beauty of the arctic, the colors reflect the cold, yet -harmonious world of ice and the colorfulness of the Aurora Borealis."* - [Nord -Theme](https://www.nordtheme.com/) - -There are also many [ports](https://www.nordtheme.com/ports) for other programs -like [vim](https://www.nordtheme.com/ports/vim) and -[tmux](https://www.nordtheme.com/ports/tmux) to make the overall appearance -coherent. I would recommend to use it in combination with the arc-theme for gtk -(fits perfectly). - -The default behaviour of st is to reverse the fore- and background colors of -each selected cell. If you want that the selection-colors are not reveresed but -instead have fixed fore- and background colors apply on top of this patch the -[selectioncolors](../selectioncolors/)-patch. Then set the following settings -in your config.h: - -static unsigned int defaultcs = 257; -static unsigned int defaultrcs = 257; -unsigned int selectionfg = 257; -unsigned int selectionbg = 0; ---- - config.def.h | 44 ++++++++++++++++++++++---------------------- - 1 file changed, 22 insertions(+), 22 deletions(-) - -diff --git a/config.def.h b/config.def.h -index 0e01717..fb504d4 100644 ---- a/config.def.h -+++ b/config.def.h -@@ -85,30 +85,30 @@ unsigned int tabspaces = 8; - /* Terminal colors (16 first used in escape sequence) */ - static const char *colorname[] = { - /* 8 normal colors */ -- "black", -- "red3", -- "green3", -- "yellow3", -- "blue2", -- "magenta3", -- "cyan3", -- "gray90", -+ "#3b4252", /* black */ -+ "#bf616a", /* red */ -+ "#a3be8c", /* green */ -+ "#ebcb8b", /* yellow */ -+ "#81a1c1", /* blue */ -+ "#b48ead", /* magenta */ -+ "#88c0d0", /* cyan */ -+ "#e5e9f0", /* white */ - - /* 8 bright colors */ -- "gray50", -- "red", -- "green", -- "yellow", -- "#5c5cff", -- "magenta", -- "cyan", -- "white", -+ "#4c566a", /* black */ -+ "#bf616a", /* red */ -+ "#a3be8c", /* green */ -+ "#ebcb8b", /* yellow */ -+ "#81a1c1", /* blue */ -+ "#b48ead", /* magenta */ -+ "#8fbcbb", /* cyan */ -+ "#eceff4", /* white */ - - [255] = 0, - - /* more colors can be added after 255 to use with DefaultXX */ -- "#cccccc", -- "#555555", -+ "#2e3440", /* background */ -+ "#d8dee9", /* foreground */ - }; - - -@@ -116,10 +116,10 @@ static const char *colorname[] = { - * Default colors (colorname index) - * foreground, background, cursor, reverse cursor - */ --unsigned int defaultfg = 7; --unsigned int defaultbg = 0; --static unsigned int defaultcs = 256; --static unsigned int defaultrcs = 257; -+unsigned int defaultfg = 257; -+unsigned int defaultbg = 256; -+static unsigned int defaultcs = 257; -+static unsigned int defaultrcs = 256; - - /* - * Default shape of cursor --- -2.22.0 - diff --git a/.suckless/st/st b/.suckless/st/st deleted file mode 100755 index bea976a238f8d30bf73659c7327ed6aa4ceac6f8..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 102984 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 Y6-~vf{CI8FW%v#J-aJ_sK{xX}][{+-}{+-}]. See -.BR XParseGeometry (3) -for further details. -.TP -.B \-i -will fixate the position given with the -g option. -.TP -.BI \-n " name" -defines the window instance name (default $TERM). -.TP -.BI \-o " iofile" -writes all the I/O to -.I iofile. -This feature is useful when recording st sessions. A value of "-" means -standard output. -.TP -.BI \-T " title" -defines the window title (default 'st'). -.TP -.BI \-t " title" -defines the window title (default 'st'). -.TP -.BI \-w " windowid" -embeds st within the window identified by -.I windowid -.TP -.BI \-l " line" -use a tty -.I line -instead of a pseudo terminal. -.I line -should be a (pseudo-)serial device (e.g. /dev/ttyS0 on Linux for serial port -0). -When this flag is given -remaining arguments are used as flags for -.BR stty(1). -By default st initializes the serial line to 8 bits, no parity, 1 stop bit -and a 38400 baud rate. The speed is set by appending it as last argument -(e.g. 'st -l /dev/ttyS0 115200'). Arguments before the last one are -.BR stty(1) -flags. If you want to set odd parity on 115200 baud use for example 'st -l -/dev/ttyS0 parenb parodd 115200'. Set the number of bits by using for -example 'st -l /dev/ttyS0 cs7 115200'. See -.BR stty(1) -for more arguments and cases. -.TP -.B \-v -prints version information to stderr, then exits. -.TP -.BI \-e " command " [ " arguments " "... ]" -st executes -.I command -instead of the shell. If this is used it -.B must be the last option -on the command line, as in xterm / rxvt. -This option is only intended for compatibility, -and all the remaining arguments are used as a command -even without it. -.SH SHORTCUTS -.TP -.B Break -Send a break in the serial line. -Break key is obtained in PC keyboards -pressing at the same time control and pause. -.TP -.B Ctrl-Print Screen -Toggle if st should print to the -.I iofile. -.TP -.B Shift-Print Screen -Print the full screen to the -.I iofile. -.TP -.B Print Screen -Print the selection to the -.I iofile. -.TP -.B Ctrl-Shift-Page Up -Increase font size. -.TP -.B Ctrl-Shift-Page Down -Decrease font size. -.TP -.B Ctrl-Shift-Home -Reset to default font size. -.TP -.B Ctrl-Shift-y -Paste from primary selection (middle mouse button). -.TP -.B Ctrl-Shift-c -Copy the selected text to the clipboard selection. -.TP -.B Ctrl-Shift-v -Paste from the clipboard selection. -.SH CUSTOMIZATION -.B st -can be customized by creating a custom config.h and (re)compiling the source -code. This keeps it fast, secure and simple. -.SH AUTHORS -See the LICENSE file for the authors. -.SH LICENSE -See the LICENSE file for the terms of redistribution. -.SH SEE ALSO -.BR tabbed (1), -.BR utmp (1), -.BR stty (1) -.SH BUGS -See the TODO file in the distribution. - diff --git a/.suckless/st/st.c b/.suckless/st/st.c deleted file mode 100644 index f1c58f0..0000000 --- a/.suckless/st/st.c +++ /dev/null @@ -1,2600 +0,0 @@ -/* See LICENSE for license details. */ -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "st.h" -#include "win.h" - -#if defined(__linux) - #include -#elif defined(__OpenBSD__) || defined(__NetBSD__) || defined(__APPLE__) - #include -#elif defined(__FreeBSD__) || defined(__DragonFly__) - #include -#endif - -/* Arbitrary sizes */ -#define UTF_INVALID 0xFFFD -#define UTF_SIZ 4 -#define ESC_BUF_SIZ (128*UTF_SIZ) -#define ESC_ARG_SIZ 16 -#define STR_BUF_SIZ ESC_BUF_SIZ -#define STR_ARG_SIZ ESC_ARG_SIZ - -/* macros */ -#define IS_SET(flag) ((term.mode & (flag)) != 0) -#define ISCONTROLC0(c) (BETWEEN(c, 0, 0x1f) || (c) == '\177') -#define ISCONTROLC1(c) (BETWEEN(c, 0x80, 0x9f)) -#define ISCONTROL(c) (ISCONTROLC0(c) || ISCONTROLC1(c)) -#define ISDELIM(u) (u && wcschr(worddelimiters, u)) - -enum term_mode { - MODE_WRAP = 1 << 0, - MODE_INSERT = 1 << 1, - MODE_ALTSCREEN = 1 << 2, - MODE_CRLF = 1 << 3, - MODE_ECHO = 1 << 4, - MODE_PRINT = 1 << 5, - MODE_UTF8 = 1 << 6, - MODE_SIXEL = 1 << 7, -}; - -enum cursor_movement { - CURSOR_SAVE, - CURSOR_LOAD -}; - -enum cursor_state { - CURSOR_DEFAULT = 0, - CURSOR_WRAPNEXT = 1, - CURSOR_ORIGIN = 2 -}; - -enum charset { - CS_GRAPHIC0, - CS_GRAPHIC1, - CS_UK, - CS_USA, - CS_MULTI, - CS_GER, - CS_FIN -}; - -enum escape_state { - ESC_START = 1, - ESC_CSI = 2, - ESC_STR = 4, /* OSC, PM, APC */ - ESC_ALTCHARSET = 8, - ESC_STR_END = 16, /* a final string was encountered */ - ESC_TEST = 32, /* Enter in test mode */ - ESC_UTF8 = 64, - ESC_DCS =128, -}; - -typedef struct { - Glyph attr; /* current char attributes */ - int x; - int y; - char state; -} TCursor; - -typedef struct { - int mode; - int type; - int snap; - /* - * Selection variables: - * nb – normalized coordinates of the beginning of the selection - * ne – normalized coordinates of the end of the selection - * ob – original coordinates of the beginning of the selection - * oe – original coordinates of the end of the selection - */ - struct { - int x, y; - } nb, ne, ob, oe; - - int alt; -} Selection; - -/* Internal representation of the screen */ -typedef struct { - int row; /* nb row */ - int col; /* nb col */ - Line *line; /* screen */ - Line *alt; /* alternate screen */ - int *dirty; /* dirtyness of lines */ - TCursor c; /* cursor */ - int ocx; /* old cursor col */ - int ocy; /* old cursor row */ - int top; /* top scroll limit */ - int bot; /* bottom scroll limit */ - int mode; /* terminal mode flags */ - int esc; /* escape state flags */ - char trantbl[4]; /* charset table translation */ - int charset; /* current charset */ - int icharset; /* selected charset for sequence */ - int *tabs; -} Term; - -/* CSI Escape sequence structs */ -/* ESC '[' [[ [] [;]] []] */ -typedef struct { - char buf[ESC_BUF_SIZ]; /* raw string */ - size_t len; /* raw string length */ - char priv; - int arg[ESC_ARG_SIZ]; - int narg; /* nb of args */ - char mode[2]; -} CSIEscape; - -/* STR Escape sequence structs */ -/* ESC type [[ [] [;]] ] ESC '\' */ -typedef struct { - char type; /* ESC type ... */ - char *buf; /* allocated raw string */ - size_t siz; /* allocation size */ - size_t len; /* raw string length */ - char *args[STR_ARG_SIZ]; - int narg; /* nb of args */ -} STREscape; - -static void execsh(char *, char **); -static void stty(char **); -static void sigchld(int); -static void ttywriteraw(const char *, size_t); - -static void csidump(void); -static void csihandle(void); -static void csiparse(void); -static void csireset(void); -static int eschandle(uchar); -static void strdump(void); -static void strhandle(void); -static void strparse(void); -static void strreset(void); - -static void tprinter(char *, size_t); -static void tdumpsel(void); -static void tdumpline(int); -static void tdump(void); -static void tclearregion(int, int, int, int); -static void tcursor(int); -static void tdeletechar(int); -static void tdeleteline(int); -static void tinsertblank(int); -static void tinsertblankline(int); -static int tlinelen(int); -static void tmoveto(int, int); -static void tmoveato(int, int); -static void tnewline(int); -static void tputtab(int); -static void tputc(Rune); -static void treset(void); -static void tscrollup(int, int); -static void tscrolldown(int, int); -static void tsetattr(int *, int); -static void tsetchar(Rune, Glyph *, int, int); -static void tsetdirt(int, int); -static void tsetscroll(int, int); -static void tswapscreen(void); -static void tsetmode(int, int, int *, int); -static int twrite(const char *, int, int); -static void tfulldirt(void); -static void tcontrolcode(uchar ); -static void tdectest(char ); -static void tdefutf8(char); -static int32_t tdefcolor(int *, int *, int); -static void tdeftran(char); -static void tstrsequence(uchar); - -static void drawregion(int, int, int, int); - -static void selnormalize(void); -static void selscroll(int, int); -static void selsnap(int *, int *, int); - -static size_t utf8decode(const char *, Rune *, size_t); -static Rune utf8decodebyte(char, size_t *); -static char utf8encodebyte(Rune, size_t); -static size_t utf8validate(Rune *, size_t); - -static char *base64dec(const char *); -static char base64dec_getc(const char **); - -static ssize_t xwrite(int, const char *, size_t); - -/* Globals */ -static Term term; -static Selection sel; -static CSIEscape csiescseq; -static STREscape strescseq; -static int iofd = 1; -static int cmdfd; -static pid_t pid; - -static uchar utfbyte[UTF_SIZ + 1] = {0x80, 0, 0xC0, 0xE0, 0xF0}; -static uchar utfmask[UTF_SIZ + 1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8}; -static Rune utfmin[UTF_SIZ + 1] = { 0, 0, 0x80, 0x800, 0x10000}; -static Rune utfmax[UTF_SIZ + 1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF}; - -ssize_t -xwrite(int fd, const char *s, size_t len) -{ - size_t aux = len; - ssize_t r; - - while (len > 0) { - r = write(fd, s, len); - if (r < 0) - return r; - len -= r; - s += r; - } - - return aux; -} - -void * -xmalloc(size_t len) -{ - void *p; - - if (!(p = malloc(len))) - die("malloc: %s\n", strerror(errno)); - - return p; -} - -void * -xrealloc(void *p, size_t len) -{ - if ((p = realloc(p, len)) == NULL) - die("realloc: %s\n", strerror(errno)); - - return p; -} - -char * -xstrdup(char *s) -{ - if ((s = strdup(s)) == NULL) - die("strdup: %s\n", strerror(errno)); - - return s; -} - -size_t -utf8decode(const char *c, Rune *u, size_t clen) -{ - size_t i, j, len, type; - Rune udecoded; - - *u = UTF_INVALID; - if (!clen) - return 0; - udecoded = utf8decodebyte(c[0], &len); - if (!BETWEEN(len, 1, UTF_SIZ)) - return 1; - for (i = 1, j = 1; i < clen && j < len; ++i, ++j) { - udecoded = (udecoded << 6) | utf8decodebyte(c[i], &type); - if (type != 0) - return j; - } - if (j < len) - return 0; - *u = udecoded; - utf8validate(u, len); - - return len; -} - -Rune -utf8decodebyte(char c, size_t *i) -{ - for (*i = 0; *i < LEN(utfmask); ++(*i)) - if (((uchar)c & utfmask[*i]) == utfbyte[*i]) - return (uchar)c & ~utfmask[*i]; - - return 0; -} - -size_t -utf8encode(Rune u, char *c) -{ - size_t len, i; - - len = utf8validate(&u, 0); - if (len > UTF_SIZ) - return 0; - - for (i = len - 1; i != 0; --i) { - c[i] = utf8encodebyte(u, 0); - u >>= 6; - } - c[0] = utf8encodebyte(u, len); - - return len; -} - -char -utf8encodebyte(Rune u, size_t i) -{ - return utfbyte[i] | (u & ~utfmask[i]); -} - -size_t -utf8validate(Rune *u, size_t i) -{ - if (!BETWEEN(*u, utfmin[i], utfmax[i]) || BETWEEN(*u, 0xD800, 0xDFFF)) - *u = UTF_INVALID; - for (i = 1; *u > utfmax[i]; ++i) - ; - - return i; -} - -static const char base64_digits[] = { - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 0, 0, - 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 0, 0, 0, -1, 0, 0, 0, 0, 1, - 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, - 22, 23, 24, 25, 0, 0, 0, 0, 0, 0, 26, 27, 28, 29, 30, 31, 32, 33, 34, - 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 -}; - -char -base64dec_getc(const char **src) -{ - while (**src && !isprint(**src)) (*src)++; - return **src ? *((*src)++) : '='; /* emulate padding if string ends */ -} - -char * -base64dec(const char *src) -{ - size_t in_len = strlen(src); - char *result, *dst; - - if (in_len % 4) - in_len += 4 - (in_len % 4); - result = dst = xmalloc(in_len / 4 * 3 + 1); - while (*src) { - int a = base64_digits[(unsigned char) base64dec_getc(&src)]; - int b = base64_digits[(unsigned char) base64dec_getc(&src)]; - int c = base64_digits[(unsigned char) base64dec_getc(&src)]; - int d = base64_digits[(unsigned char) base64dec_getc(&src)]; - - /* invalid input. 'a' can be -1, e.g. if src is "\n" (c-str) */ - if (a == -1 || b == -1) - break; - - *dst++ = (a << 2) | ((b & 0x30) >> 4); - if (c == -1) - break; - *dst++ = ((b & 0x0f) << 4) | ((c & 0x3c) >> 2); - if (d == -1) - break; - *dst++ = ((c & 0x03) << 6) | d; - } - *dst = '\0'; - return result; -} - -void -selinit(void) -{ - sel.mode = SEL_IDLE; - sel.snap = 0; - sel.ob.x = -1; -} - -int -tlinelen(int y) -{ - int i = term.col; - - if (term.line[y][i - 1].mode & ATTR_WRAP) - return i; - - while (i > 0 && term.line[y][i - 1].u == ' ') - --i; - - return i; -} - -void -selstart(int col, int row, int snap) -{ - selclear(); - sel.mode = SEL_EMPTY; - sel.type = SEL_REGULAR; - sel.alt = IS_SET(MODE_ALTSCREEN); - sel.snap = snap; - sel.oe.x = sel.ob.x = col; - sel.oe.y = sel.ob.y = row; - selnormalize(); - - if (sel.snap != 0) - sel.mode = SEL_READY; - tsetdirt(sel.nb.y, sel.ne.y); -} - -void -selextend(int col, int row, int type, int done) -{ - int oldey, oldex, oldsby, oldsey, oldtype; - - if (sel.mode == SEL_IDLE) - return; - if (done && sel.mode == SEL_EMPTY) { - selclear(); - return; - } - - oldey = sel.oe.y; - oldex = sel.oe.x; - oldsby = sel.nb.y; - oldsey = sel.ne.y; - oldtype = sel.type; - - sel.oe.x = col; - sel.oe.y = row; - selnormalize(); - sel.type = type; - - if (oldey != sel.oe.y || oldex != sel.oe.x || oldtype != sel.type || sel.mode == SEL_EMPTY) - tsetdirt(MIN(sel.nb.y, oldsby), MAX(sel.ne.y, oldsey)); - - sel.mode = done ? SEL_IDLE : SEL_READY; -} - -void -selnormalize(void) -{ - int i; - - if (sel.type == SEL_REGULAR && sel.ob.y != sel.oe.y) { - sel.nb.x = sel.ob.y < sel.oe.y ? sel.ob.x : sel.oe.x; - sel.ne.x = sel.ob.y < sel.oe.y ? sel.oe.x : sel.ob.x; - } else { - sel.nb.x = MIN(sel.ob.x, sel.oe.x); - sel.ne.x = MAX(sel.ob.x, sel.oe.x); - } - sel.nb.y = MIN(sel.ob.y, sel.oe.y); - sel.ne.y = MAX(sel.ob.y, sel.oe.y); - - selsnap(&sel.nb.x, &sel.nb.y, -1); - selsnap(&sel.ne.x, &sel.ne.y, +1); - - /* expand selection over line breaks */ - if (sel.type == SEL_RECTANGULAR) - return; - i = tlinelen(sel.nb.y); - if (i < sel.nb.x) - sel.nb.x = i; - if (tlinelen(sel.ne.y) <= sel.ne.x) - sel.ne.x = term.col - 1; -} - -int -selected(int x, int y) -{ - if (sel.mode == SEL_EMPTY || sel.ob.x == -1 || - sel.alt != IS_SET(MODE_ALTSCREEN)) - return 0; - - if (sel.type == SEL_RECTANGULAR) - return BETWEEN(y, sel.nb.y, sel.ne.y) - && BETWEEN(x, sel.nb.x, sel.ne.x); - - return BETWEEN(y, sel.nb.y, sel.ne.y) - && (y != sel.nb.y || x >= sel.nb.x) - && (y != sel.ne.y || x <= sel.ne.x); -} - -void -selsnap(int *x, int *y, int direction) -{ - int newx, newy, xt, yt; - int delim, prevdelim; - Glyph *gp, *prevgp; - - switch (sel.snap) { - case SNAP_WORD: - /* - * Snap around if the word wraps around at the end or - * beginning of a line. - */ - prevgp = &term.line[*y][*x]; - prevdelim = ISDELIM(prevgp->u); - for (;;) { - newx = *x + direction; - newy = *y; - if (!BETWEEN(newx, 0, term.col - 1)) { - newy += direction; - newx = (newx + term.col) % term.col; - if (!BETWEEN(newy, 0, term.row - 1)) - break; - - if (direction > 0) - yt = *y, xt = *x; - else - yt = newy, xt = newx; - if (!(term.line[yt][xt].mode & ATTR_WRAP)) - break; - } - - if (newx >= tlinelen(newy)) - break; - - gp = &term.line[newy][newx]; - delim = ISDELIM(gp->u); - if (!(gp->mode & ATTR_WDUMMY) && (delim != prevdelim - || (delim && gp->u != prevgp->u))) - break; - - *x = newx; - *y = newy; - prevgp = gp; - prevdelim = delim; - } - break; - case SNAP_LINE: - /* - * Snap around if the the previous line or the current one - * has set ATTR_WRAP at its end. Then the whole next or - * previous line will be selected. - */ - *x = (direction < 0) ? 0 : term.col - 1; - if (direction < 0) { - for (; *y > 0; *y += direction) { - if (!(term.line[*y-1][term.col-1].mode - & ATTR_WRAP)) { - break; - } - } - } else if (direction > 0) { - for (; *y < term.row-1; *y += direction) { - if (!(term.line[*y][term.col-1].mode - & ATTR_WRAP)) { - break; - } - } - } - break; - } -} - -char * -getsel(void) -{ - char *str, *ptr; - int y, bufsize, lastx, linelen; - Glyph *gp, *last; - - if (sel.ob.x == -1) - return NULL; - - bufsize = (term.col+1) * (sel.ne.y-sel.nb.y+1) * UTF_SIZ; - ptr = str = xmalloc(bufsize); - - /* append every set & selected glyph to the selection */ - for (y = sel.nb.y; y <= sel.ne.y; y++) { - if ((linelen = tlinelen(y)) == 0) { - *ptr++ = '\n'; - continue; - } - - if (sel.type == SEL_RECTANGULAR) { - gp = &term.line[y][sel.nb.x]; - lastx = sel.ne.x; - } else { - gp = &term.line[y][sel.nb.y == y ? sel.nb.x : 0]; - lastx = (sel.ne.y == y) ? sel.ne.x : term.col-1; - } - last = &term.line[y][MIN(lastx, linelen-1)]; - while (last >= gp && last->u == ' ') - --last; - - for ( ; gp <= last; ++gp) { - if (gp->mode & ATTR_WDUMMY) - continue; - - ptr += utf8encode(gp->u, ptr); - } - - /* - * Copy and pasting of line endings is inconsistent - * in the inconsistent terminal and GUI world. - * The best solution seems like to produce '\n' when - * something is copied from st and convert '\n' to - * '\r', when something to be pasted is received by - * st. - * FIXME: Fix the computer world. - */ - if ((y < sel.ne.y || lastx >= linelen) && !(last->mode & ATTR_WRAP)) - *ptr++ = '\n'; - } - *ptr = 0; - return str; -} - -void -selclear(void) -{ - if (sel.ob.x == -1) - return; - sel.mode = SEL_IDLE; - sel.ob.x = -1; - tsetdirt(sel.nb.y, sel.ne.y); -} - -void -die(const char *errstr, ...) -{ - va_list ap; - - va_start(ap, errstr); - vfprintf(stderr, errstr, ap); - va_end(ap); - exit(1); -} - -void -execsh(char *cmd, char **args) -{ - char *sh, *prog; - const struct passwd *pw; - - errno = 0; - if ((pw = getpwuid(getuid())) == NULL) { - if (errno) - die("getpwuid: %s\n", strerror(errno)); - else - die("who are you?\n"); - } - - if ((sh = getenv("SHELL")) == NULL) - sh = (pw->pw_shell[0]) ? pw->pw_shell : cmd; - - if (args) - prog = args[0]; - else if (utmp) - prog = utmp; - else - prog = sh; - DEFAULT(args, ((char *[]) {prog, NULL})); - - unsetenv("COLUMNS"); - unsetenv("LINES"); - unsetenv("TERMCAP"); - setenv("LOGNAME", pw->pw_name, 1); - setenv("USER", pw->pw_name, 1); - setenv("SHELL", sh, 1); - setenv("HOME", pw->pw_dir, 1); - setenv("TERM", termname, 1); - - signal(SIGCHLD, SIG_DFL); - signal(SIGHUP, SIG_DFL); - signal(SIGINT, SIG_DFL); - signal(SIGQUIT, SIG_DFL); - signal(SIGTERM, SIG_DFL); - signal(SIGALRM, SIG_DFL); - - execvp(prog, args); - _exit(1); -} - -void -sigchld(int a) -{ - int stat; - pid_t p; - - if ((p = waitpid(pid, &stat, WNOHANG)) < 0) - die("waiting for pid %hd failed: %s\n", pid, strerror(errno)); - - if (pid != p) - return; - - if (WIFEXITED(stat) && WEXITSTATUS(stat)) - die("child exited with status %d\n", WEXITSTATUS(stat)); - else if (WIFSIGNALED(stat)) - die("child terminated due to signal %d\n", WTERMSIG(stat)); - exit(0); -} - -void -stty(char **args) -{ - char cmd[_POSIX_ARG_MAX], **p, *q, *s; - size_t n, siz; - - if ((n = strlen(stty_args)) > sizeof(cmd)-1) - die("incorrect stty parameters\n"); - memcpy(cmd, stty_args, n); - q = cmd + n; - siz = sizeof(cmd) - n; - for (p = args; p && (s = *p); ++p) { - if ((n = strlen(s)) > siz-1) - die("stty parameter length too long\n"); - *q++ = ' '; - memcpy(q, s, n); - q += n; - siz -= n + 1; - } - *q = '\0'; - if (system(cmd) != 0) - perror("Couldn't call stty"); -} - -int -ttynew(char *line, char *cmd, char *out, char **args) -{ - int m, s; - - if (out) { - term.mode |= MODE_PRINT; - iofd = (!strcmp(out, "-")) ? - 1 : open(out, O_WRONLY | O_CREAT, 0666); - if (iofd < 0) { - fprintf(stderr, "Error opening %s:%s\n", - out, strerror(errno)); - } - } - - if (line) { - if ((cmdfd = open(line, O_RDWR)) < 0) - die("open line '%s' failed: %s\n", - line, strerror(errno)); - dup2(cmdfd, 0); - stty(args); - return cmdfd; - } - - /* seems to work fine on linux, openbsd and freebsd */ - if (openpty(&m, &s, NULL, NULL, NULL) < 0) - die("openpty failed: %s\n", strerror(errno)); - - switch (pid = fork()) { - case -1: - die("fork failed: %s\n", strerror(errno)); - break; - case 0: - close(iofd); - setsid(); /* create a new process group */ - dup2(s, 0); - dup2(s, 1); - dup2(s, 2); - if (ioctl(s, TIOCSCTTY, NULL) < 0) - die("ioctl TIOCSCTTY failed: %s\n", strerror(errno)); - close(s); - close(m); -#ifdef __OpenBSD__ - if (pledge("stdio getpw proc exec", NULL) == -1) - die("pledge\n"); -#endif - execsh(cmd, args); - break; - default: -#ifdef __OpenBSD__ - if (pledge("stdio rpath tty proc", NULL) == -1) - die("pledge\n"); -#endif - close(s); - cmdfd = m; - signal(SIGCHLD, sigchld); - break; - } - return cmdfd; -} - -size_t -ttyread(void) -{ - static char buf[BUFSIZ]; - static int buflen = 0; - int written; - int ret; - - /* append read bytes to unprocessed bytes */ - if ((ret = read(cmdfd, buf+buflen, LEN(buf)-buflen)) < 0) - die("couldn't read from shell: %s\n", strerror(errno)); - buflen += ret; - - written = twrite(buf, buflen, 0); - buflen -= written; - /* keep any uncomplete utf8 char for the next call */ - if (buflen > 0) - memmove(buf, buf + written, buflen); - - return ret; -} - -void -ttywrite(const char *s, size_t n, int may_echo) -{ - const char *next; - - if (may_echo && IS_SET(MODE_ECHO)) - twrite(s, n, 1); - - if (!IS_SET(MODE_CRLF)) { - ttywriteraw(s, n); - return; - } - - /* This is similar to how the kernel handles ONLCR for ttys */ - while (n > 0) { - if (*s == '\r') { - next = s + 1; - ttywriteraw("\r\n", 2); - } else { - next = memchr(s, '\r', n); - DEFAULT(next, s + n); - ttywriteraw(s, next - s); - } - n -= next - s; - s = next; - } -} - -void -ttywriteraw(const char *s, size_t n) -{ - fd_set wfd, rfd; - ssize_t r; - size_t lim = 256; - - /* - * Remember that we are using a pty, which might be a modem line. - * Writing too much will clog the line. That's why we are doing this - * dance. - * FIXME: Migrate the world to Plan 9. - */ - while (n > 0) { - FD_ZERO(&wfd); - FD_ZERO(&rfd); - FD_SET(cmdfd, &wfd); - FD_SET(cmdfd, &rfd); - - /* Check if we can write. */ - if (pselect(cmdfd+1, &rfd, &wfd, NULL, NULL, NULL) < 0) { - if (errno == EINTR) - continue; - die("select failed: %s\n", strerror(errno)); - } - if (FD_ISSET(cmdfd, &wfd)) { - /* - * Only write the bytes written by ttywrite() or the - * default of 256. This seems to be a reasonable value - * for a serial line. Bigger values might clog the I/O. - */ - if ((r = write(cmdfd, s, (n < lim)? n : lim)) < 0) - goto write_error; - if (r < n) { - /* - * We weren't able to write out everything. - * This means the buffer is getting full - * again. Empty it. - */ - if (n < lim) - lim = ttyread(); - n -= r; - s += r; - } else { - /* All bytes have been written. */ - break; - } - } - if (FD_ISSET(cmdfd, &rfd)) - lim = ttyread(); - } - return; - -write_error: - die("write error on tty: %s\n", strerror(errno)); -} - -void -ttyresize(int tw, int th) -{ - struct winsize w; - - w.ws_row = term.row; - w.ws_col = term.col; - w.ws_xpixel = tw; - w.ws_ypixel = th; - if (ioctl(cmdfd, TIOCSWINSZ, &w) < 0) - fprintf(stderr, "Couldn't set window size: %s\n", strerror(errno)); -} - -void -ttyhangup() -{ - /* Send SIGHUP to shell */ - kill(pid, SIGHUP); -} - -int -tattrset(int attr) -{ - int i, j; - - for (i = 0; i < term.row-1; i++) { - for (j = 0; j < term.col-1; j++) { - if (term.line[i][j].mode & attr) - return 1; - } - } - - return 0; -} - -void -tsetdirt(int top, int bot) -{ - int i; - - LIMIT(top, 0, term.row-1); - LIMIT(bot, 0, term.row-1); - - for (i = top; i <= bot; i++) - term.dirty[i] = 1; -} - -void -tsetdirtattr(int attr) -{ - int i, j; - - for (i = 0; i < term.row-1; i++) { - for (j = 0; j < term.col-1; j++) { - if (term.line[i][j].mode & attr) { - tsetdirt(i, i); - break; - } - } - } -} - -void -tfulldirt(void) -{ - tsetdirt(0, term.row-1); -} - -void -tcursor(int mode) -{ - static TCursor c[2]; - int alt = IS_SET(MODE_ALTSCREEN); - - if (mode == CURSOR_SAVE) { - c[alt] = term.c; - } else if (mode == CURSOR_LOAD) { - term.c = c[alt]; - tmoveto(c[alt].x, c[alt].y); - } -} - -void -treset(void) -{ - uint i; - - term.c = (TCursor){{ - .mode = ATTR_NULL, - .fg = defaultfg, - .bg = defaultbg - }, .x = 0, .y = 0, .state = CURSOR_DEFAULT}; - - memset(term.tabs, 0, term.col * sizeof(*term.tabs)); - for (i = tabspaces; i < term.col; i += tabspaces) - term.tabs[i] = 1; - term.top = 0; - term.bot = term.row - 1; - term.mode = MODE_WRAP|MODE_UTF8; - memset(term.trantbl, CS_USA, sizeof(term.trantbl)); - term.charset = 0; - - for (i = 0; i < 2; i++) { - tmoveto(0, 0); - tcursor(CURSOR_SAVE); - tclearregion(0, 0, term.col-1, term.row-1); - tswapscreen(); - } -} - -void -tnew(int col, int row) -{ - term = (Term){ .c = { .attr = { .fg = defaultfg, .bg = defaultbg } } }; - tresize(col, row); - treset(); -} - -void -tswapscreen(void) -{ - Line *tmp = term.line; - - term.line = term.alt; - term.alt = tmp; - term.mode ^= MODE_ALTSCREEN; - tfulldirt(); -} - -void -tscrolldown(int orig, int n) -{ - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - - tsetdirt(orig, term.bot-n); - tclearregion(0, term.bot-n+1, term.col-1, term.bot); - - for (i = term.bot; i >= orig+n; i--) { - temp = term.line[i]; - term.line[i] = term.line[i-n]; - term.line[i-n] = temp; - } - - selscroll(orig, n); -} - -void -tscrollup(int orig, int n) -{ - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - - tclearregion(0, orig, term.col-1, orig+n-1); - tsetdirt(orig+n, term.bot); - - for (i = orig; i <= term.bot-n; i++) { - temp = term.line[i]; - term.line[i] = term.line[i+n]; - term.line[i+n] = temp; - } - - selscroll(orig, -n); -} - -void -selscroll(int orig, int n) -{ - if (sel.ob.x == -1) - return; - - if (BETWEEN(sel.ob.y, orig, term.bot) || BETWEEN(sel.oe.y, orig, term.bot)) { - if ((sel.ob.y += n) > term.bot || (sel.oe.y += n) < term.top) { - selclear(); - return; - } - if (sel.type == SEL_RECTANGULAR) { - if (sel.ob.y < term.top) - sel.ob.y = term.top; - if (sel.oe.y > term.bot) - sel.oe.y = term.bot; - } else { - if (sel.ob.y < term.top) { - sel.ob.y = term.top; - sel.ob.x = 0; - } - if (sel.oe.y > term.bot) { - sel.oe.y = term.bot; - sel.oe.x = term.col; - } - } - selnormalize(); - } -} - -void -tnewline(int first_col) -{ - int y = term.c.y; - - if (y == term.bot) { - tscrollup(term.top, 1); - } else { - y++; - } - tmoveto(first_col ? 0 : term.c.x, y); -} - -void -csiparse(void) -{ - char *p = csiescseq.buf, *np; - long int v; - - csiescseq.narg = 0; - if (*p == '?') { - csiescseq.priv = 1; - p++; - } - - csiescseq.buf[csiescseq.len] = '\0'; - while (p < csiescseq.buf+csiescseq.len) { - np = NULL; - v = strtol(p, &np, 10); - if (np == p) - v = 0; - if (v == LONG_MAX || v == LONG_MIN) - v = -1; - csiescseq.arg[csiescseq.narg++] = v; - p = np; - if (*p != ';' || csiescseq.narg == ESC_ARG_SIZ) - break; - p++; - } - csiescseq.mode[0] = *p++; - csiescseq.mode[1] = (p < csiescseq.buf+csiescseq.len) ? *p : '\0'; -} - -/* for absolute user moves, when decom is set */ -void -tmoveato(int x, int y) -{ - tmoveto(x, y + ((term.c.state & CURSOR_ORIGIN) ? term.top: 0)); -} - -void -tmoveto(int x, int y) -{ - int miny, maxy; - - if (term.c.state & CURSOR_ORIGIN) { - miny = term.top; - maxy = term.bot; - } else { - miny = 0; - maxy = term.row - 1; - } - term.c.state &= ~CURSOR_WRAPNEXT; - term.c.x = LIMIT(x, 0, term.col-1); - term.c.y = LIMIT(y, miny, maxy); -} - -void -tsetchar(Rune u, Glyph *attr, int x, int y) -{ - static char *vt100_0[62] = { /* 0x41 - 0x7e */ - "↑", "↓", "→", "←", "█", "▚", "☃", /* A - G */ - 0, 0, 0, 0, 0, 0, 0, 0, /* H - O */ - 0, 0, 0, 0, 0, 0, 0, 0, /* P - W */ - 0, 0, 0, 0, 0, 0, 0, " ", /* X - _ */ - "◆", "▒", "␉", "␌", "␍", "␊", "°", "±", /* ` - g */ - "␤", "␋", "┘", "┐", "┌", "└", "┼", "⎺", /* h - o */ - "⎻", "─", "⎼", "⎽", "├", "┤", "┴", "┬", /* p - w */ - "│", "≤", "≥", "π", "≠", "£", "·", /* x - ~ */ - }; - - /* - * The table is proudly stolen from rxvt. - */ - if (term.trantbl[term.charset] == CS_GRAPHIC0 && - BETWEEN(u, 0x41, 0x7e) && vt100_0[u - 0x41]) - utf8decode(vt100_0[u - 0x41], &u, UTF_SIZ); - - if (term.line[y][x].mode & ATTR_WIDE) { - if (x+1 < term.col) { - term.line[y][x+1].u = ' '; - term.line[y][x+1].mode &= ~ATTR_WDUMMY; - } - } else if (term.line[y][x].mode & ATTR_WDUMMY) { - term.line[y][x-1].u = ' '; - term.line[y][x-1].mode &= ~ATTR_WIDE; - } - - term.dirty[y] = 1; - term.line[y][x] = *attr; - term.line[y][x].u = u; -} - -void -tclearregion(int x1, int y1, int x2, int y2) -{ - int x, y, temp; - Glyph *gp; - - if (x1 > x2) - temp = x1, x1 = x2, x2 = temp; - if (y1 > y2) - temp = y1, y1 = y2, y2 = temp; - - LIMIT(x1, 0, term.col-1); - LIMIT(x2, 0, term.col-1); - LIMIT(y1, 0, term.row-1); - LIMIT(y2, 0, term.row-1); - - for (y = y1; y <= y2; y++) { - term.dirty[y] = 1; - for (x = x1; x <= x2; x++) { - gp = &term.line[y][x]; - if (selected(x, y)) - selclear(); - gp->fg = term.c.attr.fg; - gp->bg = term.c.attr.bg; - gp->mode = 0; - gp->u = ' '; - } - } -} - -void -tdeletechar(int n) -{ - int dst, src, size; - Glyph *line; - - LIMIT(n, 0, term.col - term.c.x); - - dst = term.c.x; - src = term.c.x + n; - size = term.col - src; - line = term.line[term.c.y]; - - memmove(&line[dst], &line[src], size * sizeof(Glyph)); - tclearregion(term.col-n, term.c.y, term.col-1, term.c.y); -} - -void -tinsertblank(int n) -{ - int dst, src, size; - Glyph *line; - - LIMIT(n, 0, term.col - term.c.x); - - dst = term.c.x + n; - src = term.c.x; - size = term.col - dst; - line = term.line[term.c.y]; - - memmove(&line[dst], &line[src], size * sizeof(Glyph)); - tclearregion(src, term.c.y, dst - 1, term.c.y); -} - -void -tinsertblankline(int n) -{ - if (BETWEEN(term.c.y, term.top, term.bot)) - tscrolldown(term.c.y, n); -} - -void -tdeleteline(int n) -{ - if (BETWEEN(term.c.y, term.top, term.bot)) - tscrollup(term.c.y, n); -} - -int32_t -tdefcolor(int *attr, int *npar, int l) -{ - int32_t idx = -1; - uint r, g, b; - - switch (attr[*npar + 1]) { - case 2: /* direct color in RGB space */ - if (*npar + 4 >= l) { - fprintf(stderr, - "erresc(38): Incorrect number of parameters (%d)\n", - *npar); - break; - } - r = attr[*npar + 2]; - g = attr[*npar + 3]; - b = attr[*npar + 4]; - *npar += 4; - if (!BETWEEN(r, 0, 255) || !BETWEEN(g, 0, 255) || !BETWEEN(b, 0, 255)) - fprintf(stderr, "erresc: bad rgb color (%u,%u,%u)\n", - r, g, b); - else - idx = TRUECOLOR(r, g, b); - break; - case 5: /* indexed color */ - if (*npar + 2 >= l) { - fprintf(stderr, - "erresc(38): Incorrect number of parameters (%d)\n", - *npar); - break; - } - *npar += 2; - if (!BETWEEN(attr[*npar], 0, 255)) - fprintf(stderr, "erresc: bad fgcolor %d\n", attr[*npar]); - else - idx = attr[*npar]; - break; - case 0: /* implemented defined (only foreground) */ - case 1: /* transparent */ - case 3: /* direct color in CMY space */ - case 4: /* direct color in CMYK space */ - default: - fprintf(stderr, - "erresc(38): gfx attr %d unknown\n", attr[*npar]); - break; - } - - return idx; -} - -void -tsetattr(int *attr, int l) -{ - int i; - int32_t idx; - - for (i = 0; i < l; i++) { - switch (attr[i]) { - case 0: - term.c.attr.mode &= ~( - ATTR_BOLD | - ATTR_FAINT | - ATTR_ITALIC | - ATTR_UNDERLINE | - ATTR_BLINK | - ATTR_REVERSE | - ATTR_INVISIBLE | - ATTR_STRUCK ); - term.c.attr.fg = defaultfg; - term.c.attr.bg = defaultbg; - break; - case 1: - term.c.attr.mode |= ATTR_BOLD; - break; - case 2: - term.c.attr.mode |= ATTR_FAINT; - break; - case 3: - term.c.attr.mode |= ATTR_ITALIC; - break; - case 4: - term.c.attr.mode |= ATTR_UNDERLINE; - break; - case 5: /* slow blink */ - /* FALLTHROUGH */ - case 6: /* rapid blink */ - term.c.attr.mode |= ATTR_BLINK; - break; - case 7: - term.c.attr.mode |= ATTR_REVERSE; - break; - case 8: - term.c.attr.mode |= ATTR_INVISIBLE; - break; - case 9: - term.c.attr.mode |= ATTR_STRUCK; - break; - case 22: - term.c.attr.mode &= ~(ATTR_BOLD | ATTR_FAINT); - break; - case 23: - term.c.attr.mode &= ~ATTR_ITALIC; - break; - case 24: - term.c.attr.mode &= ~ATTR_UNDERLINE; - break; - case 25: - term.c.attr.mode &= ~ATTR_BLINK; - break; - case 27: - term.c.attr.mode &= ~ATTR_REVERSE; - break; - case 28: - term.c.attr.mode &= ~ATTR_INVISIBLE; - break; - case 29: - term.c.attr.mode &= ~ATTR_STRUCK; - break; - case 38: - if ((idx = tdefcolor(attr, &i, l)) >= 0) - term.c.attr.fg = idx; - break; - case 39: - term.c.attr.fg = defaultfg; - break; - case 48: - if ((idx = tdefcolor(attr, &i, l)) >= 0) - term.c.attr.bg = idx; - break; - case 49: - term.c.attr.bg = defaultbg; - break; - default: - if (BETWEEN(attr[i], 30, 37)) { - term.c.attr.fg = attr[i] - 30; - } else if (BETWEEN(attr[i], 40, 47)) { - term.c.attr.bg = attr[i] - 40; - } else if (BETWEEN(attr[i], 90, 97)) { - term.c.attr.fg = attr[i] - 90 + 8; - } else if (BETWEEN(attr[i], 100, 107)) { - term.c.attr.bg = attr[i] - 100 + 8; - } else { - fprintf(stderr, - "erresc(default): gfx attr %d unknown\n", - attr[i]); - csidump(); - } - break; - } - } -} - -void -tsetscroll(int t, int b) -{ - int temp; - - LIMIT(t, 0, term.row-1); - LIMIT(b, 0, term.row-1); - if (t > b) { - temp = t; - t = b; - b = temp; - } - term.top = t; - term.bot = b; -} - -void -tsetmode(int priv, int set, int *args, int narg) -{ - int alt, *lim; - - for (lim = args + narg; args < lim; ++args) { - if (priv) { - switch (*args) { - case 1: /* DECCKM -- Cursor key */ - xsetmode(set, MODE_APPCURSOR); - break; - case 5: /* DECSCNM -- Reverse video */ - xsetmode(set, MODE_REVERSE); - break; - case 6: /* DECOM -- Origin */ - MODBIT(term.c.state, set, CURSOR_ORIGIN); - tmoveato(0, 0); - break; - case 7: /* DECAWM -- Auto wrap */ - MODBIT(term.mode, set, MODE_WRAP); - break; - case 0: /* Error (IGNORED) */ - case 2: /* DECANM -- ANSI/VT52 (IGNORED) */ - case 3: /* DECCOLM -- Column (IGNORED) */ - case 4: /* DECSCLM -- Scroll (IGNORED) */ - case 8: /* DECARM -- Auto repeat (IGNORED) */ - case 18: /* DECPFF -- Printer feed (IGNORED) */ - case 19: /* DECPEX -- Printer extent (IGNORED) */ - case 42: /* DECNRCM -- National characters (IGNORED) */ - case 12: /* att610 -- Start blinking cursor (IGNORED) */ - break; - case 25: /* DECTCEM -- Text Cursor Enable Mode */ - xsetmode(!set, MODE_HIDE); - break; - case 9: /* X10 mouse compatibility mode */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEX10); - break; - case 1000: /* 1000: report button press */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEBTN); - break; - case 1002: /* 1002: report motion on button press */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEMOTION); - break; - case 1003: /* 1003: enable all mouse motions */ - xsetpointermotion(set); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEMANY); - break; - case 1004: /* 1004: send focus events to tty */ - xsetmode(set, MODE_FOCUS); - break; - case 1006: /* 1006: extended reporting mode */ - xsetmode(set, MODE_MOUSESGR); - break; - case 1034: - xsetmode(set, MODE_8BIT); - break; - case 1049: /* swap screen & set/restore cursor as xterm */ - if (!allowaltscreen) - break; - tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD); - /* FALLTHROUGH */ - case 47: /* swap screen */ - case 1047: - if (!allowaltscreen) - break; - alt = IS_SET(MODE_ALTSCREEN); - if (alt) { - tclearregion(0, 0, term.col-1, - term.row-1); - } - if (set ^ alt) /* set is always 1 or 0 */ - tswapscreen(); - if (*args != 1049) - break; - /* FALLTHROUGH */ - case 1048: - tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD); - break; - case 2004: /* 2004: bracketed paste mode */ - xsetmode(set, MODE_BRCKTPASTE); - break; - /* Not implemented mouse modes. See comments there. */ - case 1001: /* mouse highlight mode; can hang the - terminal by design when implemented. */ - case 1005: /* UTF-8 mouse mode; will confuse - applications not supporting UTF-8 - and luit. */ - case 1015: /* urxvt mangled mouse mode; incompatible - and can be mistaken for other control - codes. */ - break; - default: - fprintf(stderr, - "erresc: unknown private set/reset mode %d\n", - *args); - break; - } - } else { - switch (*args) { - case 0: /* Error (IGNORED) */ - break; - case 2: - xsetmode(set, MODE_KBDLOCK); - break; - case 4: /* IRM -- Insertion-replacement */ - MODBIT(term.mode, set, MODE_INSERT); - break; - case 12: /* SRM -- Send/Receive */ - MODBIT(term.mode, !set, MODE_ECHO); - break; - case 20: /* LNM -- Linefeed/new line */ - MODBIT(term.mode, set, MODE_CRLF); - break; - default: - fprintf(stderr, - "erresc: unknown set/reset mode %d\n", - *args); - break; - } - } - } -} - -void -csihandle(void) -{ - char buf[40]; - int len; - - switch (csiescseq.mode[0]) { - default: - unknown: - fprintf(stderr, "erresc: unknown csi "); - csidump(); - /* die(""); */ - break; - case '@': /* ICH -- Insert blank char */ - DEFAULT(csiescseq.arg[0], 1); - tinsertblank(csiescseq.arg[0]); - break; - case 'A': /* CUU -- Cursor Up */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x, term.c.y-csiescseq.arg[0]); - break; - case 'B': /* CUD -- Cursor Down */ - case 'e': /* VPR --Cursor Down */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x, term.c.y+csiescseq.arg[0]); - break; - case 'i': /* MC -- Media Copy */ - switch (csiescseq.arg[0]) { - case 0: - tdump(); - break; - case 1: - tdumpline(term.c.y); - break; - case 2: - tdumpsel(); - break; - case 4: - term.mode &= ~MODE_PRINT; - break; - case 5: - term.mode |= MODE_PRINT; - break; - } - break; - case 'c': /* DA -- Device Attributes */ - if (csiescseq.arg[0] == 0) - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 'C': /* CUF -- Cursor Forward */ - case 'a': /* HPR -- Cursor Forward */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x+csiescseq.arg[0], term.c.y); - break; - case 'D': /* CUB -- Cursor Backward */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x-csiescseq.arg[0], term.c.y); - break; - case 'E': /* CNL -- Cursor Down and first col */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(0, term.c.y+csiescseq.arg[0]); - break; - case 'F': /* CPL -- Cursor Up and first col */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(0, term.c.y-csiescseq.arg[0]); - break; - case 'g': /* TBC -- Tabulation clear */ - switch (csiescseq.arg[0]) { - case 0: /* clear current tab stop */ - term.tabs[term.c.x] = 0; - break; - case 3: /* clear all the tabs */ - memset(term.tabs, 0, term.col * sizeof(*term.tabs)); - break; - default: - goto unknown; - } - break; - case 'G': /* CHA -- Move to */ - case '`': /* HPA */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(csiescseq.arg[0]-1, term.c.y); - break; - case 'H': /* CUP -- Move to */ - case 'f': /* HVP */ - DEFAULT(csiescseq.arg[0], 1); - DEFAULT(csiescseq.arg[1], 1); - tmoveato(csiescseq.arg[1]-1, csiescseq.arg[0]-1); - break; - case 'I': /* CHT -- Cursor Forward Tabulation tab stops */ - DEFAULT(csiescseq.arg[0], 1); - tputtab(csiescseq.arg[0]); - break; - case 'J': /* ED -- Clear screen */ - switch (csiescseq.arg[0]) { - case 0: /* below */ - tclearregion(term.c.x, term.c.y, term.col-1, term.c.y); - if (term.c.y < term.row-1) { - tclearregion(0, term.c.y+1, term.col-1, - term.row-1); - } - break; - case 1: /* above */ - if (term.c.y > 1) - tclearregion(0, 0, term.col-1, term.c.y-1); - tclearregion(0, term.c.y, term.c.x, term.c.y); - break; - case 2: /* all */ - tclearregion(0, 0, term.col-1, term.row-1); - break; - default: - goto unknown; - } - break; - case 'K': /* EL -- Clear line */ - switch (csiescseq.arg[0]) { - case 0: /* right */ - tclearregion(term.c.x, term.c.y, term.col-1, - term.c.y); - break; - case 1: /* left */ - tclearregion(0, term.c.y, term.c.x, term.c.y); - break; - case 2: /* all */ - tclearregion(0, term.c.y, term.col-1, term.c.y); - break; - } - break; - case 'S': /* SU -- Scroll line up */ - DEFAULT(csiescseq.arg[0], 1); - tscrollup(term.top, csiescseq.arg[0]); - break; - case 'T': /* SD -- Scroll line down */ - DEFAULT(csiescseq.arg[0], 1); - tscrolldown(term.top, csiescseq.arg[0]); - break; - case 'L': /* IL -- Insert blank lines */ - DEFAULT(csiescseq.arg[0], 1); - tinsertblankline(csiescseq.arg[0]); - break; - case 'l': /* RM -- Reset Mode */ - tsetmode(csiescseq.priv, 0, csiescseq.arg, csiescseq.narg); - break; - case 'M': /* DL -- Delete lines */ - DEFAULT(csiescseq.arg[0], 1); - tdeleteline(csiescseq.arg[0]); - break; - case 'X': /* ECH -- Erase char */ - DEFAULT(csiescseq.arg[0], 1); - tclearregion(term.c.x, term.c.y, - term.c.x + csiescseq.arg[0] - 1, term.c.y); - break; - case 'P': /* DCH -- Delete char */ - DEFAULT(csiescseq.arg[0], 1); - tdeletechar(csiescseq.arg[0]); - break; - case 'Z': /* CBT -- Cursor Backward Tabulation tab stops */ - DEFAULT(csiescseq.arg[0], 1); - tputtab(-csiescseq.arg[0]); - break; - case 'd': /* VPA -- Move to */ - DEFAULT(csiescseq.arg[0], 1); - tmoveato(term.c.x, csiescseq.arg[0]-1); - break; - case 'h': /* SM -- Set terminal mode */ - tsetmode(csiescseq.priv, 1, csiescseq.arg, csiescseq.narg); - break; - case 'm': /* SGR -- Terminal attribute (color) */ - tsetattr(csiescseq.arg, csiescseq.narg); - break; - case 'n': /* DSR – Device Status Report (cursor position) */ - if (csiescseq.arg[0] == 6) { - len = snprintf(buf, sizeof(buf),"\033[%i;%iR", - term.c.y+1, term.c.x+1); - ttywrite(buf, len, 0); - } - break; - case 'r': /* DECSTBM -- Set Scrolling Region */ - if (csiescseq.priv) { - goto unknown; - } else { - DEFAULT(csiescseq.arg[0], 1); - DEFAULT(csiescseq.arg[1], term.row); - tsetscroll(csiescseq.arg[0]-1, csiescseq.arg[1]-1); - tmoveato(0, 0); - } - break; - case 's': /* DECSC -- Save cursor position (ANSI.SYS) */ - tcursor(CURSOR_SAVE); - break; - case 'u': /* DECRC -- Restore cursor position (ANSI.SYS) */ - tcursor(CURSOR_LOAD); - break; - case ' ': - switch (csiescseq.mode[1]) { - case 'q': /* DECSCUSR -- Set Cursor Style */ - if (xsetcursor(csiescseq.arg[0])) - goto unknown; - break; - default: - goto unknown; - } - break; - } -} - -void -csidump(void) -{ - size_t i; - uint c; - - fprintf(stderr, "ESC["); - for (i = 0; i < csiescseq.len; i++) { - c = csiescseq.buf[i] & 0xff; - if (isprint(c)) { - putc(c, stderr); - } else if (c == '\n') { - fprintf(stderr, "(\\n)"); - } else if (c == '\r') { - fprintf(stderr, "(\\r)"); - } else if (c == 0x1b) { - fprintf(stderr, "(\\e)"); - } else { - fprintf(stderr, "(%02x)", c); - } - } - putc('\n', stderr); -} - -void -csireset(void) -{ - memset(&csiescseq, 0, sizeof(csiescseq)); -} - -void -strhandle(void) -{ - char *p = NULL, *dec; - int j, narg, par; - - term.esc &= ~(ESC_STR_END|ESC_STR); - strparse(); - par = (narg = strescseq.narg) ? atoi(strescseq.args[0]) : 0; - - switch (strescseq.type) { - case ']': /* OSC -- Operating System Command */ - switch (par) { - case 0: - case 1: - case 2: - if (narg > 1) - xsettitle(strescseq.args[1]); - return; - case 52: - if (narg > 2) { - dec = base64dec(strescseq.args[2]); - if (dec) { - xsetsel(dec); - xclipcopy(); - } else { - fprintf(stderr, "erresc: invalid base64\n"); - } - } - return; - case 4: /* color set */ - if (narg < 3) - break; - p = strescseq.args[2]; - /* FALLTHROUGH */ - case 104: /* color reset, here p = NULL */ - j = (narg > 1) ? atoi(strescseq.args[1]) : -1; - if (xsetcolorname(j, p)) { - if (par == 104 && narg <= 1) - return; /* color reset without parameter */ - fprintf(stderr, "erresc: invalid color j=%d, p=%s\n", - j, p ? p : "(null)"); - } else { - /* - * TODO if defaultbg color is changed, borders - * are dirty - */ - redraw(); - } - return; - } - break; - case 'k': /* old title set compatibility */ - xsettitle(strescseq.args[0]); - return; - case 'P': /* DCS -- Device Control String */ - term.mode |= ESC_DCS; - case '_': /* APC -- Application Program Command */ - case '^': /* PM -- Privacy Message */ - return; - } - - fprintf(stderr, "erresc: unknown str "); - strdump(); -} - -void -strparse(void) -{ - int c; - char *p = strescseq.buf; - - strescseq.narg = 0; - strescseq.buf[strescseq.len] = '\0'; - - if (*p == '\0') - return; - - while (strescseq.narg < STR_ARG_SIZ) { - strescseq.args[strescseq.narg++] = p; - while ((c = *p) != ';' && c != '\0') - ++p; - if (c == '\0') - return; - *p++ = '\0'; - } -} - -void -strdump(void) -{ - size_t i; - uint c; - - fprintf(stderr, "ESC%c", strescseq.type); - for (i = 0; i < strescseq.len; i++) { - c = strescseq.buf[i] & 0xff; - if (c == '\0') { - putc('\n', stderr); - return; - } else if (isprint(c)) { - putc(c, stderr); - } else if (c == '\n') { - fprintf(stderr, "(\\n)"); - } else if (c == '\r') { - fprintf(stderr, "(\\r)"); - } else if (c == 0x1b) { - fprintf(stderr, "(\\e)"); - } else { - fprintf(stderr, "(%02x)", c); - } - } - fprintf(stderr, "ESC\\\n"); -} - -void -strreset(void) -{ - strescseq = (STREscape){ - .buf = xrealloc(strescseq.buf, STR_BUF_SIZ), - .siz = STR_BUF_SIZ, - }; -} - -void -sendbreak(const Arg *arg) -{ - if (tcsendbreak(cmdfd, 0)) - perror("Error sending break"); -} - -void -tprinter(char *s, size_t len) -{ - if (iofd != -1 && xwrite(iofd, s, len) < 0) { - perror("Error writing to output file"); - close(iofd); - iofd = -1; - } -} - -void -toggleprinter(const Arg *arg) -{ - term.mode ^= MODE_PRINT; -} - -void -printscreen(const Arg *arg) -{ - tdump(); -} - -void -printsel(const Arg *arg) -{ - tdumpsel(); -} - -void -tdumpsel(void) -{ - char *ptr; - - if ((ptr = getsel())) { - tprinter(ptr, strlen(ptr)); - free(ptr); - } -} - -void -tdumpline(int n) -{ - char buf[UTF_SIZ]; - Glyph *bp, *end; - - bp = &term.line[n][0]; - end = &bp[MIN(tlinelen(n), term.col) - 1]; - if (bp != end || bp->u != ' ') { - for ( ;bp <= end; ++bp) - tprinter(buf, utf8encode(bp->u, buf)); - } - tprinter("\n", 1); -} - -void -tdump(void) -{ - int i; - - for (i = 0; i < term.row; ++i) - tdumpline(i); -} - -void -tputtab(int n) -{ - uint x = term.c.x; - - if (n > 0) { - while (x < term.col && n--) - for (++x; x < term.col && !term.tabs[x]; ++x) - /* nothing */ ; - } else if (n < 0) { - while (x > 0 && n++) - for (--x; x > 0 && !term.tabs[x]; --x) - /* nothing */ ; - } - term.c.x = LIMIT(x, 0, term.col-1); -} - -void -tdefutf8(char ascii) -{ - if (ascii == 'G') - term.mode |= MODE_UTF8; - else if (ascii == '@') - term.mode &= ~MODE_UTF8; -} - -void -tdeftran(char ascii) -{ - static char cs[] = "0B"; - static int vcs[] = {CS_GRAPHIC0, CS_USA}; - char *p; - - if ((p = strchr(cs, ascii)) == NULL) { - fprintf(stderr, "esc unhandled charset: ESC ( %c\n", ascii); - } else { - term.trantbl[term.icharset] = vcs[p - cs]; - } -} - -void -tdectest(char c) -{ - int x, y; - - if (c == '8') { /* DEC screen alignment test. */ - for (x = 0; x < term.col; ++x) { - for (y = 0; y < term.row; ++y) - tsetchar('E', &term.c.attr, x, y); - } - } -} - -void -tstrsequence(uchar c) -{ - strreset(); - - switch (c) { - case 0x90: /* DCS -- Device Control String */ - c = 'P'; - term.esc |= ESC_DCS; - break; - case 0x9f: /* APC -- Application Program Command */ - c = '_'; - break; - case 0x9e: /* PM -- Privacy Message */ - c = '^'; - break; - case 0x9d: /* OSC -- Operating System Command */ - c = ']'; - break; - } - strescseq.type = c; - term.esc |= ESC_STR; -} - -void -tcontrolcode(uchar ascii) -{ - switch (ascii) { - case '\t': /* HT */ - tputtab(1); - return; - case '\b': /* BS */ - tmoveto(term.c.x-1, term.c.y); - return; - case '\r': /* CR */ - tmoveto(0, term.c.y); - return; - case '\f': /* LF */ - case '\v': /* VT */ - case '\n': /* LF */ - /* go to first col if the mode is set */ - tnewline(IS_SET(MODE_CRLF)); - return; - case '\a': /* BEL */ - if (term.esc & ESC_STR_END) { - /* backwards compatibility to xterm */ - strhandle(); - } else { - xbell(); - } - break; - case '\033': /* ESC */ - csireset(); - term.esc &= ~(ESC_CSI|ESC_ALTCHARSET|ESC_TEST); - term.esc |= ESC_START; - return; - case '\016': /* SO (LS1 -- Locking shift 1) */ - case '\017': /* SI (LS0 -- Locking shift 0) */ - term.charset = 1 - (ascii - '\016'); - return; - case '\032': /* SUB */ - tsetchar('?', &term.c.attr, term.c.x, term.c.y); - case '\030': /* CAN */ - csireset(); - break; - case '\005': /* ENQ (IGNORED) */ - case '\000': /* NUL (IGNORED) */ - case '\021': /* XON (IGNORED) */ - case '\023': /* XOFF (IGNORED) */ - case 0177: /* DEL (IGNORED) */ - return; - case 0x80: /* TODO: PAD */ - case 0x81: /* TODO: HOP */ - case 0x82: /* TODO: BPH */ - case 0x83: /* TODO: NBH */ - case 0x84: /* TODO: IND */ - break; - case 0x85: /* NEL -- Next line */ - tnewline(1); /* always go to first col */ - break; - case 0x86: /* TODO: SSA */ - case 0x87: /* TODO: ESA */ - break; - case 0x88: /* HTS -- Horizontal tab stop */ - term.tabs[term.c.x] = 1; - break; - case 0x89: /* TODO: HTJ */ - case 0x8a: /* TODO: VTS */ - case 0x8b: /* TODO: PLD */ - case 0x8c: /* TODO: PLU */ - case 0x8d: /* TODO: RI */ - case 0x8e: /* TODO: SS2 */ - case 0x8f: /* TODO: SS3 */ - case 0x91: /* TODO: PU1 */ - case 0x92: /* TODO: PU2 */ - case 0x93: /* TODO: STS */ - case 0x94: /* TODO: CCH */ - case 0x95: /* TODO: MW */ - case 0x96: /* TODO: SPA */ - case 0x97: /* TODO: EPA */ - case 0x98: /* TODO: SOS */ - case 0x99: /* TODO: SGCI */ - break; - case 0x9a: /* DECID -- Identify Terminal */ - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 0x9b: /* TODO: CSI */ - case 0x9c: /* TODO: ST */ - break; - case 0x90: /* DCS -- Device Control String */ - case 0x9d: /* OSC -- Operating System Command */ - case 0x9e: /* PM -- Privacy Message */ - case 0x9f: /* APC -- Application Program Command */ - tstrsequence(ascii); - return; - } - /* only CAN, SUB, \a and C1 chars interrupt a sequence */ - term.esc &= ~(ESC_STR_END|ESC_STR); -} - -/* - * returns 1 when the sequence is finished and it hasn't to read - * more characters for this sequence, otherwise 0 - */ -int -eschandle(uchar ascii) -{ - switch (ascii) { - case '[': - term.esc |= ESC_CSI; - return 0; - case '#': - term.esc |= ESC_TEST; - return 0; - case '%': - term.esc |= ESC_UTF8; - return 0; - case 'P': /* DCS -- Device Control String */ - case '_': /* APC -- Application Program Command */ - case '^': /* PM -- Privacy Message */ - case ']': /* OSC -- Operating System Command */ - case 'k': /* old title set compatibility */ - tstrsequence(ascii); - return 0; - case 'n': /* LS2 -- Locking shift 2 */ - case 'o': /* LS3 -- Locking shift 3 */ - term.charset = 2 + (ascii - 'n'); - break; - case '(': /* GZD4 -- set primary charset G0 */ - case ')': /* G1D4 -- set secondary charset G1 */ - case '*': /* G2D4 -- set tertiary charset G2 */ - case '+': /* G3D4 -- set quaternary charset G3 */ - term.icharset = ascii - '('; - term.esc |= ESC_ALTCHARSET; - return 0; - case 'D': /* IND -- Linefeed */ - if (term.c.y == term.bot) { - tscrollup(term.top, 1); - } else { - tmoveto(term.c.x, term.c.y+1); - } - break; - case 'E': /* NEL -- Next line */ - tnewline(1); /* always go to first col */ - break; - case 'H': /* HTS -- Horizontal tab stop */ - term.tabs[term.c.x] = 1; - break; - case 'M': /* RI -- Reverse index */ - if (term.c.y == term.top) { - tscrolldown(term.top, 1); - } else { - tmoveto(term.c.x, term.c.y-1); - } - break; - case 'Z': /* DECID -- Identify Terminal */ - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 'c': /* RIS -- Reset to initial state */ - treset(); - resettitle(); - xloadcols(); - break; - case '=': /* DECPAM -- Application keypad */ - xsetmode(1, MODE_APPKEYPAD); - break; - case '>': /* DECPNM -- Normal keypad */ - xsetmode(0, MODE_APPKEYPAD); - break; - case '7': /* DECSC -- Save Cursor */ - tcursor(CURSOR_SAVE); - break; - case '8': /* DECRC -- Restore Cursor */ - tcursor(CURSOR_LOAD); - break; - case '\\': /* ST -- String Terminator */ - if (term.esc & ESC_STR_END) - strhandle(); - break; - default: - fprintf(stderr, "erresc: unknown sequence ESC 0x%02X '%c'\n", - (uchar) ascii, isprint(ascii)? ascii:'.'); - break; - } - return 1; -} - -void -tputc(Rune u) -{ - char c[UTF_SIZ]; - int control; - int width, len; - Glyph *gp; - - control = ISCONTROL(u); - if (!IS_SET(MODE_UTF8) && !IS_SET(MODE_SIXEL)) { - c[0] = u; - width = len = 1; - } else { - len = utf8encode(u, c); - if (!control && (width = wcwidth(u)) == -1) { - memcpy(c, "\357\277\275", 4); /* UTF_INVALID */ - width = 1; - } - } - - if (IS_SET(MODE_PRINT)) - tprinter(c, len); - - /* - * STR sequence must be checked before anything else - * because it uses all following characters until it - * receives a ESC, a SUB, a ST or any other C1 control - * character. - */ - if (term.esc & ESC_STR) { - if (u == '\a' || u == 030 || u == 032 || u == 033 || - ISCONTROLC1(u)) { - term.esc &= ~(ESC_START|ESC_STR|ESC_DCS); - if (IS_SET(MODE_SIXEL)) { - /* TODO: render sixel */; - term.mode &= ~MODE_SIXEL; - return; - } - term.esc |= ESC_STR_END; - goto check_control_code; - } - - if (IS_SET(MODE_SIXEL)) { - /* TODO: implement sixel mode */ - return; - } - if (term.esc&ESC_DCS && strescseq.len == 0 && u == 'q') - term.mode |= MODE_SIXEL; - - if (strescseq.len+len >= strescseq.siz) { - /* - * Here is a bug in terminals. If the user never sends - * some code to stop the str or esc command, then st - * will stop responding. But this is better than - * silently failing with unknown characters. At least - * then users will report back. - * - * In the case users ever get fixed, here is the code: - */ - /* - * term.esc = 0; - * strhandle(); - */ - if (strescseq.siz > (SIZE_MAX - UTF_SIZ) / 2) - return; - strescseq.siz *= 2; - strescseq.buf = xrealloc(strescseq.buf, strescseq.siz); - } - - memmove(&strescseq.buf[strescseq.len], c, len); - strescseq.len += len; - return; - } - -check_control_code: - /* - * Actions of control codes must be performed as soon they arrive - * because they can be embedded inside a control sequence, and - * they must not cause conflicts with sequences. - */ - if (control) { - tcontrolcode(u); - /* - * control codes are not shown ever - */ - return; - } else if (term.esc & ESC_START) { - if (term.esc & ESC_CSI) { - csiescseq.buf[csiescseq.len++] = u; - if (BETWEEN(u, 0x40, 0x7E) - || csiescseq.len >= \ - sizeof(csiescseq.buf)-1) { - term.esc = 0; - csiparse(); - csihandle(); - } - return; - } else if (term.esc & ESC_UTF8) { - tdefutf8(u); - } else if (term.esc & ESC_ALTCHARSET) { - tdeftran(u); - } else if (term.esc & ESC_TEST) { - tdectest(u); - } else { - if (!eschandle(u)) - return; - /* sequence already finished */ - } - term.esc = 0; - /* - * All characters which form part of a sequence are not - * printed - */ - return; - } - if (sel.ob.x != -1 && BETWEEN(term.c.y, sel.ob.y, sel.oe.y)) - selclear(); - - gp = &term.line[term.c.y][term.c.x]; - if (IS_SET(MODE_WRAP) && (term.c.state & CURSOR_WRAPNEXT)) { - gp->mode |= ATTR_WRAP; - tnewline(1); - gp = &term.line[term.c.y][term.c.x]; - } - - if (IS_SET(MODE_INSERT) && term.c.x+width < term.col) - memmove(gp+width, gp, (term.col - term.c.x - width) * sizeof(Glyph)); - - if (term.c.x+width > term.col) { - tnewline(1); - gp = &term.line[term.c.y][term.c.x]; - } - - tsetchar(u, &term.c.attr, term.c.x, term.c.y); - - if (width == 2) { - gp->mode |= ATTR_WIDE; - if (term.c.x+1 < term.col) { - gp[1].u = '\0'; - gp[1].mode = ATTR_WDUMMY; - } - } - if (term.c.x+width < term.col) { - tmoveto(term.c.x+width, term.c.y); - } else { - term.c.state |= CURSOR_WRAPNEXT; - } -} - -int -twrite(const char *buf, int buflen, int show_ctrl) -{ - int charsize; - Rune u; - int n; - - for (n = 0; n < buflen; n += charsize) { - if (IS_SET(MODE_UTF8) && !IS_SET(MODE_SIXEL)) { - /* process a complete utf8 char */ - charsize = utf8decode(buf + n, &u, buflen - n); - if (charsize == 0) - break; - } else { - u = buf[n] & 0xFF; - charsize = 1; - } - if (show_ctrl && ISCONTROL(u)) { - if (u & 0x80) { - u &= 0x7f; - tputc('^'); - tputc('['); - } else if (u != '\n' && u != '\r' && u != '\t') { - u ^= 0x40; - tputc('^'); - } - } - tputc(u); - } - return n; -} - -void -tresize(int col, int row) -{ - int i; - int minrow = MIN(row, term.row); - int mincol = MIN(col, term.col); - int *bp; - TCursor c; - - if (col < 1 || row < 1) { - fprintf(stderr, - "tresize: error resizing to %dx%d\n", col, row); - return; - } - - /* - * slide screen to keep cursor where we expect it - - * tscrollup would work here, but we can optimize to - * memmove because we're freeing the earlier lines - */ - for (i = 0; i <= term.c.y - row; i++) { - free(term.line[i]); - free(term.alt[i]); - } - /* ensure that both src and dst are not NULL */ - if (i > 0) { - memmove(term.line, term.line + i, row * sizeof(Line)); - memmove(term.alt, term.alt + i, row * sizeof(Line)); - } - for (i += row; i < term.row; i++) { - free(term.line[i]); - free(term.alt[i]); - } - - /* resize to new height */ - term.line = xrealloc(term.line, row * sizeof(Line)); - term.alt = xrealloc(term.alt, row * sizeof(Line)); - term.dirty = xrealloc(term.dirty, row * sizeof(*term.dirty)); - term.tabs = xrealloc(term.tabs, col * sizeof(*term.tabs)); - - /* resize each row to new width, zero-pad if needed */ - for (i = 0; i < minrow; i++) { - term.line[i] = xrealloc(term.line[i], col * sizeof(Glyph)); - term.alt[i] = xrealloc(term.alt[i], col * sizeof(Glyph)); - } - - /* allocate any new rows */ - for (/* i = minrow */; i < row; i++) { - term.line[i] = xmalloc(col * sizeof(Glyph)); - term.alt[i] = xmalloc(col * sizeof(Glyph)); - } - if (col > term.col) { - bp = term.tabs + term.col; - - memset(bp, 0, sizeof(*term.tabs) * (col - term.col)); - while (--bp > term.tabs && !*bp) - /* nothing */ ; - for (bp += tabspaces; bp < term.tabs + col; bp += tabspaces) - *bp = 1; - } - /* update terminal size */ - term.col = col; - term.row = row; - /* reset scrolling region */ - tsetscroll(0, row-1); - /* make use of the LIMIT in tmoveto */ - tmoveto(term.c.x, term.c.y); - /* Clearing both screens (it makes dirty all lines) */ - c = term.c; - for (i = 0; i < 2; i++) { - if (mincol < col && 0 < minrow) { - tclearregion(mincol, 0, col - 1, minrow - 1); - } - if (0 < col && minrow < row) { - tclearregion(0, minrow, col - 1, row - 1); - } - tswapscreen(); - tcursor(CURSOR_LOAD); - } - term.c = c; -} - -void -resettitle(void) -{ - xsettitle(NULL); -} - -void -drawregion(int x1, int y1, int x2, int y2) -{ - int y; - for (y = y1; y < y2; y++) { - if (!term.dirty[y]) - continue; - - term.dirty[y] = 0; - xdrawline(term.line[y], x1, y, x2); - } -} - -void -draw(void) -{ - int cx = term.c.x; - - if (!xstartdraw()) - return; - - /* adjust cursor position */ - LIMIT(term.ocx, 0, term.col-1); - LIMIT(term.ocy, 0, term.row-1); - if (term.line[term.ocy][term.ocx].mode & ATTR_WDUMMY) - term.ocx--; - if (term.line[term.c.y][cx].mode & ATTR_WDUMMY) - cx--; - - drawregion(0, 0, term.col, term.row); - xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], - term.ocx, term.ocy, term.line[term.ocy][term.ocx], - term.line[term.ocy], term.col); - term.ocx = cx, term.ocy = term.c.y; - xfinishdraw(); - xximspot(term.ocx, term.ocy); -} - -void -redraw(void) -{ - tfulldirt(); - draw(); -} diff --git a/.suckless/st/st.c.orig b/.suckless/st/st.c.orig deleted file mode 100644 index 3e48410..0000000 --- a/.suckless/st/st.c.orig +++ /dev/null @@ -1,2599 +0,0 @@ -/* See LICENSE for license details. */ -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -#include "st.h" -#include "win.h" - -#if defined(__linux) - #include -#elif defined(__OpenBSD__) || defined(__NetBSD__) || defined(__APPLE__) - #include -#elif defined(__FreeBSD__) || defined(__DragonFly__) - #include -#endif - -/* Arbitrary sizes */ -#define UTF_INVALID 0xFFFD -#define UTF_SIZ 4 -#define ESC_BUF_SIZ (128*UTF_SIZ) -#define ESC_ARG_SIZ 16 -#define STR_BUF_SIZ ESC_BUF_SIZ -#define STR_ARG_SIZ ESC_ARG_SIZ - -/* macros */ -#define IS_SET(flag) ((term.mode & (flag)) != 0) -#define ISCONTROLC0(c) (BETWEEN(c, 0, 0x1f) || (c) == '\177') -#define ISCONTROLC1(c) (BETWEEN(c, 0x80, 0x9f)) -#define ISCONTROL(c) (ISCONTROLC0(c) || ISCONTROLC1(c)) -#define ISDELIM(u) (u && wcschr(worddelimiters, u)) - -enum term_mode { - MODE_WRAP = 1 << 0, - MODE_INSERT = 1 << 1, - MODE_ALTSCREEN = 1 << 2, - MODE_CRLF = 1 << 3, - MODE_ECHO = 1 << 4, - MODE_PRINT = 1 << 5, - MODE_UTF8 = 1 << 6, - MODE_SIXEL = 1 << 7, -}; - -enum cursor_movement { - CURSOR_SAVE, - CURSOR_LOAD -}; - -enum cursor_state { - CURSOR_DEFAULT = 0, - CURSOR_WRAPNEXT = 1, - CURSOR_ORIGIN = 2 -}; - -enum charset { - CS_GRAPHIC0, - CS_GRAPHIC1, - CS_UK, - CS_USA, - CS_MULTI, - CS_GER, - CS_FIN -}; - -enum escape_state { - ESC_START = 1, - ESC_CSI = 2, - ESC_STR = 4, /* OSC, PM, APC */ - ESC_ALTCHARSET = 8, - ESC_STR_END = 16, /* a final string was encountered */ - ESC_TEST = 32, /* Enter in test mode */ - ESC_UTF8 = 64, - ESC_DCS =128, -}; - -typedef struct { - Glyph attr; /* current char attributes */ - int x; - int y; - char state; -} TCursor; - -typedef struct { - int mode; - int type; - int snap; - /* - * Selection variables: - * nb – normalized coordinates of the beginning of the selection - * ne – normalized coordinates of the end of the selection - * ob – original coordinates of the beginning of the selection - * oe – original coordinates of the end of the selection - */ - struct { - int x, y; - } nb, ne, ob, oe; - - int alt; -} Selection; - -/* Internal representation of the screen */ -typedef struct { - int row; /* nb row */ - int col; /* nb col */ - Line *line; /* screen */ - Line *alt; /* alternate screen */ - int *dirty; /* dirtyness of lines */ - TCursor c; /* cursor */ - int ocx; /* old cursor col */ - int ocy; /* old cursor row */ - int top; /* top scroll limit */ - int bot; /* bottom scroll limit */ - int mode; /* terminal mode flags */ - int esc; /* escape state flags */ - char trantbl[4]; /* charset table translation */ - int charset; /* current charset */ - int icharset; /* selected charset for sequence */ - int *tabs; -} Term; - -/* CSI Escape sequence structs */ -/* ESC '[' [[ [] [;]] []] */ -typedef struct { - char buf[ESC_BUF_SIZ]; /* raw string */ - size_t len; /* raw string length */ - char priv; - int arg[ESC_ARG_SIZ]; - int narg; /* nb of args */ - char mode[2]; -} CSIEscape; - -/* STR Escape sequence structs */ -/* ESC type [[ [] [;]] ] ESC '\' */ -typedef struct { - char type; /* ESC type ... */ - char *buf; /* allocated raw string */ - size_t siz; /* allocation size */ - size_t len; /* raw string length */ - char *args[STR_ARG_SIZ]; - int narg; /* nb of args */ -} STREscape; - -static void execsh(char *, char **); -static void stty(char **); -static void sigchld(int); -static void ttywriteraw(const char *, size_t); - -static void csidump(void); -static void csihandle(void); -static void csiparse(void); -static void csireset(void); -static int eschandle(uchar); -static void strdump(void); -static void strhandle(void); -static void strparse(void); -static void strreset(void); - -static void tprinter(char *, size_t); -static void tdumpsel(void); -static void tdumpline(int); -static void tdump(void); -static void tclearregion(int, int, int, int); -static void tcursor(int); -static void tdeletechar(int); -static void tdeleteline(int); -static void tinsertblank(int); -static void tinsertblankline(int); -static int tlinelen(int); -static void tmoveto(int, int); -static void tmoveato(int, int); -static void tnewline(int); -static void tputtab(int); -static void tputc(Rune); -static void treset(void); -static void tscrollup(int, int); -static void tscrolldown(int, int); -static void tsetattr(int *, int); -static void tsetchar(Rune, Glyph *, int, int); -static void tsetdirt(int, int); -static void tsetscroll(int, int); -static void tswapscreen(void); -static void tsetmode(int, int, int *, int); -static int twrite(const char *, int, int); -static void tfulldirt(void); -static void tcontrolcode(uchar ); -static void tdectest(char ); -static void tdefutf8(char); -static int32_t tdefcolor(int *, int *, int); -static void tdeftran(char); -static void tstrsequence(uchar); - -static void drawregion(int, int, int, int); - -static void selnormalize(void); -static void selscroll(int, int); -static void selsnap(int *, int *, int); - -static size_t utf8decode(const char *, Rune *, size_t); -static Rune utf8decodebyte(char, size_t *); -static char utf8encodebyte(Rune, size_t); -static size_t utf8validate(Rune *, size_t); - -static char *base64dec(const char *); -static char base64dec_getc(const char **); - -static ssize_t xwrite(int, const char *, size_t); - -/* Globals */ -static Term term; -static Selection sel; -static CSIEscape csiescseq; -static STREscape strescseq; -static int iofd = 1; -static int cmdfd; -static pid_t pid; - -static uchar utfbyte[UTF_SIZ + 1] = {0x80, 0, 0xC0, 0xE0, 0xF0}; -static uchar utfmask[UTF_SIZ + 1] = {0xC0, 0x80, 0xE0, 0xF0, 0xF8}; -static Rune utfmin[UTF_SIZ + 1] = { 0, 0, 0x80, 0x800, 0x10000}; -static Rune utfmax[UTF_SIZ + 1] = {0x10FFFF, 0x7F, 0x7FF, 0xFFFF, 0x10FFFF}; - -ssize_t -xwrite(int fd, const char *s, size_t len) -{ - size_t aux = len; - ssize_t r; - - while (len > 0) { - r = write(fd, s, len); - if (r < 0) - return r; - len -= r; - s += r; - } - - return aux; -} - -void * -xmalloc(size_t len) -{ - void *p; - - if (!(p = malloc(len))) - die("malloc: %s\n", strerror(errno)); - - return p; -} - -void * -xrealloc(void *p, size_t len) -{ - if ((p = realloc(p, len)) == NULL) - die("realloc: %s\n", strerror(errno)); - - return p; -} - -char * -xstrdup(char *s) -{ - if ((s = strdup(s)) == NULL) - die("strdup: %s\n", strerror(errno)); - - return s; -} - -size_t -utf8decode(const char *c, Rune *u, size_t clen) -{ - size_t i, j, len, type; - Rune udecoded; - - *u = UTF_INVALID; - if (!clen) - return 0; - udecoded = utf8decodebyte(c[0], &len); - if (!BETWEEN(len, 1, UTF_SIZ)) - return 1; - for (i = 1, j = 1; i < clen && j < len; ++i, ++j) { - udecoded = (udecoded << 6) | utf8decodebyte(c[i], &type); - if (type != 0) - return j; - } - if (j < len) - return 0; - *u = udecoded; - utf8validate(u, len); - - return len; -} - -Rune -utf8decodebyte(char c, size_t *i) -{ - for (*i = 0; *i < LEN(utfmask); ++(*i)) - if (((uchar)c & utfmask[*i]) == utfbyte[*i]) - return (uchar)c & ~utfmask[*i]; - - return 0; -} - -size_t -utf8encode(Rune u, char *c) -{ - size_t len, i; - - len = utf8validate(&u, 0); - if (len > UTF_SIZ) - return 0; - - for (i = len - 1; i != 0; --i) { - c[i] = utf8encodebyte(u, 0); - u >>= 6; - } - c[0] = utf8encodebyte(u, len); - - return len; -} - -char -utf8encodebyte(Rune u, size_t i) -{ - return utfbyte[i] | (u & ~utfmask[i]); -} - -size_t -utf8validate(Rune *u, size_t i) -{ - if (!BETWEEN(*u, utfmin[i], utfmax[i]) || BETWEEN(*u, 0xD800, 0xDFFF)) - *u = UTF_INVALID; - for (i = 1; *u > utfmax[i]; ++i) - ; - - return i; -} - -static const char base64_digits[] = { - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 62, 0, 0, 0, - 63, 52, 53, 54, 55, 56, 57, 58, 59, 60, 61, 0, 0, 0, -1, 0, 0, 0, 0, 1, - 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 16, 17, 18, 19, 20, 21, - 22, 23, 24, 25, 0, 0, 0, 0, 0, 0, 26, 27, 28, 29, 30, 31, 32, 33, 34, - 35, 36, 37, 38, 39, 40, 41, 42, 43, 44, 45, 46, 47, 48, 49, 50, 51, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, - 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 -}; - -char -base64dec_getc(const char **src) -{ - while (**src && !isprint(**src)) (*src)++; - return **src ? *((*src)++) : '='; /* emulate padding if string ends */ -} - -char * -base64dec(const char *src) -{ - size_t in_len = strlen(src); - char *result, *dst; - - if (in_len % 4) - in_len += 4 - (in_len % 4); - result = dst = xmalloc(in_len / 4 * 3 + 1); - while (*src) { - int a = base64_digits[(unsigned char) base64dec_getc(&src)]; - int b = base64_digits[(unsigned char) base64dec_getc(&src)]; - int c = base64_digits[(unsigned char) base64dec_getc(&src)]; - int d = base64_digits[(unsigned char) base64dec_getc(&src)]; - - /* invalid input. 'a' can be -1, e.g. if src is "\n" (c-str) */ - if (a == -1 || b == -1) - break; - - *dst++ = (a << 2) | ((b & 0x30) >> 4); - if (c == -1) - break; - *dst++ = ((b & 0x0f) << 4) | ((c & 0x3c) >> 2); - if (d == -1) - break; - *dst++ = ((c & 0x03) << 6) | d; - } - *dst = '\0'; - return result; -} - -void -selinit(void) -{ - sel.mode = SEL_IDLE; - sel.snap = 0; - sel.ob.x = -1; -} - -int -tlinelen(int y) -{ - int i = term.col; - - if (term.line[y][i - 1].mode & ATTR_WRAP) - return i; - - while (i > 0 && term.line[y][i - 1].u == ' ') - --i; - - return i; -} - -void -selstart(int col, int row, int snap) -{ - selclear(); - sel.mode = SEL_EMPTY; - sel.type = SEL_REGULAR; - sel.alt = IS_SET(MODE_ALTSCREEN); - sel.snap = snap; - sel.oe.x = sel.ob.x = col; - sel.oe.y = sel.ob.y = row; - selnormalize(); - - if (sel.snap != 0) - sel.mode = SEL_READY; - tsetdirt(sel.nb.y, sel.ne.y); -} - -void -selextend(int col, int row, int type, int done) -{ - int oldey, oldex, oldsby, oldsey, oldtype; - - if (sel.mode == SEL_IDLE) - return; - if (done && sel.mode == SEL_EMPTY) { - selclear(); - return; - } - - oldey = sel.oe.y; - oldex = sel.oe.x; - oldsby = sel.nb.y; - oldsey = sel.ne.y; - oldtype = sel.type; - - sel.oe.x = col; - sel.oe.y = row; - selnormalize(); - sel.type = type; - - if (oldey != sel.oe.y || oldex != sel.oe.x || oldtype != sel.type || sel.mode == SEL_EMPTY) - tsetdirt(MIN(sel.nb.y, oldsby), MAX(sel.ne.y, oldsey)); - - sel.mode = done ? SEL_IDLE : SEL_READY; -} - -void -selnormalize(void) -{ - int i; - - if (sel.type == SEL_REGULAR && sel.ob.y != sel.oe.y) { - sel.nb.x = sel.ob.y < sel.oe.y ? sel.ob.x : sel.oe.x; - sel.ne.x = sel.ob.y < sel.oe.y ? sel.oe.x : sel.ob.x; - } else { - sel.nb.x = MIN(sel.ob.x, sel.oe.x); - sel.ne.x = MAX(sel.ob.x, sel.oe.x); - } - sel.nb.y = MIN(sel.ob.y, sel.oe.y); - sel.ne.y = MAX(sel.ob.y, sel.oe.y); - - selsnap(&sel.nb.x, &sel.nb.y, -1); - selsnap(&sel.ne.x, &sel.ne.y, +1); - - /* expand selection over line breaks */ - if (sel.type == SEL_RECTANGULAR) - return; - i = tlinelen(sel.nb.y); - if (i < sel.nb.x) - sel.nb.x = i; - if (tlinelen(sel.ne.y) <= sel.ne.x) - sel.ne.x = term.col - 1; -} - -int -selected(int x, int y) -{ - if (sel.mode == SEL_EMPTY || sel.ob.x == -1 || - sel.alt != IS_SET(MODE_ALTSCREEN)) - return 0; - - if (sel.type == SEL_RECTANGULAR) - return BETWEEN(y, sel.nb.y, sel.ne.y) - && BETWEEN(x, sel.nb.x, sel.ne.x); - - return BETWEEN(y, sel.nb.y, sel.ne.y) - && (y != sel.nb.y || x >= sel.nb.x) - && (y != sel.ne.y || x <= sel.ne.x); -} - -void -selsnap(int *x, int *y, int direction) -{ - int newx, newy, xt, yt; - int delim, prevdelim; - Glyph *gp, *prevgp; - - switch (sel.snap) { - case SNAP_WORD: - /* - * Snap around if the word wraps around at the end or - * beginning of a line. - */ - prevgp = &term.line[*y][*x]; - prevdelim = ISDELIM(prevgp->u); - for (;;) { - newx = *x + direction; - newy = *y; - if (!BETWEEN(newx, 0, term.col - 1)) { - newy += direction; - newx = (newx + term.col) % term.col; - if (!BETWEEN(newy, 0, term.row - 1)) - break; - - if (direction > 0) - yt = *y, xt = *x; - else - yt = newy, xt = newx; - if (!(term.line[yt][xt].mode & ATTR_WRAP)) - break; - } - - if (newx >= tlinelen(newy)) - break; - - gp = &term.line[newy][newx]; - delim = ISDELIM(gp->u); - if (!(gp->mode & ATTR_WDUMMY) && (delim != prevdelim - || (delim && gp->u != prevgp->u))) - break; - - *x = newx; - *y = newy; - prevgp = gp; - prevdelim = delim; - } - break; - case SNAP_LINE: - /* - * Snap around if the the previous line or the current one - * has set ATTR_WRAP at its end. Then the whole next or - * previous line will be selected. - */ - *x = (direction < 0) ? 0 : term.col - 1; - if (direction < 0) { - for (; *y > 0; *y += direction) { - if (!(term.line[*y-1][term.col-1].mode - & ATTR_WRAP)) { - break; - } - } - } else if (direction > 0) { - for (; *y < term.row-1; *y += direction) { - if (!(term.line[*y][term.col-1].mode - & ATTR_WRAP)) { - break; - } - } - } - break; - } -} - -char * -getsel(void) -{ - char *str, *ptr; - int y, bufsize, lastx, linelen; - Glyph *gp, *last; - - if (sel.ob.x == -1) - return NULL; - - bufsize = (term.col+1) * (sel.ne.y-sel.nb.y+1) * UTF_SIZ; - ptr = str = xmalloc(bufsize); - - /* append every set & selected glyph to the selection */ - for (y = sel.nb.y; y <= sel.ne.y; y++) { - if ((linelen = tlinelen(y)) == 0) { - *ptr++ = '\n'; - continue; - } - - if (sel.type == SEL_RECTANGULAR) { - gp = &term.line[y][sel.nb.x]; - lastx = sel.ne.x; - } else { - gp = &term.line[y][sel.nb.y == y ? sel.nb.x : 0]; - lastx = (sel.ne.y == y) ? sel.ne.x : term.col-1; - } - last = &term.line[y][MIN(lastx, linelen-1)]; - while (last >= gp && last->u == ' ') - --last; - - for ( ; gp <= last; ++gp) { - if (gp->mode & ATTR_WDUMMY) - continue; - - ptr += utf8encode(gp->u, ptr); - } - - /* - * Copy and pasting of line endings is inconsistent - * in the inconsistent terminal and GUI world. - * The best solution seems like to produce '\n' when - * something is copied from st and convert '\n' to - * '\r', when something to be pasted is received by - * st. - * FIXME: Fix the computer world. - */ - if ((y < sel.ne.y || lastx >= linelen) && !(last->mode & ATTR_WRAP)) - *ptr++ = '\n'; - } - *ptr = 0; - return str; -} - -void -selclear(void) -{ - if (sel.ob.x == -1) - return; - sel.mode = SEL_IDLE; - sel.ob.x = -1; - tsetdirt(sel.nb.y, sel.ne.y); -} - -void -die(const char *errstr, ...) -{ - va_list ap; - - va_start(ap, errstr); - vfprintf(stderr, errstr, ap); - va_end(ap); - exit(1); -} - -void -execsh(char *cmd, char **args) -{ - char *sh, *prog; - const struct passwd *pw; - - errno = 0; - if ((pw = getpwuid(getuid())) == NULL) { - if (errno) - die("getpwuid: %s\n", strerror(errno)); - else - die("who are you?\n"); - } - - if ((sh = getenv("SHELL")) == NULL) - sh = (pw->pw_shell[0]) ? pw->pw_shell : cmd; - - if (args) - prog = args[0]; - else if (utmp) - prog = utmp; - else - prog = sh; - DEFAULT(args, ((char *[]) {prog, NULL})); - - unsetenv("COLUMNS"); - unsetenv("LINES"); - unsetenv("TERMCAP"); - setenv("LOGNAME", pw->pw_name, 1); - setenv("USER", pw->pw_name, 1); - setenv("SHELL", sh, 1); - setenv("HOME", pw->pw_dir, 1); - setenv("TERM", termname, 1); - - signal(SIGCHLD, SIG_DFL); - signal(SIGHUP, SIG_DFL); - signal(SIGINT, SIG_DFL); - signal(SIGQUIT, SIG_DFL); - signal(SIGTERM, SIG_DFL); - signal(SIGALRM, SIG_DFL); - - execvp(prog, args); - _exit(1); -} - -void -sigchld(int a) -{ - int stat; - pid_t p; - - if ((p = waitpid(pid, &stat, WNOHANG)) < 0) - die("waiting for pid %hd failed: %s\n", pid, strerror(errno)); - - if (pid != p) - return; - - if (WIFEXITED(stat) && WEXITSTATUS(stat)) - die("child exited with status %d\n", WEXITSTATUS(stat)); - else if (WIFSIGNALED(stat)) - die("child terminated due to signal %d\n", WTERMSIG(stat)); - exit(0); -} - -void -stty(char **args) -{ - char cmd[_POSIX_ARG_MAX], **p, *q, *s; - size_t n, siz; - - if ((n = strlen(stty_args)) > sizeof(cmd)-1) - die("incorrect stty parameters\n"); - memcpy(cmd, stty_args, n); - q = cmd + n; - siz = sizeof(cmd) - n; - for (p = args; p && (s = *p); ++p) { - if ((n = strlen(s)) > siz-1) - die("stty parameter length too long\n"); - *q++ = ' '; - memcpy(q, s, n); - q += n; - siz -= n + 1; - } - *q = '\0'; - if (system(cmd) != 0) - perror("Couldn't call stty"); -} - -int -ttynew(char *line, char *cmd, char *out, char **args) -{ - int m, s; - - if (out) { - term.mode |= MODE_PRINT; - iofd = (!strcmp(out, "-")) ? - 1 : open(out, O_WRONLY | O_CREAT, 0666); - if (iofd < 0) { - fprintf(stderr, "Error opening %s:%s\n", - out, strerror(errno)); - } - } - - if (line) { - if ((cmdfd = open(line, O_RDWR)) < 0) - die("open line '%s' failed: %s\n", - line, strerror(errno)); - dup2(cmdfd, 0); - stty(args); - return cmdfd; - } - - /* seems to work fine on linux, openbsd and freebsd */ - if (openpty(&m, &s, NULL, NULL, NULL) < 0) - die("openpty failed: %s\n", strerror(errno)); - - switch (pid = fork()) { - case -1: - die("fork failed: %s\n", strerror(errno)); - break; - case 0: - close(iofd); - setsid(); /* create a new process group */ - dup2(s, 0); - dup2(s, 1); - dup2(s, 2); - if (ioctl(s, TIOCSCTTY, NULL) < 0) - die("ioctl TIOCSCTTY failed: %s\n", strerror(errno)); - close(s); - close(m); -#ifdef __OpenBSD__ - if (pledge("stdio getpw proc exec", NULL) == -1) - die("pledge\n"); -#endif - execsh(cmd, args); - break; - default: -#ifdef __OpenBSD__ - if (pledge("stdio rpath tty proc", NULL) == -1) - die("pledge\n"); -#endif - close(s); - cmdfd = m; - signal(SIGCHLD, sigchld); - break; - } - return cmdfd; -} - -size_t -ttyread(void) -{ - static char buf[BUFSIZ]; - static int buflen = 0; - int written; - int ret; - - /* append read bytes to unprocessed bytes */ - if ((ret = read(cmdfd, buf+buflen, LEN(buf)-buflen)) < 0) - die("couldn't read from shell: %s\n", strerror(errno)); - buflen += ret; - - written = twrite(buf, buflen, 0); - buflen -= written; - /* keep any uncomplete utf8 char for the next call */ - if (buflen > 0) - memmove(buf, buf + written, buflen); - - return ret; -} - -void -ttywrite(const char *s, size_t n, int may_echo) -{ - const char *next; - - if (may_echo && IS_SET(MODE_ECHO)) - twrite(s, n, 1); - - if (!IS_SET(MODE_CRLF)) { - ttywriteraw(s, n); - return; - } - - /* This is similar to how the kernel handles ONLCR for ttys */ - while (n > 0) { - if (*s == '\r') { - next = s + 1; - ttywriteraw("\r\n", 2); - } else { - next = memchr(s, '\r', n); - DEFAULT(next, s + n); - ttywriteraw(s, next - s); - } - n -= next - s; - s = next; - } -} - -void -ttywriteraw(const char *s, size_t n) -{ - fd_set wfd, rfd; - ssize_t r; - size_t lim = 256; - - /* - * Remember that we are using a pty, which might be a modem line. - * Writing too much will clog the line. That's why we are doing this - * dance. - * FIXME: Migrate the world to Plan 9. - */ - while (n > 0) { - FD_ZERO(&wfd); - FD_ZERO(&rfd); - FD_SET(cmdfd, &wfd); - FD_SET(cmdfd, &rfd); - - /* Check if we can write. */ - if (pselect(cmdfd+1, &rfd, &wfd, NULL, NULL, NULL) < 0) { - if (errno == EINTR) - continue; - die("select failed: %s\n", strerror(errno)); - } - if (FD_ISSET(cmdfd, &wfd)) { - /* - * Only write the bytes written by ttywrite() or the - * default of 256. This seems to be a reasonable value - * for a serial line. Bigger values might clog the I/O. - */ - if ((r = write(cmdfd, s, (n < lim)? n : lim)) < 0) - goto write_error; - if (r < n) { - /* - * We weren't able to write out everything. - * This means the buffer is getting full - * again. Empty it. - */ - if (n < lim) - lim = ttyread(); - n -= r; - s += r; - } else { - /* All bytes have been written. */ - break; - } - } - if (FD_ISSET(cmdfd, &rfd)) - lim = ttyread(); - } - return; - -write_error: - die("write error on tty: %s\n", strerror(errno)); -} - -void -ttyresize(int tw, int th) -{ - struct winsize w; - - w.ws_row = term.row; - w.ws_col = term.col; - w.ws_xpixel = tw; - w.ws_ypixel = th; - if (ioctl(cmdfd, TIOCSWINSZ, &w) < 0) - fprintf(stderr, "Couldn't set window size: %s\n", strerror(errno)); -} - -void -ttyhangup() -{ - /* Send SIGHUP to shell */ - kill(pid, SIGHUP); -} - -int -tattrset(int attr) -{ - int i, j; - - for (i = 0; i < term.row-1; i++) { - for (j = 0; j < term.col-1; j++) { - if (term.line[i][j].mode & attr) - return 1; - } - } - - return 0; -} - -void -tsetdirt(int top, int bot) -{ - int i; - - LIMIT(top, 0, term.row-1); - LIMIT(bot, 0, term.row-1); - - for (i = top; i <= bot; i++) - term.dirty[i] = 1; -} - -void -tsetdirtattr(int attr) -{ - int i, j; - - for (i = 0; i < term.row-1; i++) { - for (j = 0; j < term.col-1; j++) { - if (term.line[i][j].mode & attr) { - tsetdirt(i, i); - break; - } - } - } -} - -void -tfulldirt(void) -{ - tsetdirt(0, term.row-1); -} - -void -tcursor(int mode) -{ - static TCursor c[2]; - int alt = IS_SET(MODE_ALTSCREEN); - - if (mode == CURSOR_SAVE) { - c[alt] = term.c; - } else if (mode == CURSOR_LOAD) { - term.c = c[alt]; - tmoveto(c[alt].x, c[alt].y); - } -} - -void -treset(void) -{ - uint i; - - term.c = (TCursor){{ - .mode = ATTR_NULL, - .fg = defaultfg, - .bg = defaultbg - }, .x = 0, .y = 0, .state = CURSOR_DEFAULT}; - - memset(term.tabs, 0, term.col * sizeof(*term.tabs)); - for (i = tabspaces; i < term.col; i += tabspaces) - term.tabs[i] = 1; - term.top = 0; - term.bot = term.row - 1; - term.mode = MODE_WRAP|MODE_UTF8; - memset(term.trantbl, CS_USA, sizeof(term.trantbl)); - term.charset = 0; - - for (i = 0; i < 2; i++) { - tmoveto(0, 0); - tcursor(CURSOR_SAVE); - tclearregion(0, 0, term.col-1, term.row-1); - tswapscreen(); - } -} - -void -tnew(int col, int row) -{ - term = (Term){ .c = { .attr = { .fg = defaultfg, .bg = defaultbg } } }; - tresize(col, row); - treset(); -} - -void -tswapscreen(void) -{ - Line *tmp = term.line; - - term.line = term.alt; - term.alt = tmp; - term.mode ^= MODE_ALTSCREEN; - tfulldirt(); -} - -void -tscrolldown(int orig, int n) -{ - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - - tsetdirt(orig, term.bot-n); - tclearregion(0, term.bot-n+1, term.col-1, term.bot); - - for (i = term.bot; i >= orig+n; i--) { - temp = term.line[i]; - term.line[i] = term.line[i-n]; - term.line[i-n] = temp; - } - - selscroll(orig, n); -} - -void -tscrollup(int orig, int n) -{ - int i; - Line temp; - - LIMIT(n, 0, term.bot-orig+1); - - tclearregion(0, orig, term.col-1, orig+n-1); - tsetdirt(orig+n, term.bot); - - for (i = orig; i <= term.bot-n; i++) { - temp = term.line[i]; - term.line[i] = term.line[i+n]; - term.line[i+n] = temp; - } - - selscroll(orig, -n); -} - -void -selscroll(int orig, int n) -{ - if (sel.ob.x == -1) - return; - - if (BETWEEN(sel.ob.y, orig, term.bot) || BETWEEN(sel.oe.y, orig, term.bot)) { - if ((sel.ob.y += n) > term.bot || (sel.oe.y += n) < term.top) { - selclear(); - return; - } - if (sel.type == SEL_RECTANGULAR) { - if (sel.ob.y < term.top) - sel.ob.y = term.top; - if (sel.oe.y > term.bot) - sel.oe.y = term.bot; - } else { - if (sel.ob.y < term.top) { - sel.ob.y = term.top; - sel.ob.x = 0; - } - if (sel.oe.y > term.bot) { - sel.oe.y = term.bot; - sel.oe.x = term.col; - } - } - selnormalize(); - } -} - -void -tnewline(int first_col) -{ - int y = term.c.y; - - if (y == term.bot) { - tscrollup(term.top, 1); - } else { - y++; - } - tmoveto(first_col ? 0 : term.c.x, y); -} - -void -csiparse(void) -{ - char *p = csiescseq.buf, *np; - long int v; - - csiescseq.narg = 0; - if (*p == '?') { - csiescseq.priv = 1; - p++; - } - - csiescseq.buf[csiescseq.len] = '\0'; - while (p < csiescseq.buf+csiescseq.len) { - np = NULL; - v = strtol(p, &np, 10); - if (np == p) - v = 0; - if (v == LONG_MAX || v == LONG_MIN) - v = -1; - csiescseq.arg[csiescseq.narg++] = v; - p = np; - if (*p != ';' || csiescseq.narg == ESC_ARG_SIZ) - break; - p++; - } - csiescseq.mode[0] = *p++; - csiescseq.mode[1] = (p < csiescseq.buf+csiescseq.len) ? *p : '\0'; -} - -/* for absolute user moves, when decom is set */ -void -tmoveato(int x, int y) -{ - tmoveto(x, y + ((term.c.state & CURSOR_ORIGIN) ? term.top: 0)); -} - -void -tmoveto(int x, int y) -{ - int miny, maxy; - - if (term.c.state & CURSOR_ORIGIN) { - miny = term.top; - maxy = term.bot; - } else { - miny = 0; - maxy = term.row - 1; - } - term.c.state &= ~CURSOR_WRAPNEXT; - term.c.x = LIMIT(x, 0, term.col-1); - term.c.y = LIMIT(y, miny, maxy); -} - -void -tsetchar(Rune u, Glyph *attr, int x, int y) -{ - static char *vt100_0[62] = { /* 0x41 - 0x7e */ - "↑", "↓", "→", "←", "█", "▚", "☃", /* A - G */ - 0, 0, 0, 0, 0, 0, 0, 0, /* H - O */ - 0, 0, 0, 0, 0, 0, 0, 0, /* P - W */ - 0, 0, 0, 0, 0, 0, 0, " ", /* X - _ */ - "◆", "▒", "␉", "␌", "␍", "␊", "°", "±", /* ` - g */ - "␤", "␋", "┘", "┐", "┌", "└", "┼", "⎺", /* h - o */ - "⎻", "─", "⎼", "⎽", "├", "┤", "┴", "┬", /* p - w */ - "│", "≤", "≥", "π", "≠", "£", "·", /* x - ~ */ - }; - - /* - * The table is proudly stolen from rxvt. - */ - if (term.trantbl[term.charset] == CS_GRAPHIC0 && - BETWEEN(u, 0x41, 0x7e) && vt100_0[u - 0x41]) - utf8decode(vt100_0[u - 0x41], &u, UTF_SIZ); - - if (term.line[y][x].mode & ATTR_WIDE) { - if (x+1 < term.col) { - term.line[y][x+1].u = ' '; - term.line[y][x+1].mode &= ~ATTR_WDUMMY; - } - } else if (term.line[y][x].mode & ATTR_WDUMMY) { - term.line[y][x-1].u = ' '; - term.line[y][x-1].mode &= ~ATTR_WIDE; - } - - term.dirty[y] = 1; - term.line[y][x] = *attr; - term.line[y][x].u = u; -} - -void -tclearregion(int x1, int y1, int x2, int y2) -{ - int x, y, temp; - Glyph *gp; - - if (x1 > x2) - temp = x1, x1 = x2, x2 = temp; - if (y1 > y2) - temp = y1, y1 = y2, y2 = temp; - - LIMIT(x1, 0, term.col-1); - LIMIT(x2, 0, term.col-1); - LIMIT(y1, 0, term.row-1); - LIMIT(y2, 0, term.row-1); - - for (y = y1; y <= y2; y++) { - term.dirty[y] = 1; - for (x = x1; x <= x2; x++) { - gp = &term.line[y][x]; - if (selected(x, y)) - selclear(); - gp->fg = term.c.attr.fg; - gp->bg = term.c.attr.bg; - gp->mode = 0; - gp->u = ' '; - } - } -} - -void -tdeletechar(int n) -{ - int dst, src, size; - Glyph *line; - - LIMIT(n, 0, term.col - term.c.x); - - dst = term.c.x; - src = term.c.x + n; - size = term.col - src; - line = term.line[term.c.y]; - - memmove(&line[dst], &line[src], size * sizeof(Glyph)); - tclearregion(term.col-n, term.c.y, term.col-1, term.c.y); -} - -void -tinsertblank(int n) -{ - int dst, src, size; - Glyph *line; - - LIMIT(n, 0, term.col - term.c.x); - - dst = term.c.x + n; - src = term.c.x; - size = term.col - dst; - line = term.line[term.c.y]; - - memmove(&line[dst], &line[src], size * sizeof(Glyph)); - tclearregion(src, term.c.y, dst - 1, term.c.y); -} - -void -tinsertblankline(int n) -{ - if (BETWEEN(term.c.y, term.top, term.bot)) - tscrolldown(term.c.y, n); -} - -void -tdeleteline(int n) -{ - if (BETWEEN(term.c.y, term.top, term.bot)) - tscrollup(term.c.y, n); -} - -int32_t -tdefcolor(int *attr, int *npar, int l) -{ - int32_t idx = -1; - uint r, g, b; - - switch (attr[*npar + 1]) { - case 2: /* direct color in RGB space */ - if (*npar + 4 >= l) { - fprintf(stderr, - "erresc(38): Incorrect number of parameters (%d)\n", - *npar); - break; - } - r = attr[*npar + 2]; - g = attr[*npar + 3]; - b = attr[*npar + 4]; - *npar += 4; - if (!BETWEEN(r, 0, 255) || !BETWEEN(g, 0, 255) || !BETWEEN(b, 0, 255)) - fprintf(stderr, "erresc: bad rgb color (%u,%u,%u)\n", - r, g, b); - else - idx = TRUECOLOR(r, g, b); - break; - case 5: /* indexed color */ - if (*npar + 2 >= l) { - fprintf(stderr, - "erresc(38): Incorrect number of parameters (%d)\n", - *npar); - break; - } - *npar += 2; - if (!BETWEEN(attr[*npar], 0, 255)) - fprintf(stderr, "erresc: bad fgcolor %d\n", attr[*npar]); - else - idx = attr[*npar]; - break; - case 0: /* implemented defined (only foreground) */ - case 1: /* transparent */ - case 3: /* direct color in CMY space */ - case 4: /* direct color in CMYK space */ - default: - fprintf(stderr, - "erresc(38): gfx attr %d unknown\n", attr[*npar]); - break; - } - - return idx; -} - -void -tsetattr(int *attr, int l) -{ - int i; - int32_t idx; - - for (i = 0; i < l; i++) { - switch (attr[i]) { - case 0: - term.c.attr.mode &= ~( - ATTR_BOLD | - ATTR_FAINT | - ATTR_ITALIC | - ATTR_UNDERLINE | - ATTR_BLINK | - ATTR_REVERSE | - ATTR_INVISIBLE | - ATTR_STRUCK ); - term.c.attr.fg = defaultfg; - term.c.attr.bg = defaultbg; - break; - case 1: - term.c.attr.mode |= ATTR_BOLD; - break; - case 2: - term.c.attr.mode |= ATTR_FAINT; - break; - case 3: - term.c.attr.mode |= ATTR_ITALIC; - break; - case 4: - term.c.attr.mode |= ATTR_UNDERLINE; - break; - case 5: /* slow blink */ - /* FALLTHROUGH */ - case 6: /* rapid blink */ - term.c.attr.mode |= ATTR_BLINK; - break; - case 7: - term.c.attr.mode |= ATTR_REVERSE; - break; - case 8: - term.c.attr.mode |= ATTR_INVISIBLE; - break; - case 9: - term.c.attr.mode |= ATTR_STRUCK; - break; - case 22: - term.c.attr.mode &= ~(ATTR_BOLD | ATTR_FAINT); - break; - case 23: - term.c.attr.mode &= ~ATTR_ITALIC; - break; - case 24: - term.c.attr.mode &= ~ATTR_UNDERLINE; - break; - case 25: - term.c.attr.mode &= ~ATTR_BLINK; - break; - case 27: - term.c.attr.mode &= ~ATTR_REVERSE; - break; - case 28: - term.c.attr.mode &= ~ATTR_INVISIBLE; - break; - case 29: - term.c.attr.mode &= ~ATTR_STRUCK; - break; - case 38: - if ((idx = tdefcolor(attr, &i, l)) >= 0) - term.c.attr.fg = idx; - break; - case 39: - term.c.attr.fg = defaultfg; - break; - case 48: - if ((idx = tdefcolor(attr, &i, l)) >= 0) - term.c.attr.bg = idx; - break; - case 49: - term.c.attr.bg = defaultbg; - break; - default: - if (BETWEEN(attr[i], 30, 37)) { - term.c.attr.fg = attr[i] - 30; - } else if (BETWEEN(attr[i], 40, 47)) { - term.c.attr.bg = attr[i] - 40; - } else if (BETWEEN(attr[i], 90, 97)) { - term.c.attr.fg = attr[i] - 90 + 8; - } else if (BETWEEN(attr[i], 100, 107)) { - term.c.attr.bg = attr[i] - 100 + 8; - } else { - fprintf(stderr, - "erresc(default): gfx attr %d unknown\n", - attr[i]); - csidump(); - } - break; - } - } -} - -void -tsetscroll(int t, int b) -{ - int temp; - - LIMIT(t, 0, term.row-1); - LIMIT(b, 0, term.row-1); - if (t > b) { - temp = t; - t = b; - b = temp; - } - term.top = t; - term.bot = b; -} - -void -tsetmode(int priv, int set, int *args, int narg) -{ - int alt, *lim; - - for (lim = args + narg; args < lim; ++args) { - if (priv) { - switch (*args) { - case 1: /* DECCKM -- Cursor key */ - xsetmode(set, MODE_APPCURSOR); - break; - case 5: /* DECSCNM -- Reverse video */ - xsetmode(set, MODE_REVERSE); - break; - case 6: /* DECOM -- Origin */ - MODBIT(term.c.state, set, CURSOR_ORIGIN); - tmoveato(0, 0); - break; - case 7: /* DECAWM -- Auto wrap */ - MODBIT(term.mode, set, MODE_WRAP); - break; - case 0: /* Error (IGNORED) */ - case 2: /* DECANM -- ANSI/VT52 (IGNORED) */ - case 3: /* DECCOLM -- Column (IGNORED) */ - case 4: /* DECSCLM -- Scroll (IGNORED) */ - case 8: /* DECARM -- Auto repeat (IGNORED) */ - case 18: /* DECPFF -- Printer feed (IGNORED) */ - case 19: /* DECPEX -- Printer extent (IGNORED) */ - case 42: /* DECNRCM -- National characters (IGNORED) */ - case 12: /* att610 -- Start blinking cursor (IGNORED) */ - break; - case 25: /* DECTCEM -- Text Cursor Enable Mode */ - xsetmode(!set, MODE_HIDE); - break; - case 9: /* X10 mouse compatibility mode */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEX10); - break; - case 1000: /* 1000: report button press */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEBTN); - break; - case 1002: /* 1002: report motion on button press */ - xsetpointermotion(0); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEMOTION); - break; - case 1003: /* 1003: enable all mouse motions */ - xsetpointermotion(set); - xsetmode(0, MODE_MOUSE); - xsetmode(set, MODE_MOUSEMANY); - break; - case 1004: /* 1004: send focus events to tty */ - xsetmode(set, MODE_FOCUS); - break; - case 1006: /* 1006: extended reporting mode */ - xsetmode(set, MODE_MOUSESGR); - break; - case 1034: - xsetmode(set, MODE_8BIT); - break; - case 1049: /* swap screen & set/restore cursor as xterm */ - if (!allowaltscreen) - break; - tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD); - /* FALLTHROUGH */ - case 47: /* swap screen */ - case 1047: - if (!allowaltscreen) - break; - alt = IS_SET(MODE_ALTSCREEN); - if (alt) { - tclearregion(0, 0, term.col-1, - term.row-1); - } - if (set ^ alt) /* set is always 1 or 0 */ - tswapscreen(); - if (*args != 1049) - break; - /* FALLTHROUGH */ - case 1048: - tcursor((set) ? CURSOR_SAVE : CURSOR_LOAD); - break; - case 2004: /* 2004: bracketed paste mode */ - xsetmode(set, MODE_BRCKTPASTE); - break; - /* Not implemented mouse modes. See comments there. */ - case 1001: /* mouse highlight mode; can hang the - terminal by design when implemented. */ - case 1005: /* UTF-8 mouse mode; will confuse - applications not supporting UTF-8 - and luit. */ - case 1015: /* urxvt mangled mouse mode; incompatible - and can be mistaken for other control - codes. */ - break; - default: - fprintf(stderr, - "erresc: unknown private set/reset mode %d\n", - *args); - break; - } - } else { - switch (*args) { - case 0: /* Error (IGNORED) */ - break; - case 2: - xsetmode(set, MODE_KBDLOCK); - break; - case 4: /* IRM -- Insertion-replacement */ - MODBIT(term.mode, set, MODE_INSERT); - break; - case 12: /* SRM -- Send/Receive */ - MODBIT(term.mode, !set, MODE_ECHO); - break; - case 20: /* LNM -- Linefeed/new line */ - MODBIT(term.mode, set, MODE_CRLF); - break; - default: - fprintf(stderr, - "erresc: unknown set/reset mode %d\n", - *args); - break; - } - } - } -} - -void -csihandle(void) -{ - char buf[40]; - int len; - - switch (csiescseq.mode[0]) { - default: - unknown: - fprintf(stderr, "erresc: unknown csi "); - csidump(); - /* die(""); */ - break; - case '@': /* ICH -- Insert blank char */ - DEFAULT(csiescseq.arg[0], 1); - tinsertblank(csiescseq.arg[0]); - break; - case 'A': /* CUU -- Cursor Up */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x, term.c.y-csiescseq.arg[0]); - break; - case 'B': /* CUD -- Cursor Down */ - case 'e': /* VPR --Cursor Down */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x, term.c.y+csiescseq.arg[0]); - break; - case 'i': /* MC -- Media Copy */ - switch (csiescseq.arg[0]) { - case 0: - tdump(); - break; - case 1: - tdumpline(term.c.y); - break; - case 2: - tdumpsel(); - break; - case 4: - term.mode &= ~MODE_PRINT; - break; - case 5: - term.mode |= MODE_PRINT; - break; - } - break; - case 'c': /* DA -- Device Attributes */ - if (csiescseq.arg[0] == 0) - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 'C': /* CUF -- Cursor Forward */ - case 'a': /* HPR -- Cursor Forward */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x+csiescseq.arg[0], term.c.y); - break; - case 'D': /* CUB -- Cursor Backward */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(term.c.x-csiescseq.arg[0], term.c.y); - break; - case 'E': /* CNL -- Cursor Down and first col */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(0, term.c.y+csiescseq.arg[0]); - break; - case 'F': /* CPL -- Cursor Up and first col */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(0, term.c.y-csiescseq.arg[0]); - break; - case 'g': /* TBC -- Tabulation clear */ - switch (csiescseq.arg[0]) { - case 0: /* clear current tab stop */ - term.tabs[term.c.x] = 0; - break; - case 3: /* clear all the tabs */ - memset(term.tabs, 0, term.col * sizeof(*term.tabs)); - break; - default: - goto unknown; - } - break; - case 'G': /* CHA -- Move to */ - case '`': /* HPA */ - DEFAULT(csiescseq.arg[0], 1); - tmoveto(csiescseq.arg[0]-1, term.c.y); - break; - case 'H': /* CUP -- Move to */ - case 'f': /* HVP */ - DEFAULT(csiescseq.arg[0], 1); - DEFAULT(csiescseq.arg[1], 1); - tmoveato(csiescseq.arg[1]-1, csiescseq.arg[0]-1); - break; - case 'I': /* CHT -- Cursor Forward Tabulation tab stops */ - DEFAULT(csiescseq.arg[0], 1); - tputtab(csiescseq.arg[0]); - break; - case 'J': /* ED -- Clear screen */ - switch (csiescseq.arg[0]) { - case 0: /* below */ - tclearregion(term.c.x, term.c.y, term.col-1, term.c.y); - if (term.c.y < term.row-1) { - tclearregion(0, term.c.y+1, term.col-1, - term.row-1); - } - break; - case 1: /* above */ - if (term.c.y > 1) - tclearregion(0, 0, term.col-1, term.c.y-1); - tclearregion(0, term.c.y, term.c.x, term.c.y); - break; - case 2: /* all */ - tclearregion(0, 0, term.col-1, term.row-1); - break; - default: - goto unknown; - } - break; - case 'K': /* EL -- Clear line */ - switch (csiescseq.arg[0]) { - case 0: /* right */ - tclearregion(term.c.x, term.c.y, term.col-1, - term.c.y); - break; - case 1: /* left */ - tclearregion(0, term.c.y, term.c.x, term.c.y); - break; - case 2: /* all */ - tclearregion(0, term.c.y, term.col-1, term.c.y); - break; - } - break; - case 'S': /* SU -- Scroll line up */ - DEFAULT(csiescseq.arg[0], 1); - tscrollup(term.top, csiescseq.arg[0]); - break; - case 'T': /* SD -- Scroll line down */ - DEFAULT(csiescseq.arg[0], 1); - tscrolldown(term.top, csiescseq.arg[0]); - break; - case 'L': /* IL -- Insert blank lines */ - DEFAULT(csiescseq.arg[0], 1); - tinsertblankline(csiescseq.arg[0]); - break; - case 'l': /* RM -- Reset Mode */ - tsetmode(csiescseq.priv, 0, csiescseq.arg, csiescseq.narg); - break; - case 'M': /* DL -- Delete lines */ - DEFAULT(csiescseq.arg[0], 1); - tdeleteline(csiescseq.arg[0]); - break; - case 'X': /* ECH -- Erase char */ - DEFAULT(csiescseq.arg[0], 1); - tclearregion(term.c.x, term.c.y, - term.c.x + csiescseq.arg[0] - 1, term.c.y); - break; - case 'P': /* DCH -- Delete char */ - DEFAULT(csiescseq.arg[0], 1); - tdeletechar(csiescseq.arg[0]); - break; - case 'Z': /* CBT -- Cursor Backward Tabulation tab stops */ - DEFAULT(csiescseq.arg[0], 1); - tputtab(-csiescseq.arg[0]); - break; - case 'd': /* VPA -- Move to */ - DEFAULT(csiescseq.arg[0], 1); - tmoveato(term.c.x, csiescseq.arg[0]-1); - break; - case 'h': /* SM -- Set terminal mode */ - tsetmode(csiescseq.priv, 1, csiescseq.arg, csiescseq.narg); - break; - case 'm': /* SGR -- Terminal attribute (color) */ - tsetattr(csiescseq.arg, csiescseq.narg); - break; - case 'n': /* DSR – Device Status Report (cursor position) */ - if (csiescseq.arg[0] == 6) { - len = snprintf(buf, sizeof(buf),"\033[%i;%iR", - term.c.y+1, term.c.x+1); - ttywrite(buf, len, 0); - } - break; - case 'r': /* DECSTBM -- Set Scrolling Region */ - if (csiescseq.priv) { - goto unknown; - } else { - DEFAULT(csiescseq.arg[0], 1); - DEFAULT(csiescseq.arg[1], term.row); - tsetscroll(csiescseq.arg[0]-1, csiescseq.arg[1]-1); - tmoveato(0, 0); - } - break; - case 's': /* DECSC -- Save cursor position (ANSI.SYS) */ - tcursor(CURSOR_SAVE); - break; - case 'u': /* DECRC -- Restore cursor position (ANSI.SYS) */ - tcursor(CURSOR_LOAD); - break; - case ' ': - switch (csiescseq.mode[1]) { - case 'q': /* DECSCUSR -- Set Cursor Style */ - if (xsetcursor(csiescseq.arg[0])) - goto unknown; - break; - default: - goto unknown; - } - break; - } -} - -void -csidump(void) -{ - size_t i; - uint c; - - fprintf(stderr, "ESC["); - for (i = 0; i < csiescseq.len; i++) { - c = csiescseq.buf[i] & 0xff; - if (isprint(c)) { - putc(c, stderr); - } else if (c == '\n') { - fprintf(stderr, "(\\n)"); - } else if (c == '\r') { - fprintf(stderr, "(\\r)"); - } else if (c == 0x1b) { - fprintf(stderr, "(\\e)"); - } else { - fprintf(stderr, "(%02x)", c); - } - } - putc('\n', stderr); -} - -void -csireset(void) -{ - memset(&csiescseq, 0, sizeof(csiescseq)); -} - -void -strhandle(void) -{ - char *p = NULL, *dec; - int j, narg, par; - - term.esc &= ~(ESC_STR_END|ESC_STR); - strparse(); - par = (narg = strescseq.narg) ? atoi(strescseq.args[0]) : 0; - - switch (strescseq.type) { - case ']': /* OSC -- Operating System Command */ - switch (par) { - case 0: - case 1: - case 2: - if (narg > 1) - xsettitle(strescseq.args[1]); - return; - case 52: - if (narg > 2) { - dec = base64dec(strescseq.args[2]); - if (dec) { - xsetsel(dec); - xclipcopy(); - } else { - fprintf(stderr, "erresc: invalid base64\n"); - } - } - return; - case 4: /* color set */ - if (narg < 3) - break; - p = strescseq.args[2]; - /* FALLTHROUGH */ - case 104: /* color reset, here p = NULL */ - j = (narg > 1) ? atoi(strescseq.args[1]) : -1; - if (xsetcolorname(j, p)) { - if (par == 104 && narg <= 1) - return; /* color reset without parameter */ - fprintf(stderr, "erresc: invalid color j=%d, p=%s\n", - j, p ? p : "(null)"); - } else { - /* - * TODO if defaultbg color is changed, borders - * are dirty - */ - redraw(); - } - return; - } - break; - case 'k': /* old title set compatibility */ - xsettitle(strescseq.args[0]); - return; - case 'P': /* DCS -- Device Control String */ - term.mode |= ESC_DCS; - case '_': /* APC -- Application Program Command */ - case '^': /* PM -- Privacy Message */ - return; - } - - fprintf(stderr, "erresc: unknown str "); - strdump(); -} - -void -strparse(void) -{ - int c; - char *p = strescseq.buf; - - strescseq.narg = 0; - strescseq.buf[strescseq.len] = '\0'; - - if (*p == '\0') - return; - - while (strescseq.narg < STR_ARG_SIZ) { - strescseq.args[strescseq.narg++] = p; - while ((c = *p) != ';' && c != '\0') - ++p; - if (c == '\0') - return; - *p++ = '\0'; - } -} - -void -strdump(void) -{ - size_t i; - uint c; - - fprintf(stderr, "ESC%c", strescseq.type); - for (i = 0; i < strescseq.len; i++) { - c = strescseq.buf[i] & 0xff; - if (c == '\0') { - putc('\n', stderr); - return; - } else if (isprint(c)) { - putc(c, stderr); - } else if (c == '\n') { - fprintf(stderr, "(\\n)"); - } else if (c == '\r') { - fprintf(stderr, "(\\r)"); - } else if (c == 0x1b) { - fprintf(stderr, "(\\e)"); - } else { - fprintf(stderr, "(%02x)", c); - } - } - fprintf(stderr, "ESC\\\n"); -} - -void -strreset(void) -{ - strescseq = (STREscape){ - .buf = xrealloc(strescseq.buf, STR_BUF_SIZ), - .siz = STR_BUF_SIZ, - }; -} - -void -sendbreak(const Arg *arg) -{ - if (tcsendbreak(cmdfd, 0)) - perror("Error sending break"); -} - -void -tprinter(char *s, size_t len) -{ - if (iofd != -1 && xwrite(iofd, s, len) < 0) { - perror("Error writing to output file"); - close(iofd); - iofd = -1; - } -} - -void -toggleprinter(const Arg *arg) -{ - term.mode ^= MODE_PRINT; -} - -void -printscreen(const Arg *arg) -{ - tdump(); -} - -void -printsel(const Arg *arg) -{ - tdumpsel(); -} - -void -tdumpsel(void) -{ - char *ptr; - - if ((ptr = getsel())) { - tprinter(ptr, strlen(ptr)); - free(ptr); - } -} - -void -tdumpline(int n) -{ - char buf[UTF_SIZ]; - Glyph *bp, *end; - - bp = &term.line[n][0]; - end = &bp[MIN(tlinelen(n), term.col) - 1]; - if (bp != end || bp->u != ' ') { - for ( ;bp <= end; ++bp) - tprinter(buf, utf8encode(bp->u, buf)); - } - tprinter("\n", 1); -} - -void -tdump(void) -{ - int i; - - for (i = 0; i < term.row; ++i) - tdumpline(i); -} - -void -tputtab(int n) -{ - uint x = term.c.x; - - if (n > 0) { - while (x < term.col && n--) - for (++x; x < term.col && !term.tabs[x]; ++x) - /* nothing */ ; - } else if (n < 0) { - while (x > 0 && n++) - for (--x; x > 0 && !term.tabs[x]; --x) - /* nothing */ ; - } - term.c.x = LIMIT(x, 0, term.col-1); -} - -void -tdefutf8(char ascii) -{ - if (ascii == 'G') - term.mode |= MODE_UTF8; - else if (ascii == '@') - term.mode &= ~MODE_UTF8; -} - -void -tdeftran(char ascii) -{ - static char cs[] = "0B"; - static int vcs[] = {CS_GRAPHIC0, CS_USA}; - char *p; - - if ((p = strchr(cs, ascii)) == NULL) { - fprintf(stderr, "esc unhandled charset: ESC ( %c\n", ascii); - } else { - term.trantbl[term.icharset] = vcs[p - cs]; - } -} - -void -tdectest(char c) -{ - int x, y; - - if (c == '8') { /* DEC screen alignment test. */ - for (x = 0; x < term.col; ++x) { - for (y = 0; y < term.row; ++y) - tsetchar('E', &term.c.attr, x, y); - } - } -} - -void -tstrsequence(uchar c) -{ - strreset(); - - switch (c) { - case 0x90: /* DCS -- Device Control String */ - c = 'P'; - term.esc |= ESC_DCS; - break; - case 0x9f: /* APC -- Application Program Command */ - c = '_'; - break; - case 0x9e: /* PM -- Privacy Message */ - c = '^'; - break; - case 0x9d: /* OSC -- Operating System Command */ - c = ']'; - break; - } - strescseq.type = c; - term.esc |= ESC_STR; -} - -void -tcontrolcode(uchar ascii) -{ - switch (ascii) { - case '\t': /* HT */ - tputtab(1); - return; - case '\b': /* BS */ - tmoveto(term.c.x-1, term.c.y); - return; - case '\r': /* CR */ - tmoveto(0, term.c.y); - return; - case '\f': /* LF */ - case '\v': /* VT */ - case '\n': /* LF */ - /* go to first col if the mode is set */ - tnewline(IS_SET(MODE_CRLF)); - return; - case '\a': /* BEL */ - if (term.esc & ESC_STR_END) { - /* backwards compatibility to xterm */ - strhandle(); - } else { - xbell(); - } - break; - case '\033': /* ESC */ - csireset(); - term.esc &= ~(ESC_CSI|ESC_ALTCHARSET|ESC_TEST); - term.esc |= ESC_START; - return; - case '\016': /* SO (LS1 -- Locking shift 1) */ - case '\017': /* SI (LS0 -- Locking shift 0) */ - term.charset = 1 - (ascii - '\016'); - return; - case '\032': /* SUB */ - tsetchar('?', &term.c.attr, term.c.x, term.c.y); - case '\030': /* CAN */ - csireset(); - break; - case '\005': /* ENQ (IGNORED) */ - case '\000': /* NUL (IGNORED) */ - case '\021': /* XON (IGNORED) */ - case '\023': /* XOFF (IGNORED) */ - case 0177: /* DEL (IGNORED) */ - return; - case 0x80: /* TODO: PAD */ - case 0x81: /* TODO: HOP */ - case 0x82: /* TODO: BPH */ - case 0x83: /* TODO: NBH */ - case 0x84: /* TODO: IND */ - break; - case 0x85: /* NEL -- Next line */ - tnewline(1); /* always go to first col */ - break; - case 0x86: /* TODO: SSA */ - case 0x87: /* TODO: ESA */ - break; - case 0x88: /* HTS -- Horizontal tab stop */ - term.tabs[term.c.x] = 1; - break; - case 0x89: /* TODO: HTJ */ - case 0x8a: /* TODO: VTS */ - case 0x8b: /* TODO: PLD */ - case 0x8c: /* TODO: PLU */ - case 0x8d: /* TODO: RI */ - case 0x8e: /* TODO: SS2 */ - case 0x8f: /* TODO: SS3 */ - case 0x91: /* TODO: PU1 */ - case 0x92: /* TODO: PU2 */ - case 0x93: /* TODO: STS */ - case 0x94: /* TODO: CCH */ - case 0x95: /* TODO: MW */ - case 0x96: /* TODO: SPA */ - case 0x97: /* TODO: EPA */ - case 0x98: /* TODO: SOS */ - case 0x99: /* TODO: SGCI */ - break; - case 0x9a: /* DECID -- Identify Terminal */ - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 0x9b: /* TODO: CSI */ - case 0x9c: /* TODO: ST */ - break; - case 0x90: /* DCS -- Device Control String */ - case 0x9d: /* OSC -- Operating System Command */ - case 0x9e: /* PM -- Privacy Message */ - case 0x9f: /* APC -- Application Program Command */ - tstrsequence(ascii); - return; - } - /* only CAN, SUB, \a and C1 chars interrupt a sequence */ - term.esc &= ~(ESC_STR_END|ESC_STR); -} - -/* - * returns 1 when the sequence is finished and it hasn't to read - * more characters for this sequence, otherwise 0 - */ -int -eschandle(uchar ascii) -{ - switch (ascii) { - case '[': - term.esc |= ESC_CSI; - return 0; - case '#': - term.esc |= ESC_TEST; - return 0; - case '%': - term.esc |= ESC_UTF8; - return 0; - case 'P': /* DCS -- Device Control String */ - case '_': /* APC -- Application Program Command */ - case '^': /* PM -- Privacy Message */ - case ']': /* OSC -- Operating System Command */ - case 'k': /* old title set compatibility */ - tstrsequence(ascii); - return 0; - case 'n': /* LS2 -- Locking shift 2 */ - case 'o': /* LS3 -- Locking shift 3 */ - term.charset = 2 + (ascii - 'n'); - break; - case '(': /* GZD4 -- set primary charset G0 */ - case ')': /* G1D4 -- set secondary charset G1 */ - case '*': /* G2D4 -- set tertiary charset G2 */ - case '+': /* G3D4 -- set quaternary charset G3 */ - term.icharset = ascii - '('; - term.esc |= ESC_ALTCHARSET; - return 0; - case 'D': /* IND -- Linefeed */ - if (term.c.y == term.bot) { - tscrollup(term.top, 1); - } else { - tmoveto(term.c.x, term.c.y+1); - } - break; - case 'E': /* NEL -- Next line */ - tnewline(1); /* always go to first col */ - break; - case 'H': /* HTS -- Horizontal tab stop */ - term.tabs[term.c.x] = 1; - break; - case 'M': /* RI -- Reverse index */ - if (term.c.y == term.top) { - tscrolldown(term.top, 1); - } else { - tmoveto(term.c.x, term.c.y-1); - } - break; - case 'Z': /* DECID -- Identify Terminal */ - ttywrite(vtiden, strlen(vtiden), 0); - break; - case 'c': /* RIS -- Reset to initial state */ - treset(); - resettitle(); - xloadcols(); - break; - case '=': /* DECPAM -- Application keypad */ - xsetmode(1, MODE_APPKEYPAD); - break; - case '>': /* DECPNM -- Normal keypad */ - xsetmode(0, MODE_APPKEYPAD); - break; - case '7': /* DECSC -- Save Cursor */ - tcursor(CURSOR_SAVE); - break; - case '8': /* DECRC -- Restore Cursor */ - tcursor(CURSOR_LOAD); - break; - case '\\': /* ST -- String Terminator */ - if (term.esc & ESC_STR_END) - strhandle(); - break; - default: - fprintf(stderr, "erresc: unknown sequence ESC 0x%02X '%c'\n", - (uchar) ascii, isprint(ascii)? ascii:'.'); - break; - } - return 1; -} - -void -tputc(Rune u) -{ - char c[UTF_SIZ]; - int control; - int width, len; - Glyph *gp; - - control = ISCONTROL(u); - if (!IS_SET(MODE_UTF8) && !IS_SET(MODE_SIXEL)) { - c[0] = u; - width = len = 1; - } else { - len = utf8encode(u, c); - if (!control && (width = wcwidth(u)) == -1) { - memcpy(c, "\357\277\275", 4); /* UTF_INVALID */ - width = 1; - } - } - - if (IS_SET(MODE_PRINT)) - tprinter(c, len); - - /* - * STR sequence must be checked before anything else - * because it uses all following characters until it - * receives a ESC, a SUB, a ST or any other C1 control - * character. - */ - if (term.esc & ESC_STR) { - if (u == '\a' || u == 030 || u == 032 || u == 033 || - ISCONTROLC1(u)) { - term.esc &= ~(ESC_START|ESC_STR|ESC_DCS); - if (IS_SET(MODE_SIXEL)) { - /* TODO: render sixel */; - term.mode &= ~MODE_SIXEL; - return; - } - term.esc |= ESC_STR_END; - goto check_control_code; - } - - if (IS_SET(MODE_SIXEL)) { - /* TODO: implement sixel mode */ - return; - } - if (term.esc&ESC_DCS && strescseq.len == 0 && u == 'q') - term.mode |= MODE_SIXEL; - - if (strescseq.len+len >= strescseq.siz) { - /* - * Here is a bug in terminals. If the user never sends - * some code to stop the str or esc command, then st - * will stop responding. But this is better than - * silently failing with unknown characters. At least - * then users will report back. - * - * In the case users ever get fixed, here is the code: - */ - /* - * term.esc = 0; - * strhandle(); - */ - if (strescseq.siz > (SIZE_MAX - UTF_SIZ) / 2) - return; - strescseq.siz *= 2; - strescseq.buf = xrealloc(strescseq.buf, strescseq.siz); - } - - memmove(&strescseq.buf[strescseq.len], c, len); - strescseq.len += len; - return; - } - -check_control_code: - /* - * Actions of control codes must be performed as soon they arrive - * because they can be embedded inside a control sequence, and - * they must not cause conflicts with sequences. - */ - if (control) { - tcontrolcode(u); - /* - * control codes are not shown ever - */ - return; - } else if (term.esc & ESC_START) { - if (term.esc & ESC_CSI) { - csiescseq.buf[csiescseq.len++] = u; - if (BETWEEN(u, 0x40, 0x7E) - || csiescseq.len >= \ - sizeof(csiescseq.buf)-1) { - term.esc = 0; - csiparse(); - csihandle(); - } - return; - } else if (term.esc & ESC_UTF8) { - tdefutf8(u); - } else if (term.esc & ESC_ALTCHARSET) { - tdeftran(u); - } else if (term.esc & ESC_TEST) { - tdectest(u); - } else { - if (!eschandle(u)) - return; - /* sequence already finished */ - } - term.esc = 0; - /* - * All characters which form part of a sequence are not - * printed - */ - return; - } - if (sel.ob.x != -1 && BETWEEN(term.c.y, sel.ob.y, sel.oe.y)) - selclear(); - - gp = &term.line[term.c.y][term.c.x]; - if (IS_SET(MODE_WRAP) && (term.c.state & CURSOR_WRAPNEXT)) { - gp->mode |= ATTR_WRAP; - tnewline(1); - gp = &term.line[term.c.y][term.c.x]; - } - - if (IS_SET(MODE_INSERT) && term.c.x+width < term.col) - memmove(gp+width, gp, (term.col - term.c.x - width) * sizeof(Glyph)); - - if (term.c.x+width > term.col) { - tnewline(1); - gp = &term.line[term.c.y][term.c.x]; - } - - tsetchar(u, &term.c.attr, term.c.x, term.c.y); - - if (width == 2) { - gp->mode |= ATTR_WIDE; - if (term.c.x+1 < term.col) { - gp[1].u = '\0'; - gp[1].mode = ATTR_WDUMMY; - } - } - if (term.c.x+width < term.col) { - tmoveto(term.c.x+width, term.c.y); - } else { - term.c.state |= CURSOR_WRAPNEXT; - } -} - -int -twrite(const char *buf, int buflen, int show_ctrl) -{ - int charsize; - Rune u; - int n; - - for (n = 0; n < buflen; n += charsize) { - if (IS_SET(MODE_UTF8) && !IS_SET(MODE_SIXEL)) { - /* process a complete utf8 char */ - charsize = utf8decode(buf + n, &u, buflen - n); - if (charsize == 0) - break; - } else { - u = buf[n] & 0xFF; - charsize = 1; - } - if (show_ctrl && ISCONTROL(u)) { - if (u & 0x80) { - u &= 0x7f; - tputc('^'); - tputc('['); - } else if (u != '\n' && u != '\r' && u != '\t') { - u ^= 0x40; - tputc('^'); - } - } - tputc(u); - } - return n; -} - -void -tresize(int col, int row) -{ - int i; - int minrow = MIN(row, term.row); - int mincol = MIN(col, term.col); - int *bp; - TCursor c; - - if (col < 1 || row < 1) { - fprintf(stderr, - "tresize: error resizing to %dx%d\n", col, row); - return; - } - - /* - * slide screen to keep cursor where we expect it - - * tscrollup would work here, but we can optimize to - * memmove because we're freeing the earlier lines - */ - for (i = 0; i <= term.c.y - row; i++) { - free(term.line[i]); - free(term.alt[i]); - } - /* ensure that both src and dst are not NULL */ - if (i > 0) { - memmove(term.line, term.line + i, row * sizeof(Line)); - memmove(term.alt, term.alt + i, row * sizeof(Line)); - } - for (i += row; i < term.row; i++) { - free(term.line[i]); - free(term.alt[i]); - } - - /* resize to new height */ - term.line = xrealloc(term.line, row * sizeof(Line)); - term.alt = xrealloc(term.alt, row * sizeof(Line)); - term.dirty = xrealloc(term.dirty, row * sizeof(*term.dirty)); - term.tabs = xrealloc(term.tabs, col * sizeof(*term.tabs)); - - /* resize each row to new width, zero-pad if needed */ - for (i = 0; i < minrow; i++) { - term.line[i] = xrealloc(term.line[i], col * sizeof(Glyph)); - term.alt[i] = xrealloc(term.alt[i], col * sizeof(Glyph)); - } - - /* allocate any new rows */ - for (/* i = minrow */; i < row; i++) { - term.line[i] = xmalloc(col * sizeof(Glyph)); - term.alt[i] = xmalloc(col * sizeof(Glyph)); - } - if (col > term.col) { - bp = term.tabs + term.col; - - memset(bp, 0, sizeof(*term.tabs) * (col - term.col)); - while (--bp > term.tabs && !*bp) - /* nothing */ ; - for (bp += tabspaces; bp < term.tabs + col; bp += tabspaces) - *bp = 1; - } - /* update terminal size */ - term.col = col; - term.row = row; - /* reset scrolling region */ - tsetscroll(0, row-1); - /* make use of the LIMIT in tmoveto */ - tmoveto(term.c.x, term.c.y); - /* Clearing both screens (it makes dirty all lines) */ - c = term.c; - for (i = 0; i < 2; i++) { - if (mincol < col && 0 < minrow) { - tclearregion(mincol, 0, col - 1, minrow - 1); - } - if (0 < col && minrow < row) { - tclearregion(0, minrow, col - 1, row - 1); - } - tswapscreen(); - tcursor(CURSOR_LOAD); - } - term.c = c; -} - -void -resettitle(void) -{ - xsettitle(NULL); -} - -void -drawregion(int x1, int y1, int x2, int y2) -{ - int y; - for (y = y1; y < y2; y++) { - if (!term.dirty[y]) - continue; - - term.dirty[y] = 0; - xdrawline(term.line[y], x1, y, x2); - } -} - -void -draw(void) -{ - int cx = term.c.x; - - if (!xstartdraw()) - return; - - /* adjust cursor position */ - LIMIT(term.ocx, 0, term.col-1); - LIMIT(term.ocy, 0, term.row-1); - if (term.line[term.ocy][term.ocx].mode & ATTR_WDUMMY) - term.ocx--; - if (term.line[term.c.y][cx].mode & ATTR_WDUMMY) - cx--; - - drawregion(0, 0, term.col, term.row); - xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], - term.ocx, term.ocy, term.line[term.ocy][term.ocx]); - term.ocx = cx, term.ocy = term.c.y; - xfinishdraw(); - xximspot(term.ocx, term.ocy); -} - -void -redraw(void) -{ - tfulldirt(); - draw(); -} diff --git a/.suckless/st/st.c.rej b/.suckless/st/st.c.rej deleted file mode 100644 index 9a79b8b..0000000 --- a/.suckless/st/st.c.rej +++ /dev/null @@ -1,12 +0,0 @@ ---- st.c -+++ st.c -@@ -2599,7 +2599,8 @@ draw(void) - - drawregion(0, 0, term.col, term.row); - xdrawcursor(cx, term.c.y, term.line[term.c.y][cx], -- term.ocx, term.ocy, term.line[term.ocy][term.ocx]); -+ term.ocx, term.ocy, term.line[term.ocy][term.ocx], -+ term.line[term.ocy], term.col); - term.ocx = cx; - term.ocy = term.c.y; - xfinishdraw(); diff --git a/.suckless/st/st.h b/.suckless/st/st.h deleted file mode 100644 index 07ba678..0000000 --- a/.suckless/st/st.h +++ /dev/null @@ -1,125 +0,0 @@ -/* See LICENSE for license details. */ - -#include -#include - -/* macros */ -#define MIN(a, b) ((a) < (b) ? (a) : (b)) -#define MAX(a, b) ((a) < (b) ? (b) : (a)) -#define LEN(a) (sizeof(a) / sizeof(a)[0]) -#define BETWEEN(x, a, b) ((a) <= (x) && (x) <= (b)) -#define DIVCEIL(n, d) (((n) + ((d) - 1)) / (d)) -#define DEFAULT(a, b) (a) = (a) ? (a) : (b) -#define LIMIT(x, a, b) (x) = (x) < (a) ? (a) : (x) > (b) ? (b) : (x) -#define ATTRCMP(a, b) (((a).mode & (~ATTR_WRAP) & (~ATTR_LIGA)) != ((b).mode & (~ATTR_WRAP) & (~ATTR_LIGA)) || \ - (a).fg != (b).fg || \ - (a).bg != (b).bg) -#define TIMEDIFF(t1, t2) ((t1.tv_sec-t2.tv_sec)*1000 + \ - (t1.tv_nsec-t2.tv_nsec)/1E6) -#define MODBIT(x, set, bit) ((set) ? ((x) |= (bit)) : ((x) &= ~(bit))) - -#define TRUECOLOR(r,g,b) (1 << 24 | (r) << 16 | (g) << 8 | (b)) -#define IS_TRUECOL(x) (1 << 24 & (x)) - -enum glyph_attribute { - ATTR_NULL = 0, - ATTR_BOLD = 1 << 0, - ATTR_FAINT = 1 << 1, - ATTR_ITALIC = 1 << 2, - ATTR_UNDERLINE = 1 << 3, - ATTR_BLINK = 1 << 4, - ATTR_REVERSE = 1 << 5, - ATTR_INVISIBLE = 1 << 6, - ATTR_STRUCK = 1 << 7, - ATTR_WRAP = 1 << 8, - ATTR_WIDE = 1 << 9, - ATTR_WDUMMY = 1 << 10, - ATTR_LIGA = 1 << 11, - ATTR_BOLD_FAINT = ATTR_BOLD | ATTR_FAINT, -}; - -enum selection_mode { - SEL_IDLE = 0, - SEL_EMPTY = 1, - SEL_READY = 2 -}; - -enum selection_type { - SEL_REGULAR = 1, - SEL_RECTANGULAR = 2 -}; - -enum selection_snap { - SNAP_WORD = 1, - SNAP_LINE = 2 -}; - -typedef unsigned char uchar; -typedef unsigned int uint; -typedef unsigned long ulong; -typedef unsigned short ushort; - -typedef uint_least32_t Rune; - -#define Glyph Glyph_ -typedef struct { - Rune u; /* character code */ - ushort mode; /* attribute flags */ - uint32_t fg; /* foreground */ - uint32_t bg; /* background */ -} Glyph; - -typedef Glyph *Line; - -typedef union { - int i; - uint ui; - float f; - const void *v; - const char *s; -} Arg; - -void die(const char *, ...); -void redraw(void); -void draw(void); - -void printscreen(const Arg *); -void printsel(const Arg *); -void sendbreak(const Arg *); -void toggleprinter(const Arg *); - -int tattrset(int); -void tnew(int, int); -void tresize(int, int); -void tsetdirtattr(int); -void ttyhangup(void); -int ttynew(char *, char *, char *, char **); -size_t ttyread(void); -void ttyresize(int, int); -void ttywrite(const char *, size_t, int); - -void resettitle(void); - -void selclear(void); -void selinit(void); -void selstart(int, int, int); -void selextend(int, int, int, int); -int selected(int, int); -char *getsel(void); - -size_t utf8encode(Rune, char *); - -void *xmalloc(size_t); -void *xrealloc(void *, size_t); -char *xstrdup(char *); - -/* config.h globals */ -extern char *utmp; -extern char *stty_args; -extern char *vtiden; -extern wchar_t *worddelimiters; -extern int allowaltscreen; -extern char *termname; -extern unsigned int tabspaces; -extern unsigned int defaultfg; -extern unsigned int defaultbg; diff --git a/.suckless/st/st.info b/.suckless/st/st.info deleted file mode 100644 index 78ffd30..0000000 --- a/.suckless/st/st.info +++ /dev/null @@ -1,222 +0,0 @@ -st| simpleterm, - acsc=+C\,D-A.B0E``aaffgghFiGjjkkllmmnnooppqqrrssttuuvvwwxxyyzz{{||}}~~, - am, - bce, - bel=^G, - blink=\E[5m, - bold=\E[1m, - cbt=\E[Z, - cvvis=\E[?25h, - civis=\E[?25l, - clear=\E[H\E[2J, - cnorm=\E[?12l\E[?25h, - colors#8, - cols#80, - cr=^M, - csr=\E[%i%p1%d;%p2%dr, - cub=\E[%p1%dD, - cub1=^H, - cud1=^J, - cud=\E[%p1%dB, - cuf1=\E[C, - cuf=\E[%p1%dC, - cup=\E[%i%p1%d;%p2%dH, - cuu1=\E[A, - cuu=\E[%p1%dA, - dch=\E[%p1%dP, - dch1=\E[P, - dim=\E[2m, - dl=\E[%p1%dM, - dl1=\E[M, - ech=\E[%p1%dX, - ed=\E[J, - el=\E[K, - el1=\E[1K, - enacs=\E)0, - flash=\E[?5h$<80/>\E[?5l, - fsl=^G, - home=\E[H, - hpa=\E[%i%p1%dG, - hs, - ht=^I, - hts=\EH, - ich=\E[%p1%d@, - il1=\E[L, - il=\E[%p1%dL, - ind=^J, - indn=\E[%p1%dS, - invis=\E[8m, - is2=\E[4l\E>\E[?1034l, - it#8, - kel=\E[1;2F, - ked=\E[1;5F, - ka1=\E[1~, - ka3=\E[5~, - kc1=\E[4~, - kc3=\E[6~, - kbs=\177, - kcbt=\E[Z, - kb2=\EOu, - kcub1=\EOD, - kcud1=\EOB, - kcuf1=\EOC, - kcuu1=\EOA, - kDC=\E[3;2~, - kent=\EOM, - kEND=\E[1;2F, - kIC=\E[2;2~, - kNXT=\E[6;2~, - kPRV=\E[5;2~, - kHOM=\E[1;2H, - kLFT=\E[1;2D, - kRIT=\E[1;2C, - kind=\E[1;2B, - kri=\E[1;2A, - kclr=\E[3;5~, - kdl1=\E[3;2~, - kdch1=\E[3~, - kich1=\E[2~, - kend=\E[4~, - kf1=\EOP, - kf2=\EOQ, - kf3=\EOR, - kf4=\EOS, - kf5=\E[15~, - kf6=\E[17~, - kf7=\E[18~, - kf8=\E[19~, - kf9=\E[20~, - kf10=\E[21~, - kf11=\E[23~, - kf12=\E[24~, - kf13=\E[1;2P, - kf14=\E[1;2Q, - kf15=\E[1;2R, - kf16=\E[1;2S, - kf17=\E[15;2~, - kf18=\E[17;2~, - kf19=\E[18;2~, - kf20=\E[19;2~, - kf21=\E[20;2~, - kf22=\E[21;2~, - kf23=\E[23;2~, - kf24=\E[24;2~, - kf25=\E[1;5P, - kf26=\E[1;5Q, - kf27=\E[1;5R, - kf28=\E[1;5S, - kf29=\E[15;5~, - kf30=\E[17;5~, - kf31=\E[18;5~, - kf32=\E[19;5~, - kf33=\E[20;5~, - kf34=\E[21;5~, - kf35=\E[23;5~, - kf36=\E[24;5~, - kf37=\E[1;6P, - kf38=\E[1;6Q, - kf39=\E[1;6R, - kf40=\E[1;6S, - kf41=\E[15;6~, - kf42=\E[17;6~, - kf43=\E[18;6~, - kf44=\E[19;6~, - kf45=\E[20;6~, - kf46=\E[21;6~, - kf47=\E[23;6~, - kf48=\E[24;6~, - kf49=\E[1;3P, - kf50=\E[1;3Q, - kf51=\E[1;3R, - kf52=\E[1;3S, - kf53=\E[15;3~, - kf54=\E[17;3~, - kf55=\E[18;3~, - kf56=\E[19;3~, - kf57=\E[20;3~, - kf58=\E[21;3~, - kf59=\E[23;3~, - kf60=\E[24;3~, - kf61=\E[1;4P, - kf62=\E[1;4Q, - kf63=\E[1;4R, - khome=\E[1~, - kil1=\E[2;5~, - krmir=\E[2;2~, - knp=\E[6~, - kmous=\E[M, - kpp=\E[5~, - lines#24, - mir, - msgr, - npc, - op=\E[39;49m, - pairs#64, - mc0=\E[i, - mc4=\E[4i, - mc5=\E[5i, - rc=\E8, - rev=\E[7m, - ri=\EM, - ritm=\E[23m, - rmacs=\E(B, - rmcup=\E[?1049l, - rmir=\E[4l, - rmkx=\E[?1l\E>, - rmso=\E[27m, - rmul=\E[24m, - rs1=\Ec, - rs2=\E[4l\E>\E[?1034l, - sc=\E7, - setab=\E[4%p1%dm, - setaf=\E[3%p1%dm, - setb=\E[4%?%p1%{1}%=%t4%e%p1%{3}%=%t6%e%p1%{4}%=%t1%e%p1%{6}%=%t3%e%p1%d%;m, - setf=\E[3%?%p1%{1}%=%t4%e%p1%{3}%=%t6%e%p1%{4}%=%t1%e%p1%{6}%=%t3%e%p1%d%;m, - sgr0=\E[0m, - sgr=%?%p9%t\E(0%e\E(B%;\E[0%?%p6%t;1%;%?%p2%t;4%;%?%p1%p3%|%t;7%;%?%p4%t;5%;%?%p7%t;8%;m, - sitm=\E[3m, - smacs=\E(0, - smcup=\E[?1049h, - smir=\E[4h, - smkx=\E[?1h\E=, - smso=\E[7m, - smul=\E[4m, - tbc=\E[3g, - tsl=\E]0;, - xenl, - vpa=\E[%i%p1%dd, -# XTerm extensions - rmxx=\E[29m, - smxx=\E[9m, -# tmux extensions, see TERMINFO EXTENSIONS in tmux(1) - Tc, - Ms=\E]52;%p1%s;%p2%s\007, - Se=\E[2 q, - Ss=\E[%p1%d q, - -st-256color| simpleterm with 256 colors, - use=st, - ccc, - colors#256, - oc=\E]104\007, - pairs#32767, -# Nicked from xterm-256color - initc=\E]4;%p1%d;rgb\:%p2%{255}%*%{1000}%/%2.2X/%p3%{255}%*%{1000}%/%2.2X/%p4%{255}%*%{1000}%/%2.2X\E\\, - setab=\E[%?%p1%{8}%<%t4%p1%d%e%p1%{16}%<%t10%p1%{8}%-%d%e48;5;%p1%d%;m, - setaf=\E[%?%p1%{8}%<%t3%p1%d%e%p1%{16}%<%t9%p1%{8}%-%d%e38;5;%p1%d%;m, - -st-meta| simpleterm with meta key, - use=st, - km, - rmm=\E[?1034l, - smm=\E[?1034h, - rs2=\E[4l\E>\E[?1034h, - is2=\E[4l\E>\E[?1034h, - -st-meta-256color| simpleterm with meta key and 256 colors, - use=st-256color, - km, - rmm=\E[?1034l, - smm=\E[?1034h, - rs2=\E[4l\E>\E[?1034h, - is2=\E[4l\E>\E[?1034h, diff --git a/.suckless/st/st.o b/.suckless/st/st.o deleted file mode 100644 index f37cd09ec49fd2cee347cc72b86bd8c23586a099..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 73248 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 P -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -static char *argv0; -#include "arg.h" -#include "st.h" -#include "win.h" -#include "hb.h" - -/* types used in config.h */ -typedef struct { - uint mod; - KeySym keysym; - void (*func)(const Arg *); - const Arg arg; -} Shortcut; - -typedef struct { - uint mod; - uint button; - void (*func)(const Arg *); - const Arg arg; - uint release; -} MouseShortcut; - -typedef struct { - KeySym k; - uint mask; - char *s; - /* three-valued logic variables: 0 indifferent, 1 on, -1 off */ - signed char appkey; /* application keypad */ - signed char appcursor; /* application cursor */ -} Key; - -/* X modifiers */ -#define XK_ANY_MOD UINT_MAX -#define XK_NO_MOD 0 -#define XK_SWITCH_MOD (1<<13) - -/* function definitions used in config.h */ -static void clipcopy(const Arg *); -static void clippaste(const Arg *); -static void numlock(const Arg *); -static void selpaste(const Arg *); -static void zoom(const Arg *); -static void zoomabs(const Arg *); -static void zoomreset(const Arg *); -static void ttysend(const Arg *); - -/* config.h for applying patches and the configuration. */ -#include "config.h" - -/* XEMBED messages */ -#define XEMBED_FOCUS_IN 4 -#define XEMBED_FOCUS_OUT 5 - -/* macros */ -#define IS_SET(flag) ((win.mode & (flag)) != 0) -#define TRUERED(x) (((x) & 0xff0000) >> 8) -#define TRUEGREEN(x) (((x) & 0xff00)) -#define TRUEBLUE(x) (((x) & 0xff) << 8) - -typedef XftDraw *Draw; -typedef XftColor Color; -typedef XftGlyphFontSpec GlyphFontSpec; - -/* Purely graphic info */ -typedef struct { - int tw, th; /* tty width and height */ - int w, h; /* window width and height */ - int hborderpx, vborderpx; - int ch; /* char height */ - int cw; /* char width */ - int mode; /* window state/mode flags */ - int cursor; /* cursor style */ -} TermWindow; - -typedef struct { - Display *dpy; - Colormap cmap; - Window win; - Drawable buf; - GlyphFontSpec *specbuf; /* font spec buffer used for rendering */ - Atom xembed, wmdeletewin, netwmname, netwmpid; - struct { - XIM xim; - XIC xic; - XPoint spot; - XVaNestedList spotlist; - } ime; - Draw draw; - Visual *vis; - XSetWindowAttributes attrs; - int scr; - int isfixed; /* is fixed geometry? */ - int l, t; /* left and top offset */ - int gm; /* geometry mask */ -} XWindow; - -typedef struct { - Atom xtarget; - char *primary, *clipboard; - struct timespec tclick1; - struct timespec tclick2; -} XSelection; - -/* Font structure */ -#define Font Font_ -typedef struct { - int height; - int width; - int ascent; - int descent; - int badslant; - int badweight; - short lbearing; - short rbearing; - XftFont *match; - FcFontSet *set; - FcPattern *pattern; -} Font; - -/* Drawing Context */ -typedef struct { - Color *col; - size_t collen; - Font font, bfont, ifont, ibfont; - GC gc; -} DC; - -static inline ushort sixd_to_16bit(int); -static int xmakeglyphfontspecs(XftGlyphFontSpec *, const Glyph *, int, int, int); -static void xdrawglyphfontspecs(const XftGlyphFontSpec *, Glyph, int, int, int); -static void xdrawglyph(Glyph, int, int); -static void xclear(int, int, int, int); -static int xgeommasktogravity(int); -static int ximopen(Display *); -static void ximinstantiate(Display *, XPointer, XPointer); -static void ximdestroy(XIM, XPointer, XPointer); -static int xicdestroy(XIC, XPointer, XPointer); -static void xinit(int, int); -static void cresize(int, int); -static void xresize(int, int); -static void xhints(void); -static int xloadcolor(int, const char *, Color *); -static int xloadfont(Font *, FcPattern *); -static void xloadfonts(char *, double); -static void xunloadfont(Font *); -static void xunloadfonts(void); -static void xsetenv(void); -static void xseturgency(int); -static int evcol(XEvent *); -static int evrow(XEvent *); - -static void expose(XEvent *); -static void visibility(XEvent *); -static void unmap(XEvent *); -static void kpress(XEvent *); -static void cmessage(XEvent *); -static void resize(XEvent *); -static void focus(XEvent *); -static int mouseaction(XEvent *, uint); -static void brelease(XEvent *); -static void bpress(XEvent *); -static void bmotion(XEvent *); -static void propnotify(XEvent *); -static void selnotify(XEvent *); -static void selclear_(XEvent *); -static void selrequest(XEvent *); -static void setsel(char *, Time); -static void mousesel(XEvent *, int); -static void mousereport(XEvent *); -static char *kmap(KeySym, uint); -static int match(uint, uint); - -static void run(void); -static void usage(void); - -static void (*handler[LASTEvent])(XEvent *) = { - [KeyPress] = kpress, - [ClientMessage] = cmessage, - [ConfigureNotify] = resize, - [VisibilityNotify] = visibility, - [UnmapNotify] = unmap, - [Expose] = expose, - [FocusIn] = focus, - [FocusOut] = focus, - [MotionNotify] = bmotion, - [ButtonPress] = bpress, - [ButtonRelease] = brelease, -/* - * Uncomment if you want the selection to disappear when you select something - * different in another window. - */ -/* [SelectionClear] = selclear_, */ - [SelectionNotify] = selnotify, -/* - * PropertyNotify is only turned on when there is some INCR transfer happening - * for the selection retrieval. - */ - [PropertyNotify] = propnotify, - [SelectionRequest] = selrequest, -}; - -/* Globals */ -static DC dc; -static XWindow xw; -static XSelection xsel; -static TermWindow win; - -/* Font Ring Cache */ -enum { - FRC_NORMAL, - FRC_ITALIC, - FRC_BOLD, - FRC_ITALICBOLD -}; - -typedef struct { - XftFont *font; - int flags; - Rune unicodep; -} Fontcache; - -/* Fontcache is an array now. A new font will be appended to the array. */ -static Fontcache *frc = NULL; -static int frclen = 0; -static int frccap = 0; -static char *usedfont = NULL; -static double usedfontsize = 0; -static double defaultfontsize = 0; - -static char *opt_class = NULL; -static char **opt_cmd = NULL; -static char *opt_embed = NULL; -static char *opt_font = NULL; -static char *opt_io = NULL; -static char *opt_line = NULL; -static char *opt_name = NULL; -static char *opt_title = NULL; - -static int oldbutton = 3; /* button event on startup: 3 = release */ - -void -clipcopy(const Arg *dummy) -{ - Atom clipboard; - - free(xsel.clipboard); - xsel.clipboard = NULL; - - if (xsel.primary != NULL) { - xsel.clipboard = xstrdup(xsel.primary); - clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - XSetSelectionOwner(xw.dpy, clipboard, xw.win, CurrentTime); - } -} - -void -clippaste(const Arg *dummy) -{ - Atom clipboard; - - clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - XConvertSelection(xw.dpy, clipboard, xsel.xtarget, clipboard, - xw.win, CurrentTime); -} - -void -selpaste(const Arg *dummy) -{ - XConvertSelection(xw.dpy, XA_PRIMARY, xsel.xtarget, XA_PRIMARY, - xw.win, CurrentTime); -} - -void -numlock(const Arg *dummy) -{ - win.mode ^= MODE_NUMLOCK; -} - -void -zoom(const Arg *arg) -{ - Arg larg; - - larg.f = usedfontsize + arg->f; - zoomabs(&larg); -} - -void -zoomabs(const Arg *arg) -{ - xunloadfonts(); - xloadfonts(usedfont, arg->f); - cresize(0, 0); - redraw(); - xhints(); -} - -void -zoomreset(const Arg *arg) -{ - Arg larg; - - if (defaultfontsize > 0) { - larg.f = defaultfontsize; - zoomabs(&larg); - } -} - -void -ttysend(const Arg *arg) -{ - ttywrite(arg->s, strlen(arg->s), 1); -} - -int -evcol(XEvent *e) -{ - int x = e->xbutton.x - win.hborderpx; - LIMIT(x, 0, win.tw - 1); - return x / win.cw; -} - -int -evrow(XEvent *e) -{ - int y = e->xbutton.y - win.vborderpx; - LIMIT(y, 0, win.th - 1); - return y / win.ch; -} - -void -mousesel(XEvent *e, int done) -{ - int type, seltype = SEL_REGULAR; - uint state = e->xbutton.state & ~(Button1Mask | forcemousemod); - - for (type = 1; type < LEN(selmasks); ++type) { - if (match(selmasks[type], state)) { - seltype = type; - break; - } - } - selextend(evcol(e), evrow(e), seltype, done); - if (done) - setsel(getsel(), e->xbutton.time); -} - -void -mousereport(XEvent *e) -{ - int len, x = evcol(e), y = evrow(e), - button = e->xbutton.button, state = e->xbutton.state; - char buf[40]; - static int ox, oy; - - /* from urxvt */ - if (e->xbutton.type == MotionNotify) { - if (x == ox && y == oy) - return; - if (!IS_SET(MODE_MOUSEMOTION) && !IS_SET(MODE_MOUSEMANY)) - return; - /* MOUSE_MOTION: no reporting if no button is pressed */ - if (IS_SET(MODE_MOUSEMOTION) && oldbutton == 3) - return; - - button = oldbutton + 32; - ox = x; - oy = y; - } else { - if (!IS_SET(MODE_MOUSESGR) && e->xbutton.type == ButtonRelease) { - button = 3; - } else { - button -= Button1; - if (button >= 3) - button += 64 - 3; - } - if (e->xbutton.type == ButtonPress) { - oldbutton = button; - ox = x; - oy = y; - } else if (e->xbutton.type == ButtonRelease) { - oldbutton = 3; - /* MODE_MOUSEX10: no button release reporting */ - if (IS_SET(MODE_MOUSEX10)) - return; - if (button == 64 || button == 65) - return; - } - } - - if (!IS_SET(MODE_MOUSEX10)) { - button += ((state & ShiftMask ) ? 4 : 0) - + ((state & Mod4Mask ) ? 8 : 0) - + ((state & ControlMask) ? 16 : 0); - } - - if (IS_SET(MODE_MOUSESGR)) { - len = snprintf(buf, sizeof(buf), "\033[<%d;%d;%d%c", - button, x+1, y+1, - e->xbutton.type == ButtonRelease ? 'm' : 'M'); - } else if (x < 223 && y < 223) { - len = snprintf(buf, sizeof(buf), "\033[M%c%c%c", - 32+button, 32+x+1, 32+y+1); - } else { - return; - } - - ttywrite(buf, len, 0); -} - -int -mouseaction(XEvent *e, uint release) -{ - MouseShortcut *ms; - - for (ms = mshortcuts; ms < mshortcuts + LEN(mshortcuts); ms++) { - if (ms->release == release && - ms->button == e->xbutton.button && - (match(ms->mod, e->xbutton.state) || /* exact or forced */ - match(ms->mod, e->xbutton.state & ~forcemousemod))) { - ms->func(&(ms->arg)); - return 1; - } - } - - return 0; -} - -void -bpress(XEvent *e) -{ - struct timespec now; - int snap; - - if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forcemousemod)) { - mousereport(e); - return; - } - - if (mouseaction(e, 0)) - return; - - if (e->xbutton.button == Button1) { - /* - * If the user clicks below predefined timeouts specific - * snapping behaviour is exposed. - */ - clock_gettime(CLOCK_MONOTONIC, &now); - if (TIMEDIFF(now, xsel.tclick2) <= tripleclicktimeout) { - snap = SNAP_LINE; - } else if (TIMEDIFF(now, xsel.tclick1) <= doubleclicktimeout) { - snap = SNAP_WORD; - } else { - snap = 0; - } - xsel.tclick2 = xsel.tclick1; - xsel.tclick1 = now; - - selstart(evcol(e), evrow(e), snap); - } -} - -void -propnotify(XEvent *e) -{ - XPropertyEvent *xpev; - Atom clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - - xpev = &e->xproperty; - if (xpev->state == PropertyNewValue && - (xpev->atom == XA_PRIMARY || - xpev->atom == clipboard)) { - selnotify(e); - } -} - -void -selnotify(XEvent *e) -{ - ulong nitems, ofs, rem; - int format; - uchar *data, *last, *repl; - Atom type, incratom, property = None; - - incratom = XInternAtom(xw.dpy, "INCR", 0); - - ofs = 0; - if (e->type == SelectionNotify) - property = e->xselection.property; - else if (e->type == PropertyNotify) - property = e->xproperty.atom; - - if (property == None) - return; - - do { - if (XGetWindowProperty(xw.dpy, xw.win, property, ofs, - BUFSIZ/4, False, AnyPropertyType, - &type, &format, &nitems, &rem, - &data)) { - fprintf(stderr, "Clipboard allocation failed\n"); - return; - } - - if (e->type == PropertyNotify && nitems == 0 && rem == 0) { - /* - * If there is some PropertyNotify with no data, then - * this is the signal of the selection owner that all - * data has been transferred. We won't need to receive - * PropertyNotify events anymore. - */ - MODBIT(xw.attrs.event_mask, 0, PropertyChangeMask); - XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask, - &xw.attrs); - } - - if (type == incratom) { - /* - * Activate the PropertyNotify events so we receive - * when the selection owner does send us the next - * chunk of data. - */ - MODBIT(xw.attrs.event_mask, 1, PropertyChangeMask); - XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask, - &xw.attrs); - - /* - * Deleting the property is the transfer start signal. - */ - XDeleteProperty(xw.dpy, xw.win, (int)property); - continue; - } - - /* - * As seen in getsel: - * Line endings are inconsistent in the terminal and GUI world - * copy and pasting. When receiving some selection data, - * replace all '\n' with '\r'. - * FIXME: Fix the computer world. - */ - repl = data; - last = data + nitems * format / 8; - while ((repl = memchr(repl, '\n', last - repl))) { - *repl++ = '\r'; - } - - if (IS_SET(MODE_BRCKTPASTE) && ofs == 0) - ttywrite("\033[200~", 6, 0); - ttywrite((char *)data, nitems * format / 8, 1); - if (IS_SET(MODE_BRCKTPASTE) && rem == 0) - ttywrite("\033[201~", 6, 0); - XFree(data); - /* number of 32-bit chunks returned */ - ofs += nitems * format / 32; - } while (rem > 0); - - /* - * Deleting the property again tells the selection owner to send the - * next data chunk in the property. - */ - XDeleteProperty(xw.dpy, xw.win, (int)property); -} - -void -xclipcopy(void) -{ - clipcopy(NULL); -} - -void -selclear_(XEvent *e) -{ - selclear(); -} - -void -selrequest(XEvent *e) -{ - XSelectionRequestEvent *xsre; - XSelectionEvent xev; - Atom xa_targets, string, clipboard; - char *seltext; - - xsre = (XSelectionRequestEvent *) e; - xev.type = SelectionNotify; - xev.requestor = xsre->requestor; - xev.selection = xsre->selection; - xev.target = xsre->target; - xev.time = xsre->time; - if (xsre->property == None) - xsre->property = xsre->target; - - /* reject */ - xev.property = None; - - xa_targets = XInternAtom(xw.dpy, "TARGETS", 0); - if (xsre->target == xa_targets) { - /* respond with the supported type */ - string = xsel.xtarget; - XChangeProperty(xsre->display, xsre->requestor, xsre->property, - XA_ATOM, 32, PropModeReplace, - (uchar *) &string, 1); - xev.property = xsre->property; - } else if (xsre->target == xsel.xtarget || xsre->target == XA_STRING) { - /* - * xith XA_STRING non ascii characters may be incorrect in the - * requestor. It is not our problem, use utf8. - */ - clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - if (xsre->selection == XA_PRIMARY) { - seltext = xsel.primary; - } else if (xsre->selection == clipboard) { - seltext = xsel.clipboard; - } else { - fprintf(stderr, - "Unhandled clipboard selection 0x%lx\n", - xsre->selection); - return; - } - if (seltext != NULL) { - XChangeProperty(xsre->display, xsre->requestor, - xsre->property, xsre->target, - 8, PropModeReplace, - (uchar *)seltext, strlen(seltext)); - xev.property = xsre->property; - } - } - - /* all done, send a notification to the listener */ - if (!XSendEvent(xsre->display, xsre->requestor, 1, 0, (XEvent *) &xev)) - fprintf(stderr, "Error sending SelectionNotify event\n"); -} - -void -setsel(char *str, Time t) -{ - if (!str) - return; - - free(xsel.primary); - xsel.primary = str; - - XSetSelectionOwner(xw.dpy, XA_PRIMARY, xw.win, t); - if (XGetSelectionOwner(xw.dpy, XA_PRIMARY) != xw.win) - selclear(); -} - -void -xsetsel(char *str) -{ - setsel(str, CurrentTime); -} - -void -brelease(XEvent *e) -{ - if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forcemousemod)) { - mousereport(e); - return; - } - - if (mouseaction(e, 1)) - return; - if (e->xbutton.button == Button1) - mousesel(e, 1); -} - -void -bmotion(XEvent *e) -{ - if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forcemousemod)) { - mousereport(e); - return; - } - - mousesel(e, 0); -} - -void -cresize(int width, int height) -{ - int col, row; - - if (width != 0) - win.w = width; - if (height != 0) - win.h = height; - - col = (win.w - 2 * borderpx) / win.cw; - row = (win.h - 2 * borderpx) / win.ch; - col = MAX(1, col); - row = MAX(1, row); - - win.hborderpx = (win.w - col * win.cw) / 2; - win.vborderpx = (win.h - row * win.ch) / 2; - - tresize(col, row); - xresize(col, row); - ttyresize(win.tw, win.th); -} - -void -xresize(int col, int row) -{ - win.tw = col * win.cw; - win.th = row * win.ch; - - XFreePixmap(xw.dpy, xw.buf); - xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, - DefaultDepth(xw.dpy, xw.scr)); - XftDrawChange(xw.draw, xw.buf); - xclear(0, 0, win.w, win.h); - - /* resize to new width */ - xw.specbuf = xrealloc(xw.specbuf, col * sizeof(GlyphFontSpec)); -} - -ushort -sixd_to_16bit(int x) -{ - return x == 0 ? 0 : 0x3737 + 0x2828 * x; -} - -int -xloadcolor(int i, const char *name, Color *ncolor) -{ - XRenderColor color = { .alpha = 0xffff }; - - if (!name) { - if (BETWEEN(i, 16, 255)) { /* 256 color */ - if (i < 6*6*6+16) { /* same colors as xterm */ - color.red = sixd_to_16bit( ((i-16)/36)%6 ); - color.green = sixd_to_16bit( ((i-16)/6) %6 ); - color.blue = sixd_to_16bit( ((i-16)/1) %6 ); - } else { /* greyscale */ - color.red = 0x0808 + 0x0a0a * (i - (6*6*6+16)); - color.green = color.blue = color.red; - } - return XftColorAllocValue(xw.dpy, xw.vis, - xw.cmap, &color, ncolor); - } else - name = colorname[i]; - } - - return XftColorAllocName(xw.dpy, xw.vis, xw.cmap, name, ncolor); -} - -void -xloadcols(void) -{ - int i; - static int loaded; - Color *cp; - - if (loaded) { - for (cp = dc.col; cp < &dc.col[dc.collen]; ++cp) - XftColorFree(xw.dpy, xw.vis, xw.cmap, cp); - } else { - dc.collen = MAX(LEN(colorname), 256); - dc.col = xmalloc(dc.collen * sizeof(Color)); - } - - for (i = 0; i < dc.collen; i++) - if (!xloadcolor(i, NULL, &dc.col[i])) { - if (colorname[i]) - die("could not allocate color '%s'\n", colorname[i]); - else - die("could not allocate color %d\n", i); - } - loaded = 1; -} - -int -xsetcolorname(int x, const char *name) -{ - Color ncolor; - - if (!BETWEEN(x, 0, dc.collen)) - return 1; - - if (!xloadcolor(x, name, &ncolor)) - return 1; - - XftColorFree(xw.dpy, xw.vis, xw.cmap, &dc.col[x]); - dc.col[x] = ncolor; - - return 0; -} - -/* - * Absolute coordinates. - */ -void -xclear(int x1, int y1, int x2, int y2) -{ - XftDrawRect(xw.draw, - &dc.col[IS_SET(MODE_REVERSE)? defaultfg : defaultbg], - x1, y1, x2-x1, y2-y1); -} - -void -xhints(void) -{ - XClassHint class = {opt_name ? opt_name : termname, - opt_class ? opt_class : termname}; - XWMHints wm = {.flags = InputHint, .input = 1}; - XSizeHints *sizeh; - - sizeh = XAllocSizeHints(); - - sizeh->flags = PSize | PResizeInc | PBaseSize | PMinSize; - sizeh->height = win.h; - sizeh->width = win.w; - sizeh->height_inc = 1; - sizeh->width_inc = 1; - sizeh->base_height = 2 * borderpx; - sizeh->base_width = 2 * borderpx; - sizeh->min_height = win.ch + 2 * borderpx; - sizeh->min_width = win.cw + 2 * borderpx; - if (xw.isfixed) { - sizeh->flags |= PMaxSize; - sizeh->min_width = sizeh->max_width = win.w; - sizeh->min_height = sizeh->max_height = win.h; - } - if (xw.gm & (XValue|YValue)) { - sizeh->flags |= USPosition | PWinGravity; - sizeh->x = xw.l; - sizeh->y = xw.t; - sizeh->win_gravity = xgeommasktogravity(xw.gm); - } - - XSetWMProperties(xw.dpy, xw.win, NULL, NULL, NULL, 0, sizeh, &wm, - &class); - XFree(sizeh); -} - -int -xgeommasktogravity(int mask) -{ - switch (mask & (XNegative|YNegative)) { - case 0: - return NorthWestGravity; - case XNegative: - return NorthEastGravity; - case YNegative: - return SouthWestGravity; - } - - return SouthEastGravity; -} - -int -xloadfont(Font *f, FcPattern *pattern) -{ - FcPattern *configured; - FcPattern *match; - FcResult result; - XGlyphInfo extents; - int wantattr, haveattr; - - /* - * Manually configure instead of calling XftMatchFont - * so that we can use the configured pattern for - * "missing glyph" lookups. - */ - configured = FcPatternDuplicate(pattern); - if (!configured) - return 1; - - FcConfigSubstitute(NULL, configured, FcMatchPattern); - XftDefaultSubstitute(xw.dpy, xw.scr, configured); - - match = FcFontMatch(NULL, configured, &result); - if (!match) { - FcPatternDestroy(configured); - return 1; - } - - if (!(f->match = XftFontOpenPattern(xw.dpy, match))) { - FcPatternDestroy(configured); - FcPatternDestroy(match); - return 1; - } - - if ((XftPatternGetInteger(pattern, "slant", 0, &wantattr) == - XftResultMatch)) { - /* - * Check if xft was unable to find a font with the appropriate - * slant but gave us one anyway. Try to mitigate. - */ - if ((XftPatternGetInteger(f->match->pattern, "slant", 0, - &haveattr) != XftResultMatch) || haveattr < wantattr) { - f->badslant = 1; - fputs("font slant does not match\n", stderr); - } - } - - if ((XftPatternGetInteger(pattern, "weight", 0, &wantattr) == - XftResultMatch)) { - if ((XftPatternGetInteger(f->match->pattern, "weight", 0, - &haveattr) != XftResultMatch) || haveattr != wantattr) { - f->badweight = 1; - fputs("font weight does not match\n", stderr); - } - } - - XftTextExtentsUtf8(xw.dpy, f->match, - (const FcChar8 *) ascii_printable, - strlen(ascii_printable), &extents); - - f->set = NULL; - f->pattern = configured; - - f->ascent = f->match->ascent; - f->descent = f->match->descent; - f->lbearing = 0; - f->rbearing = f->match->max_advance_width; - - f->height = f->ascent + f->descent; - f->width = DIVCEIL(extents.xOff, strlen(ascii_printable)); - - return 0; -} - -void -xloadfonts(char *fontstr, double fontsize) -{ - FcPattern *pattern; - double fontval; - - if (fontstr[0] == '-') - pattern = XftXlfdParse(fontstr, False, False); - else - pattern = FcNameParse((FcChar8 *)fontstr); - - if (!pattern) - die("can't open font %s\n", fontstr); - - if (fontsize > 1) { - FcPatternDel(pattern, FC_PIXEL_SIZE); - FcPatternDel(pattern, FC_SIZE); - FcPatternAddDouble(pattern, FC_PIXEL_SIZE, (double)fontsize); - usedfontsize = fontsize; - } else { - if (FcPatternGetDouble(pattern, FC_PIXEL_SIZE, 0, &fontval) == - FcResultMatch) { - usedfontsize = fontval; - } else if (FcPatternGetDouble(pattern, FC_SIZE, 0, &fontval) == - FcResultMatch) { - usedfontsize = -1; - } else { - /* - * Default font size is 12, if none given. This is to - * have a known usedfontsize value. - */ - FcPatternAddDouble(pattern, FC_PIXEL_SIZE, 12); - usedfontsize = 12; - } - defaultfontsize = usedfontsize; - } - - if (xloadfont(&dc.font, pattern)) - die("can't open font %s\n", fontstr); - - if (usedfontsize < 0) { - FcPatternGetDouble(dc.font.match->pattern, - FC_PIXEL_SIZE, 0, &fontval); - usedfontsize = fontval; - if (fontsize == 0) - defaultfontsize = fontval; - } - - /* Setting character width and height. */ - win.cw = ceilf(dc.font.width * cwscale); - win.ch = ceilf(dc.font.height * chscale); - - FcPatternDel(pattern, FC_SLANT); - FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ITALIC); - if (xloadfont(&dc.ifont, pattern)) - die("can't open font %s\n", fontstr); - - FcPatternDel(pattern, FC_WEIGHT); - FcPatternAddInteger(pattern, FC_WEIGHT, FC_WEIGHT_BOLD); - if (xloadfont(&dc.ibfont, pattern)) - die("can't open font %s\n", fontstr); - - FcPatternDel(pattern, FC_SLANT); - FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ROMAN); - if (xloadfont(&dc.bfont, pattern)) - die("can't open font %s\n", fontstr); - - FcPatternDestroy(pattern); -} - -void -xunloadfont(Font *f) -{ - XftFontClose(xw.dpy, f->match); - FcPatternDestroy(f->pattern); - if (f->set) - FcFontSetDestroy(f->set); -} - -void -xunloadfonts(void) -{ - /* Clear Harfbuzz font cache. */ - hbunloadfonts(); - - /* Free the loaded fonts in the font cache. */ - while (frclen > 0) - XftFontClose(xw.dpy, frc[--frclen].font); - - xunloadfont(&dc.font); - xunloadfont(&dc.bfont); - xunloadfont(&dc.ifont); - xunloadfont(&dc.ibfont); -} - -int -ximopen(Display *dpy) -{ - XIMCallback imdestroy = { .client_data = NULL, .callback = ximdestroy }; - XICCallback icdestroy = { .client_data = NULL, .callback = xicdestroy }; - - xw.ime.xim = XOpenIM(xw.dpy, NULL, NULL, NULL); - if (xw.ime.xim == NULL) - return 0; - - if (XSetIMValues(xw.ime.xim, XNDestroyCallback, &imdestroy, NULL)) - fprintf(stderr, "XSetIMValues: " - "Could not set XNDestroyCallback.\n"); - - xw.ime.spotlist = XVaCreateNestedList(0, XNSpotLocation, &xw.ime.spot, - NULL); - - if (xw.ime.xic == NULL) { - xw.ime.xic = XCreateIC(xw.ime.xim, XNInputStyle, - XIMPreeditNothing | XIMStatusNothing, - XNClientWindow, xw.win, - XNFocusWindow, xw.win, - XNDestroyCallback, &icdestroy, - NULL); - } - if (xw.ime.xic == NULL) - fprintf(stderr, "XCreateIC: Could not create input context.\n"); - - return 1; -} - -void -ximinstantiate(Display *dpy, XPointer client, XPointer call) -{ - if (ximopen(dpy)) - XUnregisterIMInstantiateCallback(xw.dpy, NULL, NULL, NULL, - ximinstantiate, NULL); -} - -void -ximdestroy(XIM xim, XPointer client, XPointer call) -{ - xw.ime.xim = NULL; - XRegisterIMInstantiateCallback(xw.dpy, NULL, NULL, NULL, - ximinstantiate, NULL); - XFree(xw.ime.spotlist); -} - -int -xicdestroy(XIC xim, XPointer client, XPointer call) -{ - xw.ime.xic = NULL; - return 1; -} - -void -xinit(int cols, int rows) -{ - XGCValues gcvalues; - Cursor cursor; - Window parent; - pid_t thispid = getpid(); - XColor xmousefg, xmousebg; - - if (!(xw.dpy = XOpenDisplay(NULL))) - die("can't open display\n"); - xw.scr = XDefaultScreen(xw.dpy); - xw.vis = XDefaultVisual(xw.dpy, xw.scr); - - /* font */ - if (!FcInit()) - die("could not init fontconfig.\n"); - - usedfont = (opt_font == NULL)? font : opt_font; - xloadfonts(usedfont, 0); - - /* colors */ - xw.cmap = XDefaultColormap(xw.dpy, xw.scr); - xloadcols(); - - /* adjust fixed window geometry */ - win.w = 2 * win.hborderpx + cols * win.cw; - win.h = 2 * win.vborderpx + rows * win.ch; - if (xw.gm & XNegative) - xw.l += DisplayWidth(xw.dpy, xw.scr) - win.w - 2; - if (xw.gm & YNegative) - xw.t += DisplayHeight(xw.dpy, xw.scr) - win.h - 2; - - /* Events */ - xw.attrs.background_pixel = dc.col[defaultbg].pixel; - xw.attrs.border_pixel = dc.col[defaultbg].pixel; - xw.attrs.bit_gravity = NorthWestGravity; - xw.attrs.event_mask = FocusChangeMask | KeyPressMask | KeyReleaseMask - | ExposureMask | VisibilityChangeMask | StructureNotifyMask - | ButtonMotionMask | ButtonPressMask | ButtonReleaseMask; - xw.attrs.colormap = xw.cmap; - - if (!(opt_embed && (parent = strtol(opt_embed, NULL, 0)))) - parent = XRootWindow(xw.dpy, xw.scr); - xw.win = XCreateWindow(xw.dpy, parent, xw.l, xw.t, - win.w, win.h, 0, XDefaultDepth(xw.dpy, xw.scr), InputOutput, - xw.vis, CWBackPixel | CWBorderPixel | CWBitGravity - | CWEventMask | CWColormap, &xw.attrs); - - memset(&gcvalues, 0, sizeof(gcvalues)); - gcvalues.graphics_exposures = False; - dc.gc = XCreateGC(xw.dpy, parent, GCGraphicsExposures, - &gcvalues); - xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, - DefaultDepth(xw.dpy, xw.scr)); - XSetForeground(xw.dpy, dc.gc, dc.col[defaultbg].pixel); - XFillRectangle(xw.dpy, xw.buf, dc.gc, 0, 0, win.w, win.h); - - /* font spec buffer */ - xw.specbuf = xmalloc(cols * sizeof(GlyphFontSpec)); - - /* Xft rendering context */ - xw.draw = XftDrawCreate(xw.dpy, xw.buf, xw.vis, xw.cmap); - - /* input methods */ - if (!ximopen(xw.dpy)) { - XRegisterIMInstantiateCallback(xw.dpy, NULL, NULL, NULL, - ximinstantiate, NULL); - } - - /* white cursor, black outline */ - cursor = XCreateFontCursor(xw.dpy, mouseshape); - XDefineCursor(xw.dpy, xw.win, cursor); - - if (XParseColor(xw.dpy, xw.cmap, colorname[mousefg], &xmousefg) == 0) { - xmousefg.red = 0xffff; - xmousefg.green = 0xffff; - xmousefg.blue = 0xffff; - } - - if (XParseColor(xw.dpy, xw.cmap, colorname[mousebg], &xmousebg) == 0) { - xmousebg.red = 0x0000; - xmousebg.green = 0x0000; - xmousebg.blue = 0x0000; - } - - XRecolorCursor(xw.dpy, cursor, &xmousefg, &xmousebg); - - xw.xembed = XInternAtom(xw.dpy, "_XEMBED", False); - xw.wmdeletewin = XInternAtom(xw.dpy, "WM_DELETE_WINDOW", False); - xw.netwmname = XInternAtom(xw.dpy, "_NET_WM_NAME", False); - XSetWMProtocols(xw.dpy, xw.win, &xw.wmdeletewin, 1); - - xw.netwmpid = XInternAtom(xw.dpy, "_NET_WM_PID", False); - XChangeProperty(xw.dpy, xw.win, xw.netwmpid, XA_CARDINAL, 32, - PropModeReplace, (uchar *)&thispid, 1); - - win.mode = MODE_NUMLOCK; - resettitle(); - xhints(); - XMapWindow(xw.dpy, xw.win); - XSync(xw.dpy, False); - - clock_gettime(CLOCK_MONOTONIC, &xsel.tclick1); - clock_gettime(CLOCK_MONOTONIC, &xsel.tclick2); - xsel.primary = NULL; - xsel.clipboard = NULL; - xsel.xtarget = XInternAtom(xw.dpy, "UTF8_STRING", 0); - if (xsel.xtarget == None) - xsel.xtarget = XA_STRING; -} - -int -xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x, int y) -{ - float winx = win.hborderpx + x * win.cw, winy = win.vborderpx + y * win.ch, xp, yp; - ushort mode, prevmode = USHRT_MAX; - Font *font = &dc.font; - int frcflags = FRC_NORMAL; - float runewidth = win.cw; - Rune rune; - FT_UInt glyphidx; - FcResult fcres; - FcPattern *fcpattern, *fontpattern; - FcFontSet *fcsets[] = { NULL }; - FcCharSet *fccharset; - int i, f, numspecs = 0; - - for (i = 0, xp = winx, yp = winy + font->ascent; i < len; ++i) { - /* Fetch rune and mode for current glyph. */ - rune = glyphs[i].u; - mode = glyphs[i].mode; - - /* Skip dummy wide-character spacing. */ - if (mode & ATTR_WDUMMY) - continue; - - /* Determine font for glyph if different from previous glyph. */ - if (prevmode != mode) { - prevmode = mode; - font = &dc.font; - frcflags = FRC_NORMAL; - runewidth = win.cw * ((mode & ATTR_WIDE) ? 2.0f : 1.0f); - if ((mode & ATTR_ITALIC) && (mode & ATTR_BOLD)) { - font = &dc.ibfont; - frcflags = FRC_ITALICBOLD; - } else if (mode & ATTR_ITALIC) { - font = &dc.ifont; - frcflags = FRC_ITALIC; - } else if (mode & ATTR_BOLD) { - font = &dc.bfont; - frcflags = FRC_BOLD; - } - yp = winy + font->ascent; - } - - /* Lookup character index with default font. */ - glyphidx = XftCharIndex(xw.dpy, font->match, rune); - if (glyphidx) { - specs[numspecs].font = font->match; - specs[numspecs].glyph = glyphidx; - specs[numspecs].x = (short)xp; - specs[numspecs].y = (short)yp; - xp += runewidth; - numspecs++; - continue; - } - - /* Fallback on font cache, search the font cache for match. */ - for (f = 0; f < frclen; f++) { - glyphidx = XftCharIndex(xw.dpy, frc[f].font, rune); - /* Everything correct. */ - if (glyphidx && frc[f].flags == frcflags) - break; - /* We got a default font for a not found glyph. */ - if (!glyphidx && frc[f].flags == frcflags - && frc[f].unicodep == rune) { - break; - } - } - - /* Nothing was found. Use fontconfig to find matching font. */ - if (f >= frclen) { - if (!font->set) - font->set = FcFontSort(0, font->pattern, - 1, 0, &fcres); - fcsets[0] = font->set; - - /* - * Nothing was found in the cache. Now use - * some dozen of Fontconfig calls to get the - * font for one single character. - * - * Xft and fontconfig are design failures. - */ - fcpattern = FcPatternDuplicate(font->pattern); - fccharset = FcCharSetCreate(); - - FcCharSetAddChar(fccharset, rune); - FcPatternAddCharSet(fcpattern, FC_CHARSET, - fccharset); - FcPatternAddBool(fcpattern, FC_SCALABLE, 1); - - FcConfigSubstitute(0, fcpattern, - FcMatchPattern); - FcDefaultSubstitute(fcpattern); - - fontpattern = FcFontSetMatch(0, fcsets, 1, - fcpattern, &fcres); - - /* Allocate memory for the new cache entry. */ - if (frclen >= frccap) { - frccap += 16; - frc = xrealloc(frc, frccap * sizeof(Fontcache)); - } - - frc[frclen].font = XftFontOpenPattern(xw.dpy, - fontpattern); - if (!frc[frclen].font) - die("XftFontOpenPattern failed seeking fallback font: %s\n", - strerror(errno)); - frc[frclen].flags = frcflags; - frc[frclen].unicodep = rune; - - glyphidx = XftCharIndex(xw.dpy, frc[frclen].font, rune); - - f = frclen; - frclen++; - - FcPatternDestroy(fcpattern); - FcCharSetDestroy(fccharset); - } - - specs[numspecs].font = frc[f].font; - specs[numspecs].glyph = glyphidx; - specs[numspecs].x = (short)xp; - specs[numspecs].y = (short)yp; - xp += runewidth; - numspecs++; - } - - /* Harfbuzz transformation for ligatures. */ - hbtransform(specs, glyphs, len, x, y); - - return numspecs; -} - -void -xdrawglyphfontspecs(const XftGlyphFontSpec *specs, Glyph base, int len, int x, int y) -{ - int charlen = len * ((base.mode & ATTR_WIDE) ? 2 : 1); - int winx = win.hborderpx + x * win.cw, winy = win.vborderpx + y * win.ch, - width = charlen * win.cw; - Color *fg, *bg, *temp, revfg, revbg, truefg, truebg; - XRenderColor colfg, colbg; - XRectangle r; - - /* Fallback on color display for attributes not supported by the font */ - if (base.mode & ATTR_ITALIC && base.mode & ATTR_BOLD) { - if (dc.ibfont.badslant || dc.ibfont.badweight) - base.fg = defaultattr; - } else if ((base.mode & ATTR_ITALIC && dc.ifont.badslant) || - (base.mode & ATTR_BOLD && dc.bfont.badweight)) { - base.fg = defaultattr; - } - - if (IS_TRUECOL(base.fg)) { - colfg.alpha = 0xffff; - colfg.red = TRUERED(base.fg); - colfg.green = TRUEGREEN(base.fg); - colfg.blue = TRUEBLUE(base.fg); - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, &truefg); - fg = &truefg; - } else { - fg = &dc.col[base.fg]; - } - - if (IS_TRUECOL(base.bg)) { - colbg.alpha = 0xffff; - colbg.green = TRUEGREEN(base.bg); - colbg.red = TRUERED(base.bg); - colbg.blue = TRUEBLUE(base.bg); - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colbg, &truebg); - bg = &truebg; - } else { - bg = &dc.col[base.bg]; - } - - /* Change basic system colors [0-7] to bright system colors [8-15] */ - if ((base.mode & ATTR_BOLD_FAINT) == ATTR_BOLD && BETWEEN(base.fg, 0, 7)) - fg = &dc.col[base.fg + 8]; - - if (IS_SET(MODE_REVERSE)) { - if (fg == &dc.col[defaultfg]) { - fg = &dc.col[defaultbg]; - } else { - colfg.red = ~fg->color.red; - colfg.green = ~fg->color.green; - colfg.blue = ~fg->color.blue; - colfg.alpha = fg->color.alpha; - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, - &revfg); - fg = &revfg; - } - - if (bg == &dc.col[defaultbg]) { - bg = &dc.col[defaultfg]; - } else { - colbg.red = ~bg->color.red; - colbg.green = ~bg->color.green; - colbg.blue = ~bg->color.blue; - colbg.alpha = bg->color.alpha; - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colbg, - &revbg); - bg = &revbg; - } - } - - if ((base.mode & ATTR_BOLD_FAINT) == ATTR_FAINT) { - colfg.red = fg->color.red / 2; - colfg.green = fg->color.green / 2; - colfg.blue = fg->color.blue / 2; - colfg.alpha = fg->color.alpha; - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, &revfg); - fg = &revfg; - } - - if (base.mode & ATTR_REVERSE) { - temp = fg; - fg = bg; - bg = temp; - } - - if (base.mode & ATTR_BLINK && win.mode & MODE_BLINK) - fg = bg; - - if (base.mode & ATTR_INVISIBLE) - fg = bg; - - /* Intelligent cleaning up of the borders. */ - if (x == 0) { - xclear(0, (y == 0)? 0 : winy, win.vborderpx, - winy + win.ch + - ((winy + win.ch >= win.vborderpx + win.th)? win.h : 0)); - } - if (winx + width >= win.hborderpx + win.tw) { - xclear(winx + width, (y == 0)? 0 : winy, win.w, - ((winy + win.ch >= win.vborderpx + win.th)? win.h : (winy + win.ch))); - } - if (y == 0) - xclear(winx, 0, winx + width, win.hborderpx); - if (winy + win.ch >= win.vborderpx + win.th) - xclear(winx, winy + win.ch, winx + width, win.h); - - /* Clean up the region we want to draw to. */ - XftDrawRect(xw.draw, bg, winx, winy, width, win.ch); - - /* Set the clip region because Xft is sometimes dirty. */ - r.x = 0; - r.y = 0; - r.height = win.ch; - r.width = width; - XftDrawSetClipRectangles(xw.draw, winx, winy, &r, 1); - - /* Render the glyphs. */ - XftDrawGlyphFontSpec(xw.draw, fg, specs, len); - - /* Render underline and strikethrough. */ - if (base.mode & ATTR_UNDERLINE) { - XftDrawRect(xw.draw, fg, winx, winy + dc.font.ascent + 1, - width, 1); - } - - if (base.mode & ATTR_STRUCK) { - XftDrawRect(xw.draw, fg, winx, winy + 2 * dc.font.ascent / 3, - width, 1); - } - - /* Reset clip to none. */ - XftDrawSetClip(xw.draw, 0); -} - -void -xdrawglyph(Glyph g, int x, int y) -{ - int numspecs; - XftGlyphFontSpec spec; - - numspecs = xmakeglyphfontspecs(&spec, &g, 1, x, y); - xdrawglyphfontspecs(&spec, g, numspecs, x, y); -} - -void -xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og, Line line, int len) -{ - Color drawcol; - - /* remove the old cursor */ - if (selected(ox, oy)) - og.mode ^= ATTR_REVERSE; - - /* Redraw the line where cursor was previously. - * It will restore the ligatures broken by the cursor. */ - xdrawline(line, 0, oy, len); - - if (IS_SET(MODE_HIDE)) - return; - - /* - * Select the right color for the right mode. - */ - g.mode &= ATTR_BOLD|ATTR_ITALIC|ATTR_UNDERLINE|ATTR_STRUCK|ATTR_WIDE; - - if (IS_SET(MODE_REVERSE)) { - g.mode |= ATTR_REVERSE; - g.bg = defaultfg; - if (selected(cx, cy)) { - drawcol = dc.col[defaultcs]; - g.fg = defaultrcs; - } else { - drawcol = dc.col[defaultrcs]; - g.fg = defaultcs; - } - } else { - if (selected(cx, cy)) { - g.fg = defaultfg; - g.bg = defaultrcs; - } else { - g.fg = defaultbg; - g.bg = defaultcs; - } - drawcol = dc.col[g.bg]; - } - - /* draw the new one */ - if (IS_SET(MODE_FOCUSED)) { - switch (win.cursor) { - case 7: /* st extension: snowman (U+2603) */ - g.u = 0x2603; - case 0: /* Blinking Block */ - case 1: /* Blinking Block (Default) */ - case 2: /* Steady Block */ - xdrawglyph(g, cx, cy); - break; - case 3: /* Blinking Underline */ - case 4: /* Steady Underline */ - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + (cy + 1) * win.ch - \ - cursorthickness, - win.cw, cursorthickness); - break; - case 5: /* Blinking bar */ - case 6: /* Steady bar */ - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + cy * win.ch, - cursorthickness, win.ch); - break; - } - } else { - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + cy * win.ch, - win.cw - 1, 1); - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + cy * win.ch, - 1, win.ch - 1); - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + (cx + 1) * win.cw - 1, - win.vborderpx + cy * win.ch, - 1, win.ch - 1); - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + (cy + 1) * win.ch - 1, - win.cw, 1); - } -} - -void -xsetenv(void) -{ - char buf[sizeof(long) * 8 + 1]; - - snprintf(buf, sizeof(buf), "%lu", xw.win); - setenv("WINDOWID", buf, 1); -} - -void -xsettitle(char *p) -{ - XTextProperty prop; - DEFAULT(p, opt_title); - - Xutf8TextListToTextProperty(xw.dpy, &p, 1, XUTF8StringStyle, - &prop); - XSetWMName(xw.dpy, xw.win, &prop); - XSetTextProperty(xw.dpy, xw.win, &prop, xw.netwmname); - XFree(prop.value); -} - -int -xstartdraw(void) -{ - return IS_SET(MODE_VISIBLE); -} - -void -xdrawline(Line line, int x1, int y1, int x2) -{ - int i, x, ox, numspecs; - Glyph base, new; - XftGlyphFontSpec *specs = xw.specbuf; - - numspecs = xmakeglyphfontspecs(specs, &line[x1], x2 - x1, x1, y1); - i = ox = 0; - for (x = x1; x < x2 && i < numspecs; x++) { - new = line[x]; - if (new.mode == ATTR_WDUMMY) - continue; - if (selected(x, y1)) - new.mode ^= ATTR_REVERSE; - if (i > 0 && ATTRCMP(base, new)) { - xdrawglyphfontspecs(specs, base, i, ox, y1); - specs += i; - numspecs -= i; - i = 0; - } - if (i == 0) { - ox = x; - base = new; - } - i++; - } - if (i > 0) - xdrawglyphfontspecs(specs, base, i, ox, y1); -} - -void -xfinishdraw(void) -{ - XCopyArea(xw.dpy, xw.buf, xw.win, dc.gc, 0, 0, win.w, - win.h, 0, 0); - XSetForeground(xw.dpy, dc.gc, - dc.col[IS_SET(MODE_REVERSE)? - defaultfg : defaultbg].pixel); -} - -void -xximspot(int x, int y) -{ - if (xw.ime.xic == NULL) - return; - - xw.ime.spot.x = borderpx + x * win.cw; - xw.ime.spot.y = borderpx + (y + 1) * win.ch; - - XSetICValues(xw.ime.xic, XNPreeditAttributes, xw.ime.spotlist, NULL); -} - -void -expose(XEvent *ev) -{ - redraw(); -} - -void -visibility(XEvent *ev) -{ - XVisibilityEvent *e = &ev->xvisibility; - - MODBIT(win.mode, e->state != VisibilityFullyObscured, MODE_VISIBLE); -} - -void -unmap(XEvent *ev) -{ - win.mode &= ~MODE_VISIBLE; -} - -void -xsetpointermotion(int set) -{ - MODBIT(xw.attrs.event_mask, set, PointerMotionMask); - XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask, &xw.attrs); -} - -void -xsetmode(int set, unsigned int flags) -{ - int mode = win.mode; - MODBIT(win.mode, set, flags); - if ((win.mode & MODE_REVERSE) != (mode & MODE_REVERSE)) - redraw(); -} - -int -xsetcursor(int cursor) -{ - DEFAULT(cursor, 1); - if (!BETWEEN(cursor, 0, 6)) - return 1; - win.cursor = cursor; - return 0; -} - -void -xseturgency(int add) -{ - XWMHints *h = XGetWMHints(xw.dpy, xw.win); - - MODBIT(h->flags, add, XUrgencyHint); - XSetWMHints(xw.dpy, xw.win, h); - XFree(h); -} - -void -xbell(void) -{ - if (!(IS_SET(MODE_FOCUSED))) - xseturgency(1); - if (bellvolume) - XkbBell(xw.dpy, xw.win, bellvolume, (Atom)NULL); -} - -void -focus(XEvent *ev) -{ - XFocusChangeEvent *e = &ev->xfocus; - - if (e->mode == NotifyGrab) - return; - - if (ev->type == FocusIn) { - if (xw.ime.xic) - XSetICFocus(xw.ime.xic); - win.mode |= MODE_FOCUSED; - xseturgency(0); - if (IS_SET(MODE_FOCUS)) - ttywrite("\033[I", 3, 0); - } else { - if (xw.ime.xic) - XUnsetICFocus(xw.ime.xic); - win.mode &= ~MODE_FOCUSED; - if (IS_SET(MODE_FOCUS)) - ttywrite("\033[O", 3, 0); - } -} - -int -match(uint mask, uint state) -{ - return mask == XK_ANY_MOD || mask == (state & ~ignoremod); -} - -char* -kmap(KeySym k, uint state) -{ - Key *kp; - int i; - - /* Check for mapped keys out of X11 function keys. */ - for (i = 0; i < LEN(mappedkeys); i++) { - if (mappedkeys[i] == k) - break; - } - if (i == LEN(mappedkeys)) { - if ((k & 0xFFFF) < 0xFD00) - return NULL; - } - - for (kp = key; kp < key + LEN(key); kp++) { - if (kp->k != k) - continue; - - if (!match(kp->mask, state)) - continue; - - if (IS_SET(MODE_APPKEYPAD) ? kp->appkey < 0 : kp->appkey > 0) - continue; - if (IS_SET(MODE_NUMLOCK) && kp->appkey == 2) - continue; - - if (IS_SET(MODE_APPCURSOR) ? kp->appcursor < 0 : kp->appcursor > 0) - continue; - - return kp->s; - } - - return NULL; -} - -void -kpress(XEvent *ev) -{ - XKeyEvent *e = &ev->xkey; - KeySym ksym; - char buf[64], *customkey; - int len; - Rune c; - Status status; - Shortcut *bp; - - if (IS_SET(MODE_KBDLOCK)) - return; - - if (xw.ime.xic) - len = XmbLookupString(xw.ime.xic, e, buf, sizeof buf, &ksym, &status); - else - len = XLookupString(e, buf, sizeof buf, &ksym, NULL); - /* 1. shortcuts */ - for (bp = shortcuts; bp < shortcuts + LEN(shortcuts); bp++) { - if (ksym == bp->keysym && match(bp->mod, e->state)) { - bp->func(&(bp->arg)); - return; - } - } - - /* 2. custom keys from config.h */ - if ((customkey = kmap(ksym, e->state))) { - ttywrite(customkey, strlen(customkey), 1); - return; - } - - /* 3. composed string from input method */ - if (len == 0) - return; - if (len == 1 && e->state & Mod1Mask) { - if (IS_SET(MODE_8BIT)) { - if (*buf < 0177) { - c = *buf | 0x80; - len = utf8encode(c, buf); - } - } else { - buf[1] = buf[0]; - buf[0] = '\033'; - len = 2; - } - } - ttywrite(buf, len, 1); -} - -void -cmessage(XEvent *e) -{ - /* - * See xembed specs - * http://standards.freedesktop.org/xembed-spec/xembed-spec-latest.html - */ - if (e->xclient.message_type == xw.xembed && e->xclient.format == 32) { - if (e->xclient.data.l[1] == XEMBED_FOCUS_IN) { - win.mode |= MODE_FOCUSED; - xseturgency(0); - } else if (e->xclient.data.l[1] == XEMBED_FOCUS_OUT) { - win.mode &= ~MODE_FOCUSED; - } - } else if (e->xclient.data.l[0] == xw.wmdeletewin) { - ttyhangup(); - exit(0); - } -} - -void -resize(XEvent *e) -{ - if (e->xconfigure.width == win.w && e->xconfigure.height == win.h) - return; - - cresize(e->xconfigure.width, e->xconfigure.height); -} - -void -run(void) -{ - XEvent ev; - int w = win.w, h = win.h; - fd_set rfd; - int xfd = XConnectionNumber(xw.dpy), xev, blinkset = 0, dodraw = 0; - int ttyfd; - struct timespec drawtimeout, *tv = NULL, now, last, lastblink; - long deltatime; - - /* Waiting for window mapping */ - do { - XNextEvent(xw.dpy, &ev); - /* - * This XFilterEvent call is required because of XOpenIM. It - * does filter out the key event and some client message for - * the input method too. - */ - if (XFilterEvent(&ev, None)) - continue; - if (ev.type == ConfigureNotify) { - w = ev.xconfigure.width; - h = ev.xconfigure.height; - } - } while (ev.type != MapNotify); - - ttyfd = ttynew(opt_line, shell, opt_io, opt_cmd); - cresize(w, h); - - clock_gettime(CLOCK_MONOTONIC, &last); - lastblink = last; - - for (xev = actionfps;;) { - FD_ZERO(&rfd); - FD_SET(ttyfd, &rfd); - FD_SET(xfd, &rfd); - - if (pselect(MAX(xfd, ttyfd)+1, &rfd, NULL, NULL, tv, NULL) < 0) { - if (errno == EINTR) - continue; - die("select failed: %s\n", strerror(errno)); - } - if (FD_ISSET(ttyfd, &rfd)) { - ttyread(); - if (blinktimeout) { - blinkset = tattrset(ATTR_BLINK); - if (!blinkset) - MODBIT(win.mode, 0, MODE_BLINK); - } - } - - if (FD_ISSET(xfd, &rfd)) - xev = actionfps; - - clock_gettime(CLOCK_MONOTONIC, &now); - drawtimeout.tv_sec = 0; - drawtimeout.tv_nsec = (1000 * 1E6)/ xfps; - tv = &drawtimeout; - - dodraw = 0; - if (blinktimeout && TIMEDIFF(now, lastblink) > blinktimeout) { - tsetdirtattr(ATTR_BLINK); - win.mode ^= MODE_BLINK; - lastblink = now; - dodraw = 1; - } - deltatime = TIMEDIFF(now, last); - if (deltatime > 1000 / (xev ? xfps : actionfps)) { - dodraw = 1; - last = now; - } - - if (dodraw) { - while (XPending(xw.dpy)) { - XNextEvent(xw.dpy, &ev); - if (XFilterEvent(&ev, None)) - continue; - if (handler[ev.type]) - (handler[ev.type])(&ev); - } - - draw(); - XFlush(xw.dpy); - - if (xev && !FD_ISSET(xfd, &rfd)) - xev--; - if (!FD_ISSET(ttyfd, &rfd) && !FD_ISSET(xfd, &rfd)) { - if (blinkset) { - if (TIMEDIFF(now, lastblink) \ - > blinktimeout) { - drawtimeout.tv_nsec = 1000; - } else { - drawtimeout.tv_nsec = (1E6 * \ - (blinktimeout - \ - TIMEDIFF(now, - lastblink))); - } - drawtimeout.tv_sec = \ - drawtimeout.tv_nsec / 1E9; - drawtimeout.tv_nsec %= (long)1E9; - } else { - tv = NULL; - } - } - } - } -} - -void -usage(void) -{ - die("usage: %s [-aiv] [-c class] [-f font] [-g geometry]" - " [-n name] [-o file]\n" - " [-T title] [-t title] [-w windowid]" - " [[-e] command [args ...]]\n" - " %s [-aiv] [-c class] [-f font] [-g geometry]" - " [-n name] [-o file]\n" - " [-T title] [-t title] [-w windowid] -l line" - " [stty_args ...]\n", argv0, argv0); -} - -int -main(int argc, char *argv[]) -{ - xw.l = xw.t = 0; - xw.isfixed = False; - win.cursor = cursorshape; - - ARGBEGIN { - case 'a': - allowaltscreen = 0; - break; - case 'c': - opt_class = EARGF(usage()); - break; - case 'e': - if (argc > 0) - --argc, ++argv; - goto run; - case 'f': - opt_font = EARGF(usage()); - break; - case 'g': - xw.gm = XParseGeometry(EARGF(usage()), - &xw.l, &xw.t, &cols, &rows); - break; - case 'i': - xw.isfixed = 1; - break; - case 'o': - opt_io = EARGF(usage()); - break; - case 'l': - opt_line = EARGF(usage()); - break; - case 'n': - opt_name = EARGF(usage()); - break; - case 't': - case 'T': - opt_title = EARGF(usage()); - break; - case 'w': - opt_embed = EARGF(usage()); - break; - case 'v': - die("%s " VERSION "\n", argv0); - break; - default: - usage(); - } ARGEND; - -run: - if (argc > 0) /* eat all remaining arguments */ - opt_cmd = argv; - - if (!opt_title) - opt_title = (opt_line || !opt_cmd) ? "st" : opt_cmd[0]; - - setlocale(LC_CTYPE, ""); - XSetLocaleModifiers(""); - cols = MAX(cols, 1); - rows = MAX(rows, 1); - tnew(cols, rows); - xinit(cols, rows); - xsetenv(); - selinit(); - run(); - - return 0; -} diff --git a/.suckless/st/x.c.orig b/.suckless/st/x.c.orig deleted file mode 100644 index 78a4044..0000000 --- a/.suckless/st/x.c.orig +++ /dev/null @@ -1,2045 +0,0 @@ -/* See LICENSE for license details. */ -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include -#include - -static char *argv0; -#include "arg.h" -#include "st.h" -#include "win.h" - -/* types used in config.h */ -typedef struct { - uint mod; - KeySym keysym; - void (*func)(const Arg *); - const Arg arg; -} Shortcut; - -typedef struct { - uint mod; - uint button; - void (*func)(const Arg *); - const Arg arg; - uint release; -} MouseShortcut; - -typedef struct { - KeySym k; - uint mask; - char *s; - /* three-valued logic variables: 0 indifferent, 1 on, -1 off */ - signed char appkey; /* application keypad */ - signed char appcursor; /* application cursor */ -} Key; - -/* X modifiers */ -#define XK_ANY_MOD UINT_MAX -#define XK_NO_MOD 0 -#define XK_SWITCH_MOD (1<<13) - -/* function definitions used in config.h */ -static void clipcopy(const Arg *); -static void clippaste(const Arg *); -static void numlock(const Arg *); -static void selpaste(const Arg *); -static void zoom(const Arg *); -static void zoomabs(const Arg *); -static void zoomreset(const Arg *); -static void ttysend(const Arg *); - -/* config.h for applying patches and the configuration. */ -#include "config.h" - -/* XEMBED messages */ -#define XEMBED_FOCUS_IN 4 -#define XEMBED_FOCUS_OUT 5 - -/* macros */ -#define IS_SET(flag) ((win.mode & (flag)) != 0) -#define TRUERED(x) (((x) & 0xff0000) >> 8) -#define TRUEGREEN(x) (((x) & 0xff00)) -#define TRUEBLUE(x) (((x) & 0xff) << 8) - -typedef XftDraw *Draw; -typedef XftColor Color; -typedef XftGlyphFontSpec GlyphFontSpec; - -/* Purely graphic info */ -typedef struct { - int tw, th; /* tty width and height */ - int w, h; /* window width and height */ - int hborderpx, vborderpx; - int ch; /* char height */ - int cw; /* char width */ - int mode; /* window state/mode flags */ - int cursor; /* cursor style */ -} TermWindow; - -typedef struct { - Display *dpy; - Colormap cmap; - Window win; - Drawable buf; - GlyphFontSpec *specbuf; /* font spec buffer used for rendering */ - Atom xembed, wmdeletewin, netwmname, netwmpid; - struct { - XIM xim; - XIC xic; - XPoint spot; - XVaNestedList spotlist; - } ime; - Draw draw; - Visual *vis; - XSetWindowAttributes attrs; - int scr; - int isfixed; /* is fixed geometry? */ - int l, t; /* left and top offset */ - int gm; /* geometry mask */ -} XWindow; - -typedef struct { - Atom xtarget; - char *primary, *clipboard; - struct timespec tclick1; - struct timespec tclick2; -} XSelection; - -/* Font structure */ -#define Font Font_ -typedef struct { - int height; - int width; - int ascent; - int descent; - int badslant; - int badweight; - short lbearing; - short rbearing; - XftFont *match; - FcFontSet *set; - FcPattern *pattern; -} Font; - -/* Drawing Context */ -typedef struct { - Color *col; - size_t collen; - Font font, bfont, ifont, ibfont; - GC gc; -} DC; - -static inline ushort sixd_to_16bit(int); -static int xmakeglyphfontspecs(XftGlyphFontSpec *, const Glyph *, int, int, int); -static void xdrawglyphfontspecs(const XftGlyphFontSpec *, Glyph, int, int, int); -static void xdrawglyph(Glyph, int, int); -static void xclear(int, int, int, int); -static int xgeommasktogravity(int); -static int ximopen(Display *); -static void ximinstantiate(Display *, XPointer, XPointer); -static void ximdestroy(XIM, XPointer, XPointer); -static int xicdestroy(XIC, XPointer, XPointer); -static void xinit(int, int); -static void cresize(int, int); -static void xresize(int, int); -static void xhints(void); -static int xloadcolor(int, const char *, Color *); -static int xloadfont(Font *, FcPattern *); -static void xloadfonts(char *, double); -static void xunloadfont(Font *); -static void xunloadfonts(void); -static void xsetenv(void); -static void xseturgency(int); -static int evcol(XEvent *); -static int evrow(XEvent *); - -static void expose(XEvent *); -static void visibility(XEvent *); -static void unmap(XEvent *); -static void kpress(XEvent *); -static void cmessage(XEvent *); -static void resize(XEvent *); -static void focus(XEvent *); -static int mouseaction(XEvent *, uint); -static void brelease(XEvent *); -static void bpress(XEvent *); -static void bmotion(XEvent *); -static void propnotify(XEvent *); -static void selnotify(XEvent *); -static void selclear_(XEvent *); -static void selrequest(XEvent *); -static void setsel(char *, Time); -static void mousesel(XEvent *, int); -static void mousereport(XEvent *); -static char *kmap(KeySym, uint); -static int match(uint, uint); - -static void run(void); -static void usage(void); - -static void (*handler[LASTEvent])(XEvent *) = { - [KeyPress] = kpress, - [ClientMessage] = cmessage, - [ConfigureNotify] = resize, - [VisibilityNotify] = visibility, - [UnmapNotify] = unmap, - [Expose] = expose, - [FocusIn] = focus, - [FocusOut] = focus, - [MotionNotify] = bmotion, - [ButtonPress] = bpress, - [ButtonRelease] = brelease, -/* - * Uncomment if you want the selection to disappear when you select something - * different in another window. - */ -/* [SelectionClear] = selclear_, */ - [SelectionNotify] = selnotify, -/* - * PropertyNotify is only turned on when there is some INCR transfer happening - * for the selection retrieval. - */ - [PropertyNotify] = propnotify, - [SelectionRequest] = selrequest, -}; - -/* Globals */ -static DC dc; -static XWindow xw; -static XSelection xsel; -static TermWindow win; - -/* Font Ring Cache */ -enum { - FRC_NORMAL, - FRC_ITALIC, - FRC_BOLD, - FRC_ITALICBOLD -}; - -typedef struct { - XftFont *font; - int flags; - Rune unicodep; -} Fontcache; - -/* Fontcache is an array now. A new font will be appended to the array. */ -static Fontcache *frc = NULL; -static int frclen = 0; -static int frccap = 0; -static char *usedfont = NULL; -static double usedfontsize = 0; -static double defaultfontsize = 0; - -static char *opt_class = NULL; -static char **opt_cmd = NULL; -static char *opt_embed = NULL; -static char *opt_font = NULL; -static char *opt_io = NULL; -static char *opt_line = NULL; -static char *opt_name = NULL; -static char *opt_title = NULL; - -static int oldbutton = 3; /* button event on startup: 3 = release */ - -void -clipcopy(const Arg *dummy) -{ - Atom clipboard; - - free(xsel.clipboard); - xsel.clipboard = NULL; - - if (xsel.primary != NULL) { - xsel.clipboard = xstrdup(xsel.primary); - clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - XSetSelectionOwner(xw.dpy, clipboard, xw.win, CurrentTime); - } -} - -void -clippaste(const Arg *dummy) -{ - Atom clipboard; - - clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - XConvertSelection(xw.dpy, clipboard, xsel.xtarget, clipboard, - xw.win, CurrentTime); -} - -void -selpaste(const Arg *dummy) -{ - XConvertSelection(xw.dpy, XA_PRIMARY, xsel.xtarget, XA_PRIMARY, - xw.win, CurrentTime); -} - -void -numlock(const Arg *dummy) -{ - win.mode ^= MODE_NUMLOCK; -} - -void -zoom(const Arg *arg) -{ - Arg larg; - - larg.f = usedfontsize + arg->f; - zoomabs(&larg); -} - -void -zoomabs(const Arg *arg) -{ - xunloadfonts(); - xloadfonts(usedfont, arg->f); - cresize(0, 0); - redraw(); - xhints(); -} - -void -zoomreset(const Arg *arg) -{ - Arg larg; - - if (defaultfontsize > 0) { - larg.f = defaultfontsize; - zoomabs(&larg); - } -} - -void -ttysend(const Arg *arg) -{ - ttywrite(arg->s, strlen(arg->s), 1); -} - -int -evcol(XEvent *e) -{ - int x = e->xbutton.x - win.hborderpx; - LIMIT(x, 0, win.tw - 1); - return x / win.cw; -} - -int -evrow(XEvent *e) -{ - int y = e->xbutton.y - win.vborderpx; - LIMIT(y, 0, win.th - 1); - return y / win.ch; -} - -void -mousesel(XEvent *e, int done) -{ - int type, seltype = SEL_REGULAR; - uint state = e->xbutton.state & ~(Button1Mask | forcemousemod); - - for (type = 1; type < LEN(selmasks); ++type) { - if (match(selmasks[type], state)) { - seltype = type; - break; - } - } - selextend(evcol(e), evrow(e), seltype, done); - if (done) - setsel(getsel(), e->xbutton.time); -} - -void -mousereport(XEvent *e) -{ - int len, x = evcol(e), y = evrow(e), - button = e->xbutton.button, state = e->xbutton.state; - char buf[40]; - static int ox, oy; - - /* from urxvt */ - if (e->xbutton.type == MotionNotify) { - if (x == ox && y == oy) - return; - if (!IS_SET(MODE_MOUSEMOTION) && !IS_SET(MODE_MOUSEMANY)) - return; - /* MOUSE_MOTION: no reporting if no button is pressed */ - if (IS_SET(MODE_MOUSEMOTION) && oldbutton == 3) - return; - - button = oldbutton + 32; - ox = x; - oy = y; - } else { - if (!IS_SET(MODE_MOUSESGR) && e->xbutton.type == ButtonRelease) { - button = 3; - } else { - button -= Button1; - if (button >= 3) - button += 64 - 3; - } - if (e->xbutton.type == ButtonPress) { - oldbutton = button; - ox = x; - oy = y; - } else if (e->xbutton.type == ButtonRelease) { - oldbutton = 3; - /* MODE_MOUSEX10: no button release reporting */ - if (IS_SET(MODE_MOUSEX10)) - return; - if (button == 64 || button == 65) - return; - } - } - - if (!IS_SET(MODE_MOUSEX10)) { - button += ((state & ShiftMask ) ? 4 : 0) - + ((state & Mod4Mask ) ? 8 : 0) - + ((state & ControlMask) ? 16 : 0); - } - - if (IS_SET(MODE_MOUSESGR)) { - len = snprintf(buf, sizeof(buf), "\033[<%d;%d;%d%c", - button, x+1, y+1, - e->xbutton.type == ButtonRelease ? 'm' : 'M'); - } else if (x < 223 && y < 223) { - len = snprintf(buf, sizeof(buf), "\033[M%c%c%c", - 32+button, 32+x+1, 32+y+1); - } else { - return; - } - - ttywrite(buf, len, 0); -} - -int -mouseaction(XEvent *e, uint release) -{ - MouseShortcut *ms; - - for (ms = mshortcuts; ms < mshortcuts + LEN(mshortcuts); ms++) { - if (ms->release == release && - ms->button == e->xbutton.button && - (match(ms->mod, e->xbutton.state) || /* exact or forced */ - match(ms->mod, e->xbutton.state & ~forcemousemod))) { - ms->func(&(ms->arg)); - return 1; - } - } - - return 0; -} - -void -bpress(XEvent *e) -{ - struct timespec now; - int snap; - - if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forcemousemod)) { - mousereport(e); - return; - } - - if (mouseaction(e, 0)) - return; - - if (e->xbutton.button == Button1) { - /* - * If the user clicks below predefined timeouts specific - * snapping behaviour is exposed. - */ - clock_gettime(CLOCK_MONOTONIC, &now); - if (TIMEDIFF(now, xsel.tclick2) <= tripleclicktimeout) { - snap = SNAP_LINE; - } else if (TIMEDIFF(now, xsel.tclick1) <= doubleclicktimeout) { - snap = SNAP_WORD; - } else { - snap = 0; - } - xsel.tclick2 = xsel.tclick1; - xsel.tclick1 = now; - - selstart(evcol(e), evrow(e), snap); - } -} - -void -propnotify(XEvent *e) -{ - XPropertyEvent *xpev; - Atom clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - - xpev = &e->xproperty; - if (xpev->state == PropertyNewValue && - (xpev->atom == XA_PRIMARY || - xpev->atom == clipboard)) { - selnotify(e); - } -} - -void -selnotify(XEvent *e) -{ - ulong nitems, ofs, rem; - int format; - uchar *data, *last, *repl; - Atom type, incratom, property = None; - - incratom = XInternAtom(xw.dpy, "INCR", 0); - - ofs = 0; - if (e->type == SelectionNotify) - property = e->xselection.property; - else if (e->type == PropertyNotify) - property = e->xproperty.atom; - - if (property == None) - return; - - do { - if (XGetWindowProperty(xw.dpy, xw.win, property, ofs, - BUFSIZ/4, False, AnyPropertyType, - &type, &format, &nitems, &rem, - &data)) { - fprintf(stderr, "Clipboard allocation failed\n"); - return; - } - - if (e->type == PropertyNotify && nitems == 0 && rem == 0) { - /* - * If there is some PropertyNotify with no data, then - * this is the signal of the selection owner that all - * data has been transferred. We won't need to receive - * PropertyNotify events anymore. - */ - MODBIT(xw.attrs.event_mask, 0, PropertyChangeMask); - XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask, - &xw.attrs); - } - - if (type == incratom) { - /* - * Activate the PropertyNotify events so we receive - * when the selection owner does send us the next - * chunk of data. - */ - MODBIT(xw.attrs.event_mask, 1, PropertyChangeMask); - XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask, - &xw.attrs); - - /* - * Deleting the property is the transfer start signal. - */ - XDeleteProperty(xw.dpy, xw.win, (int)property); - continue; - } - - /* - * As seen in getsel: - * Line endings are inconsistent in the terminal and GUI world - * copy and pasting. When receiving some selection data, - * replace all '\n' with '\r'. - * FIXME: Fix the computer world. - */ - repl = data; - last = data + nitems * format / 8; - while ((repl = memchr(repl, '\n', last - repl))) { - *repl++ = '\r'; - } - - if (IS_SET(MODE_BRCKTPASTE) && ofs == 0) - ttywrite("\033[200~", 6, 0); - ttywrite((char *)data, nitems * format / 8, 1); - if (IS_SET(MODE_BRCKTPASTE) && rem == 0) - ttywrite("\033[201~", 6, 0); - XFree(data); - /* number of 32-bit chunks returned */ - ofs += nitems * format / 32; - } while (rem > 0); - - /* - * Deleting the property again tells the selection owner to send the - * next data chunk in the property. - */ - XDeleteProperty(xw.dpy, xw.win, (int)property); -} - -void -xclipcopy(void) -{ - clipcopy(NULL); -} - -void -selclear_(XEvent *e) -{ - selclear(); -} - -void -selrequest(XEvent *e) -{ - XSelectionRequestEvent *xsre; - XSelectionEvent xev; - Atom xa_targets, string, clipboard; - char *seltext; - - xsre = (XSelectionRequestEvent *) e; - xev.type = SelectionNotify; - xev.requestor = xsre->requestor; - xev.selection = xsre->selection; - xev.target = xsre->target; - xev.time = xsre->time; - if (xsre->property == None) - xsre->property = xsre->target; - - /* reject */ - xev.property = None; - - xa_targets = XInternAtom(xw.dpy, "TARGETS", 0); - if (xsre->target == xa_targets) { - /* respond with the supported type */ - string = xsel.xtarget; - XChangeProperty(xsre->display, xsre->requestor, xsre->property, - XA_ATOM, 32, PropModeReplace, - (uchar *) &string, 1); - xev.property = xsre->property; - } else if (xsre->target == xsel.xtarget || xsre->target == XA_STRING) { - /* - * xith XA_STRING non ascii characters may be incorrect in the - * requestor. It is not our problem, use utf8. - */ - clipboard = XInternAtom(xw.dpy, "CLIPBOARD", 0); - if (xsre->selection == XA_PRIMARY) { - seltext = xsel.primary; - } else if (xsre->selection == clipboard) { - seltext = xsel.clipboard; - } else { - fprintf(stderr, - "Unhandled clipboard selection 0x%lx\n", - xsre->selection); - return; - } - if (seltext != NULL) { - XChangeProperty(xsre->display, xsre->requestor, - xsre->property, xsre->target, - 8, PropModeReplace, - (uchar *)seltext, strlen(seltext)); - xev.property = xsre->property; - } - } - - /* all done, send a notification to the listener */ - if (!XSendEvent(xsre->display, xsre->requestor, 1, 0, (XEvent *) &xev)) - fprintf(stderr, "Error sending SelectionNotify event\n"); -} - -void -setsel(char *str, Time t) -{ - if (!str) - return; - - free(xsel.primary); - xsel.primary = str; - - XSetSelectionOwner(xw.dpy, XA_PRIMARY, xw.win, t); - if (XGetSelectionOwner(xw.dpy, XA_PRIMARY) != xw.win) - selclear(); -} - -void -xsetsel(char *str) -{ - setsel(str, CurrentTime); -} - -void -brelease(XEvent *e) -{ - if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forcemousemod)) { - mousereport(e); - return; - } - - if (mouseaction(e, 1)) - return; - if (e->xbutton.button == Button1) - mousesel(e, 1); -} - -void -bmotion(XEvent *e) -{ - if (IS_SET(MODE_MOUSE) && !(e->xbutton.state & forcemousemod)) { - mousereport(e); - return; - } - - mousesel(e, 0); -} - -void -cresize(int width, int height) -{ - int col, row; - - if (width != 0) - win.w = width; - if (height != 0) - win.h = height; - - col = (win.w - 2 * borderpx) / win.cw; - row = (win.h - 2 * borderpx) / win.ch; - col = MAX(1, col); - row = MAX(1, row); - - win.hborderpx = (win.w - col * win.cw) / 2; - win.vborderpx = (win.h - row * win.ch) / 2; - - tresize(col, row); - xresize(col, row); - ttyresize(win.tw, win.th); -} - -void -xresize(int col, int row) -{ - win.tw = col * win.cw; - win.th = row * win.ch; - - XFreePixmap(xw.dpy, xw.buf); - xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, - DefaultDepth(xw.dpy, xw.scr)); - XftDrawChange(xw.draw, xw.buf); - xclear(0, 0, win.w, win.h); - - /* resize to new width */ - xw.specbuf = xrealloc(xw.specbuf, col * sizeof(GlyphFontSpec)); -} - -ushort -sixd_to_16bit(int x) -{ - return x == 0 ? 0 : 0x3737 + 0x2828 * x; -} - -int -xloadcolor(int i, const char *name, Color *ncolor) -{ - XRenderColor color = { .alpha = 0xffff }; - - if (!name) { - if (BETWEEN(i, 16, 255)) { /* 256 color */ - if (i < 6*6*6+16) { /* same colors as xterm */ - color.red = sixd_to_16bit( ((i-16)/36)%6 ); - color.green = sixd_to_16bit( ((i-16)/6) %6 ); - color.blue = sixd_to_16bit( ((i-16)/1) %6 ); - } else { /* greyscale */ - color.red = 0x0808 + 0x0a0a * (i - (6*6*6+16)); - color.green = color.blue = color.red; - } - return XftColorAllocValue(xw.dpy, xw.vis, - xw.cmap, &color, ncolor); - } else - name = colorname[i]; - } - - return XftColorAllocName(xw.dpy, xw.vis, xw.cmap, name, ncolor); -} - -void -xloadcols(void) -{ - int i; - static int loaded; - Color *cp; - - if (loaded) { - for (cp = dc.col; cp < &dc.col[dc.collen]; ++cp) - XftColorFree(xw.dpy, xw.vis, xw.cmap, cp); - } else { - dc.collen = MAX(LEN(colorname), 256); - dc.col = xmalloc(dc.collen * sizeof(Color)); - } - - for (i = 0; i < dc.collen; i++) - if (!xloadcolor(i, NULL, &dc.col[i])) { - if (colorname[i]) - die("could not allocate color '%s'\n", colorname[i]); - else - die("could not allocate color %d\n", i); - } - loaded = 1; -} - -int -xsetcolorname(int x, const char *name) -{ - Color ncolor; - - if (!BETWEEN(x, 0, dc.collen)) - return 1; - - if (!xloadcolor(x, name, &ncolor)) - return 1; - - XftColorFree(xw.dpy, xw.vis, xw.cmap, &dc.col[x]); - dc.col[x] = ncolor; - - return 0; -} - -/* - * Absolute coordinates. - */ -void -xclear(int x1, int y1, int x2, int y2) -{ - XftDrawRect(xw.draw, - &dc.col[IS_SET(MODE_REVERSE)? defaultfg : defaultbg], - x1, y1, x2-x1, y2-y1); -} - -void -xhints(void) -{ - XClassHint class = {opt_name ? opt_name : termname, - opt_class ? opt_class : termname}; - XWMHints wm = {.flags = InputHint, .input = 1}; - XSizeHints *sizeh; - - sizeh = XAllocSizeHints(); - - sizeh->flags = PSize | PResizeInc | PBaseSize | PMinSize; - sizeh->height = win.h; - sizeh->width = win.w; - sizeh->height_inc = 1; - sizeh->width_inc = 1; - sizeh->base_height = 2 * borderpx; - sizeh->base_width = 2 * borderpx; - sizeh->min_height = win.ch + 2 * borderpx; - sizeh->min_width = win.cw + 2 * borderpx; - if (xw.isfixed) { - sizeh->flags |= PMaxSize; - sizeh->min_width = sizeh->max_width = win.w; - sizeh->min_height = sizeh->max_height = win.h; - } - if (xw.gm & (XValue|YValue)) { - sizeh->flags |= USPosition | PWinGravity; - sizeh->x = xw.l; - sizeh->y = xw.t; - sizeh->win_gravity = xgeommasktogravity(xw.gm); - } - - XSetWMProperties(xw.dpy, xw.win, NULL, NULL, NULL, 0, sizeh, &wm, - &class); - XFree(sizeh); -} - -int -xgeommasktogravity(int mask) -{ - switch (mask & (XNegative|YNegative)) { - case 0: - return NorthWestGravity; - case XNegative: - return NorthEastGravity; - case YNegative: - return SouthWestGravity; - } - - return SouthEastGravity; -} - -int -xloadfont(Font *f, FcPattern *pattern) -{ - FcPattern *configured; - FcPattern *match; - FcResult result; - XGlyphInfo extents; - int wantattr, haveattr; - - /* - * Manually configure instead of calling XftMatchFont - * so that we can use the configured pattern for - * "missing glyph" lookups. - */ - configured = FcPatternDuplicate(pattern); - if (!configured) - return 1; - - FcConfigSubstitute(NULL, configured, FcMatchPattern); - XftDefaultSubstitute(xw.dpy, xw.scr, configured); - - match = FcFontMatch(NULL, configured, &result); - if (!match) { - FcPatternDestroy(configured); - return 1; - } - - if (!(f->match = XftFontOpenPattern(xw.dpy, match))) { - FcPatternDestroy(configured); - FcPatternDestroy(match); - return 1; - } - - if ((XftPatternGetInteger(pattern, "slant", 0, &wantattr) == - XftResultMatch)) { - /* - * Check if xft was unable to find a font with the appropriate - * slant but gave us one anyway. Try to mitigate. - */ - if ((XftPatternGetInteger(f->match->pattern, "slant", 0, - &haveattr) != XftResultMatch) || haveattr < wantattr) { - f->badslant = 1; - fputs("font slant does not match\n", stderr); - } - } - - if ((XftPatternGetInteger(pattern, "weight", 0, &wantattr) == - XftResultMatch)) { - if ((XftPatternGetInteger(f->match->pattern, "weight", 0, - &haveattr) != XftResultMatch) || haveattr != wantattr) { - f->badweight = 1; - fputs("font weight does not match\n", stderr); - } - } - - XftTextExtentsUtf8(xw.dpy, f->match, - (const FcChar8 *) ascii_printable, - strlen(ascii_printable), &extents); - - f->set = NULL; - f->pattern = configured; - - f->ascent = f->match->ascent; - f->descent = f->match->descent; - f->lbearing = 0; - f->rbearing = f->match->max_advance_width; - - f->height = f->ascent + f->descent; - f->width = DIVCEIL(extents.xOff, strlen(ascii_printable)); - - return 0; -} - -void -xloadfonts(char *fontstr, double fontsize) -{ - FcPattern *pattern; - double fontval; - - if (fontstr[0] == '-') - pattern = XftXlfdParse(fontstr, False, False); - else - pattern = FcNameParse((FcChar8 *)fontstr); - - if (!pattern) - die("can't open font %s\n", fontstr); - - if (fontsize > 1) { - FcPatternDel(pattern, FC_PIXEL_SIZE); - FcPatternDel(pattern, FC_SIZE); - FcPatternAddDouble(pattern, FC_PIXEL_SIZE, (double)fontsize); - usedfontsize = fontsize; - } else { - if (FcPatternGetDouble(pattern, FC_PIXEL_SIZE, 0, &fontval) == - FcResultMatch) { - usedfontsize = fontval; - } else if (FcPatternGetDouble(pattern, FC_SIZE, 0, &fontval) == - FcResultMatch) { - usedfontsize = -1; - } else { - /* - * Default font size is 12, if none given. This is to - * have a known usedfontsize value. - */ - FcPatternAddDouble(pattern, FC_PIXEL_SIZE, 12); - usedfontsize = 12; - } - defaultfontsize = usedfontsize; - } - - if (xloadfont(&dc.font, pattern)) - die("can't open font %s\n", fontstr); - - if (usedfontsize < 0) { - FcPatternGetDouble(dc.font.match->pattern, - FC_PIXEL_SIZE, 0, &fontval); - usedfontsize = fontval; - if (fontsize == 0) - defaultfontsize = fontval; - } - - /* Setting character width and height. */ - win.cw = ceilf(dc.font.width * cwscale); - win.ch = ceilf(dc.font.height * chscale); - - FcPatternDel(pattern, FC_SLANT); - FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ITALIC); - if (xloadfont(&dc.ifont, pattern)) - die("can't open font %s\n", fontstr); - - FcPatternDel(pattern, FC_WEIGHT); - FcPatternAddInteger(pattern, FC_WEIGHT, FC_WEIGHT_BOLD); - if (xloadfont(&dc.ibfont, pattern)) - die("can't open font %s\n", fontstr); - - FcPatternDel(pattern, FC_SLANT); - FcPatternAddInteger(pattern, FC_SLANT, FC_SLANT_ROMAN); - if (xloadfont(&dc.bfont, pattern)) - die("can't open font %s\n", fontstr); - - FcPatternDestroy(pattern); -} - -void -xunloadfont(Font *f) -{ - XftFontClose(xw.dpy, f->match); - FcPatternDestroy(f->pattern); - if (f->set) - FcFontSetDestroy(f->set); -} - -void -xunloadfonts(void) -{ - /* Free the loaded fonts in the font cache. */ - while (frclen > 0) - XftFontClose(xw.dpy, frc[--frclen].font); - - xunloadfont(&dc.font); - xunloadfont(&dc.bfont); - xunloadfont(&dc.ifont); - xunloadfont(&dc.ibfont); -} - -int -ximopen(Display *dpy) -{ - XIMCallback imdestroy = { .client_data = NULL, .callback = ximdestroy }; - XICCallback icdestroy = { .client_data = NULL, .callback = xicdestroy }; - - xw.ime.xim = XOpenIM(xw.dpy, NULL, NULL, NULL); - if (xw.ime.xim == NULL) - return 0; - - if (XSetIMValues(xw.ime.xim, XNDestroyCallback, &imdestroy, NULL)) - fprintf(stderr, "XSetIMValues: " - "Could not set XNDestroyCallback.\n"); - - xw.ime.spotlist = XVaCreateNestedList(0, XNSpotLocation, &xw.ime.spot, - NULL); - - if (xw.ime.xic == NULL) { - xw.ime.xic = XCreateIC(xw.ime.xim, XNInputStyle, - XIMPreeditNothing | XIMStatusNothing, - XNClientWindow, xw.win, - XNFocusWindow, xw.win, - XNDestroyCallback, &icdestroy, - NULL); - } - if (xw.ime.xic == NULL) - fprintf(stderr, "XCreateIC: Could not create input context.\n"); - - return 1; -} - -void -ximinstantiate(Display *dpy, XPointer client, XPointer call) -{ - if (ximopen(dpy)) - XUnregisterIMInstantiateCallback(xw.dpy, NULL, NULL, NULL, - ximinstantiate, NULL); -} - -void -ximdestroy(XIM xim, XPointer client, XPointer call) -{ - xw.ime.xim = NULL; - XRegisterIMInstantiateCallback(xw.dpy, NULL, NULL, NULL, - ximinstantiate, NULL); - XFree(xw.ime.spotlist); -} - -int -xicdestroy(XIC xim, XPointer client, XPointer call) -{ - xw.ime.xic = NULL; - return 1; -} - -void -xinit(int cols, int rows) -{ - XGCValues gcvalues; - Cursor cursor; - Window parent; - pid_t thispid = getpid(); - XColor xmousefg, xmousebg; - - if (!(xw.dpy = XOpenDisplay(NULL))) - die("can't open display\n"); - xw.scr = XDefaultScreen(xw.dpy); - xw.vis = XDefaultVisual(xw.dpy, xw.scr); - - /* font */ - if (!FcInit()) - die("could not init fontconfig.\n"); - - usedfont = (opt_font == NULL)? font : opt_font; - xloadfonts(usedfont, 0); - - /* colors */ - xw.cmap = XDefaultColormap(xw.dpy, xw.scr); - xloadcols(); - - /* adjust fixed window geometry */ - win.w = 2 * win.hborderpx + cols * win.cw; - win.h = 2 * win.vborderpx + rows * win.ch; - if (xw.gm & XNegative) - xw.l += DisplayWidth(xw.dpy, xw.scr) - win.w - 2; - if (xw.gm & YNegative) - xw.t += DisplayHeight(xw.dpy, xw.scr) - win.h - 2; - - /* Events */ - xw.attrs.background_pixel = dc.col[defaultbg].pixel; - xw.attrs.border_pixel = dc.col[defaultbg].pixel; - xw.attrs.bit_gravity = NorthWestGravity; - xw.attrs.event_mask = FocusChangeMask | KeyPressMask | KeyReleaseMask - | ExposureMask | VisibilityChangeMask | StructureNotifyMask - | ButtonMotionMask | ButtonPressMask | ButtonReleaseMask; - xw.attrs.colormap = xw.cmap; - - if (!(opt_embed && (parent = strtol(opt_embed, NULL, 0)))) - parent = XRootWindow(xw.dpy, xw.scr); - xw.win = XCreateWindow(xw.dpy, parent, xw.l, xw.t, - win.w, win.h, 0, XDefaultDepth(xw.dpy, xw.scr), InputOutput, - xw.vis, CWBackPixel | CWBorderPixel | CWBitGravity - | CWEventMask | CWColormap, &xw.attrs); - - memset(&gcvalues, 0, sizeof(gcvalues)); - gcvalues.graphics_exposures = False; - dc.gc = XCreateGC(xw.dpy, parent, GCGraphicsExposures, - &gcvalues); - xw.buf = XCreatePixmap(xw.dpy, xw.win, win.w, win.h, - DefaultDepth(xw.dpy, xw.scr)); - XSetForeground(xw.dpy, dc.gc, dc.col[defaultbg].pixel); - XFillRectangle(xw.dpy, xw.buf, dc.gc, 0, 0, win.w, win.h); - - /* font spec buffer */ - xw.specbuf = xmalloc(cols * sizeof(GlyphFontSpec)); - - /* Xft rendering context */ - xw.draw = XftDrawCreate(xw.dpy, xw.buf, xw.vis, xw.cmap); - - /* input methods */ - if (!ximopen(xw.dpy)) { - XRegisterIMInstantiateCallback(xw.dpy, NULL, NULL, NULL, - ximinstantiate, NULL); - } - - /* white cursor, black outline */ - cursor = XCreateFontCursor(xw.dpy, mouseshape); - XDefineCursor(xw.dpy, xw.win, cursor); - - if (XParseColor(xw.dpy, xw.cmap, colorname[mousefg], &xmousefg) == 0) { - xmousefg.red = 0xffff; - xmousefg.green = 0xffff; - xmousefg.blue = 0xffff; - } - - if (XParseColor(xw.dpy, xw.cmap, colorname[mousebg], &xmousebg) == 0) { - xmousebg.red = 0x0000; - xmousebg.green = 0x0000; - xmousebg.blue = 0x0000; - } - - XRecolorCursor(xw.dpy, cursor, &xmousefg, &xmousebg); - - xw.xembed = XInternAtom(xw.dpy, "_XEMBED", False); - xw.wmdeletewin = XInternAtom(xw.dpy, "WM_DELETE_WINDOW", False); - xw.netwmname = XInternAtom(xw.dpy, "_NET_WM_NAME", False); - XSetWMProtocols(xw.dpy, xw.win, &xw.wmdeletewin, 1); - - xw.netwmpid = XInternAtom(xw.dpy, "_NET_WM_PID", False); - XChangeProperty(xw.dpy, xw.win, xw.netwmpid, XA_CARDINAL, 32, - PropModeReplace, (uchar *)&thispid, 1); - - win.mode = MODE_NUMLOCK; - resettitle(); - xhints(); - XMapWindow(xw.dpy, xw.win); - XSync(xw.dpy, False); - - clock_gettime(CLOCK_MONOTONIC, &xsel.tclick1); - clock_gettime(CLOCK_MONOTONIC, &xsel.tclick2); - xsel.primary = NULL; - xsel.clipboard = NULL; - xsel.xtarget = XInternAtom(xw.dpy, "UTF8_STRING", 0); - if (xsel.xtarget == None) - xsel.xtarget = XA_STRING; -} - -int -xmakeglyphfontspecs(XftGlyphFontSpec *specs, const Glyph *glyphs, int len, int x, int y) -{ - float winx = win.hborderpx + x * win.cw, winy = win.vborderpx + y * win.ch, xp, yp; - ushort mode, prevmode = USHRT_MAX; - Font *font = &dc.font; - int frcflags = FRC_NORMAL; - float runewidth = win.cw; - Rune rune; - FT_UInt glyphidx; - FcResult fcres; - FcPattern *fcpattern, *fontpattern; - FcFontSet *fcsets[] = { NULL }; - FcCharSet *fccharset; - int i, f, numspecs = 0; - - for (i = 0, xp = winx, yp = winy + font->ascent; i < len; ++i) { - /* Fetch rune and mode for current glyph. */ - rune = glyphs[i].u; - mode = glyphs[i].mode; - - /* Skip dummy wide-character spacing. */ - if (mode == ATTR_WDUMMY) - continue; - - /* Determine font for glyph if different from previous glyph. */ - if (prevmode != mode) { - prevmode = mode; - font = &dc.font; - frcflags = FRC_NORMAL; - runewidth = win.cw * ((mode & ATTR_WIDE) ? 2.0f : 1.0f); - if ((mode & ATTR_ITALIC) && (mode & ATTR_BOLD)) { - font = &dc.ibfont; - frcflags = FRC_ITALICBOLD; - } else if (mode & ATTR_ITALIC) { - font = &dc.ifont; - frcflags = FRC_ITALIC; - } else if (mode & ATTR_BOLD) { - font = &dc.bfont; - frcflags = FRC_BOLD; - } - yp = winy + font->ascent; - } - - /* Lookup character index with default font. */ - glyphidx = XftCharIndex(xw.dpy, font->match, rune); - if (glyphidx) { - specs[numspecs].font = font->match; - specs[numspecs].glyph = glyphidx; - specs[numspecs].x = (short)xp; - specs[numspecs].y = (short)yp; - xp += runewidth; - numspecs++; - continue; - } - - /* Fallback on font cache, search the font cache for match. */ - for (f = 0; f < frclen; f++) { - glyphidx = XftCharIndex(xw.dpy, frc[f].font, rune); - /* Everything correct. */ - if (glyphidx && frc[f].flags == frcflags) - break; - /* We got a default font for a not found glyph. */ - if (!glyphidx && frc[f].flags == frcflags - && frc[f].unicodep == rune) { - break; - } - } - - /* Nothing was found. Use fontconfig to find matching font. */ - if (f >= frclen) { - if (!font->set) - font->set = FcFontSort(0, font->pattern, - 1, 0, &fcres); - fcsets[0] = font->set; - - /* - * Nothing was found in the cache. Now use - * some dozen of Fontconfig calls to get the - * font for one single character. - * - * Xft and fontconfig are design failures. - */ - fcpattern = FcPatternDuplicate(font->pattern); - fccharset = FcCharSetCreate(); - - FcCharSetAddChar(fccharset, rune); - FcPatternAddCharSet(fcpattern, FC_CHARSET, - fccharset); - FcPatternAddBool(fcpattern, FC_SCALABLE, 1); - - FcConfigSubstitute(0, fcpattern, - FcMatchPattern); - FcDefaultSubstitute(fcpattern); - - fontpattern = FcFontSetMatch(0, fcsets, 1, - fcpattern, &fcres); - - /* Allocate memory for the new cache entry. */ - if (frclen >= frccap) { - frccap += 16; - frc = xrealloc(frc, frccap * sizeof(Fontcache)); - } - - frc[frclen].font = XftFontOpenPattern(xw.dpy, - fontpattern); - if (!frc[frclen].font) - die("XftFontOpenPattern failed seeking fallback font: %s\n", - strerror(errno)); - frc[frclen].flags = frcflags; - frc[frclen].unicodep = rune; - - glyphidx = XftCharIndex(xw.dpy, frc[frclen].font, rune); - - f = frclen; - frclen++; - - FcPatternDestroy(fcpattern); - FcCharSetDestroy(fccharset); - } - - specs[numspecs].font = frc[f].font; - specs[numspecs].glyph = glyphidx; - specs[numspecs].x = (short)xp; - specs[numspecs].y = (short)yp; - xp += runewidth; - numspecs++; - } - - return numspecs; -} - -void -xdrawglyphfontspecs(const XftGlyphFontSpec *specs, Glyph base, int len, int x, int y) -{ - int charlen = len * ((base.mode & ATTR_WIDE) ? 2 : 1); - int winx = win.hborderpx + x * win.cw, winy = win.vborderpx + y * win.ch, - width = charlen * win.cw; - Color *fg, *bg, *temp, revfg, revbg, truefg, truebg; - XRenderColor colfg, colbg; - XRectangle r; - - /* Fallback on color display for attributes not supported by the font */ - if (base.mode & ATTR_ITALIC && base.mode & ATTR_BOLD) { - if (dc.ibfont.badslant || dc.ibfont.badweight) - base.fg = defaultattr; - } else if ((base.mode & ATTR_ITALIC && dc.ifont.badslant) || - (base.mode & ATTR_BOLD && dc.bfont.badweight)) { - base.fg = defaultattr; - } - - if (IS_TRUECOL(base.fg)) { - colfg.alpha = 0xffff; - colfg.red = TRUERED(base.fg); - colfg.green = TRUEGREEN(base.fg); - colfg.blue = TRUEBLUE(base.fg); - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, &truefg); - fg = &truefg; - } else { - fg = &dc.col[base.fg]; - } - - if (IS_TRUECOL(base.bg)) { - colbg.alpha = 0xffff; - colbg.green = TRUEGREEN(base.bg); - colbg.red = TRUERED(base.bg); - colbg.blue = TRUEBLUE(base.bg); - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colbg, &truebg); - bg = &truebg; - } else { - bg = &dc.col[base.bg]; - } - - /* Change basic system colors [0-7] to bright system colors [8-15] */ - if ((base.mode & ATTR_BOLD_FAINT) == ATTR_BOLD && BETWEEN(base.fg, 0, 7)) - fg = &dc.col[base.fg + 8]; - - if (IS_SET(MODE_REVERSE)) { - if (fg == &dc.col[defaultfg]) { - fg = &dc.col[defaultbg]; - } else { - colfg.red = ~fg->color.red; - colfg.green = ~fg->color.green; - colfg.blue = ~fg->color.blue; - colfg.alpha = fg->color.alpha; - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, - &revfg); - fg = &revfg; - } - - if (bg == &dc.col[defaultbg]) { - bg = &dc.col[defaultfg]; - } else { - colbg.red = ~bg->color.red; - colbg.green = ~bg->color.green; - colbg.blue = ~bg->color.blue; - colbg.alpha = bg->color.alpha; - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colbg, - &revbg); - bg = &revbg; - } - } - - if ((base.mode & ATTR_BOLD_FAINT) == ATTR_FAINT) { - colfg.red = fg->color.red / 2; - colfg.green = fg->color.green / 2; - colfg.blue = fg->color.blue / 2; - colfg.alpha = fg->color.alpha; - XftColorAllocValue(xw.dpy, xw.vis, xw.cmap, &colfg, &revfg); - fg = &revfg; - } - - if (base.mode & ATTR_REVERSE) { - temp = fg; - fg = bg; - bg = temp; - } - - if (base.mode & ATTR_BLINK && win.mode & MODE_BLINK) - fg = bg; - - if (base.mode & ATTR_INVISIBLE) - fg = bg; - - /* Intelligent cleaning up of the borders. */ - if (x == 0) { - xclear(0, (y == 0)? 0 : winy, win.vborderpx, - winy + win.ch + - ((winy + win.ch >= win.vborderpx + win.th)? win.h : 0)); - } - if (winx + width >= win.hborderpx + win.tw) { - xclear(winx + width, (y == 0)? 0 : winy, win.w, - ((winy + win.ch >= win.vborderpx + win.th)? win.h : (winy + win.ch))); - } - if (y == 0) - xclear(winx, 0, winx + width, win.hborderpx); - if (winy + win.ch >= win.vborderpx + win.th) - xclear(winx, winy + win.ch, winx + width, win.h); - - /* Clean up the region we want to draw to. */ - XftDrawRect(xw.draw, bg, winx, winy, width, win.ch); - - /* Set the clip region because Xft is sometimes dirty. */ - r.x = 0; - r.y = 0; - r.height = win.ch; - r.width = width; - XftDrawSetClipRectangles(xw.draw, winx, winy, &r, 1); - - /* Render the glyphs. */ - XftDrawGlyphFontSpec(xw.draw, fg, specs, len); - - /* Render underline and strikethrough. */ - if (base.mode & ATTR_UNDERLINE) { - XftDrawRect(xw.draw, fg, winx, winy + dc.font.ascent + 1, - width, 1); - } - - if (base.mode & ATTR_STRUCK) { - XftDrawRect(xw.draw, fg, winx, winy + 2 * dc.font.ascent / 3, - width, 1); - } - - /* Reset clip to none. */ - XftDrawSetClip(xw.draw, 0); -} - -void -xdrawglyph(Glyph g, int x, int y) -{ - int numspecs; - XftGlyphFontSpec spec; - - numspecs = xmakeglyphfontspecs(&spec, &g, 1, x, y); - xdrawglyphfontspecs(&spec, g, numspecs, x, y); -} - -void -xdrawcursor(int cx, int cy, Glyph g, int ox, int oy, Glyph og) -{ - Color drawcol; - - /* remove the old cursor */ - if (selected(ox, oy)) - og.mode ^= ATTR_REVERSE; - xdrawglyph(og, ox, oy); - - if (IS_SET(MODE_HIDE)) - return; - - /* - * Select the right color for the right mode. - */ - g.mode &= ATTR_BOLD|ATTR_ITALIC|ATTR_UNDERLINE|ATTR_STRUCK|ATTR_WIDE; - - if (IS_SET(MODE_REVERSE)) { - g.mode |= ATTR_REVERSE; - g.bg = defaultfg; - if (selected(cx, cy)) { - drawcol = dc.col[defaultcs]; - g.fg = defaultrcs; - } else { - drawcol = dc.col[defaultrcs]; - g.fg = defaultcs; - } - } else { - if (selected(cx, cy)) { - g.fg = defaultfg; - g.bg = defaultrcs; - } else { - g.fg = defaultbg; - g.bg = defaultcs; - } - drawcol = dc.col[g.bg]; - } - - /* draw the new one */ - if (IS_SET(MODE_FOCUSED)) { - switch (win.cursor) { - case 7: /* st extension: snowman (U+2603) */ - g.u = 0x2603; - case 0: /* Blinking Block */ - case 1: /* Blinking Block (Default) */ - case 2: /* Steady Block */ - xdrawglyph(g, cx, cy); - break; - case 3: /* Blinking Underline */ - case 4: /* Steady Underline */ - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + (cy + 1) * win.ch - \ - cursorthickness, - win.cw, cursorthickness); - break; - case 5: /* Blinking bar */ - case 6: /* Steady bar */ - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + cy * win.ch, - cursorthickness, win.ch); - break; - } - } else { - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + cy * win.ch, - win.cw - 1, 1); - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + cy * win.ch, - 1, win.ch - 1); - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + (cx + 1) * win.cw - 1, - win.vborderpx + cy * win.ch, - 1, win.ch - 1); - XftDrawRect(xw.draw, &drawcol, - win.hborderpx + cx * win.cw, - win.vborderpx + (cy + 1) * win.ch - 1, - win.cw, 1); - } -} - -void -xsetenv(void) -{ - char buf[sizeof(long) * 8 + 1]; - - snprintf(buf, sizeof(buf), "%lu", xw.win); - setenv("WINDOWID", buf, 1); -} - -void -xsettitle(char *p) -{ - XTextProperty prop; - DEFAULT(p, opt_title); - - Xutf8TextListToTextProperty(xw.dpy, &p, 1, XUTF8StringStyle, - &prop); - XSetWMName(xw.dpy, xw.win, &prop); - XSetTextProperty(xw.dpy, xw.win, &prop, xw.netwmname); - XFree(prop.value); -} - -int -xstartdraw(void) -{ - return IS_SET(MODE_VISIBLE); -} - -void -xdrawline(Line line, int x1, int y1, int x2) -{ - int i, x, ox, numspecs; - Glyph base, new; - XftGlyphFontSpec *specs = xw.specbuf; - - numspecs = xmakeglyphfontspecs(specs, &line[x1], x2 - x1, x1, y1); - i = ox = 0; - for (x = x1; x < x2 && i < numspecs; x++) { - new = line[x]; - if (new.mode == ATTR_WDUMMY) - continue; - if (selected(x, y1)) - new.mode ^= ATTR_REVERSE; - if (i > 0 && ATTRCMP(base, new)) { - xdrawglyphfontspecs(specs, base, i, ox, y1); - specs += i; - numspecs -= i; - i = 0; - } - if (i == 0) { - ox = x; - base = new; - } - i++; - } - if (i > 0) - xdrawglyphfontspecs(specs, base, i, ox, y1); -} - -void -xfinishdraw(void) -{ - XCopyArea(xw.dpy, xw.buf, xw.win, dc.gc, 0, 0, win.w, - win.h, 0, 0); - XSetForeground(xw.dpy, dc.gc, - dc.col[IS_SET(MODE_REVERSE)? - defaultfg : defaultbg].pixel); -} - -void -xximspot(int x, int y) -{ - if (xw.ime.xic == NULL) - return; - - xw.ime.spot.x = borderpx + x * win.cw; - xw.ime.spot.y = borderpx + (y + 1) * win.ch; - - XSetICValues(xw.ime.xic, XNPreeditAttributes, xw.ime.spotlist, NULL); -} - -void -expose(XEvent *ev) -{ - redraw(); -} - -void -visibility(XEvent *ev) -{ - XVisibilityEvent *e = &ev->xvisibility; - - MODBIT(win.mode, e->state != VisibilityFullyObscured, MODE_VISIBLE); -} - -void -unmap(XEvent *ev) -{ - win.mode &= ~MODE_VISIBLE; -} - -void -xsetpointermotion(int set) -{ - MODBIT(xw.attrs.event_mask, set, PointerMotionMask); - XChangeWindowAttributes(xw.dpy, xw.win, CWEventMask, &xw.attrs); -} - -void -xsetmode(int set, unsigned int flags) -{ - int mode = win.mode; - MODBIT(win.mode, set, flags); - if ((win.mode & MODE_REVERSE) != (mode & MODE_REVERSE)) - redraw(); -} - -int -xsetcursor(int cursor) -{ - DEFAULT(cursor, 1); - if (!BETWEEN(cursor, 0, 6)) - return 1; - win.cursor = cursor; - return 0; -} - -void -xseturgency(int add) -{ - XWMHints *h = XGetWMHints(xw.dpy, xw.win); - - MODBIT(h->flags, add, XUrgencyHint); - XSetWMHints(xw.dpy, xw.win, h); - XFree(h); -} - -void -xbell(void) -{ - if (!(IS_SET(MODE_FOCUSED))) - xseturgency(1); - if (bellvolume) - XkbBell(xw.dpy, xw.win, bellvolume, (Atom)NULL); -} - -void -focus(XEvent *ev) -{ - XFocusChangeEvent *e = &ev->xfocus; - - if (e->mode == NotifyGrab) - return; - - if (ev->type == FocusIn) { - if (xw.ime.xic) - XSetICFocus(xw.ime.xic); - win.mode |= MODE_FOCUSED; - xseturgency(0); - if (IS_SET(MODE_FOCUS)) - ttywrite("\033[I", 3, 0); - } else { - if (xw.ime.xic) - XUnsetICFocus(xw.ime.xic); - win.mode &= ~MODE_FOCUSED; - if (IS_SET(MODE_FOCUS)) - ttywrite("\033[O", 3, 0); - } -} - -int -match(uint mask, uint state) -{ - return mask == XK_ANY_MOD || mask == (state & ~ignoremod); -} - -char* -kmap(KeySym k, uint state) -{ - Key *kp; - int i; - - /* Check for mapped keys out of X11 function keys. */ - for (i = 0; i < LEN(mappedkeys); i++) { - if (mappedkeys[i] == k) - break; - } - if (i == LEN(mappedkeys)) { - if ((k & 0xFFFF) < 0xFD00) - return NULL; - } - - for (kp = key; kp < key + LEN(key); kp++) { - if (kp->k != k) - continue; - - if (!match(kp->mask, state)) - continue; - - if (IS_SET(MODE_APPKEYPAD) ? kp->appkey < 0 : kp->appkey > 0) - continue; - if (IS_SET(MODE_NUMLOCK) && kp->appkey == 2) - continue; - - if (IS_SET(MODE_APPCURSOR) ? kp->appcursor < 0 : kp->appcursor > 0) - continue; - - return kp->s; - } - - return NULL; -} - -void -kpress(XEvent *ev) -{ - XKeyEvent *e = &ev->xkey; - KeySym ksym; - char buf[64], *customkey; - int len; - Rune c; - Status status; - Shortcut *bp; - - if (IS_SET(MODE_KBDLOCK)) - return; - - if (xw.ime.xic) - len = XmbLookupString(xw.ime.xic, e, buf, sizeof buf, &ksym, &status); - else - len = XLookupString(e, buf, sizeof buf, &ksym, NULL); - /* 1. shortcuts */ - for (bp = shortcuts; bp < shortcuts + LEN(shortcuts); bp++) { - if (ksym == bp->keysym && match(bp->mod, e->state)) { - bp->func(&(bp->arg)); - return; - } - } - - /* 2. custom keys from config.h */ - if ((customkey = kmap(ksym, e->state))) { - ttywrite(customkey, strlen(customkey), 1); - return; - } - - /* 3. composed string from input method */ - if (len == 0) - return; - if (len == 1 && e->state & Mod1Mask) { - if (IS_SET(MODE_8BIT)) { - if (*buf < 0177) { - c = *buf | 0x80; - len = utf8encode(c, buf); - } - } else { - buf[1] = buf[0]; - buf[0] = '\033'; - len = 2; - } - } - ttywrite(buf, len, 1); -} - -void -cmessage(XEvent *e) -{ - /* - * See xembed specs - * http://standards.freedesktop.org/xembed-spec/xembed-spec-latest.html - */ - if (e->xclient.message_type == xw.xembed && e->xclient.format == 32) { - if (e->xclient.data.l[1] == XEMBED_FOCUS_IN) { - win.mode |= MODE_FOCUSED; - xseturgency(0); - } else if (e->xclient.data.l[1] == XEMBED_FOCUS_OUT) { - win.mode &= ~MODE_FOCUSED; - } - } else if (e->xclient.data.l[0] == xw.wmdeletewin) { - ttyhangup(); - exit(0); - } -} - -void -resize(XEvent *e) -{ - if (e->xconfigure.width == win.w && e->xconfigure.height == win.h) - return; - - cresize(e->xconfigure.width, e->xconfigure.height); -} - -void -run(void) -{ - XEvent ev; - int w = win.w, h = win.h; - fd_set rfd; - int xfd = XConnectionNumber(xw.dpy), xev, blinkset = 0, dodraw = 0; - int ttyfd; - struct timespec drawtimeout, *tv = NULL, now, last, lastblink; - long deltatime; - - /* Waiting for window mapping */ - do { - XNextEvent(xw.dpy, &ev); - /* - * This XFilterEvent call is required because of XOpenIM. It - * does filter out the key event and some client message for - * the input method too. - */ - if (XFilterEvent(&ev, None)) - continue; - if (ev.type == ConfigureNotify) { - w = ev.xconfigure.width; - h = ev.xconfigure.height; - } - } while (ev.type != MapNotify); - - ttyfd = ttynew(opt_line, shell, opt_io, opt_cmd); - cresize(w, h); - - clock_gettime(CLOCK_MONOTONIC, &last); - lastblink = last; - - for (xev = actionfps;;) { - FD_ZERO(&rfd); - FD_SET(ttyfd, &rfd); - FD_SET(xfd, &rfd); - - if (pselect(MAX(xfd, ttyfd)+1, &rfd, NULL, NULL, tv, NULL) < 0) { - if (errno == EINTR) - continue; - die("select failed: %s\n", strerror(errno)); - } - if (FD_ISSET(ttyfd, &rfd)) { - ttyread(); - if (blinktimeout) { - blinkset = tattrset(ATTR_BLINK); - if (!blinkset) - MODBIT(win.mode, 0, MODE_BLINK); - } - } - - if (FD_ISSET(xfd, &rfd)) - xev = actionfps; - - clock_gettime(CLOCK_MONOTONIC, &now); - drawtimeout.tv_sec = 0; - drawtimeout.tv_nsec = (1000 * 1E6)/ xfps; - tv = &drawtimeout; - - dodraw = 0; - if (blinktimeout && TIMEDIFF(now, lastblink) > blinktimeout) { - tsetdirtattr(ATTR_BLINK); - win.mode ^= MODE_BLINK; - lastblink = now; - dodraw = 1; - } - deltatime = TIMEDIFF(now, last); - if (deltatime > 1000 / (xev ? xfps : actionfps)) { - dodraw = 1; - last = now; - } - - if (dodraw) { - while (XPending(xw.dpy)) { - XNextEvent(xw.dpy, &ev); - if (XFilterEvent(&ev, None)) - continue; - if (handler[ev.type]) - (handler[ev.type])(&ev); - } - - draw(); - XFlush(xw.dpy); - - if (xev && !FD_ISSET(xfd, &rfd)) - xev--; - if (!FD_ISSET(ttyfd, &rfd) && !FD_ISSET(xfd, &rfd)) { - if (blinkset) { - if (TIMEDIFF(now, lastblink) \ - > blinktimeout) { - drawtimeout.tv_nsec = 1000; - } else { - drawtimeout.tv_nsec = (1E6 * \ - (blinktimeout - \ - TIMEDIFF(now, - lastblink))); - } - drawtimeout.tv_sec = \ - drawtimeout.tv_nsec / 1E9; - drawtimeout.tv_nsec %= (long)1E9; - } else { - tv = NULL; - } - } - } - } -} - -void -usage(void) -{ - die("usage: %s [-aiv] [-c class] [-f font] [-g geometry]" - " [-n name] [-o file]\n" - " [-T title] [-t title] [-w windowid]" - " [[-e] command [args ...]]\n" - " %s [-aiv] [-c class] [-f font] [-g geometry]" - " [-n name] [-o file]\n" - " [-T title] [-t title] [-w windowid] -l line" - " [stty_args ...]\n", argv0, argv0); -} - -int -main(int argc, char *argv[]) -{ - xw.l = xw.t = 0; - xw.isfixed = False; - win.cursor = cursorshape; - - ARGBEGIN { - case 'a': - allowaltscreen = 0; - break; - case 'c': - opt_class = EARGF(usage()); - break; - case 'e': - if (argc > 0) - --argc, ++argv; - goto run; - case 'f': - opt_font = EARGF(usage()); - break; - case 'g': - xw.gm = XParseGeometry(EARGF(usage()), - &xw.l, &xw.t, &cols, &rows); - break; - case 'i': - xw.isfixed = 1; - break; - case 'o': - opt_io = EARGF(usage()); - break; - case 'l': - opt_line = EARGF(usage()); - break; - case 'n': - opt_name = EARGF(usage()); - break; - case 't': - case 'T': - opt_title = EARGF(usage()); - break; - case 'w': - opt_embed = EARGF(usage()); - break; - case 'v': - die("%s " VERSION "\n", argv0); - break; - default: - usage(); - } ARGEND; - -run: - if (argc > 0) /* eat all remaining arguments */ - opt_cmd = argv; - - if (!opt_title) - opt_title = (opt_line || !opt_cmd) ? "st" : opt_cmd[0]; - - setlocale(LC_CTYPE, ""); - XSetLocaleModifiers(""); - cols = MAX(cols, 1); - rows = MAX(rows, 1); - tnew(cols, rows); - xinit(cols, rows); - xsetenv(); - selinit(); - run(); - - return 0; -} diff --git a/.suckless/st/x.o b/.suckless/st/x.o deleted file mode 100644 index 28870e4a01ec1386168be51b6154e881ea896f30..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 73808 zcmeIb3v^Z05%_)b00LreRMe^+lAh8)xWzi)kOeQR~= zx#yhUo;`c^?B{uKvMM`wYHUo5!y(4`uG4kK)Nztd@1j`-XE~=i$?Fn8V@ zH}X+JPW0>O=WfH9m?*cQU~yZ2H?rNIoB6MBFmqXQ^}#@W_T0=V@zt%__0@U7`ia5l zpC&Fpu{cn)0GbG&PHe8p~F(<)f-`2+z*NyB*Sa3yPVc?3um4QXu zF&lYLBv9awS70H?xJ?J)=!kMz$QW#qzA0@ldPjaHo zg>HTP&{WJT8cazT)wnx81q)HMJD3u0cYQlL+`2>IF=Uuqe{9L?!zl@c?wE9=$X7QO zYzL!~w>L~{zv$nfZAFq>(BU>1d)@khN80|frsD(PzFR(o-{@|)pxG^XA@k;cWw=dW zp3`*XtYO>Sf^BX|<9&bl!})Ggi$D5&(-&tA+tV;GE9S|qki>~L58D$yuBfCjHML>$ z`0-$AG(E*Fcoo#xG#~n-NB%Drg;QePg55IrFPpwRYglu`psbh|Och6u3~T1x*V^23 zQ{vs|yWT!(ef67;jy3AJk<6sw`p&DZcKCZljxUjbeC)<*chk_qEOrG-MUzf3mK)l(Qrye z^wsFxlwdSB#SKEu=XPrc#iJT+w$VMl2e-wUyo-GQKJ=;Y{Ps_MCxlZ51*46=2MzCx z?SR-L5PN+}7RY`KXCJ~@Y06+1+R|dIG{B93O|nmY^LImZFeSOJ!*}x_Xlpsq!wu6q z&}g?I*f$B=qaRWxVakB1bZ$Y>_>{+>hj+Bz0a4JNQb=(i0kj5FvaH@fQw&{UX?q-aim|KNeVw z+c0yz+wi>%*B5-=jSfm_-3lUqLv0Z=OhAt;!p@6+i`1k#zS{U$Je>r7nFj3!D$=cA zm1f#$PIQNDuhH!okrs>|aHG2kz2VvcGB^5xA&;5iOfk$iH~Rh)xHnQP?KXr{7Uwnu zQx@h%-!)0QWZdXZ+Xpg@n|uxRXmEiYKAZjpHAy4s0a@j4G|nwB2}}dCjn<81Knbf% zdL0*xRwM+Y#RE;#Nt3EL|LFEeBh7i8c0-G@&S@*ijT?&P%Fcwb(O3Hb9KvK4lWC0e z-TX7~drtJtoTmNpt}V_b_36j9)9|A-qD;*UpPf+scyuQ#_&zAm?Kk?GdxtDuJ+zV7 z=vCOH7Mhm4{g{@#(6nTTl`Xk-)Ylyy*_ov&1ATSVpuI+Swykxcmg_6hOr@AH@R(|u zlG^$rq>D5r*?H0}Xm^|5hvJ!*MeV}4xpf8@Z$Nzp@mBir}L{y=0$ zjIGJ&wt(-!<~XzD0>#Ne--D*ZW=Ee3#+k_z$h)xZ4Mtlcd;1hcj`Rs%5P&fvd^~r< zx+T!l0mXy|6AGUMQ}c0PN?W;0sQOCqK|*v=%HiF>W5So>to>c6uOm$@37cU`48kyf z#2!olxECgzV9F%7VPQ&z+mM@*kpnG;S1ZTN>TM@ehV_F}+^&uVWrnMFY|1Um_t$L7Z898EZb3LSxlMH& zN@3swk7eaVTf9A16uCZSWUQ|?7nDQY`NREPaB#E@hAcQu3Lj@Zk>L8W-vW?@`|P@o z>IuQ2j>lH^b&5l-WFS*?c1?gdz-vL2(}pim;xx%~dMT+~sR{7bEMbO+zF8 zBna&=If!eKZi6m1N7pVQFM-+C7=pDovnUuXOvy!ux&gB|4@M6Lquw?S+-qBCZ959J6Q9yBKS6Y!&aUtlWZ2q{B6haI z{uT3LYcAmW5s-oHS}`xT-U|e~MbYyuo&AE2O~;_)|GHc?eyQ!wU}Z%v*G$<3&ZS7>wqD_eX06#crVtB~mSx{t#n-Fj#BGch>dN8jjC>YYw) z3$B1U%~`$wo3>_i9oTh!8!%H`ZbP4W{T#BGiwoLQuz{;w6-YG6*gVV!p9uID9BNCEi@sy7dcFlA)&$rmN5QFq6m*KX-Fjotn!o zd(#4{nohxPQxW#=z@u3F1h*bMR$r4W{bYpT{z4**wegujSj{y&4~=v8lqB#8%x9*b zJa#%9(-OR^NwnSzg8zl$+B=QnK~nJp&m1Ik*e)Phag%037rsL4k{W7aIG0H{A}57VU17`Uu_Rpi@WzIVc&c1~NBla1gLJ z99j;-;08^r?p4r>*JqHE84Y|te49wurQq;JyJYNsfyh+l_*fyjy}M~5-)kw~=Z-DU zV|%_eWfrCUV-G1I;kV@3mr|mmJDrD9vzUOQ7#G=*3WJVmU1ludT4@boLxI^-->qq# zM=dBxHc8F{TID&WR+69sjCbLT_V&E>s$SA3(dCD0YoKi_ofo;jvPSJ?JD#)Xtvwo? z#2K>Up1RX9M#K8M$T!j~nSDQc2j{%H*Ws6o)%pya^qAdgUXGjK6|n#!tluLaWpwK= zsW3Xz{V%xH(E0^ti7g%$2IejVz#|`8nEY*%0^HED-n8Qr-1jSUqx(z>@~F+ft}%Qj zj~^F813PB?pd`jEl+q0RMlZD;Q{8qZ;VXsHTwFat+Z#wj;*YTPMs{Ru&$Tsp6|`k& zov?s5sq+GPxpYs#?jB2k`!QVkc3pK$mN>-19UOj+40-PjYw1)j+c7bboYua&%5gbk1PAevAtT-8IL>z*UA#hZCtO zW^~FxYeQsT+7`T36!|hKeDW4p!@!Y0d|c$qn9~{~dyhmK`$wm^X3YcbAOYF|cp-0& zt$~24>aGgtX%U5W&yxg=R}WoYZuU?yN#RT z8X*TRrC@y?IdWEbVC2Z)>WgO7m!_n)^}$`n3|L7He;nKfiyl}Nw_XDE!0mi6&|`Y4 z3tqq;7f(H|6%#;@<3$KQY5WttIGz`)$LDfh$&Aj=lkLZb2As zY5N|8+gEPWfjD~u&s+x)7b{m7136+h&;w2;Mp`>Ta>`~5r7*FgMGEXfWut4*XH-tz5s{D z6t`fZv0@=w(H7^%eAU_kad!Iciw4tL1rD9o<6zkFh&l(NdiH8-94ItyoCC{!ueKJs zF2kwz!Kyda&D<7FaG@WDV;8`}^qrPTFtN3LW>#?-ph36Zmht=X{u@fpgQNO6dG!+` z9eu68gSOWP4mUb57F73-boBAnRzh2;ubSI1vm>%=Qsj$R->l~%9r3=}*P)C?-~%`i zDYM|VOYQd|5vlRjwpq!?mcxGQ4EXV7zv4zez~N$cu)&Q9He|z%WHTvkO@-*LsR1rO zB0J!g`XLZEzOW5U^^Fs-HJiS%J#z%4cf#ZBfO)tnX*Hbv<4{vlxW63+;)YTgjl#NC`ny8;D)OkBj3dMZYqG|w){0; zzYPMuTjs&pMfUI#ID`zL6j0tFP~IUumv<{RqSk>R-Idd$-Lix3-4SdUcpJz$O<%@E z_Q6f%=OXXN_kSf2c{gct!=S{N=UZ>&HiRzDt?%dVZh~YT#;>{2-S*N7E`m%S1Yo@y zfjJXr!o|AT%!j9VFv^?-S@Ko0ThzGt2rS(W8f!_zAm zgCU)Zrg}AGH@Xcjkuu!O=I}h1Zot>B$BbYc_0`^FvaEs|H`gWOHDHpRMWDXzP=lA6 ziCFv4BIb+Hae^VtL-XiaiA}chX+5Xd=wTbG=$Yp$-*S_vH7)L z3%>I#K>eIFS_GM1IFL^=w^d(01MQmxsgJFI=pOel+$3m$%iHPT!!E;M?Zh@E^lVNW z4o7skqSsQb2QiW^AyXjzF%wH`GdB)++8*O0XMdCsxwwCLINcb-C(C_Iqo8HLw#dG( zqo#u%+TpK;i5{BDF|ys})_V?gbl5T7A8t0zqeo3$7qe|!Jou*VU61ggEk-{N)-TNq ztl9q!SRUlZJ72tGmU_7ejH6l%fV<~A;qLjzfyhp{XY)==1<*ir2OeGohtb{9R(Jvf ziA?e8KCV6q3u9;1?SaR!Wwq^g>lg5I1Xy{$-SU&Y(63*`&UDx;i;^2*ROx$YT3$@p zXD_>^cYxc`D=jyEiQ_L0-p9zEW-9}SUSn%@-^g`yVTy#+O)jQ7+!=oqO4`$Mw>csJ$KcLQA00Kz<||#0)g92$CN7FcmN%| z>TOdcwp6fE?tZ;`1Y8Ts&25zJEeqgcsO?P`DzCFMyY+)05#BV3HoF%XN5eDo78w5U zGX9|JJHOF|ZewErJ&9<~LEn{TtUl4r`sFaLe+^gB<~g^ioB*^Zn0W%x-GMc*28eZr;)_L~d0>|*{?oyf zvsPc1?D&4!*fwtUb*GtwODJbw?RNCw#0#nq)L#cP*@2d`ZRYP}!y6)de2+{?ZfxmS z+gSY%c&rib3wmP1J79r0&m_-jj!^ixhuhE+uEIYzb z^V`F912^mf+ryjo_H*NYh>C*sSs7%Z?X&j6#kvnxf>ynBj+8c|kiCpM4tk?o@Ttec zVyjgRoNyCObWeDi3oTjnPn5O!_w3XU0|c!2IU-5ZZUw}V`hO)a#;njhP4 zmK;oRNrvqgTOKUTV6l_ku=HZ+N7=rd4+2d``v%PmO-(JQ2O0+Ui@baBuqHeI&4?bz zdGURC1QXvd`Qo}4d^y{ij`WSjr$n3KK?{gC42(6>IWN9v-t>SDftRq(qo2C>@D72UvEaFK*Zl*dxvsJ1E=*hpiG6iu4Tvi0K*bic!j7?afFJY(vy>;5xNxaei0E6Y=t+oLX+#D*VwPzlaY6S z8)>T9g7cJ_!ViBEZFIp8>(-%ucs}i~JLIc<2u7RMIv}0>;+R{!`+3HB0zGZyp!(rq zFWfwbD7sAm&t_Zqz-gBdz3Q33rQz0N*7r$f!O=Ph7CJEd%}q)34V#OL{;rh+jDOi~ z!$#Bg@&37pG7iIu66|r{_P`#upw`Ic)W`3Jn||THx(&9|bTqf!XG}9;^9D#)?T-rsyGX2lE*Dmbq8gIA|u0d#I?f$M$w|A6Sd=*Tmw#H|n zVmh88j>n7R3*Z<%^QrIhIBpIb-|Ofw&B2bz}KJ#7IGMApnTNj$wsbXG(H&m*i zTfNTX`?2@vV1?BACaSrZD*K z7A)Ep!mC~|Z@{bLubJBcbepoPydG}>&~y#{>!B)PLbLNBHmZ!Sc8mA6v#4cR(<6Ih z%-b?x zvFn=27kX5OJ7kaBzZoko&24(u4|i|+KV@Ea3j)f(7wQ(`8|Ux_9F#e)<&D=MNx@Fr z=z}SFfnom!k@@CI&vYZJ^Gmn?Hhg;&vW8bs1HK*4fGN2temI9&&o^vv3cR3~1n-_h z;#7CY5ilVa&YJf5VTJ^AuuBKw9pp6F{N`PttRTEyv(S_hZUmT{0nnd<@CxFj8&(A~ zzy86s@G@3Gd{JP{Q71n9PJJAxm|s670d`5j`l^BWLST@d=SR1}>j;r{f87f~U-sU* zk8cK~a2k|blxqsxV3qx63 z@fJqi$E$GpUo39rLqSuZ^g9YoB^2Vaqr4RK4;!8WMTQFkd>=+s!c(RaU=n5@Z0p(o zm%j?5;H_e13k?eXO~j|@Y@BbvYrC_f&*auWl(HGxLbvV#cm0C!o-SCzuPMTvM$?t* z*QH?33P$&t5h9mfK(Q_O-qpUkPqAGE*1Xpcd#^t{GZ1;!AFMayQ}k`}DNOli(5*SR zw7xe9Mk?R1Qn+BkOUSlaT~V7I69x7982%%cwsfJwSNliE$WD9k_6S5nMTz%SL34s> zNA^k&UZgYkvdwrOInvj6OC2PKr$oM6@P1+BNbC()2W=r5rpM4bmUhUAw8Cqd54J=< zv@c=e2;mPridTU=t-+yL1J3X4-M+dbpjA~g8;fOWoe3=h*0Foqj*A@WSKSnNBJNy# z`w6BVcx~Lc)^?9ia{GVbM)9rma_A$_Jzj?HG1oK_7_P8;L|?$Jk?am>hlT;~07JjX z?Z4Ob3)8fG!wR9T;CeW>0oqs~`dYM=ZvKAIazQf;woQTP>j8Wh5oEBrW#DDX881VV zJr|7NYf0aOuiLkwqp$lOw3m6^T(U*Gq_qVay!9z2G1oNKrO|D^2Vb`CM4P~!;1Va; zkkz3snkaP>T$)nK4}A}s`N^h+JL(j zGgx)nc^8%jS2;B91UT4e7NZF}r-bX)&}hhG`83;rQT_CagOs9jlycKNGg zOlIU;NDSEyG^+%7gmuk{ZGFtuo-L5Eay#|>uuW($2K(crgjv&F<~1MTmA4rCYEy7x zgohr+GIWXek!|pOI}2Qv5Ns%i9y~PxoDGi-Au+AZR%eXx)op`w-y?Z3wJ-Q;AA#f5 zUycaFI%poi$uWP!9soDH#<|g!w&km9Vx8*!Ze(Xd0IqRb*Fs#E8JGlDZoWt2Pp^f1 zzLNm;)-rd}IldqEhePwYf2SJNCkJj?-gS!}G!u-1=;^Dm)AoqGGl>+G0pU zG2AT3GKSaT*wl7vi}MsUF3&~V_l2UU&Ctw@}&}6O=78Nt{`dc8a?dG zHVu!OpE-uBcGK=`otyObUW;8q8vrx(K)V75hd5jU^mQKQFRmJ+yN9$jx@_b47c8ZxXy* zT41|()4qOWx*K;ps;r-#JnW#&v|;Y(Z|uVtm`x#twGr=|+1%W?8%>I=!Nc}Y{=uYg ztaa46#03QY8xE_%?z&d*bm`#RbPl*tim+wS+fFriChX;o^*_EWT77a)?%A1ZC;cAc z?6~l~;?q-MbvpN|7)L9 zfnxTU%xaQe{7>#a8h7;IOt?fdEw-D1FdkY__&23yco%QLaGmv#tl%rR>4SLFp16$N zZvys^hN_a>bKUD^WJ|)9T?h{|x88wWq?_;H@(P}~9&0l^EPx9i+Zd0H#wy0w*^}*C z?Wfz_V8{*9`Fl?X)r`r{x4YpuPMCDCh*5WnO#ABobW?id|5>zse&$(9Z{=`xVp@QG zYCNcO3ipG5AH`?(J-Czy>sWgY-f}xu&atjFeW_a=*@;%%xI6~1YRI?i(9`+|yg^ad z_}E!!rZ31v!8>)h=I0yD^!`NvQS=?tr;Ow61#st$1HIAnyjnjz z@*i1~gz3-c^#6hR?||NxFSd1>-)ngvzyy2#iv&Nw1h@Ye334$(QD*{bQ+A^5s<_tI z;0|xg#UR%D92~)#qV*I=$a5UupBl*Z6u?NDQR-d@!=*rM+rQ+Up4MNQk^s(c{ zXJk&8IO(GAUYr$}JSCVtb())V$@JVAGiT*}Z}yzI^Dez?{`W6mu<(j27cI_TQcxHw zT3TFk)w0s%W#tuDS5}3qS6oxG^4jaJ|G^FLg#*XQLjz#(g;!+>_7I5)Q*v|iCeI4Y z4mt(J`IS|nuv1l#Uz)$9G~`s3=9h(?qVlq^-yHf2%R^QEvhuKhd49N{cz|m)CRAEgaxLV5KZT*HaAo<*Dfy+POY#erIaL+q z;oS0q{BTKm8RTA2U3FPWSz-A#PC;o&s4Pq;C1n-W;W^=zU}w(EDYKm^r6m_*{k($K<*=KrAua5iIT0siOA4a}aFJ$DX}9N1<# z7cMC&yRfR*sS1^b3c{ppf**@eP+nbHXk0oRB2%~fX)oe$nHGJ%LzL3=1$F6G-vMY zoSD;{p{3Q1m4^s$%q8$MYT{@YcDSK`X4Sw!0QR_5!trF-;}(WvNHhz#6>vNS_OLA& zJ8?9+bu9j5!)A4koeGE3VFUGen2yootmrsA9&L})OW`=zoEr`4#dwSYGhiDt(ZJED zXpBvehldb9>nhlm!8XG=9)6d@Rt8%+Y!$Fw4OXyYY6HVRxg^hmgST z((P`n-Ho%m@phMiyUU$3j(1i$XTbm9dE=dNW;Y8R10}S(v357k?#9?%I_zeB&x+aI zSi2i%cVp}>9d@&3TQR#EYj@-9Zj9Zf!)_LK7|7J_#@gLDyBlM6#==q5Mqtn9rVYS3 zR^@_GXXJ zO2OW?%`}oS1IB15AnJs@ZH=Qy3Tn-8`XwoDvtvkV3`v-M2EW$ukl}F z#_y6sh*&TJq6^BGFV8P4^e@P-Tw3KHIdbF`oqGO1wr51CzqF(*--$a&6W=MAko4_xv< zJnQ$2C;dO|MZZ@(>-UT&{de@D-z%Q=d&ZOgpOJn#?N)_&!tuI(Snw_n`#63h#pAzi zlYy|~70>o?JfE`;onFWDIj8rs|L3HLP%FFew-KCgJTpX0s!;}x&?MA4FY$lsCH~L7#Q&w2_=kFlf4GYcKIzdWnCum-yKf&+WywimqRd z$LB6R1&;6EQzS5s*ZqgZIo>>5WGMC=KTt4v$noZ(q>P|rj@Rv<$A8awa>(3X{KN5H z{^58p|8TsQe>mRDKOFDnACC9(5664?$1|S%^SxgD!|`7J;dn3qaJ-j)INr-Y9Pi~H zj`#8p$9wt5GoJi2rx*WlyqAAC-pfB6@8utk_wo@(E-|8j) z?Ox*F=_P(|FY&mp<$I})2;AQ=-m7n+IL3Qjn+tWGhnOBCmwDL9dFOk?vz=b?tlul1 zZRhKVE<;yejK>_$*8w8_2^r7VVO{a|h|^Ee5p>S+m-Z6>eUEr9e}RX8SpO=Icz>7P zEco|=rp%7h&if~9bSlJkh1Inq1Lmyp#w@$9w0kFN_MvB=4LH-Lz$HgfFLB8oS1-RZ6`g;a34%eiY-b^7!HT{Z$vQ^ zE_JG9XKw?rLF(Gi=w8Vt%yHHOC$)}(^=vkDQVE?TF@r}60 z)Jq0nIuln)akD7R4K~Wjg>(FR1BLKnWy{5DPa&9t*pl66T(?3BcWEI6(uel_l) z>^EcGHelVZi-#(_3F|h}H<*|gw<>0Usnvj})xdFR#~#u#5&yx7v?D9-K1kGAfhhYK z$^O9BQC92$QxQ3d?$pFQH*x;t#KkKTvm8@re)Dkx1M9HYJ~NgtlCqqr#f;al5yMeL6~ zYAkMGOk&Mc$a`wyTG-aZc1O(Q#7!|Ji4~?ypHV*MpZXZPuQpa-nUdnsFvsbKHb_6_ z#r`^RejssiVPamf@F%v7Af*|F3874-&^FTHa3S%JgnuexpJI1hZmldxtcWd7teFf3 zUzu2eW6lOj_jA-YGcl>cSQonh)H)e}9-{c$Fdo`V1GblSanLAk!uAsB6HH8tEgF!R z3Iuww?fldS+bJ#fd9-yg`5)_MAPv^p7-#&l+{BvpPjO=~4$_r@G03xsPa_VdB%mMd zXNTZ@Wh^^bunxzQo6yEcJPuHE2N>HR7f1lMVV_+`x`zBG^$%k}T&zb609{y~H%Z?J zIJV_khAuSy_6Wt{x7)havmd^D&D{l}uq-L~Paq*Yg1&>Yd`r88u_Vs;Wj>`j9}`}l zm^2e|HtnDIZ;=}x$KHcRVca^3n}cyMcCMhh@x$r86o-DY>2Pq(;FBLK&zd z%oFQqe-9kTfuqE~MtN07v7Z|!K;<}&AEHy?Uy@_E9e<47(8yMHW^ZJ2NoJFf$&204 zDU<}ff@I*;aucvTwZy*npVa%Nn7B7!K+8CGdg~nfUm|@gv7di?eFb4yh^zb-edwnY z@BxM>{(xW2gzBG`m^v*ntp>URRg>{`9?4#TvUxDSPESn0x-BL?9XaR#*I~0aX=*7B z+nQ5k=N9zUM&d(|L%_P+OzcbqFg9Qv7sKHm;+6QI3(9=Vm@qAIb8K8VE^$k2c48y+ z(mk*>L#KQV_IqPvb09jl4q~T4Z0t2CydVZnr{d{mIK}+FC15v-cRuNHl^2Kq_^^i% z=g>f`;^IJ)4%l5|g&v&eG2nBx7?59$ClFHY&cSsT%Cj8vQ;C~#Ovs}@QGNmJG4Q$# z)1iL&v4Qp%z!}Py5*|YPal{+woT|ma`4i>Mm<`9gwwp{G|KKHtHFoFBggsi1Uzo+P z*!K+(?6;23#@plcEof@oJ{{cP@IF^^?xLnD^**@l}q~`+jjReOrYeUeJMtncnY1Vqc@eiD# zem@*zU_DvH^DOS<_!mLY&UMK)0^1FSImFwEo3$VCD~PW=!-zQ-*_~5N{1M`2od$7X z;Aqc4lH+>Ea-iQhF#jRRv)#<`+7In~+zhbrnH)-Z2k`^M&6*D4?k4`mSys@j+2HW^ z#H;<5bHD#H@c~0DA8vOJeun__9Y;Ky_;%vg5$AF18R8ERpGNYp5dWOGS${#?Tf{Fp z#|oM?793(fi!r09<*Pvs!+zLzjh;^DIMhOVmJPLXW^DzBpAo-LKT5pyTq|$Z zPxo##C6V^-DBfKJCQMQ4h>ck!f~9rW1^tp zJiB&}i*ruu4(QCTsJlm%L%F){j&;R_4e#3p=xC+kTq|ACHxOt4a!iEu^E#FJTH+@% zpnZekeLLG9NB@N6X}DtS;2+dQ`+3~wh>gUDFrfXfh)+@ccZMJ9&p(oUm6HDp@gFE| zzV-+9EK>Xl($DRQqjwV59(|fPw{w<%p7Ons0qtKkys!CtU^410lK+W>aQrXgzf`=H z_T_@V*YtCo)u#Jgz%rFxQIE{_zu_+J#iow)fpo=|ibah&I6xSROvivOPYdx}3u{1e3=CXVx@ z41Xhz>ns_Xh=&y4O}t9+XNluFONTf|kDiY6BFO8p1k<&8@B<$FkOz;20lp{wCwuTz z4?fC+kM-cwJox1vd>QZ)VCGy$PJ>eusy*b_ko;ODU+*D*hX=pUga6HgKjXpodhky? zczmCp>m}KPpXb3Rc<|{Se4z(l>cJ~K_!zCqCSRXL#^j55CBQ7kTjI9(<(-zsZB&>B0Z#!5{PB&wB8eJ@`Hke#nF4 zr9w~s9|HUYShugH`$fE7JKsZof(M`O!SN5f_muDD9=y_n-{`@k9{eXB{5Kx_0T2GD z2Y=FoKkvcc@!)M9{BsX})Pv*w_MZGa*n#lr5-%&!B>0mn>{$* z`|imPzxCi-Jot7G-t56&_2BP#@DDtAy9fW;gTL%C-u8t{^q%~1ya&hol_$dejYPUJ z!&lTpi4ReH1o3p@U(-5byodgv2hSxvqefZ<<~b1@UQT=u@lB+^&_mB^4_@!VHgZMtXLVo-0Vt zXCCq~P!O!gv+;ofgz0wYoJ4#%@igLtiQh$h2yy&|6YAMYoNiw_BZ&W#xOr{_Y48Ju zxc=Ng>0#MsLI8MA?Qoh0|DFdAdGM<|_|+c#1`pog!8dyFyFK_r9{g_}e76UG*@OSf zgMaM7k9zO{$MswCGhQ6^Yv7x<{D>3Nma>`lG2j!N~gMPd47dsKGW(fFR!i&3%7OyO1a?kMA$;jgNrh@E;L~xKO}M&pX{fAVC78Rs)7+BfC1q7% z^C7t~NMYWXgOguXP*Sp}q7ptQXFfvP8B+zuhKvk-HMS_fx-@L8!mr4JZv4uuQ-dF( zt8&UK!i(@zh33F~*3cYk3778ZT9bD!@HXdCkaCW70=w zI_0H>@B!B_e22DtC89BD4t&-QKdOgHDnnPpNB;2pxA2Wx>t^)nlI5Trk}Ro&kM8Az z3jo4L@Sv3V#Xftq1ilelRpl(BT?l+rUDoLX`23$KV`VjD3kSHMw_&wP1?~W?P!x0>IU+K>cC%Lq)Im$(oNcQlPcYo*Q8Cik|ym~BQw^-jx~~Fjm%i1f2@@;`p2479LQZXYwFZF@WI}>fyueqi=ZEcu(yOO3#%)f`8n{-$I7xm7-|r|C^?6} z-#F`eD;{>5<{oJH==XO3PUx{)B^e}X0W=Vv;@A^90G*VV6FY6W!HJ2 zun=Rq(hvaAZftTnjBKFAeE4`y^^z)RtML6~KtXF-H#QZ93(znFzeU}{=Q_JI(?>T^ zv$d}a7#-N;Y-gHF%3sdqZ~#|h9vD&hH$tvm3Y|<*9=!!$DPMhu_er(9a>rfgGOb}jGV54 zvhyR^n2(1~tV25M2bu~trlr_yio$c@>zmp50nTvMyl_zle5SJyO6L?^Q(0mhjg2J_ zzXLAahm}IL4K)`mug0%~OC&OwUs_bC6o3VkWo%!?1D!vxP&HaHX?6w7j6W5>s0zVdt}nIz_6=%!E~h1uHAe#*eGA`c9c57MWL8 z1ywR-s+l#g;`tc5u=zG>p|dm;h7(XxUbX^^@9h0BgTOLi5k!Yepi_ekbYW<{^OrBl zEiYeIT`>nfIbH@mwcA-WR5Z-L<%MP#z*ae>6sP8?1=NMOKR`=2^Ui#1fq5k`_rpkJ z>xR#4ySJW0mv9QOvRfKM5R z;TyZ=>+?|XaA8TMId|sg+3(jn^QV?pR~17S?3yKE1%vZxL3ssCo{$I~#!DE}56uW@ z`Vthx^lcO~3mns4@XStz^U*`tGqCN=EyttIYO_7lj-hixrvG%t!DtQMH=Wt6fXu{f z3kFlEK?T*7RWNqc6v1cRtBP&Ep8_TaV7xKiU}`xu@5=IOSW?u$8U{apZ@b47`wd}J zwu*8XC7_+#g#{YE9Bb2*hcZ|aDlLTtA1n~Dtk>k1hO1y10Slu1B{;r85**Q>-(rDs z&Ebsl!jhtrP$l#XtP*Hd6(xm^>7zmWU3v%lf|(QMSY3R2X-QReK8^}G@Ui+%A>(jt zGcaU^VI-eFyS$t~*Uu?~p^7k!*0xdEUc_h9ra(29l$LgmraW$1VHPv(*A^O;!dhqw zota_B7=fdwsc4RZGG8_W>MdLjxC%yK@CViXjQk3=W6sL50`ydf03kyNxau{X7-z&l; zg-|t}ADwop!^s|Y)?cUO2h#aE#aYjfg`T^Fo(GgXmy4fAYyaT;Xc*W(EWbl>wx?Nfw&!Kys9)Oi zu7~{lLjD(!hW+rdlIMKC67u*?2+Q|J!y&N!{BK@P5_}=Vv;3LF(aueRUm@f%E}rr$ zRJ@eM1)bSI?WWR%r?f;#Uzn-KdddX2Nh>~zOOjjb0u-K=Xb)MA|=oE zT^(IAmkqw@_!Qi5yAf^^yCY^Q^;=;{NIBA zRq#{cxf2E~@7=I*c~=o<|L}N=&)+aG=kfM-#Wz?n=U0laC*GuZJt~ColH&NDGlmZo z-%flCJP*OZ_VBzsL2;fR@%fdM=XN!fIQrq&u(2Pm(DF2Y*}t07l`qeaw<&p^AMyDT z2DX#se=qnw5YPD@794Hja($yXm#YsvH^9)T-<~@7TnK|M*9lsl%5|#XzlC&cPe^cS zPpRT;PnF{AXMCQ6LEH07CBM;%IgcsM?W$36ZdZF0Url;mP<$=%FBHF?c!%QL&W|5x z1=!DQ&q<1}COxMq&UT)yINOu1crEG4Q=Id?RB_ICq2iqHGR0X>L~&lHttF22^&8l@ zJ#19+tmg^ESB`J2Gv5vf^z2S&Fm$Da6q~(*6sT z{5puoFkNwOpCQG0oxDnMuJ>Bvy8Yau2|bNM&&Pr{2_BaOClI*(vz`Rv z=+Ditu|Llh@=F9CDde{aK269M2tLb0{z@T_m#~9j!?0d(w2AG!P3YN1B|v}f5_}AC^iPrCS%NPW+!cCu3qDK87YlxwhkT)sm-&gBkdVvv7WqV5|uJcXtkRLAOWxnHtJnG{6DwhZ@%R5`> zk>$P8L%vAJ%X}+@Jmwpu^41CYKLKYyKPdQ9f;S00C4xUKp;o~?qHihN%X z9OdW2hT&borTuNhQFaq-?Ef!4?$Gu z404)hqe?zV>yuv#zFf%vPUv}7@V^N89fI!=yiD*Xg&sM7y)5L>{<&n&n}U}MJ#9kI zbAlff@~DT`b>ApHo%9bm1%x1QKjrmmn&8q869lh-bnMT7;8zPiL-0z$=L(K?vj49n zj_e;oKBVMx$^L4=cL@2L1g{djUg($pzejM{u3i^<&?atIZwvWmq5r6o9}ICA`r(IV zAfTV0hmHMqqTr}M7d8xM3NGc#h@wz7L ze?b1vX??qskr#QFIg^F{%`-r2RF9|z868vSsPoob4p?|`H`vpf`?1!Pmk!=+6 zR|xqXf|m(iEqImCBm2uAg*@i5h5WEZ@i6fnigP_atvK7?OdNH;BJ6on$#cFRD9-tQ zsyOHSrQ)1#2XUQme|nFA+b8QuQk?aiOdR#RD(oLA_-le^Dn0lt97BPSe;v-bJ(qaM z-zel|JAYZpv;A)>&i216^t>VRO*;dvvg6na*t0!jiDP-CozsQYIUGNDKMxms}9uj_=K1;U3|kc6XtDFO;BmmdqV>;9inBd$2`=^jOL2_DaKM8f5ghYn z{oe>K^~axS)OD^ySietj)N_RFzg+N@LVpQyOfUVpQpig^j|%y_g`W2Wm;Rsm9T0+m z^?NOBT&{V7W8Oz$!*DrqWE){)`Q=Lf?p4YmY3CWlwVmfG`5i{95*&3i|AWx;w&0I=$nO^X zdLjRe&?D!sR|J>!-Y)c@FLG)A`clZhBlJ6dIDw%1`*D7QkKuzAXZ`0A$2|AK#(o%1 zob~@4;xJs~H@jo>e^2lqfC%fklsM{_?YTtA-yr16Jmg;&{9Pgch2Z-HPZ|Oz5ZL}Z z?5Q)!gRfPb$NvqAbAQ>S_)*gT8{$~rf5FCncudI4e$ga2reQt1gr1E;&nrrv?cA$4 z+xdawtp6jSU)p)%*^n6o?YAQ0Xybm^xLhlhJjP+TRq>B3cJ2{;6)@KGu;8l&-zGTb zjcZm6uPgqh#mE(h)D4DXZgP?&g0B;inE@V6yIR8bz;u72VCBD#7|ayJ@M}lM}M}##(q0b$+JBp6u+MI zq$|$$Oi+9^$p;i46k{V?#ksy_5Jx+43}8FwddM$Od_U>0P@Mg_LUFE_A1Kb_y$i~>sG}J$(}oeo&!S9FO)pnd5_|3XOrTCN&haP{~@9OSr7Ra6<-Z$Fublf z*ZbQ-|Gz&17TN!B?&u$?C>&gB}cIM@3S;^^mgp?`{pyes5C5%Ti{KPdRMN>32t zFsxC0xW&#|!EXe{<-J94v^9m~HxNhmsgVDR;QIw1ooW+e|D7XvCUMmBu;8}}{+ZxA z1(*5<4g(E-2e68RQ2>DHdZxQ@+!M6#1Nbr}0 ze$1Q4r4NMsVIlvqhy0N9K?nkmlic3UC62m}z{Y-<ljgVAn|8FtDBN*{$R`-xmbG3DUFtn?lbf!9Ni4mE-N}l!4RD6R~>&#W0?Odcd+f$=B+w%kB+MbA#-$41^ ztT^X;yW(vBU5c|ke^Q+7d4#yO=kH2>1KG1%akl4q#o3-$6=!=6D9-kLN?hCXrIKg; z@nfxFT<;sm&H;+Eou?_z_6$>;?HNg2+mj)<9QURwJ-jZRp?EFXnXmXqSP%%siu1T# zt~g)suTY%F^XrJC{rJsu?x#0ud5FWXPH`T8ZWH_-kYhdf2#$HM{r4%(_G~7OvREe8 z^EV+c$BjLLW13vD^L53!J^V}Yxg>u;ac(C^6u*?@=#kpO5t@uY&-hSh4I`%*QofHfwD88Tg8H%(0!xaCF`d7B#GeIxc*Uf@s z-{R+|?+A`%&P8J&#AVn$vXeoc^$aG?_6Mz)bB^LC5r=PhnSk<9*jUd*q5l-2f3k=C zG$D_FgMjtS7V>eRAFsVJEL428q3{m82QOCqI+CwY{CeUmJot6Q(Qo)Sl-SNS9`Zj_ zd^PD`C-j^q^!!Zl(*^&P(1Uq%fBBOKe?;gxL+IJ6EXy4cl|RlE1^6;iM_f^Kz!*wIrXVIFIL373XAj|D4RmrnIFI1fUnJ)DBg`V#!dG3!v z#aaJ!#aVxz&_6`zzf8&V{Ju%$imXFabg&VD|iIQ!=_#o0e!DbDsg6Ky%LydT5HcJ?FA z?G5LC495#D`%$vssEhm2*@|;{hbhkG9Z4K@pp9%#hLY#~&6c;YuHUP`<{akg`V;%w)g#L>>7!k$e^zL50) zPI1<=!-GF5^q(vAzoO*1UA?6^>)$8zoG0`g5c2g<2Cnzd1;?_n-;OHI`eP=65X!-p zD)gUA9NRO_rL2F1;8K6O;zuD3hD_pGf02j$G9f=q*mI4LZvee)&kcfO9!D`#2oa%Y zBb>ARkCZ&NQ4Bv7{AM_3`MU&{`hQIv*+wD%dnG@d^gk^4Ekge9f=m5P#I^oCNeHaw-7l?A@DS6KKGQ~OHg+dSdnDZ_6kY6t3M+o__ zl7ETHb%T)KFUnP?cY68uFWf1BWM2#&ru2R01*73XrbD;~7sPE3|PK)?MIHnyiPan!v@@Dqun z{Oy9DN?gldEaWlnC~O#lS!QSE_b1?x^R?8D!;I5=nvy?2JYDfd zy01sKNjm*-h~&Bbvi`40K1=EOhU9~aPbbB$;ziVNa~1zN@jS&7vn-ga_$MSkU-45& zexc&06JMbRgfr#hzvsPIasE5;?TYi?Yd@lRd$I-i3>^dO=f6*n z_gI)G!NGlWj@K-bZ|Gysac-4-Bk>Hy?;)P0`2ECP#UCP`r}*EA&sUuP9_?bq@$X1t zC{~>9uTY%ruTh-sU#&RXzgBU!f4$<*kUe)O{t|J2e`_yBck{pEZ<0L9U|{)gNN<|O zot)om%2Rv*b{Ys3cGoGNNPNBG{2k%Vil0jI&5EBvyj}6Lh$ksOoI^ZK@$-o1DSiR* z3dJuZzFzS$#5XHGo_Mq3`1kHHv@3owasJ*6Mkm2Gg?Jk6nO{QZd5X`X^9sf1(D{1B z`TLNY6~B_sn-yO|=k1D@(0LLrvLUdYWyI4IuO^mqLh&L<# zW8&?K|CD%Ae0TeQPCQNVUlPw#{I|p_6u*!7dd2@te6!+@5N}reZ^YXbw|`$A+6?t$ z_WyR0Pg8t1@jS(!BwnHTv&7dcZvP%V%rA<+O!Cc&zfQbe@y`C+x4ZxC->-L^G{w1} z<|+Oz+pjqH$MuT8NAjB$=W(f7@m989@sEio_3LgwU)Q85{uRmRDIQDhvO@9x#MdjH zM0~U2rxI`0ob6Zq47NX^yZzrGJ!y&$C7!4FaN-q;k0HKZ@d?B?D;^-;thh_OU2z`g zlhnM%<9wRpvq(>#;+GPyQ2Yww>lH5~zFF~Q#G4hbB;Kw#|DDXF0p0!2^G=%LJn!Tw zzMAw`DE>p@>lMG5_-4g#Bi^j|M&j*?-%UJ;UgYNS=MTiw6rafLUvZuXD-{0|=~=J% z!^Af$&f{CN;yjPEE6(#y(!lQi=lLZ~ah_lJyRuwfo?j}IJkKxd75_Q;VYA{qzcee( z^GmzpJijCz-`#$mU(yuk`6W;BE#%J%#UEq;E6(e~&5FlSxtbN{^ygp2#>l^kb zuUpa-=XFb-;=FFDP@LB->lNp9%VxzNr}8!{&g-#u#d$r(-!*0XcaWYmU-x?9by1$; zye_Ix{Au>T;{PDNS@Boce#PG+-mdt5;z_vZguwQ<5l>V6An`oK4-v0W{A=Rt6_2NN z%x1-pBi^j|iNxC#KaF_O3Ek~KlX#lqDa7*>zkqm!;-iVLS3HyWX2r9JH!B__-mds` z;z=iVw|_SAG{wJ9JWuh(#48jpBEDYnGUA&RUqQTC@f(P@D}EDk{>~(~pLN93=!G}t z8;IvAzL7Y8*OBG#A--P8-%otA;(sRItoUQZ+ZEqNJZVsO|35`MP4O3q=PCXg@e0NF z5?`-)3-Qg0^ZLG7@gpSPuJ|{^lTPk#e_vW>rzxIDJWugch*v279pdX1Pa(cp@!`aq z6;CJLuJ|P4NvCwTKS-Rvv&Ze>65@GEem3z6#V;qmUhyTwH!FS>@n*#^LSXHIFDEB73c9}v*O&} zn-%B&*seJD7yeEf`=8r$>giqO!ll)9L+rV$_~U0Z6I zuU33NasE5qXy>J{aagb9`8uEfPB+Tq(6yM>hcu0w^~NZSq>x5?M56`I6MT%|6@v4- z6^Pagj#+LSZoy^`-mG{z$+s(x&!jOV(H?iG|A8S$;eLhUTwi$}yh3rS#c}*}fV#^tN1G7 z&nV98zaxrYK=Wb>l>?)3yO-vhEXDT|4=c{kU)C$mRs z=Y4yX{Ia3e&ZCOoK|Gc22eO{lbFKVB#re5gSn($>u=0F=lJ)TTbH9@3ag6UzvV7tQ ztEWZD4iZ<%;PTN4=MSr#Gg_8pTwP!RV$anAEJdl{8e7FFXp4lP-*_i zF#NS+XJlc1INuq$q^impSy_%JC_ZZBsP3om*X&IiD?%qm8ZaarY5vBr<~>2FH+^J! z=V?JzxZ4TjSWv!vIsQTIk!9uK(8y^s=Z!Fbx1Mqd6)!5P#J_O=|5EUS9vo_^8Fp&T zqN5pnK<6|p)0yMou!BQFRwuCs&A+z$*XG*&DX_z|RBa9|3!QU`(=ca+;dfJ(J>+?e zBm8_odz|)Br%s<0AORKOC(Ju+-e}io2DIn&hiDw(Xf8kRvmm0U^cmA^ikq=9i@@dQ z^ynTuR$|Vc9GhspNQ;Lw{C3h=#IibxJ-GL<-7lu{qyMpMbN=lAd|;Tra+9<85`YlW zXx_DFj^pP8+T(hqr}|I$iUdUds7mL*5?D|9Z<=G%Z=!W7X*(wWFhum2|2H;8#UQ)K z^5fb|=YKP>p7PHsv*}i2qzHa-jE7O!ddk1%M4Ql0=`sJ~Vbl3z`$iGn{u;{xLST7& z)>Z%94t@s_n4WI)u$Z&wKTQ9SNBU+;?{;dm2W$`Tn<#ynl1ruq>k!4!Rt&s