Migrate st to version 8.4
							parent
							
								
									fc0c03da7c
								
							
						
					
					
						commit
						1093e2f027
					
				| 
						 | 
				
			
			@ -1,54 +0,0 @@
 | 
			
		|||
*.old/
 | 
			
		||||
 | 
			
		||||
# Prerequisites
 | 
			
		||||
*.d
 | 
			
		||||
 | 
			
		||||
# Object files
 | 
			
		||||
*.o
 | 
			
		||||
*.ko
 | 
			
		||||
*.obj
 | 
			
		||||
*.elf
 | 
			
		||||
 | 
			
		||||
# Linker output
 | 
			
		||||
*.ilk
 | 
			
		||||
*.map
 | 
			
		||||
*.exp
 | 
			
		||||
 | 
			
		||||
# Precompiled Headers
 | 
			
		||||
*.gch
 | 
			
		||||
*.pch
 | 
			
		||||
 | 
			
		||||
# Libraries
 | 
			
		||||
*.lib
 | 
			
		||||
*.a
 | 
			
		||||
*.la
 | 
			
		||||
*.lo
 | 
			
		||||
 | 
			
		||||
# Shared objects (inc. Windows DLLs)
 | 
			
		||||
*.dll
 | 
			
		||||
*.so
 | 
			
		||||
*.so.*
 | 
			
		||||
*.dylib
 | 
			
		||||
 | 
			
		||||
# Executables
 | 
			
		||||
*.exe
 | 
			
		||||
*.out
 | 
			
		||||
*.app
 | 
			
		||||
*.i*86
 | 
			
		||||
*.x86_64
 | 
			
		||||
*.hex
 | 
			
		||||
 | 
			
		||||
# Debug files
 | 
			
		||||
*.dSYM/
 | 
			
		||||
*.su
 | 
			
		||||
*.idb
 | 
			
		||||
*.pdb
 | 
			
		||||
 | 
			
		||||
# Kernel Module Compile Results
 | 
			
		||||
*.mod*
 | 
			
		||||
*.cmd
 | 
			
		||||
.tmp_versions/
 | 
			
		||||
modules.order
 | 
			
		||||
Module.symvers
 | 
			
		||||
Mkfile.old
 | 
			
		||||
dkms.conf
 | 
			
		||||
| 
						 | 
				
			
			@ -1,201 +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
 | 
			
		||||
* Using the excellent tool of [scroll](https://git.suckless.org/scroll/).
 | 
			
		||||
* Using the scrollback [patch](https://st.suckless.org/patches/scrollback/).
 | 
			
		||||
 | 
			
		||||
## I would like to have utmp and/or scroll functionality by default
 | 
			
		||||
 | 
			
		||||
You can add the absolute patch of both programs in your config.h
 | 
			
		||||
file. You only have to modify the value of utmp and scroll variables.
 | 
			
		||||
 | 
			
		||||
## 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
 | 
			
		||||
<http://zsh.sourceforge.net/FAQ/zshfaq03.html#l25>:
 | 
			
		||||
 | 
			
		||||
	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
 | 
			
		||||
<http://pubs.opengroup.org/onlinepubs/9699919799/utilities/c99.html#tag_20_11_13>.
 | 
			
		||||
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
 | 
			
		||||
<https://lists.suckless.org/dev/1404/20697.html>. 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.
 | 
			
		||||
| 
						 | 
				
			
			@ -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
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -1,34 +0,0 @@
 | 
			
		|||
MIT/X Consortium License
 | 
			
		||||
 | 
			
		||||
© 2014-2018 Hiltjo Posthuma <hiltjo at codemadness dot org>
 | 
			
		||||
© 2018 Devin J. Pohly <djpohly at gmail dot com>
 | 
			
		||||
© 2014-2017 Quentin Rameau <quinq at fifth dot space>
 | 
			
		||||
© 2009-2012 Aurélien APTEL <aurelien dot aptel at gmail dot com>
 | 
			
		||||
© 2008-2017 Anselm R Garbe <garbeam at gmail dot com>
 | 
			
		||||
© 2012-2017 Roberto E. Vargas Caballero <k0ga at shike2 dot com>
 | 
			
		||||
© 2012-2016 Christoph Lohmann <20h at r-36 dot net>
 | 
			
		||||
© 2013 Eon S. Jeon <esjeon at hyunmu dot am>
 | 
			
		||||
© 2013 Alexander Sedov <alex0player at gmail dot com>
 | 
			
		||||
© 2013 Mark Edgar <medgar123 at gmail dot com>
 | 
			
		||||
© 2013-2014 Eric Pruitt <eric.pruitt at gmail dot com>
 | 
			
		||||
© 2013 Michael Forney <mforney at mforney dot org>
 | 
			
		||||
© 2013-2014 Markus Teich <markus dot teich at stusta dot mhn dot de>
 | 
			
		||||
© 2014-2015 Laslo Hunhold <dev at frign dot de>
 | 
			
		||||
 | 
			
		||||
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.
 | 
			
		||||
| 
						 | 
				
			
			@ -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
 | 
			
		||||
| 
						 | 
				
			
			@ -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 <aurelien dot aptel at gmail dot com> bt source code.
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -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
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -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
 | 
			
		||||
| 
						 | 
				
			
			@ -1,463 +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: scroll and/or utmp
 | 
			
		||||
 * 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;
 | 
			
		||||
/* scroll program: to enable use a string like "scroll" */
 | 
			
		||||
char *scroll = 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 */
 | 
			
		||||
	"black",
 | 
			
		||||
	"red3",
 | 
			
		||||
	"green3",
 | 
			
		||||
	"yellow3",
 | 
			
		||||
	"blue2",
 | 
			
		||||
	"magenta3",
 | 
			
		||||
	"cyan3",
 | 
			
		||||
	"gray90",
 | 
			
		||||
 | 
			
		||||
	/* 8 bright colors */
 | 
			
		||||
	"gray50",
 | 
			
		||||
	"red",
 | 
			
		||||
	"green",
 | 
			
		||||
	"yellow",
 | 
			
		||||
	"#5c5cff",
 | 
			
		||||
	"magenta",
 | 
			
		||||
	"cyan",
 | 
			
		||||
	"white",
 | 
			
		||||
 | 
			
		||||
	[255] = 0,
 | 
			
		||||
 | 
			
		||||
	/* more colors can be added after 255 to use with DefaultXX */
 | 
			
		||||
	"#cccccc",
 | 
			
		||||
	"#555555",
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * 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;
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * 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} },
 | 
			
		||||
	{ ShiftMask,            XK_Page_Up,     kscrollup,      {.i = -1} },
 | 
			
		||||
	{ ShiftMask,            XK_Page_Down,   kscrolldown,    {.i = -1} },
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * 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{|}~";
 | 
			
		||||
| 
						 | 
				
			
			@ -1,456 +0,0 @@
 | 
			
		|||
/* See LICENSE file for copyright and license details. */
 | 
			
		||||
 | 
			
		||||
static char *font = "Iosevka Term SS08:pixelsize=20:antialias=true:autohint=true";
 | 
			
		||||
static int borderpx = 0;
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * What program is execed by st depends of these precedence rules:
 | 
			
		||||
 * 1: program passed with -e
 | 
			
		||||
 * 2: scroll and/or utmp
 | 
			
		||||
 * 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;
 | 
			
		||||
/* scroll program: to enable use a string like "scroll" */
 | 
			
		||||
char *scroll = 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[] = {
 | 
			
		||||
 | 
			
		||||
	"#111111", /*black*/
 | 
			
		||||
	"#ac4142", /*r*/
 | 
			
		||||
	"#90a959", /*g*/
 | 
			
		||||
	"#f4bf75", /*y*/
 | 
			
		||||
	"#6a9fb5", /*b*/
 | 
			
		||||
	"#aa759f", /*m*/
 | 
			
		||||
	"#75b5aa", /*c*/
 | 
			
		||||
	"#ccc",    /*white*/
 | 
			
		||||
 | 
			
		||||
	"#111111", /*black*/
 | 
			
		||||
	"#ac4142", /*r*/
 | 
			
		||||
	"#90a959", /*g*/
 | 
			
		||||
	"#f4bf75", /*y*/
 | 
			
		||||
	"#6a9fb5", /*b*/
 | 
			
		||||
	"#aa759f", /*m*/
 | 
			
		||||
	"#75b5aa", /*c*/
 | 
			
		||||
	"#ccc",    /*white*/
 | 
			
		||||
 | 
			
		||||
	[255] = 0,
 | 
			
		||||
  [256] = "#111111", /* background */
 | 
			
		||||
  [257] = "#d0d0d0", /* 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 = 257;
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
 * 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 Mod4Mask
 | 
			
		||||
#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} },
 | 
			
		||||
	{ MODKEY|ShiftMask,     XK_Up,          zoom,           {.f = +4} },
 | 
			
		||||
	{ MODKEY|ShiftMask,     XK_Down,        zoom,           {.f = -4} },
 | 
			
		||||
	{ MODKEY|ShiftMask,     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} },
 | 
			
		||||
	{ MODKEY|ShiftMask,     XK_K,           kscrollup,      {.i =  8} },
 | 
			
		||||
	{ MODKEY|ShiftMask,     XK_J,           kscrolldown,    {.i =  8} },
 | 
			
		||||
	{ ShiftMask,            XK_Return,      newterm,        {.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{|}~";
 | 
			
		||||
| 
						 | 
				
			
			@ -1,37 +0,0 @@
 | 
			
		|||
# st version
 | 
			
		||||
VERSION = 0.8.3
 | 
			
		||||
 | 
			
		||||
# 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
 | 
			
		||||
| 
						 | 
				
			
			@ -1,136 +0,0 @@
 | 
			
		|||
#include <stdlib.h>
 | 
			
		||||
#include <stdio.h>
 | 
			
		||||
#include <math.h>
 | 
			
		||||
#include <X11/Xft/Xft.h>
 | 
			
		||||
#include <hb.h>
 | 
			
		||||
#include <hb-ft.h>
 | 
			
		||||
 | 
			
		||||
#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);
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -1,7 +0,0 @@
 | 
			
		|||
#include <X11/Xft/Xft.h>
 | 
			
		||||
#include <hb.h>
 | 
			
		||||
#include <hb-ft.h>
 | 
			
		||||
 | 
			
		||||
void hbunloadfonts();
 | 
			
		||||
void hbtransform(XftGlyphFontSpec *, const Glyph *, size_t, int, int);
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -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);
 | 
			
		||||
 	}
 | 
			
		||||
 }
 | 
			
		||||
| 
						 | 
				
			
			@ -1,12 +0,0 @@
 | 
			
		|||
diff --git a/x.c b/x.c
 | 
			
		||||
index e5f1737..5cabd60 100644
 | 
			
		||||
--- a/x.c
 | 
			
		||||
+++ b/x.c
 | 
			
		||||
@@ -673,6 +673,7 @@ setsel(char *str, Time t)
 | 
			
		||||
 	XSetSelectionOwner(xw.dpy, XA_PRIMARY, xw.win, t);
 | 
			
		||||
 	if (XGetSelectionOwner(xw.dpy, XA_PRIMARY) != xw.win)
 | 
			
		||||
 		selclear();
 | 
			
		||||
+	clipcopy(NULL);
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
 void
 | 
			
		||||
| 
						 | 
				
			
			@ -1,307 +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 <stdlib.h>
 | 
			
		||||
+#include <stdio.h>
 | 
			
		||||
+#include <math.h>
 | 
			
		||||
+#include <X11/Xft/Xft.h>
 | 
			
		||||
+#include <hb.h>
 | 
			
		||||
+#include <hb-ft.h>
 | 
			
		||||
+
 | 
			
		||||
+#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 <X11/Xft/Xft.h>
 | 
			
		||||
+#include <hb.h>
 | 
			
		||||
+#include <hb-ft.h>
 | 
			
		||||
+
 | 
			
		||||
+void hbunloadfonts();
 | 
			
		||||
+void hbtransform(XftGlyphFontSpec *, const Glyph *, size_t, int, int);
 | 
			
		||||
+
 | 
			
		||||
diff --git a/st.c b/st.c
 | 
			
		||||
index 641edc0..14a0b66 100644
 | 
			
		||||
--- a/st.c
 | 
			
		||||
+++ b/st.c
 | 
			
		||||
@@ -2670,7 +2670,8 @@ draw(void)
 | 
			
		||||
 	drawregion(0, 0, term.col, term.row);
 | 
			
		||||
 	if (term.scr == 0)
 | 
			
		||||
 		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 b9a4eeb..b49b84a 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;
 | 
			
		||||
| 
						 | 
				
			
			@ -1,79 +0,0 @@
 | 
			
		|||
From a7eedc85e0609177cdb1ed3f6203fa37e6420012 Mon Sep 17 00:00:00 2001
 | 
			
		||||
From: Matias Lang <yo@matiaslang.me>
 | 
			
		||||
Date: Wed, 17 Jul 2019 01:10:44 -0300
 | 
			
		||||
Subject: [PATCH] Add shortcut to spawn new terminal in the current dir
 | 
			
		||||
 | 
			
		||||
Ctrl-Shift-Return now creates a new ST terminal, whose CWD is the same
 | 
			
		||||
as the parent st's CWD
 | 
			
		||||
---
 | 
			
		||||
 config.h |  1 +
 | 
			
		||||
 st.c         | 21 +++++++++++++++++++++
 | 
			
		||||
 st.h         |  1 +
 | 
			
		||||
 3 files changed, 23 insertions(+)
 | 
			
		||||
 | 
			
		||||
diff --git a/config.h b/config.h
 | 
			
		||||
index 0e01717..31f26d8 100644
 | 
			
		||||
--- a/config.h
 | 
			
		||||
+++ b/config.h
 | 
			
		||||
@@ -178,6 +178,7 @@ static Shortcut shortcuts[] = {
 | 
			
		||||
 	{ TERMMOD,              XK_Y,           selpaste,       {.i =  0} },
 | 
			
		||||
 	{ ShiftMask,            XK_Insert,      selpaste,       {.i =  0} },
 | 
			
		||||
 	{ TERMMOD,              XK_Num_Lock,    numlock,        {.i =  0} },
 | 
			
		||||
+	{ TERMMOD,              XK_Return,      newterm,        {.i =  0} },
 | 
			
		||||
 };
 | 
			
		||||
 
 | 
			
		||||
 /*
 | 
			
		||||
diff --git a/st.c b/st.c
 | 
			
		||||
index b8e6077..044e29b 100644
 | 
			
		||||
--- a/st.c
 | 
			
		||||
+++ b/st.c
 | 
			
		||||
@@ -153,6 +153,7 @@ typedef struct {
 | 
			
		||||
 } STREscape;
 | 
			
		||||
 
 | 
			
		||||
 static void execsh(char *, char **);
 | 
			
		||||
+static char *getcwd_by_pid(pid_t pid);
 | 
			
		||||
 static void stty(char **);
 | 
			
		||||
 static void sigchld(int);
 | 
			
		||||
 static void ttywriteraw(const char *, size_t);
 | 
			
		||||
@@ -1059,6 +1060,26 @@ tswapscreen(void)
 | 
			
		||||
 	tfulldirt();
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
+void
 | 
			
		||||
+newterm(const Arg* a)
 | 
			
		||||
+{
 | 
			
		||||
+	switch (fork()) {
 | 
			
		||||
+	case -1:
 | 
			
		||||
+		die("fork failed: %s\n", strerror(errno));
 | 
			
		||||
+		break;
 | 
			
		||||
+	case 0:
 | 
			
		||||
+		chdir(getcwd_by_pid(pid));
 | 
			
		||||
+		execlp("st", "./st", NULL);
 | 
			
		||||
+		break;
 | 
			
		||||
+	}
 | 
			
		||||
+}
 | 
			
		||||
+
 | 
			
		||||
+static char *getcwd_by_pid(pid_t pid) {
 | 
			
		||||
+	char buf[32];
 | 
			
		||||
+	snprintf(buf, sizeof buf, "/proc/%d/cwd", pid);
 | 
			
		||||
+	return realpath(buf, NULL);
 | 
			
		||||
+}
 | 
			
		||||
+
 | 
			
		||||
 void
 | 
			
		||||
 tscrolldown(int orig, int n)
 | 
			
		||||
 {
 | 
			
		||||
diff --git a/st.h b/st.h
 | 
			
		||||
index 38c61c4..54d4a43 100644
 | 
			
		||||
--- a/st.h
 | 
			
		||||
+++ b/st.h
 | 
			
		||||
@@ -80,6 +80,7 @@ void die(const char *, ...);
 | 
			
		||||
 void redraw(void);
 | 
			
		||||
 void draw(void);
 | 
			
		||||
 
 | 
			
		||||
+void newterm(const Arg *);
 | 
			
		||||
 void printscreen(const Arg *);
 | 
			
		||||
 void printsel(const Arg *);
 | 
			
		||||
 void sendbreak(const Arg *);
 | 
			
		||||
-- 
 | 
			
		||||
2.19.2
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -1,351 +0,0 @@
 | 
			
		|||
diff --git a/config.def.h b/config.def.h
 | 
			
		||||
index 0895a1f..eef24df 100644
 | 
			
		||||
--- a/config.def.h
 | 
			
		||||
+++ b/config.def.h
 | 
			
		||||
@@ -188,6 +188,8 @@ static Shortcut shortcuts[] = {
 | 
			
		||||
 	{ TERMMOD,              XK_Y,           selpaste,       {.i =  0} },
 | 
			
		||||
 	{ ShiftMask,            XK_Insert,      selpaste,       {.i =  0} },
 | 
			
		||||
 	{ TERMMOD,              XK_Num_Lock,    numlock,        {.i =  0} },
 | 
			
		||||
+	{ ShiftMask,            XK_Page_Up,     kscrollup,      {.i = -1} },
 | 
			
		||||
+	{ ShiftMask,            XK_Page_Down,   kscrolldown,    {.i = -1} },
 | 
			
		||||
 };
 | 
			
		||||
 
 | 
			
		||||
 /*
 | 
			
		||||
diff --git a/st.c b/st.c
 | 
			
		||||
index 0ce6ac2..641edc0 100644
 | 
			
		||||
--- a/st.c
 | 
			
		||||
+++ b/st.c
 | 
			
		||||
@@ -35,6 +35,7 @@
 | 
			
		||||
 #define ESC_ARG_SIZ   16
 | 
			
		||||
 #define STR_BUF_SIZ   ESC_BUF_SIZ
 | 
			
		||||
 #define STR_ARG_SIZ   ESC_ARG_SIZ
 | 
			
		||||
+#define HISTSIZE      2000
 | 
			
		||||
 
 | 
			
		||||
 /* macros */
 | 
			
		||||
 #define IS_SET(flag)		((term.mode & (flag)) != 0)
 | 
			
		||||
@@ -42,6 +43,9 @@
 | 
			
		||||
 #define ISCONTROLC1(c)		(BETWEEN(c, 0x80, 0x9f))
 | 
			
		||||
 #define ISCONTROL(c)		(ISCONTROLC0(c) || ISCONTROLC1(c))
 | 
			
		||||
 #define ISDELIM(u)		(u && wcschr(worddelimiters, u))
 | 
			
		||||
+#define TLINE(y)		((y) < term.scr ? term.hist[((y) + term.histi - \
 | 
			
		||||
+				term.scr + HISTSIZE + 1) % HISTSIZE] : \
 | 
			
		||||
+				term.line[(y) - term.scr])
 | 
			
		||||
 
 | 
			
		||||
 enum term_mode {
 | 
			
		||||
 	MODE_WRAP        = 1 << 0,
 | 
			
		||||
@@ -117,6 +121,9 @@ typedef struct {
 | 
			
		||||
 	int col;      /* nb col */
 | 
			
		||||
 	Line *line;   /* screen */
 | 
			
		||||
 	Line *alt;    /* alternate screen */
 | 
			
		||||
+	Line hist[HISTSIZE]; /* history buffer */
 | 
			
		||||
+	int histi;    /* history index */
 | 
			
		||||
+	int scr;      /* scroll back */
 | 
			
		||||
 	int *dirty;   /* dirtyness of lines */
 | 
			
		||||
 	TCursor c;    /* cursor */
 | 
			
		||||
 	int ocx;      /* old cursor col */
 | 
			
		||||
@@ -185,8 +192,8 @@ 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 tscrollup(int, int, int);
 | 
			
		||||
+static void tscrolldown(int, int, int);
 | 
			
		||||
 static void tsetattr(int *, int);
 | 
			
		||||
 static void tsetchar(Rune, Glyph *, int, int);
 | 
			
		||||
 static void tsetdirt(int, int);
 | 
			
		||||
@@ -415,10 +422,10 @@ tlinelen(int y)
 | 
			
		||||
 {
 | 
			
		||||
 	int i = term.col;
 | 
			
		||||
 
 | 
			
		||||
-	if (term.line[y][i - 1].mode & ATTR_WRAP)
 | 
			
		||||
+	if (TLINE(y)[i - 1].mode & ATTR_WRAP)
 | 
			
		||||
 		return i;
 | 
			
		||||
 
 | 
			
		||||
-	while (i > 0 && term.line[y][i - 1].u == ' ')
 | 
			
		||||
+	while (i > 0 && TLINE(y)[i - 1].u == ' ')
 | 
			
		||||
 		--i;
 | 
			
		||||
 
 | 
			
		||||
 	return i;
 | 
			
		||||
@@ -527,7 +534,7 @@ selsnap(int *x, int *y, int direction)
 | 
			
		||||
 		 * Snap around if the word wraps around at the end or
 | 
			
		||||
 		 * beginning of a line.
 | 
			
		||||
 		 */
 | 
			
		||||
-		prevgp = &term.line[*y][*x];
 | 
			
		||||
+		prevgp = &TLINE(*y)[*x];
 | 
			
		||||
 		prevdelim = ISDELIM(prevgp->u);
 | 
			
		||||
 		for (;;) {
 | 
			
		||||
 			newx = *x + direction;
 | 
			
		||||
@@ -542,14 +549,14 @@ selsnap(int *x, int *y, int direction)
 | 
			
		||||
 					yt = *y, xt = *x;
 | 
			
		||||
 				else
 | 
			
		||||
 					yt = newy, xt = newx;
 | 
			
		||||
-				if (!(term.line[yt][xt].mode & ATTR_WRAP))
 | 
			
		||||
+				if (!(TLINE(yt)[xt].mode & ATTR_WRAP))
 | 
			
		||||
 					break;
 | 
			
		||||
 			}
 | 
			
		||||
 
 | 
			
		||||
 			if (newx >= tlinelen(newy))
 | 
			
		||||
 				break;
 | 
			
		||||
 
 | 
			
		||||
-			gp = &term.line[newy][newx];
 | 
			
		||||
+			gp = &TLINE(newy)[newx];
 | 
			
		||||
 			delim = ISDELIM(gp->u);
 | 
			
		||||
 			if (!(gp->mode & ATTR_WDUMMY) && (delim != prevdelim
 | 
			
		||||
 					|| (delim && gp->u != prevgp->u)))
 | 
			
		||||
@@ -570,14 +577,14 @@ selsnap(int *x, int *y, int direction)
 | 
			
		||||
 		*x = (direction < 0) ? 0 : term.col - 1;
 | 
			
		||||
 		if (direction < 0) {
 | 
			
		||||
 			for (; *y > 0; *y += direction) {
 | 
			
		||||
-				if (!(term.line[*y-1][term.col-1].mode
 | 
			
		||||
+				if (!(TLINE(*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
 | 
			
		||||
+				if (!(TLINE(*y)[term.col-1].mode
 | 
			
		||||
 						& ATTR_WRAP)) {
 | 
			
		||||
 					break;
 | 
			
		||||
 				}
 | 
			
		||||
@@ -608,13 +615,13 @@ getsel(void)
 | 
			
		||||
 		}
 | 
			
		||||
 
 | 
			
		||||
 		if (sel.type == SEL_RECTANGULAR) {
 | 
			
		||||
-			gp = &term.line[y][sel.nb.x];
 | 
			
		||||
+			gp = &TLINE(y)[sel.nb.x];
 | 
			
		||||
 			lastx = sel.ne.x;
 | 
			
		||||
 		} else {
 | 
			
		||||
-			gp = &term.line[y][sel.nb.y == y ? sel.nb.x : 0];
 | 
			
		||||
+			gp = &TLINE(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)];
 | 
			
		||||
+		last = &TLINE(y)[MIN(lastx, linelen-1)];
 | 
			
		||||
 		while (last >= gp && last->u == ' ')
 | 
			
		||||
 			--last;
 | 
			
		||||
 
 | 
			
		||||
@@ -849,6 +856,9 @@ void
 | 
			
		||||
 ttywrite(const char *s, size_t n, int may_echo)
 | 
			
		||||
 {
 | 
			
		||||
 	const char *next;
 | 
			
		||||
+	Arg arg = (Arg) { .i = term.scr };
 | 
			
		||||
+
 | 
			
		||||
+	kscrolldown(&arg);
 | 
			
		||||
 
 | 
			
		||||
 	if (may_echo && IS_SET(MODE_ECHO))
 | 
			
		||||
 		twrite(s, n, 1);
 | 
			
		||||
@@ -1060,13 +1070,53 @@ tswapscreen(void)
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
 void
 | 
			
		||||
-tscrolldown(int orig, int n)
 | 
			
		||||
+kscrolldown(const Arg* a)
 | 
			
		||||
+{
 | 
			
		||||
+	int n = a->i;
 | 
			
		||||
+
 | 
			
		||||
+	if (n < 0)
 | 
			
		||||
+		n = term.row + n;
 | 
			
		||||
+
 | 
			
		||||
+	if (n > term.scr)
 | 
			
		||||
+		n = term.scr;
 | 
			
		||||
+
 | 
			
		||||
+	if (term.scr > 0) {
 | 
			
		||||
+		term.scr -= n;
 | 
			
		||||
+		selscroll(0, -n);
 | 
			
		||||
+		tfulldirt();
 | 
			
		||||
+	}
 | 
			
		||||
+}
 | 
			
		||||
+
 | 
			
		||||
+void
 | 
			
		||||
+kscrollup(const Arg* a)
 | 
			
		||||
+{
 | 
			
		||||
+	int n = a->i;
 | 
			
		||||
+
 | 
			
		||||
+	if (n < 0)
 | 
			
		||||
+		n = term.row + n;
 | 
			
		||||
+
 | 
			
		||||
+	if (term.scr <= HISTSIZE-n) {
 | 
			
		||||
+		term.scr += n;
 | 
			
		||||
+		selscroll(0, n);
 | 
			
		||||
+		tfulldirt();
 | 
			
		||||
+	}
 | 
			
		||||
+}
 | 
			
		||||
+
 | 
			
		||||
+void
 | 
			
		||||
+tscrolldown(int orig, int n, int copyhist)
 | 
			
		||||
 {
 | 
			
		||||
 	int i;
 | 
			
		||||
 	Line temp;
 | 
			
		||||
 
 | 
			
		||||
 	LIMIT(n, 0, term.bot-orig+1);
 | 
			
		||||
 
 | 
			
		||||
+	if (copyhist) {
 | 
			
		||||
+		term.histi = (term.histi - 1 + HISTSIZE) % HISTSIZE;
 | 
			
		||||
+		temp = term.hist[term.histi];
 | 
			
		||||
+		term.hist[term.histi] = term.line[term.bot];
 | 
			
		||||
+		term.line[term.bot] = temp;
 | 
			
		||||
+	}
 | 
			
		||||
+
 | 
			
		||||
 	tsetdirt(orig, term.bot-n);
 | 
			
		||||
 	tclearregion(0, term.bot-n+1, term.col-1, term.bot);
 | 
			
		||||
 
 | 
			
		||||
@@ -1076,17 +1126,28 @@ tscrolldown(int orig, int n)
 | 
			
		||||
 		term.line[i-n] = temp;
 | 
			
		||||
 	}
 | 
			
		||||
 
 | 
			
		||||
-	selscroll(orig, n);
 | 
			
		||||
+	if (term.scr == 0)
 | 
			
		||||
+		selscroll(orig, n);
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
 void
 | 
			
		||||
-tscrollup(int orig, int n)
 | 
			
		||||
+tscrollup(int orig, int n, int copyhist)
 | 
			
		||||
 {
 | 
			
		||||
 	int i;
 | 
			
		||||
 	Line temp;
 | 
			
		||||
 
 | 
			
		||||
 	LIMIT(n, 0, term.bot-orig+1);
 | 
			
		||||
 
 | 
			
		||||
+	if (copyhist) {
 | 
			
		||||
+		term.histi = (term.histi + 1) % HISTSIZE;
 | 
			
		||||
+		temp = term.hist[term.histi];
 | 
			
		||||
+		term.hist[term.histi] = term.line[orig];
 | 
			
		||||
+		term.line[orig] = temp;
 | 
			
		||||
+	}
 | 
			
		||||
+
 | 
			
		||||
+	if (term.scr > 0 && term.scr < HISTSIZE)
 | 
			
		||||
+		term.scr = MIN(term.scr + n, HISTSIZE-1);
 | 
			
		||||
+
 | 
			
		||||
 	tclearregion(0, orig, term.col-1, orig+n-1);
 | 
			
		||||
 	tsetdirt(orig+n, term.bot);
 | 
			
		||||
 
 | 
			
		||||
@@ -1096,7 +1157,8 @@ tscrollup(int orig, int n)
 | 
			
		||||
 		term.line[i+n] = temp;
 | 
			
		||||
 	}
 | 
			
		||||
 
 | 
			
		||||
-	selscroll(orig, -n);
 | 
			
		||||
+	if (term.scr == 0)
 | 
			
		||||
+		selscroll(orig, -n);
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
 void
 | 
			
		||||
@@ -1135,7 +1197,7 @@ tnewline(int first_col)
 | 
			
		||||
 	int y = term.c.y;
 | 
			
		||||
 
 | 
			
		||||
 	if (y == term.bot) {
 | 
			
		||||
-		tscrollup(term.top, 1);
 | 
			
		||||
+		tscrollup(term.top, 1, 1);
 | 
			
		||||
 	} else {
 | 
			
		||||
 		y++;
 | 
			
		||||
 	}
 | 
			
		||||
@@ -1300,14 +1362,14 @@ void
 | 
			
		||||
 tinsertblankline(int n)
 | 
			
		||||
 {
 | 
			
		||||
 	if (BETWEEN(term.c.y, term.top, term.bot))
 | 
			
		||||
-		tscrolldown(term.c.y, n);
 | 
			
		||||
+		tscrolldown(term.c.y, n, 0);
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
 void
 | 
			
		||||
 tdeleteline(int n)
 | 
			
		||||
 {
 | 
			
		||||
 	if (BETWEEN(term.c.y, term.top, term.bot))
 | 
			
		||||
-		tscrollup(term.c.y, n);
 | 
			
		||||
+		tscrollup(term.c.y, n, 0);
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
 int32_t
 | 
			
		||||
@@ -1738,11 +1800,11 @@ csihandle(void)
 | 
			
		||||
 		break;
 | 
			
		||||
 	case 'S': /* SU -- Scroll <n> line up */
 | 
			
		||||
 		DEFAULT(csiescseq.arg[0], 1);
 | 
			
		||||
-		tscrollup(term.top, csiescseq.arg[0]);
 | 
			
		||||
+		tscrollup(term.top, csiescseq.arg[0], 0);
 | 
			
		||||
 		break;
 | 
			
		||||
 	case 'T': /* SD -- Scroll <n> line down */
 | 
			
		||||
 		DEFAULT(csiescseq.arg[0], 1);
 | 
			
		||||
-		tscrolldown(term.top, csiescseq.arg[0]);
 | 
			
		||||
+		tscrolldown(term.top, csiescseq.arg[0], 0);
 | 
			
		||||
 		break;
 | 
			
		||||
 	case 'L': /* IL -- Insert <n> blank lines */
 | 
			
		||||
 		DEFAULT(csiescseq.arg[0], 1);
 | 
			
		||||
@@ -2248,7 +2310,7 @@ eschandle(uchar ascii)
 | 
			
		||||
 		return 0;
 | 
			
		||||
 	case 'D': /* IND -- Linefeed */
 | 
			
		||||
 		if (term.c.y == term.bot) {
 | 
			
		||||
-			tscrollup(term.top, 1);
 | 
			
		||||
+			tscrollup(term.top, 1, 1);
 | 
			
		||||
 		} else {
 | 
			
		||||
 			tmoveto(term.c.x, term.c.y+1);
 | 
			
		||||
 		}
 | 
			
		||||
@@ -2261,7 +2323,7 @@ eschandle(uchar ascii)
 | 
			
		||||
 		break;
 | 
			
		||||
 	case 'M': /* RI -- Reverse index */
 | 
			
		||||
 		if (term.c.y == term.top) {
 | 
			
		||||
-			tscrolldown(term.top, 1);
 | 
			
		||||
+			tscrolldown(term.top, 1, 1);
 | 
			
		||||
 		} else {
 | 
			
		||||
 			tmoveto(term.c.x, term.c.y-1);
 | 
			
		||||
 		}
 | 
			
		||||
@@ -2482,7 +2544,7 @@ twrite(const char *buf, int buflen, int show_ctrl)
 | 
			
		||||
 void
 | 
			
		||||
 tresize(int col, int row)
 | 
			
		||||
 {
 | 
			
		||||
-	int i;
 | 
			
		||||
+	int i, j;
 | 
			
		||||
 	int minrow = MIN(row, term.row);
 | 
			
		||||
 	int mincol = MIN(col, term.col);
 | 
			
		||||
 	int *bp;
 | 
			
		||||
@@ -2519,6 +2581,14 @@ tresize(int col, int row)
 | 
			
		||||
 	term.dirty = xrealloc(term.dirty, row * sizeof(*term.dirty));
 | 
			
		||||
 	term.tabs = xrealloc(term.tabs, col * sizeof(*term.tabs));
 | 
			
		||||
 
 | 
			
		||||
+	for (i = 0; i < HISTSIZE; i++) {
 | 
			
		||||
+		term.hist[i] = xrealloc(term.hist[i], col * sizeof(Glyph));
 | 
			
		||||
+		for (j = mincol; j < col; j++) {
 | 
			
		||||
+			term.hist[i][j] = term.c.attr;
 | 
			
		||||
+			term.hist[i][j].u = ' ';
 | 
			
		||||
+		}
 | 
			
		||||
+	}
 | 
			
		||||
+
 | 
			
		||||
 	/* 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));
 | 
			
		||||
@@ -2577,7 +2647,7 @@ drawregion(int x1, int y1, int x2, int y2)
 | 
			
		||||
 			continue;
 | 
			
		||||
 
 | 
			
		||||
 		term.dirty[y] = 0;
 | 
			
		||||
-		xdrawline(term.line[y], x1, y, x2);
 | 
			
		||||
+		xdrawline(TLINE(y), x1, y, x2);
 | 
			
		||||
 	}
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
@@ -2598,8 +2668,9 @@ draw(void)
 | 
			
		||||
 		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]);
 | 
			
		||||
+	if (term.scr == 0)
 | 
			
		||||
+		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();
 | 
			
		||||
diff --git a/st.h b/st.h
 | 
			
		||||
index d978458..b9a4eeb 100644
 | 
			
		||||
--- a/st.h
 | 
			
		||||
+++ b/st.h
 | 
			
		||||
@@ -81,6 +81,8 @@ void die(const char *, ...);
 | 
			
		||||
 void redraw(void);
 | 
			
		||||
 void draw(void);
 | 
			
		||||
 
 | 
			
		||||
+void kscrolldown(const Arg *);
 | 
			
		||||
+void kscrollup(const Arg *);
 | 
			
		||||
 void printscreen(const Arg *);
 | 
			
		||||
 void printsel(const Arg *);
 | 
			
		||||
 void sendbreak(const Arg *);
 | 
			
		||||
										
											Binary file not shown.
										
									
								
							| 
						 | 
				
			
			@ -1,177 +0,0 @@
 | 
			
		|||
.TH ST 1 st\-VERSION
 | 
			
		||||
.SH NAME
 | 
			
		||||
st \- simple terminal
 | 
			
		||||
.SH SYNOPSIS
 | 
			
		||||
.B st
 | 
			
		||||
.RB [ \-aiv ]
 | 
			
		||||
.RB [ \-c
 | 
			
		||||
.IR class ]
 | 
			
		||||
.RB [ \-f
 | 
			
		||||
.IR font ]
 | 
			
		||||
.RB [ \-g
 | 
			
		||||
.IR geometry ]
 | 
			
		||||
.RB [ \-n
 | 
			
		||||
.IR name ]
 | 
			
		||||
.RB [ \-o
 | 
			
		||||
.IR iofile ]
 | 
			
		||||
.RB [ \-T
 | 
			
		||||
.IR title ]
 | 
			
		||||
.RB [ \-t
 | 
			
		||||
.IR title ]
 | 
			
		||||
.RB [ \-l
 | 
			
		||||
.IR line ]
 | 
			
		||||
.RB [ \-w
 | 
			
		||||
.IR windowid ]
 | 
			
		||||
.RB [[ \-e ]
 | 
			
		||||
.IR command
 | 
			
		||||
.RI [ arguments ...]]
 | 
			
		||||
.PP
 | 
			
		||||
.B st
 | 
			
		||||
.RB [ \-aiv ]
 | 
			
		||||
.RB [ \-c
 | 
			
		||||
.IR class ]
 | 
			
		||||
.RB [ \-f
 | 
			
		||||
.IR font ]
 | 
			
		||||
.RB [ \-g
 | 
			
		||||
.IR geometry ]
 | 
			
		||||
.RB [ \-n
 | 
			
		||||
.IR name ]
 | 
			
		||||
.RB [ \-o
 | 
			
		||||
.IR iofile ]
 | 
			
		||||
.RB [ \-T
 | 
			
		||||
.IR title ]
 | 
			
		||||
.RB [ \-t
 | 
			
		||||
.IR title ]
 | 
			
		||||
.RB [ \-w
 | 
			
		||||
.IR windowid ]
 | 
			
		||||
.RB \-l
 | 
			
		||||
.IR line
 | 
			
		||||
.RI [ stty_args ...]
 | 
			
		||||
.SH DESCRIPTION
 | 
			
		||||
.B st
 | 
			
		||||
is a simple terminal emulator.
 | 
			
		||||
.SH OPTIONS
 | 
			
		||||
.TP
 | 
			
		||||
.B \-a
 | 
			
		||||
disable alternate screens in terminal
 | 
			
		||||
.TP
 | 
			
		||||
.BI \-c " class"
 | 
			
		||||
defines the window class (default $TERM).
 | 
			
		||||
.TP
 | 
			
		||||
.BI \-f " font"
 | 
			
		||||
defines the
 | 
			
		||||
.I font
 | 
			
		||||
to use when st is run.
 | 
			
		||||
.TP
 | 
			
		||||
.BI \-g " geometry"
 | 
			
		||||
defines the X11 geometry string.
 | 
			
		||||
The form is [=][<cols>{xX}<rows>][{+-}<xoffset>{+-}<yoffset>]. 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),
 | 
			
		||||
.BR scroll (1)
 | 
			
		||||
.SH BUGS
 | 
			
		||||
See the TODO file in the distribution.
 | 
			
		||||
 | 
			
		||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| 
						 | 
				
			
			@ -1,129 +0,0 @@
 | 
			
		|||
/* See LICENSE for license details. */
 | 
			
		||||
 | 
			
		||||
#include <stdint.h>
 | 
			
		||||
#include <sys/types.h>
 | 
			
		||||
 | 
			
		||||
/* 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 kscrolldown(const Arg *);
 | 
			
		||||
void kscrollup(const Arg *);
 | 
			
		||||
void printscreen(const Arg *);
 | 
			
		||||
void printsel(const Arg *);
 | 
			
		||||
void sendbreak(const Arg *);
 | 
			
		||||
void toggleprinter(const Arg *);
 | 
			
		||||
void newterm(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 *scroll;
 | 
			
		||||
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;
 | 
			
		||||
| 
						 | 
				
			
			@ -1,128 +0,0 @@
 | 
			
		|||
/* See LICENSE for license details. */
 | 
			
		||||
 | 
			
		||||
#include <stdint.h>
 | 
			
		||||
#include <sys/types.h>
 | 
			
		||||
 | 
			
		||||
/* 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 kscrolldown(const Arg *);
 | 
			
		||||
void kscrollup(const Arg *);
 | 
			
		||||
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 *scroll;
 | 
			
		||||
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;
 | 
			
		||||
| 
						 | 
				
			
			@ -1,10 +0,0 @@
 | 
			
		|||
--- st.h
 | 
			
		||||
+++ st.h
 | 
			
		||||
@@ -80,6 +80,7 @@ void die(const char *, ...);
 | 
			
		||||
 void redraw(void);
 | 
			
		||||
 void draw(void);
 | 
			
		||||
 
 | 
			
		||||
+void newterm(const Arg *);
 | 
			
		||||
 void printscreen(const Arg *);
 | 
			
		||||
 void printsel(const Arg *);
 | 
			
		||||
 void sendbreak(const Arg *);
 | 
			
		||||
| 
						 | 
				
			
			@ -1,236 +0,0 @@
 | 
			
		|||
st-mono| simpleterm monocolor,
 | 
			
		||||
	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#2,
 | 
			
		||||
	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,
 | 
			
		||||
	sitm=\E[3m,
 | 
			
		||||
	sgr0=\E[0m,
 | 
			
		||||
	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| simpleterm,
 | 
			
		||||
	use=st-mono,
 | 
			
		||||
	colors#8,
 | 
			
		||||
	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,
 | 
			
		||||
	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,
 | 
			
		||||
 | 
			
		||||
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,
 | 
			
		||||
 | 
			
		||||
st-bs| simpleterm with backspace as backspace,
 | 
			
		||||
	use=st,
 | 
			
		||||
	kbs=\010,
 | 
			
		||||
	kdch1=\177,
 | 
			
		||||
 | 
			
		||||
st-bs-256color| simpleterm with backspace as backspace and 256colors,
 | 
			
		||||
	use=st-256color,
 | 
			
		||||
	kbs=\010,
 | 
			
		||||
	kdch1=\177,
 | 
			
		||||
| 
						 | 
				
			
			@ -1,39 +0,0 @@
 | 
			
		|||
/* See LICENSE for license details. */
 | 
			
		||||
 | 
			
		||||
enum win_mode {
 | 
			
		||||
	MODE_VISIBLE     = 1 << 0,
 | 
			
		||||
	MODE_FOCUSED     = 1 << 1,
 | 
			
		||||
	MODE_APPKEYPAD   = 1 << 2,
 | 
			
		||||
	MODE_MOUSEBTN    = 1 << 3,
 | 
			
		||||
	MODE_MOUSEMOTION = 1 << 4,
 | 
			
		||||
	MODE_REVERSE     = 1 << 5,
 | 
			
		||||
	MODE_KBDLOCK     = 1 << 6,
 | 
			
		||||
	MODE_HIDE        = 1 << 7,
 | 
			
		||||
	MODE_APPCURSOR   = 1 << 8,
 | 
			
		||||
	MODE_MOUSESGR    = 1 << 9,
 | 
			
		||||
	MODE_8BIT        = 1 << 10,
 | 
			
		||||
	MODE_BLINK       = 1 << 11,
 | 
			
		||||
	MODE_FBLINK      = 1 << 12,
 | 
			
		||||
	MODE_FOCUS       = 1 << 13,
 | 
			
		||||
	MODE_MOUSEX10    = 1 << 14,
 | 
			
		||||
	MODE_MOUSEMANY   = 1 << 15,
 | 
			
		||||
	MODE_BRCKTPASTE  = 1 << 16,
 | 
			
		||||
	MODE_NUMLOCK     = 1 << 17,
 | 
			
		||||
	MODE_MOUSE       = MODE_MOUSEBTN|MODE_MOUSEMOTION|MODE_MOUSEX10\
 | 
			
		||||
	                  |MODE_MOUSEMANY,
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
void xbell(void);
 | 
			
		||||
void xclipcopy(void);
 | 
			
		||||
void xdrawcursor(int, int, Glyph, int, int, Glyph, Line, int);
 | 
			
		||||
void xdrawline(Line, int, int, int);
 | 
			
		||||
void xfinishdraw(void);
 | 
			
		||||
void xloadcols(void);
 | 
			
		||||
int xsetcolorname(int, const char *);
 | 
			
		||||
void xsettitle(char *);
 | 
			
		||||
int xsetcursor(int);
 | 
			
		||||
void xsetmode(int, unsigned int);
 | 
			
		||||
void xsetpointermotion(int);
 | 
			
		||||
void xsetsel(char *);
 | 
			
		||||
int xstartdraw(void);
 | 
			
		||||
void xximspot(int, int);
 | 
			
		||||
							
								
								
									
										2070
									
								
								.suckless/st-8.3/x.c
								
								
								
								
							
							
						
						
									
										2070
									
								
								.suckless/st-8.3/x.c
								
								
								
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| 
						 | 
				
			
			@ -1,10 +0,0 @@
 | 
			
		|||
--- x.c
 | 
			
		||||
+++ x.c
 | 
			
		||||
@@ -673,6 +673,7 @@ setsel(char *str, Time t)
 | 
			
		||||
 	XSetSelectionOwner(xw.dpy, XA_PRIMARY, xw.win, t);
 | 
			
		||||
 	if (XGetSelectionOwner(xw.dpy, XA_PRIMARY) != xw.win)
 | 
			
		||||
 		selclear();
 | 
			
		||||
+	clipcopy(NULL);
 | 
			
		||||
 }
 | 
			
		||||
 
 | 
			
		||||
 void
 | 
			
		||||
		Loading…
	
		Reference in New Issue