commit 2d62a08faff524d83f58c622e0e17b986c7d0e97
parent 14ca14d4020e6d8348049ea643af4e611495c073
Author: Brian Swetland <swetland@frotz.net>
Date: Mon, 27 Feb 2023 10:13:19 -0800
Import termbox library from https://github.com/nsf/termbox
Upstream version is no longer maintained.
Discarded python module and "waf" build goop.
Trimmed the README to the essentials and added
a reference to the upstream project.
Renamed some files for clarity.
Diffstat:
12 files changed, 2928 insertions(+), 0 deletions(-)
diff --git a/termbox/LICENSE b/termbox/LICENSE
@@ -0,0 +1,19 @@
+Copyright (C) 2010-2013 nsf <no.smile.face@gmail.com>
+
+Permission is hereby granted, free of charge, to any person obtaining a copy
+of this software and associated documentation files (the "Software"), to deal
+in the Software without restriction, including without limitation the rights
+to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
+copies of the Software, and to permit persons to whom the Software is
+furnished to do so, subject to the following conditions:
+
+The above copyright notice and this permission notice shall be included in
+all copies or substantial portions of the Software.
+
+THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
+IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
+FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
+AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
+LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
+OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
+THE SOFTWARE.
diff --git a/termbox/README.md b/termbox/README.md
@@ -0,0 +1,105 @@
+
+## README
+
+This is a fork of https://github.com/nsf/termbox
+as of commit a04f34c11494a1dbfb6286fc95470dea4086c520
+"Add a note: project is no longer maintained"
+
+This README has been trimmed down to just the introduction and changelog
+(below) which remain unchanged from the upstream version:
+
+## Termbox
+
+Termbox is a library that provides minimalistic API which allows the
+programmer to write text-based user interfaces.
+
+It is based on a very simple abstraction. The main idea is viewing terminals as
+a table of fixed-size cells and input being a stream of structured
+messages. Would be fair to say that the model is inspired by windows console
+API. The abstraction itself is not perfect and it may create problems in certain
+areas. The most sensitive ones are copy & pasting and wide characters (mostly
+Chinese, Japanese, Korean (CJK) characters). When it comes to copy & pasting,
+the notion of cells is not really compatible with the idea of text. And CJK
+runes often require more than one cell to display them nicely. Despite the
+mentioned flaws, using such a simple model brings benefits in a form of
+simplicity. And KISS principle is important.
+
+At this point one should realize, that CLI (command-line interfaces) aren't
+really a thing termbox is aimed at. But rather pseudo-graphical user interfaces.
+
+### Getting started
+
+Termbox's interface only consists of 12 functions::
+
+```
+tb_init() // initialization
+tb_shutdown() // shutdown
+
+tb_width() // width of the terminal screen
+tb_height() // height of the terminal screen
+
+tb_clear() // clear buffer
+tb_present() // sync internal buffer with terminal
+
+tb_put_cell()
+tb_change_cell()
+tb_blit() // drawing functions
+
+tb_select_input_mode() // change input mode
+tb_peek_event() // peek a keyboard event
+tb_poll_event() // wait for a keyboard event
+```
+
+See src/termbox.h header file for full detail.
+
+### Changes
+
+v1.1.2:
+
+- Properly include changelog into the tagged version commit. I.e. I messed up
+ by tagging v1.1.1 and describing it in changelog after tagged commit. This
+ commit marked as v1.1.2 includes changelog for both v1.1.1 and v1.1.2. There
+ are no code changes in this minor release.
+
+v1.1.1:
+
+- Ncurses 6.1 compatibility fix. See https://github.com/nsf/termbox-go/issues/185.
+
+v1.1.0:
+
+- API: tb_width() and tb_height() are guaranteed to be negative if the termbox
+ wasn't initialized.
+- API: Output mode switching is now possible, adds 256-color and grayscale color
+ modes.
+- API: Better tb_blit() function. Thanks, Gunnar Zötl <gz@tset.de>.
+- API: New tb_cell_buffer() function for direct back buffer access.
+- API: Add new init function variants which allow using arbitrary file
+ descriptor as a terminal.
+- Improvements in input handling code.
+- Calling tb_shutdown() twice is detected and results in abort() to discourage
+ doing so.
+- Mouse event handling is ported from termbox-go.
+- Paint demo port from termbox-go to demonstrate mouse handling capabilities.
+- Bug fixes in code and documentation.
+
+v1.0.0:
+
+- Remove the Go directory. People generally know about termbox-go and where
+ to look for it.
+- Remove old terminfo-related python scripts and backport the new one from
+ termbox-go.
+- Remove cmake/make-based build scripts, use waf.
+- Add a simple terminfo database parser. Now termbox prefers using the
+ terminfo database if it can be found. Otherwise it still has a fallback
+ built-in database for most popular terminals.
+- Some internal code cleanups and refactorings. The most important change is
+ that termbox doesn't leak meaningless exported symbols like 'keys' and
+ 'funcs' now. Only the ones that have 'tb_' as a prefix are being exported.
+- API: Remove unsigned ints, use plain ints instead.
+- API: Rename UTF-8 functions 'utf8_*' -> 'tb_utf8_*'.
+- API: TB_DEFAULT equals 0 now, it means you can use attributes alones
+ assuming the default color.
+- API: Add TB_REVERSE.
+- API: Add TB_INPUT_CURRENT.
+- Move python module to its own directory and update it due to changes in the
+ termbox library.
diff --git a/termbox/bytebuffer.inl.c b/termbox/bytebuffer.inl.c
@@ -0,0 +1,70 @@
+struct bytebuffer {
+ char *buf;
+ int len;
+ int cap;
+};
+
+static void bytebuffer_reserve(struct bytebuffer *b, int cap) {
+ if (b->cap >= cap) {
+ return;
+ }
+
+ // prefer doubling capacity
+ if (b->cap * 2 >= cap) {
+ cap = b->cap * 2;
+ }
+
+ char *newbuf = realloc(b->buf, cap);
+ b->buf = newbuf;
+ b->cap = cap;
+}
+
+static void bytebuffer_init(struct bytebuffer *b, int cap) {
+ b->cap = 0;
+ b->len = 0;
+ b->buf = 0;
+
+ if (cap > 0) {
+ b->cap = cap;
+ b->buf = malloc(cap); // just assume malloc works always
+ }
+}
+
+static void bytebuffer_free(struct bytebuffer *b) {
+ if (b->buf)
+ free(b->buf);
+}
+
+static void bytebuffer_clear(struct bytebuffer *b) {
+ b->len = 0;
+}
+
+static void bytebuffer_append(struct bytebuffer *b, const char *data, int len) {
+ bytebuffer_reserve(b, b->len + len);
+ memcpy(b->buf + b->len, data, len);
+ b->len += len;
+}
+
+static void bytebuffer_puts(struct bytebuffer *b, const char *str) {
+ bytebuffer_append(b, str, strlen(str));
+}
+
+static void bytebuffer_resize(struct bytebuffer *b, int len) {
+ bytebuffer_reserve(b, len);
+ b->len = len;
+}
+
+static void bytebuffer_flush(struct bytebuffer *b, int fd) {
+ write(fd, b->buf, b->len);
+ bytebuffer_clear(b);
+}
+
+static void bytebuffer_truncate(struct bytebuffer *b, int n) {
+ if (n <= 0)
+ return;
+ if (n > b->len)
+ n = b->len;
+ const int nmove = b->len - n;
+ memmove(b->buf, b->buf+n, nmove);
+ b->len -= n;
+}
diff --git a/termbox/extras/collect_terminfo.py b/termbox/extras/collect_terminfo.py
@@ -0,0 +1,108 @@
+#!/usr/bin/env python
+
+import sys, os, subprocess
+
+def escaped(s):
+ return s.replace("\033", "\\033")
+
+def tput(term, name):
+ try:
+ return subprocess.check_output(['tput', '-T%s' % term, name]).decode()
+ except subprocess.CalledProcessError as e:
+ return e.output.decode()
+
+
+def w(s):
+ if s == None:
+ return
+ sys.stdout.write(s)
+
+terminals = {
+ 'xterm' : 'xterm',
+ 'rxvt-256color' : 'rxvt_256color',
+ 'rxvt-unicode' : 'rxvt_unicode',
+ 'linux' : 'linux',
+ 'Eterm' : 'eterm',
+ 'screen' : 'screen'
+}
+
+keys = [
+ "F1", "kf1",
+ "F2", "kf2",
+ "F3", "kf3",
+ "F4", "kf4",
+ "F5", "kf5",
+ "F6", "kf6",
+ "F7", "kf7",
+ "F8", "kf8",
+ "F9", "kf9",
+ "F10", "kf10",
+ "F11", "kf11",
+ "F12", "kf12",
+ "INSERT", "kich1",
+ "DELETE", "kdch1",
+ "HOME", "khome",
+ "END", "kend",
+ "PGUP", "kpp",
+ "PGDN", "knp",
+ "KEY_UP", "kcuu1",
+ "KEY_DOWN", "kcud1",
+ "KEY_LEFT", "kcub1",
+ "KEY_RIGHT", "kcuf1"
+]
+
+funcs = [
+ "T_ENTER_CA", "smcup",
+ "T_EXIT_CA", "rmcup",
+ "T_SHOW_CURSOR", "cnorm",
+ "T_HIDE_CURSOR", "civis",
+ "T_CLEAR_SCREEN", "clear",
+ "T_SGR0", "sgr0",
+ "T_UNDERLINE", "smul",
+ "T_BOLD", "bold",
+ "T_BLINK", "blink",
+ "T_REVERSE", "rev",
+ "T_ENTER_KEYPAD", "smkx",
+ "T_EXIT_KEYPAD", "rmkx"
+]
+
+def iter_pairs(iterable):
+ iterable = iter(iterable)
+ while True:
+ yield (next(iterable), next(iterable))
+
+def do_term(term, nick):
+ w("// %s\n" % term)
+ w("static const char *%s_keys[] = {\n\t" % nick)
+ for k, v in iter_pairs(keys):
+ w('"')
+ w(escaped(tput(term, v)))
+ w('",')
+ w(" 0\n};\n")
+ w("static const char *%s_funcs[] = {\n\t" % nick)
+ for k,v in iter_pairs(funcs):
+ w('"')
+ if v == "sgr":
+ w("\\033[3%d;4%dm")
+ elif v == "cup":
+ w("\\033[%d;%dH")
+ else:
+ w(escaped(tput(term, v)))
+ w('", ')
+ w("\n};\n\n")
+
+def do_terms(d):
+ w("static struct term {\n")
+ w("\tconst char *name;\n")
+ w("\tconst char **keys;\n")
+ w("\tconst char **funcs;\n")
+ w("} terms[] = {\n")
+ for k, v in d.items():
+ w('\t{"%s", %s_keys, %s_funcs},\n' % (k, v, v))
+ w("\t{0, 0, 0},\n")
+ w("};\n")
+
+for k,v in terminals.items():
+ do_term(k, v)
+
+do_terms(terminals)
diff --git a/termbox/extras/keyboard.demo.c b/termbox/extras/keyboard.demo.c
@@ -0,0 +1,751 @@
+#include <assert.h>
+#include <stdint.h>
+#include <stdarg.h>
+#include <stdio.h>
+#include "../termbox.h"
+
+struct key {
+ unsigned char x;
+ unsigned char y;
+ uint32_t ch;
+};
+
+#define STOP {0,0,0}
+struct key K_ESC[] = {{1,1,'E'},{2,1,'S'},{3,1,'C'},STOP};
+struct key K_F1[] = {{6,1,'F'},{7,1,'1'},STOP};
+struct key K_F2[] = {{9,1,'F'},{10,1,'2'},STOP};
+struct key K_F3[] = {{12,1,'F'},{13,1,'3'},STOP};
+struct key K_F4[] = {{15,1,'F'},{16,1,'4'},STOP};
+struct key K_F5[] = {{19,1,'F'},{20,1,'5'},STOP};
+struct key K_F6[] = {{22,1,'F'},{23,1,'6'},STOP};
+struct key K_F7[] = {{25,1,'F'},{26,1,'7'},STOP};
+struct key K_F8[] = {{28,1,'F'},{29,1,'8'},STOP};
+struct key K_F9[] = {{33,1,'F'},{34,1,'9'},STOP};
+struct key K_F10[] = {{36,1,'F'},{37,1,'1'},{38,1,'0'},STOP};
+struct key K_F11[] = {{40,1,'F'},{41,1,'1'},{42,1,'1'},STOP};
+struct key K_F12[] = {{44,1,'F'},{45,1,'1'},{46,1,'2'},STOP};
+struct key K_PRN[] = {{50,1,'P'},{51,1,'R'},{52,1,'N'},STOP};
+struct key K_SCR[] = {{54,1,'S'},{55,1,'C'},{56,1,'R'},STOP};
+struct key K_BRK[] = {{58,1,'B'},{59,1,'R'},{60,1,'K'},STOP};
+struct key K_LED1[] = {{66,1,'-'},STOP};
+struct key K_LED2[] = {{70,1,'-'},STOP};
+struct key K_LED3[] = {{74,1,'-'},STOP};
+
+struct key K_TILDE[] = {{1,4,'`'},STOP};
+struct key K_TILDE_SHIFT[] = {{1,4,'~'},STOP};
+struct key K_1[] = {{4,4,'1'},STOP};
+struct key K_1_SHIFT[] = {{4,4,'!'},STOP};
+struct key K_2[] = {{7,4,'2'},STOP};
+struct key K_2_SHIFT[] = {{7,4,'@'},STOP};
+struct key K_3[] = {{10,4,'3'},STOP};
+struct key K_3_SHIFT[] = {{10,4,'#'},STOP};
+struct key K_4[] = {{13,4,'4'},STOP};
+struct key K_4_SHIFT[] = {{13,4,'$'},STOP};
+struct key K_5[] = {{16,4,'5'},STOP};
+struct key K_5_SHIFT[] = {{16,4,'%'},STOP};
+struct key K_6[] = {{19,4,'6'},STOP};
+struct key K_6_SHIFT[] = {{19,4,'^'},STOP};
+struct key K_7[] = {{22,4,'7'},STOP};
+struct key K_7_SHIFT[] = {{22,4,'&'},STOP};
+struct key K_8[] = {{25,4,'8'},STOP};
+struct key K_8_SHIFT[] = {{25,4,'*'},STOP};
+struct key K_9[] = {{28,4,'9'},STOP};
+struct key K_9_SHIFT[] = {{28,4,'('},STOP};
+struct key K_0[] = {{31,4,'0'},STOP};
+struct key K_0_SHIFT[] = {{31,4,')'},STOP};
+struct key K_MINUS[] = {{34,4,'-'},STOP};
+struct key K_MINUS_SHIFT[] = {{34,4,'_'},STOP};
+struct key K_EQUALS[] = {{37,4,'='},STOP};
+struct key K_EQUALS_SHIFT[] = {{37,4,'+'},STOP};
+struct key K_BACKSLASH[] = {{40,4,'\\'},STOP};
+struct key K_BACKSLASH_SHIFT[] = {{40,4,'|'},STOP};
+struct key K_BACKSPACE[] = {{44,4,0x2190},{45,4,0x2500},{46,4,0x2500},STOP};
+struct key K_INS[] = {{50,4,'I'},{51,4,'N'},{52,4,'S'},STOP};
+struct key K_HOM[] = {{54,4,'H'},{55,4,'O'},{56,4,'M'},STOP};
+struct key K_PGU[] = {{58,4,'P'},{59,4,'G'},{60,4,'U'},STOP};
+struct key K_K_NUMLOCK[] = {{65,4,'N'},STOP};
+struct key K_K_SLASH[] = {{68,4,'/'},STOP};
+struct key K_K_STAR[] = {{71,4,'*'},STOP};
+struct key K_K_MINUS[] = {{74,4,'-'},STOP};
+
+struct key K_TAB[] = {{1,6,'T'},{2,6,'A'},{3,6,'B'},STOP};
+struct key K_q[] = {{6,6,'q'},STOP};
+struct key K_Q[] = {{6,6,'Q'},STOP};
+struct key K_w[] = {{9,6,'w'},STOP};
+struct key K_W[] = {{9,6,'W'},STOP};
+struct key K_e[] = {{12,6,'e'},STOP};
+struct key K_E[] = {{12,6,'E'},STOP};
+struct key K_r[] = {{15,6,'r'},STOP};
+struct key K_R[] = {{15,6,'R'},STOP};
+struct key K_t[] = {{18,6,'t'},STOP};
+struct key K_T[] = {{18,6,'T'},STOP};
+struct key K_y[] = {{21,6,'y'},STOP};
+struct key K_Y[] = {{21,6,'Y'},STOP};
+struct key K_u[] = {{24,6,'u'},STOP};
+struct key K_U[] = {{24,6,'U'},STOP};
+struct key K_i[] = {{27,6,'i'},STOP};
+struct key K_I[] = {{27,6,'I'},STOP};
+struct key K_o[] = {{30,6,'o'},STOP};
+struct key K_O[] = {{30,6,'O'},STOP};
+struct key K_p[] = {{33,6,'p'},STOP};
+struct key K_P[] = {{33,6,'P'},STOP};
+struct key K_LSQB[] = {{36,6,'['},STOP};
+struct key K_LCUB[] = {{36,6,'{'},STOP};
+struct key K_RSQB[] = {{39,6,']'},STOP};
+struct key K_RCUB[] = {{39,6,'}'},STOP};
+struct key K_ENTER[] = {
+ {43,6,0x2591},{44,6,0x2591},{45,6,0x2591},{46,6,0x2591},
+ {43,7,0x2591},{44,7,0x2591},{45,7,0x21B5},{46,7,0x2591},
+ {41,8,0x2591},{42,8,0x2591},{43,8,0x2591},{44,8,0x2591},
+ {45,8,0x2591},{46,8,0x2591},STOP
+};
+struct key K_DEL[] = {{50,6,'D'},{51,6,'E'},{52,6,'L'},STOP};
+struct key K_END[] = {{54,6,'E'},{55,6,'N'},{56,6,'D'},STOP};
+struct key K_PGD[] = {{58,6,'P'},{59,6,'G'},{60,6,'D'},STOP};
+struct key K_K_7[] = {{65,6,'7'},STOP};
+struct key K_K_8[] = {{68,6,'8'},STOP};
+struct key K_K_9[] = {{71,6,'9'},STOP};
+struct key K_K_PLUS[] = {{74,6,' '},{74,7,'+'},{74,8,' '},STOP};
+
+struct key K_CAPS[] = {{1,8,'C'},{2,8,'A'},{3,8,'P'},{4,8,'S'},STOP};
+struct key K_a[] = {{7,8,'a'},STOP};
+struct key K_A[] = {{7,8,'A'},STOP};
+struct key K_s[] = {{10,8,'s'},STOP};
+struct key K_S[] = {{10,8,'S'},STOP};
+struct key K_d[] = {{13,8,'d'},STOP};
+struct key K_D[] = {{13,8,'D'},STOP};
+struct key K_f[] = {{16,8,'f'},STOP};
+struct key K_F[] = {{16,8,'F'},STOP};
+struct key K_g[] = {{19,8,'g'},STOP};
+struct key K_G[] = {{19,8,'G'},STOP};
+struct key K_h[] = {{22,8,'h'},STOP};
+struct key K_H[] = {{22,8,'H'},STOP};
+struct key K_j[] = {{25,8,'j'},STOP};
+struct key K_J[] = {{25,8,'J'},STOP};
+struct key K_k[] = {{28,8,'k'},STOP};
+struct key K_K[] = {{28,8,'K'},STOP};
+struct key K_l[] = {{31,8,'l'},STOP};
+struct key K_L[] = {{31,8,'L'},STOP};
+struct key K_SEMICOLON[] = {{34,8,';'},STOP};
+struct key K_PARENTHESIS[] = {{34,8,':'},STOP};
+struct key K_QUOTE[] = {{37,8,'\''},STOP};
+struct key K_DOUBLEQUOTE[] = {{37,8,'"'},STOP};
+struct key K_K_4[] = {{65,8,'4'},STOP};
+struct key K_K_5[] = {{68,8,'5'},STOP};
+struct key K_K_6[] = {{71,8,'6'},STOP};
+
+struct key K_LSHIFT[] = {{1,10,'S'},{2,10,'H'},{3,10,'I'},{4,10,'F'},{5,10,'T'},STOP};
+struct key K_z[] = {{9,10,'z'},STOP};
+struct key K_Z[] = {{9,10,'Z'},STOP};
+struct key K_x[] = {{12,10,'x'},STOP};
+struct key K_X[] = {{12,10,'X'},STOP};
+struct key K_c[] = {{15,10,'c'},STOP};
+struct key K_C[] = {{15,10,'C'},STOP};
+struct key K_v[] = {{18,10,'v'},STOP};
+struct key K_V[] = {{18,10,'V'},STOP};
+struct key K_b[] = {{21,10,'b'},STOP};
+struct key K_B[] = {{21,10,'B'},STOP};
+struct key K_n[] = {{24,10,'n'},STOP};
+struct key K_N[] = {{24,10,'N'},STOP};
+struct key K_m[] = {{27,10,'m'},STOP};
+struct key K_M[] = {{27,10,'M'},STOP};
+struct key K_COMMA[] = {{30,10,','},STOP};
+struct key K_LANB[] = {{30,10,'<'},STOP};
+struct key K_PERIOD[] = {{33,10,'.'},STOP};
+struct key K_RANB[] = {{33,10,'>'},STOP};
+struct key K_SLASH[] = {{36,10,'/'},STOP};
+struct key K_QUESTION[] = {{36,10,'?'},STOP};
+struct key K_RSHIFT[] = {{42,10,'S'},{43,10,'H'},{44,10,'I'},{45,10,'F'},{46,10,'T'},STOP};
+struct key K_ARROW_UP[] = {{54,10,'('},{55,10,0x2191},{56,10,')'},STOP};
+struct key K_K_1[] = {{65,10,'1'},STOP};
+struct key K_K_2[] = {{68,10,'2'},STOP};
+struct key K_K_3[] = {{71,10,'3'},STOP};
+struct key K_K_ENTER[] = {{74,10,0x2591},{74,11,0x2591},{74,12,0x2591},STOP};
+
+struct key K_LCTRL[] = {{1,12,'C'},{2,12,'T'},{3,12,'R'},{4,12,'L'},STOP};
+struct key K_LWIN[] = {{6,12,'W'},{7,12,'I'},{8,12,'N'},STOP};
+struct key K_LALT[] = {{10,12,'A'},{11,12,'L'},{12,12,'T'},STOP};
+struct key K_SPACE[] = {
+ {14,12,' '},{15,12,' '},{16,12,' '},{17,12,' '},{18,12,' '},
+ {19,12,'S'},{20,12,'P'},{21,12,'A'},{22,12,'C'},{23,12,'E'},
+ {24,12,' '},{25,12,' '},{26,12,' '},{27,12,' '},{28,12,' '},
+ STOP
+};
+struct key K_RALT[] = {{30,12,'A'},{31,12,'L'},{32,12,'T'},STOP};
+struct key K_RWIN[] = {{34,12,'W'},{35,12,'I'},{36,12,'N'},STOP};
+struct key K_RPROP[] = {{38,12,'P'},{39,12,'R'},{40,12,'O'},{41,12,'P'},STOP};
+struct key K_RCTRL[] = {{43,12,'C'},{44,12,'T'},{45,12,'R'},{46,12,'L'},STOP};
+struct key K_ARROW_LEFT[] = {{50,12,'('},{51,12,0x2190},{52,12,')'},STOP};
+struct key K_ARROW_DOWN[] = {{54,12,'('},{55,12,0x2193},{56,12,')'},STOP};
+struct key K_ARROW_RIGHT[] = {{58,12,'('},{59,12,0x2192},{60,12,')'},STOP};
+struct key K_K_0[] = {{65,12,' '},{66,12,'0'},{67,12,' '},{68,12,' '},STOP};
+struct key K_K_PERIOD[] = {{71,12,'.'},STOP};
+
+struct combo {
+ struct key *keys[6];
+};
+
+struct combo combos[] = {
+ {{K_TILDE, K_2, K_LCTRL, K_RCTRL, 0}},
+ {{K_A, K_LCTRL, K_RCTRL, 0}},
+ {{K_B, K_LCTRL, K_RCTRL, 0}},
+ {{K_C, K_LCTRL, K_RCTRL, 0}},
+ {{K_D, K_LCTRL, K_RCTRL, 0}},
+ {{K_E, K_LCTRL, K_RCTRL, 0}},
+ {{K_F, K_LCTRL, K_RCTRL, 0}},
+ {{K_G, K_LCTRL, K_RCTRL, 0}},
+ {{K_H, K_BACKSPACE, K_LCTRL, K_RCTRL, 0}},
+ {{K_I, K_TAB, K_LCTRL, K_RCTRL, 0}},
+ {{K_J, K_LCTRL, K_RCTRL, 0}},
+ {{K_K, K_LCTRL, K_RCTRL, 0}},
+ {{K_L, K_LCTRL, K_RCTRL, 0}},
+ {{K_M, K_ENTER, K_K_ENTER, K_LCTRL, K_RCTRL, 0}},
+ {{K_N, K_LCTRL, K_RCTRL, 0}},
+ {{K_O, K_LCTRL, K_RCTRL, 0}},
+ {{K_P, K_LCTRL, K_RCTRL, 0}},
+ {{K_Q, K_LCTRL, K_RCTRL, 0}},
+ {{K_R, K_LCTRL, K_RCTRL, 0}},
+ {{K_S, K_LCTRL, K_RCTRL, 0}},
+ {{K_T, K_LCTRL, K_RCTRL, 0}},
+ {{K_U, K_LCTRL, K_RCTRL, 0}},
+ {{K_V, K_LCTRL, K_RCTRL, 0}},
+ {{K_W, K_LCTRL, K_RCTRL, 0}},
+ {{K_X, K_LCTRL, K_RCTRL, 0}},
+ {{K_Y, K_LCTRL, K_RCTRL, 0}},
+ {{K_Z, K_LCTRL, K_RCTRL, 0}},
+ {{K_LSQB, K_ESC, K_3, K_LCTRL, K_RCTRL, 0}},
+ {{K_4, K_BACKSLASH, K_LCTRL, K_RCTRL, 0}},
+ {{K_RSQB, K_5, K_LCTRL, K_RCTRL, 0}},
+ {{K_6, K_LCTRL, K_RCTRL, 0}},
+ {{K_7, K_SLASH, K_MINUS_SHIFT, K_LCTRL, K_RCTRL, 0}},
+ {{K_SPACE,0}},
+ {{K_1_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_DOUBLEQUOTE,K_LSHIFT,K_RSHIFT,0}},
+ {{K_3_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_4_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_5_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_7_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_QUOTE,0}},
+ {{K_9_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_0_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_8_SHIFT,K_K_STAR,K_LSHIFT,K_RSHIFT,0}},
+ {{K_EQUALS_SHIFT,K_K_PLUS,K_LSHIFT,K_RSHIFT,0}},
+ {{K_COMMA,0}},
+ {{K_MINUS,K_K_MINUS,0}},
+ {{K_PERIOD,K_K_PERIOD,0}},
+ {{K_SLASH,K_K_SLASH,0}},
+ {{K_0,K_K_0,0}},
+ {{K_1,K_K_1,0}},
+ {{K_2,K_K_2,0}},
+ {{K_3,K_K_3,0}},
+ {{K_4,K_K_4,0}},
+ {{K_5,K_K_5,0}},
+ {{K_6,K_K_6,0}},
+ {{K_7,K_K_7,0}},
+ {{K_8,K_K_8,0}},
+ {{K_9,K_K_9,0}},
+ {{K_PARENTHESIS,K_LSHIFT,K_RSHIFT,0}},
+ {{K_SEMICOLON,0}},
+ {{K_LANB,K_LSHIFT,K_RSHIFT,0}},
+ {{K_EQUALS,0}},
+ {{K_RANB,K_LSHIFT,K_RSHIFT,0}},
+ {{K_QUESTION,K_LSHIFT,K_RSHIFT,0}},
+ {{K_2_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_A,K_LSHIFT,K_RSHIFT,0}},
+ {{K_B,K_LSHIFT,K_RSHIFT,0}},
+ {{K_C,K_LSHIFT,K_RSHIFT,0}},
+ {{K_D,K_LSHIFT,K_RSHIFT,0}},
+ {{K_E,K_LSHIFT,K_RSHIFT,0}},
+ {{K_F,K_LSHIFT,K_RSHIFT,0}},
+ {{K_G,K_LSHIFT,K_RSHIFT,0}},
+ {{K_H,K_LSHIFT,K_RSHIFT,0}},
+ {{K_I,K_LSHIFT,K_RSHIFT,0}},
+ {{K_J,K_LSHIFT,K_RSHIFT,0}},
+ {{K_K,K_LSHIFT,K_RSHIFT,0}},
+ {{K_L,K_LSHIFT,K_RSHIFT,0}},
+ {{K_M,K_LSHIFT,K_RSHIFT,0}},
+ {{K_N,K_LSHIFT,K_RSHIFT,0}},
+ {{K_O,K_LSHIFT,K_RSHIFT,0}},
+ {{K_P,K_LSHIFT,K_RSHIFT,0}},
+ {{K_Q,K_LSHIFT,K_RSHIFT,0}},
+ {{K_R,K_LSHIFT,K_RSHIFT,0}},
+ {{K_S,K_LSHIFT,K_RSHIFT,0}},
+ {{K_T,K_LSHIFT,K_RSHIFT,0}},
+ {{K_U,K_LSHIFT,K_RSHIFT,0}},
+ {{K_V,K_LSHIFT,K_RSHIFT,0}},
+ {{K_W,K_LSHIFT,K_RSHIFT,0}},
+ {{K_X,K_LSHIFT,K_RSHIFT,0}},
+ {{K_Y,K_LSHIFT,K_RSHIFT,0}},
+ {{K_Z,K_LSHIFT,K_RSHIFT,0}},
+ {{K_LSQB,0}},
+ {{K_BACKSLASH,0}},
+ {{K_RSQB,0}},
+ {{K_6_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_MINUS_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_TILDE,0}},
+ {{K_a,0}},
+ {{K_b,0}},
+ {{K_c,0}},
+ {{K_d,0}},
+ {{K_e,0}},
+ {{K_f,0}},
+ {{K_g,0}},
+ {{K_h,0}},
+ {{K_i,0}},
+ {{K_j,0}},
+ {{K_k,0}},
+ {{K_l,0}},
+ {{K_m,0}},
+ {{K_n,0}},
+ {{K_o,0}},
+ {{K_p,0}},
+ {{K_q,0}},
+ {{K_r,0}},
+ {{K_s,0}},
+ {{K_t,0}},
+ {{K_u,0}},
+ {{K_v,0}},
+ {{K_w,0}},
+ {{K_x,0}},
+ {{K_y,0}},
+ {{K_z,0}},
+ {{K_LCUB,K_LSHIFT,K_RSHIFT,0}},
+ {{K_BACKSLASH_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_RCUB,K_LSHIFT,K_RSHIFT,0}},
+ {{K_TILDE_SHIFT,K_LSHIFT,K_RSHIFT,0}},
+ {{K_8, K_BACKSPACE, K_LCTRL, K_RCTRL, 0}}
+};
+
+struct combo func_combos[] = {
+ {{K_F1,0}},
+ {{K_F2,0}},
+ {{K_F3,0}},
+ {{K_F4,0}},
+ {{K_F5,0}},
+ {{K_F6,0}},
+ {{K_F7,0}},
+ {{K_F8,0}},
+ {{K_F9,0}},
+ {{K_F10,0}},
+ {{K_F11,0}},
+ {{K_F12,0}},
+ {{K_INS,0}},
+ {{K_DEL,0}},
+ {{K_HOM,0}},
+ {{K_END,0}},
+ {{K_PGU,0}},
+ {{K_PGD,0}},
+ {{K_ARROW_UP,0}},
+ {{K_ARROW_DOWN,0}},
+ {{K_ARROW_LEFT,0}},
+ {{K_ARROW_RIGHT,0}}
+};
+
+void print_tb(const char *str, int x, int y, uint16_t fg, uint16_t bg)
+{
+ while (*str) {
+ uint32_t uni;
+ str += tb_utf8_char_to_unicode(&uni, str);
+ tb_change_cell(x, y, uni, fg, bg);
+ x++;
+ }
+}
+
+void printf_tb(int x, int y, uint16_t fg, uint16_t bg, const char *fmt, ...)
+{
+ char buf[4096];
+ va_list vl;
+ va_start(vl, fmt);
+ vsnprintf(buf, sizeof(buf), fmt, vl);
+ va_end(vl);
+ print_tb(buf, x, y, fg, bg);
+}
+
+void draw_key(struct key *k, uint16_t fg, uint16_t bg)
+{
+ while (k->x) {
+ tb_change_cell(k->x+2, k->y+4, k->ch, fg, bg);
+ k++;
+ }
+}
+
+void draw_keyboard()
+{
+ int i;
+ tb_change_cell(0, 0, 0x250C, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(79, 0, 0x2510, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(0, 23, 0x2514, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(79, 23, 0x2518, TB_WHITE, TB_DEFAULT);
+
+ for (i = 1; i < 79; ++i) {
+ tb_change_cell(i, 0, 0x2500, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(i, 23, 0x2500, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(i, 17, 0x2500, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(i, 4, 0x2500, TB_WHITE, TB_DEFAULT);
+ }
+ for (i = 1; i < 23; ++i) {
+ tb_change_cell(0, i, 0x2502, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(79, i, 0x2502, TB_WHITE, TB_DEFAULT);
+ }
+ tb_change_cell(0, 17, 0x251C, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(79, 17, 0x2524, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(0, 4, 0x251C, TB_WHITE, TB_DEFAULT);
+ tb_change_cell(79, 4, 0x2524, TB_WHITE, TB_DEFAULT);
+ for (i = 5; i < 17; ++i) {
+ tb_change_cell(1, i, 0x2588, TB_YELLOW, TB_YELLOW);
+ tb_change_cell(78, i, 0x2588, TB_YELLOW, TB_YELLOW);
+ }
+
+ draw_key(K_ESC, TB_WHITE, TB_BLUE);
+ draw_key(K_F1, TB_WHITE, TB_BLUE);
+ draw_key(K_F2, TB_WHITE, TB_BLUE);
+ draw_key(K_F3, TB_WHITE, TB_BLUE);
+ draw_key(K_F4, TB_WHITE, TB_BLUE);
+ draw_key(K_F5, TB_WHITE, TB_BLUE);
+ draw_key(K_F6, TB_WHITE, TB_BLUE);
+ draw_key(K_F7, TB_WHITE, TB_BLUE);
+ draw_key(K_F8, TB_WHITE, TB_BLUE);
+ draw_key(K_F9, TB_WHITE, TB_BLUE);
+ draw_key(K_F10, TB_WHITE, TB_BLUE);
+ draw_key(K_F11, TB_WHITE, TB_BLUE);
+ draw_key(K_F12, TB_WHITE, TB_BLUE);
+ draw_key(K_PRN, TB_WHITE, TB_BLUE);
+ draw_key(K_SCR, TB_WHITE, TB_BLUE);
+ draw_key(K_BRK, TB_WHITE, TB_BLUE);
+ draw_key(K_LED1, TB_WHITE, TB_BLUE);
+ draw_key(K_LED2, TB_WHITE, TB_BLUE);
+ draw_key(K_LED3, TB_WHITE, TB_BLUE);
+
+ draw_key(K_TILDE, TB_WHITE, TB_BLUE);
+ draw_key(K_1, TB_WHITE, TB_BLUE);
+ draw_key(K_2, TB_WHITE, TB_BLUE);
+ draw_key(K_3, TB_WHITE, TB_BLUE);
+ draw_key(K_4, TB_WHITE, TB_BLUE);
+ draw_key(K_5, TB_WHITE, TB_BLUE);
+ draw_key(K_6, TB_WHITE, TB_BLUE);
+ draw_key(K_7, TB_WHITE, TB_BLUE);
+ draw_key(K_8, TB_WHITE, TB_BLUE);
+ draw_key(K_9, TB_WHITE, TB_BLUE);
+ draw_key(K_0, TB_WHITE, TB_BLUE);
+ draw_key(K_MINUS, TB_WHITE, TB_BLUE);
+ draw_key(K_EQUALS, TB_WHITE, TB_BLUE);
+ draw_key(K_BACKSLASH, TB_WHITE, TB_BLUE);
+ draw_key(K_BACKSPACE, TB_WHITE, TB_BLUE);
+ draw_key(K_INS, TB_WHITE, TB_BLUE);
+ draw_key(K_HOM, TB_WHITE, TB_BLUE);
+ draw_key(K_PGU, TB_WHITE, TB_BLUE);
+ draw_key(K_K_NUMLOCK, TB_WHITE, TB_BLUE);
+ draw_key(K_K_SLASH, TB_WHITE, TB_BLUE);
+ draw_key(K_K_STAR, TB_WHITE, TB_BLUE);
+ draw_key(K_K_MINUS, TB_WHITE, TB_BLUE);
+
+ draw_key(K_TAB, TB_WHITE, TB_BLUE);
+ draw_key(K_q, TB_WHITE, TB_BLUE);
+ draw_key(K_w, TB_WHITE, TB_BLUE);
+ draw_key(K_e, TB_WHITE, TB_BLUE);
+ draw_key(K_r, TB_WHITE, TB_BLUE);
+ draw_key(K_t, TB_WHITE, TB_BLUE);
+ draw_key(K_y, TB_WHITE, TB_BLUE);
+ draw_key(K_u, TB_WHITE, TB_BLUE);
+ draw_key(K_i, TB_WHITE, TB_BLUE);
+ draw_key(K_o, TB_WHITE, TB_BLUE);
+ draw_key(K_p, TB_WHITE, TB_BLUE);
+ draw_key(K_LSQB, TB_WHITE, TB_BLUE);
+ draw_key(K_RSQB, TB_WHITE, TB_BLUE);
+ draw_key(K_ENTER, TB_WHITE, TB_BLUE);
+ draw_key(K_DEL, TB_WHITE, TB_BLUE);
+ draw_key(K_END, TB_WHITE, TB_BLUE);
+ draw_key(K_PGD, TB_WHITE, TB_BLUE);
+ draw_key(K_K_7, TB_WHITE, TB_BLUE);
+ draw_key(K_K_8, TB_WHITE, TB_BLUE);
+ draw_key(K_K_9, TB_WHITE, TB_BLUE);
+ draw_key(K_K_PLUS, TB_WHITE, TB_BLUE);
+
+ draw_key(K_CAPS, TB_WHITE, TB_BLUE);
+ draw_key(K_a, TB_WHITE, TB_BLUE);
+ draw_key(K_s, TB_WHITE, TB_BLUE);
+ draw_key(K_d, TB_WHITE, TB_BLUE);
+ draw_key(K_f, TB_WHITE, TB_BLUE);
+ draw_key(K_g, TB_WHITE, TB_BLUE);
+ draw_key(K_h, TB_WHITE, TB_BLUE);
+ draw_key(K_j, TB_WHITE, TB_BLUE);
+ draw_key(K_k, TB_WHITE, TB_BLUE);
+ draw_key(K_l, TB_WHITE, TB_BLUE);
+ draw_key(K_SEMICOLON, TB_WHITE, TB_BLUE);
+ draw_key(K_QUOTE, TB_WHITE, TB_BLUE);
+ draw_key(K_K_4, TB_WHITE, TB_BLUE);
+ draw_key(K_K_5, TB_WHITE, TB_BLUE);
+ draw_key(K_K_6, TB_WHITE, TB_BLUE);
+
+ draw_key(K_LSHIFT, TB_WHITE, TB_BLUE);
+ draw_key(K_z, TB_WHITE, TB_BLUE);
+ draw_key(K_x, TB_WHITE, TB_BLUE);
+ draw_key(K_c, TB_WHITE, TB_BLUE);
+ draw_key(K_v, TB_WHITE, TB_BLUE);
+ draw_key(K_b, TB_WHITE, TB_BLUE);
+ draw_key(K_n, TB_WHITE, TB_BLUE);
+ draw_key(K_m, TB_WHITE, TB_BLUE);
+ draw_key(K_COMMA, TB_WHITE, TB_BLUE);
+ draw_key(K_PERIOD, TB_WHITE, TB_BLUE);
+ draw_key(K_SLASH, TB_WHITE, TB_BLUE);
+ draw_key(K_RSHIFT, TB_WHITE, TB_BLUE);
+ draw_key(K_ARROW_UP, TB_WHITE, TB_BLUE);
+ draw_key(K_K_1, TB_WHITE, TB_BLUE);
+ draw_key(K_K_2, TB_WHITE, TB_BLUE);
+ draw_key(K_K_3, TB_WHITE, TB_BLUE);
+ draw_key(K_K_ENTER, TB_WHITE, TB_BLUE);
+
+ draw_key(K_LCTRL, TB_WHITE, TB_BLUE);
+ draw_key(K_LWIN, TB_WHITE, TB_BLUE);
+ draw_key(K_LALT, TB_WHITE, TB_BLUE);
+ draw_key(K_SPACE, TB_WHITE, TB_BLUE);
+ draw_key(K_RCTRL, TB_WHITE, TB_BLUE);
+ draw_key(K_RPROP, TB_WHITE, TB_BLUE);
+ draw_key(K_RWIN, TB_WHITE, TB_BLUE);
+ draw_key(K_RALT, TB_WHITE, TB_BLUE);
+ draw_key(K_ARROW_LEFT, TB_WHITE, TB_BLUE);
+ draw_key(K_ARROW_DOWN, TB_WHITE, TB_BLUE);
+ draw_key(K_ARROW_RIGHT, TB_WHITE, TB_BLUE);
+ draw_key(K_K_0, TB_WHITE, TB_BLUE);
+ draw_key(K_K_PERIOD, TB_WHITE, TB_BLUE);
+
+ printf_tb(33, 1, TB_MAGENTA | TB_BOLD, TB_DEFAULT, "Keyboard demo!");
+ printf_tb(21, 2, TB_MAGENTA, TB_DEFAULT, "(press CTRL+X and then CTRL+Q to exit)");
+ printf_tb(15, 3, TB_MAGENTA, TB_DEFAULT, "(press CTRL+X and then CTRL+C to change input mode)");
+
+ int inputmode = tb_select_input_mode(0);
+ char inputmode_str[64];
+
+ if (inputmode & TB_INPUT_ESC)
+ sprintf(inputmode_str, "TB_INPUT_ESC");
+ if (inputmode & TB_INPUT_ALT)
+ sprintf(inputmode_str, "TB_INPUT_ALT");
+
+ if (inputmode & TB_INPUT_MOUSE)
+ sprintf(inputmode_str, "%s | TB_INPUT_MOUSE", inputmode_str);
+
+ printf_tb(3, 18, TB_WHITE, TB_DEFAULT, "Input mode: %s", inputmode_str);
+}
+
+const char *funckeymap(int k)
+{
+ static const char *fcmap[] = {
+ "CTRL+2, CTRL+~",
+ "CTRL+A",
+ "CTRL+B",
+ "CTRL+C",
+ "CTRL+D",
+ "CTRL+E",
+ "CTRL+F",
+ "CTRL+G",
+ "CTRL+H, BACKSPACE",
+ "CTRL+I, TAB",
+ "CTRL+J",
+ "CTRL+K",
+ "CTRL+L",
+ "CTRL+M, ENTER",
+ "CTRL+N",
+ "CTRL+O",
+ "CTRL+P",
+ "CTRL+Q",
+ "CTRL+R",
+ "CTRL+S",
+ "CTRL+T",
+ "CTRL+U",
+ "CTRL+V",
+ "CTRL+W",
+ "CTRL+X",
+ "CTRL+Y",
+ "CTRL+Z",
+ "CTRL+3, ESC, CTRL+[",
+ "CTRL+4, CTRL+\\",
+ "CTRL+5, CTRL+]",
+ "CTRL+6",
+ "CTRL+7, CTRL+/, CTRL+_",
+ "SPACE"
+ };
+ static const char *fkmap[] = {
+ "F1",
+ "F2",
+ "F3",
+ "F4",
+ "F5",
+ "F6",
+ "F7",
+ "F8",
+ "F9",
+ "F10",
+ "F11",
+ "F12",
+ "INSERT",
+ "DELETE",
+ "HOME",
+ "END",
+ "PGUP",
+ "PGDN",
+ "ARROW UP",
+ "ARROW DOWN",
+ "ARROW LEFT",
+ "ARROW RIGHT"
+ };
+
+ if (k == TB_KEY_CTRL_8)
+ return "CTRL+8, BACKSPACE 2"; /* 0x7F */
+ else if (k >= TB_KEY_ARROW_RIGHT && k <= 0xFFFF)
+ return fkmap[0xFFFF-k];
+ else if (k <= TB_KEY_SPACE)
+ return fcmap[k];
+ return "UNKNOWN";
+}
+
+void pretty_print_press(struct tb_event *ev)
+{
+ char buf[7];
+ buf[tb_utf8_unicode_to_char(buf, ev->ch)] = '\0';
+ printf_tb(3, 19, TB_WHITE , TB_DEFAULT, "Key: ");
+ printf_tb(8, 19, TB_YELLOW, TB_DEFAULT, "decimal: %d", ev->key);
+ printf_tb(8, 20, TB_GREEN , TB_DEFAULT, "hex: 0x%X", ev->key);
+ printf_tb(8, 21, TB_CYAN , TB_DEFAULT, "octal: 0%o", ev->key);
+ printf_tb(8, 22, TB_RED , TB_DEFAULT, "string: %s", funckeymap(ev->key));
+
+ printf_tb(54, 19, TB_WHITE , TB_DEFAULT, "Char: ");
+ printf_tb(60, 19, TB_YELLOW, TB_DEFAULT, "decimal: %d", ev->ch);
+ printf_tb(60, 20, TB_GREEN , TB_DEFAULT, "hex: 0x%X", ev->ch);
+ printf_tb(60, 21, TB_CYAN , TB_DEFAULT, "octal: 0%o", ev->ch);
+ printf_tb(60, 22, TB_RED , TB_DEFAULT, "string: %s", buf);
+
+ printf_tb(54, 18, TB_WHITE, TB_DEFAULT, "Modifier: %s",
+ (ev->mod) ? "TB_MOD_ALT" : "none");
+
+}
+
+void pretty_print_resize(struct tb_event *ev)
+{
+ printf_tb(3, 19, TB_WHITE, TB_DEFAULT, "Resize event: %d x %d", ev->w, ev->h);
+}
+
+int counter = 0;
+
+void pretty_print_mouse(struct tb_event *ev) {
+ printf_tb(3, 19, TB_WHITE, TB_DEFAULT, "Mouse event: %d x %d", ev->x, ev->y);
+ char *btn = "";
+ switch (ev->key) {
+ case TB_KEY_MOUSE_LEFT:
+ btn = "MouseLeft: %d";
+ break;
+ case TB_KEY_MOUSE_MIDDLE:
+ btn = "MouseMiddle: %d";
+ break;
+ case TB_KEY_MOUSE_RIGHT:
+ btn = "MouseRight: %d";
+ break;
+ case TB_KEY_MOUSE_WHEEL_UP:
+ btn = "MouseWheelUp: %d";
+ break;
+ case TB_KEY_MOUSE_WHEEL_DOWN:
+ btn = "MouseWheelDown: %d";
+ break;
+ case TB_KEY_MOUSE_RELEASE:
+ btn = "MouseRelease: %d";
+ }
+ counter++;
+ printf_tb(43, 19, TB_WHITE, TB_DEFAULT, "Key: ");
+ printf_tb(48, 19, TB_YELLOW, TB_DEFAULT, btn, counter);
+}
+
+void dispatch_press(struct tb_event *ev)
+{
+ if (ev->mod & TB_MOD_ALT) {
+ draw_key(K_LALT, TB_WHITE, TB_RED);
+ draw_key(K_RALT, TB_WHITE, TB_RED);
+ }
+
+ struct combo *k = 0;
+ if (ev->key >= TB_KEY_ARROW_RIGHT)
+ k = &func_combos[0xFFFF-ev->key];
+ else if (ev->ch < 128) {
+ if (ev->ch == 0 && ev->key < 128)
+ k = &combos[ev->key];
+ else
+ k = &combos[ev->ch];
+ }
+ if (!k)
+ return;
+
+ struct key **keys = k->keys;
+ while (*keys) {
+ draw_key(*keys, TB_WHITE, TB_RED);
+ keys++;
+ }
+}
+
+int main(int argc, char **argv)
+{
+ (void) argc; (void) argv;
+ int ret;
+
+ ret = tb_init();
+ if (ret) {
+ fprintf(stderr, "tb_init() failed with error code %d\n", ret);
+ return 1;
+ }
+
+ tb_select_input_mode(TB_INPUT_ESC | TB_INPUT_MOUSE);
+ struct tb_event ev;
+
+ tb_clear();
+ draw_keyboard();
+ tb_present();
+ int inputmode = 0;
+ int ctrlxpressed = 0;
+
+ while (tb_poll_event(&ev)) {
+ switch (ev.type) {
+ case TB_EVENT_KEY:
+ if (ev.key == TB_KEY_CTRL_Q && ctrlxpressed) {
+ tb_shutdown();
+ return 0;
+ }
+ if (ev.key == TB_KEY_CTRL_C && ctrlxpressed) {
+ static int chmap[] = {
+ TB_INPUT_ESC | TB_INPUT_MOUSE, /* 101 */
+ TB_INPUT_ALT | TB_INPUT_MOUSE, /* 110 */
+ TB_INPUT_ESC, /* 001 */
+ TB_INPUT_ALT, /* 010 */
+ };
+ inputmode++;
+ if (inputmode >= 4) {
+ inputmode = 0;
+ }
+ tb_select_input_mode(chmap[inputmode]);
+ }
+ if (ev.key == TB_KEY_CTRL_X)
+ ctrlxpressed = 1;
+ else
+ ctrlxpressed = 0;
+
+ tb_clear();
+ draw_keyboard();
+ dispatch_press(&ev);
+ pretty_print_press(&ev);
+ tb_present();
+ break;
+ case TB_EVENT_RESIZE:
+ tb_clear();
+ draw_keyboard();
+ pretty_print_resize(&ev);
+ tb_present();
+ break;
+ case TB_EVENT_MOUSE:
+ tb_clear();
+ draw_keyboard();
+ pretty_print_mouse(&ev);
+ tb_present();
+ break;
+ default:
+ break;
+ }
+ }
+ tb_shutdown();
+ return 0;
+}
diff --git a/termbox/extras/output.demo.c b/termbox/extras/output.demo.c
@@ -0,0 +1,116 @@
+#include <stdio.h>
+#include <string.h>
+#include "../termbox.h"
+
+static const char chars[] = "nnnnnnnnnbbbbbbbbbuuuuuuuuuBBBBBBBBB";
+
+static const uint16_t all_attrs[] = {
+ 0,
+ TB_BOLD,
+ TB_UNDERLINE,
+ TB_BOLD | TB_UNDERLINE,
+};
+
+static int next_char(int current) {
+ current++;
+ if (!chars[current])
+ current = 0;
+ return current;
+}
+
+static void draw_line(int x, int y, uint16_t bg) {
+ int a, c;
+ int current_char = 0;
+ for (a = 0; a < 4; a++) {
+ for (c = TB_DEFAULT; c <= TB_WHITE; c++) {
+ uint16_t fg = all_attrs[a] | c;
+ tb_change_cell(x, y, chars[current_char], fg, bg);
+ current_char = next_char(current_char);
+ x++;
+ }
+ }
+}
+
+static void print_combinations_table(int sx, int sy, const uint16_t *attrs, int attrs_n) {
+ int i, c;
+ for (i = 0; i < attrs_n; i++) {
+ for (c = TB_DEFAULT; c <= TB_WHITE; c++) {
+ uint16_t bg = attrs[i] | c;
+ draw_line(sx, sy, bg);
+ sy++;
+ }
+ }
+}
+
+static void draw_all() {
+ tb_clear();
+
+ tb_select_output_mode(TB_OUTPUT_NORMAL);
+ static const uint16_t col1[] = {0, TB_BOLD};
+ static const uint16_t col2[] = {TB_REVERSE};
+ print_combinations_table(1, 1, col1, 2);
+ print_combinations_table(2 + strlen(chars), 1, col2, 1);
+ tb_present();
+
+ tb_select_output_mode(TB_OUTPUT_GRAYSCALE);
+ int c, x, y;
+ for (x = 0, y = 23; x < 24; ++x) {
+ tb_change_cell(x, y, '@', x, 0);
+ tb_change_cell(x+25, y, ' ', 0, x);
+ }
+ tb_present();
+
+ tb_select_output_mode(TB_OUTPUT_216);
+ y++;
+ for (c = 0, x = 0; c < 216; ++c, ++x) {
+ if (!(x%24)) {
+ x = 0;
+ ++y;
+ }
+ tb_change_cell(x, y, '@', c, 0);
+ tb_change_cell(x+25, y, ' ', 0, c);
+ }
+ tb_present();
+
+ tb_select_output_mode(TB_OUTPUT_256);
+ y++;
+ for (c = 0, x = 0; c < 256; ++c, ++x) {
+ if (!(x%24)) {
+ x = 0;
+ ++y;
+ }
+ tb_change_cell(x, y, '+', c | ((y & 1) ? TB_UNDERLINE : 0), 0);
+ tb_change_cell(x+25, y, ' ', 0, c);
+ }
+ tb_present();
+}
+
+int main(int argc, char **argv) {
+ (void)argc; (void)argv;
+ int ret = tb_init();
+ if (ret) {
+ fprintf(stderr, "tb_init() failed with error code %d\n", ret);
+ return 1;
+ }
+
+ draw_all();
+
+ struct tb_event ev;
+ while (tb_poll_event(&ev)) {
+ switch (ev.type) {
+ case TB_EVENT_KEY:
+ switch (ev.key) {
+ case TB_KEY_ESC:
+ goto done;
+ break;
+ }
+ break;
+ case TB_EVENT_RESIZE:
+ draw_all();
+ break;
+ }
+ }
+done:
+ tb_shutdown();
+ return 0;
+}
diff --git a/termbox/extras/paint.demo.c b/termbox/extras/paint.demo.c
@@ -0,0 +1,143 @@
+#include "../termbox.h"
+#include <stdlib.h>
+#include <stdio.h>
+#include <string.h>
+
+static int curCol = 0;
+static int curRune = 0;
+static struct tb_cell *backbuf;
+static int bbw = 0, bbh = 0;
+
+static const uint32_t runes[] = {
+ 0x20, // ' '
+ 0x2591, // '░'
+ 0x2592, // '▒'
+ 0x2593, // '▓'
+ 0x2588, // '█'
+};
+
+#define len(a) (sizeof(a)/sizeof(a[0]))
+
+static const uint16_t colors[] = {
+ TB_BLACK,
+ TB_RED,
+ TB_GREEN,
+ TB_YELLOW,
+ TB_BLUE,
+ TB_MAGENTA,
+ TB_CYAN,
+ TB_WHITE,
+};
+
+void updateAndDrawButtons(int *current, int x, int y, int mx, int my, int n, void (*attrFunc)(int, uint32_t*, uint16_t*, uint16_t*)) {
+ int lx = x;
+ int ly = y;
+ for (int i = 0; i < n; i++) {
+ if (lx <= mx && mx <= lx+3 && ly <= my && my <= ly+1) {
+ *current = i;
+ }
+ uint32_t r;
+ uint16_t fg, bg;
+ (*attrFunc)(i, &r, &fg, &bg);
+ tb_change_cell(lx+0, ly+0, r, fg, bg);
+ tb_change_cell(lx+1, ly+0, r, fg, bg);
+ tb_change_cell(lx+2, ly+0, r, fg, bg);
+ tb_change_cell(lx+3, ly+0, r, fg, bg);
+ tb_change_cell(lx+0, ly+1, r, fg, bg);
+ tb_change_cell(lx+1, ly+1, r, fg, bg);
+ tb_change_cell(lx+2, ly+1, r, fg, bg);
+ tb_change_cell(lx+3, ly+1, r, fg, bg);
+ lx += 4;
+ }
+ lx = x;
+ ly = y;
+ for (int i = 0; i < n; i++) {
+ if (*current == i) {
+ uint16_t fg = TB_RED | TB_BOLD;
+ uint16_t bg = TB_DEFAULT;
+ tb_change_cell(lx+0, ly+2, '^', fg, bg);
+ tb_change_cell(lx+1, ly+2, '^', fg, bg);
+ tb_change_cell(lx+2, ly+2, '^', fg, bg);
+ tb_change_cell(lx+3, ly+2, '^', fg, bg);
+ }
+ lx += 4;
+ }
+}
+
+void runeAttrFunc(int i, uint32_t *r, uint16_t *fg, uint16_t *bg) {
+ *r = runes[i];
+ *fg = TB_DEFAULT;
+ *bg = TB_DEFAULT;
+}
+
+void colorAttrFunc(int i, uint32_t *r, uint16_t *fg, uint16_t *bg) {
+ *r = ' ';
+ *fg = TB_DEFAULT;
+ *bg = colors[i];
+}
+
+void updateAndRedrawAll(int mx, int my) {
+ tb_clear();
+ if (mx != -1 && my != -1) {
+ backbuf[bbw*my+mx].ch = runes[curRune];
+ backbuf[bbw*my+mx].fg = colors[curCol];
+ }
+ memcpy(tb_cell_buffer(), backbuf, sizeof(struct tb_cell)*bbw*bbh);
+ int h = tb_height();
+ updateAndDrawButtons(&curRune, 0, 0, mx, my, len(runes), runeAttrFunc);
+ updateAndDrawButtons(&curCol, 0, h-3, mx, my, len(colors), colorAttrFunc);
+ tb_present();
+}
+
+void reallocBackBuffer(int w, int h) {
+ bbw = w;
+ bbh = h;
+ if (backbuf)
+ free(backbuf);
+ backbuf = calloc(sizeof(struct tb_cell), w*h);
+}
+
+int main(int argv, char **argc) {
+ (void)argc; (void)argv;
+ int code = tb_init();
+ if (code < 0) {
+ fprintf(stderr, "termbox init failed, code: %d\n", code);
+ return -1;
+ }
+
+ tb_select_input_mode(TB_INPUT_ESC | TB_INPUT_MOUSE);
+ int w = tb_width();
+ int h = tb_height();
+ reallocBackBuffer(w, h);
+ updateAndRedrawAll(-1, -1);
+ for (;;) {
+ struct tb_event ev;
+ int mx = -1;
+ int my = -1;
+ int t = tb_poll_event(&ev);
+ if (t == -1) {
+ tb_shutdown();
+ fprintf(stderr, "termbox poll event error\n");
+ return -1;
+ }
+
+ switch (t) {
+ case TB_EVENT_KEY:
+ if (ev.key == TB_KEY_ESC) {
+ tb_shutdown();
+ return 0;
+ }
+ break;
+ case TB_EVENT_MOUSE:
+ if (ev.key == TB_KEY_MOUSE_LEFT) {
+ mx = ev.x;
+ my = ev.y;
+ }
+ break;
+ case TB_EVENT_RESIZE:
+ reallocBackBuffer(ev.w, ev.h);
+ break;
+ }
+ updateAndRedrawAll(mx, my);
+ }
+}
diff --git a/termbox/input.inl.c b/termbox/input.inl.c
@@ -0,0 +1,228 @@
+// if s1 starts with s2 returns true, else false
+// len is the length of s1
+// s2 should be null-terminated
+static bool starts_with(const char *s1, int len, const char *s2)
+{
+ int n = 0;
+ while (*s2 && n < len) {
+ if (*s1++ != *s2++)
+ return false;
+ n++;
+ }
+ return *s2 == 0;
+}
+
+static int parse_mouse_event(struct tb_event *event, const char *buf, int len) {
+ if (len >= 6 && starts_with(buf, len, "\033[M")) {
+ // X10 mouse encoding, the simplest one
+ // \033 [ M Cb Cx Cy
+ int b = buf[3] - 32;
+ switch (b & 3) {
+ case 0:
+ if ((b & 64) != 0)
+ event->key = TB_KEY_MOUSE_WHEEL_UP;
+ else
+ event->key = TB_KEY_MOUSE_LEFT;
+ break;
+ case 1:
+ if ((b & 64) != 0)
+ event->key = TB_KEY_MOUSE_WHEEL_DOWN;
+ else
+ event->key = TB_KEY_MOUSE_MIDDLE;
+ break;
+ case 2:
+ event->key = TB_KEY_MOUSE_RIGHT;
+ break;
+ case 3:
+ event->key = TB_KEY_MOUSE_RELEASE;
+ break;
+ default:
+ return -6;
+ }
+ event->type = TB_EVENT_MOUSE; // TB_EVENT_KEY by default
+ if ((b & 32) != 0)
+ event->mod |= TB_MOD_MOTION;
+
+ // the coord is 1,1 for upper left
+ event->x = (uint8_t)buf[4] - 1 - 32;
+ event->y = (uint8_t)buf[5] - 1 - 32;
+
+ return 6;
+ } else if (starts_with(buf, len, "\033[<") || starts_with(buf, len, "\033[")) {
+ // xterm 1006 extended mode or urxvt 1015 extended mode
+ // xterm: \033 [ < Cb ; Cx ; Cy (M or m)
+ // urxvt: \033 [ Cb ; Cx ; Cy M
+ int i, mi = -1, starti = -1;
+ int isM, isU, s1 = -1, s2 = -1;
+ int n1 = 0, n2 = 0, n3 = 0;
+
+ for (i = 0; i < len; i++) {
+ // We search the first (s1) and the last (s2) ';'
+ if (buf[i] == ';') {
+ if (s1 == -1)
+ s1 = i;
+ s2 = i;
+ }
+
+ // We search for the first 'm' or 'M'
+ if ((buf[i] == 'm' || buf[i] == 'M') && mi == -1) {
+ mi = i;
+ break;
+ }
+ }
+ if (mi == -1)
+ return 0;
+
+ // whether it's a capital M or not
+ isM = (buf[mi] == 'M');
+
+ if (buf[2] == '<') {
+ isU = 0;
+ starti = 3;
+ } else {
+ isU = 1;
+ starti = 2;
+ }
+
+ if (s1 == -1 || s2 == -1 || s1 == s2)
+ return 0;
+
+ n1 = strtoul(&buf[starti], NULL, 10);
+ n2 = strtoul(&buf[s1 + 1], NULL, 10);
+ n3 = strtoul(&buf[s2 + 1], NULL, 10);
+
+ if (isU)
+ n1 -= 32;
+
+ switch (n1 & 3) {
+ case 0:
+ if ((n1&64) != 0) {
+ event->key = TB_KEY_MOUSE_WHEEL_UP;
+ } else {
+ event->key = TB_KEY_MOUSE_LEFT;
+ }
+ break;
+ case 1:
+ if ((n1&64) != 0) {
+ event->key = TB_KEY_MOUSE_WHEEL_DOWN;
+ } else {
+ event->key = TB_KEY_MOUSE_MIDDLE;
+ }
+ break;
+ case 2:
+ event->key = TB_KEY_MOUSE_RIGHT;
+ break;
+ case 3:
+ event->key = TB_KEY_MOUSE_RELEASE;
+ break;
+ default:
+ return mi + 1;
+ }
+
+ if (!isM) {
+ // on xterm mouse release is signaled by lowercase m
+ event->key = TB_KEY_MOUSE_RELEASE;
+ }
+
+ event->type = TB_EVENT_MOUSE; // TB_EVENT_KEY by default
+ if ((n1&32) != 0)
+ event->mod |= TB_MOD_MOTION;
+
+ event->x = (uint8_t)n2 - 1;
+ event->y = (uint8_t)n3 - 1;
+
+ return mi + 1;
+ }
+
+ return 0;
+}
+
+// convert escape sequence to event, and return consumed bytes on success (failure == 0)
+static int parse_escape_seq(struct tb_event *event, const char *buf, int len)
+{
+ int mouse_parsed = parse_mouse_event(event, buf, len);
+
+ if (mouse_parsed != 0)
+ return mouse_parsed;
+
+ // it's pretty simple here, find 'starts_with' match and return
+ // success, else return failure
+ int i;
+ for (i = 0; keys[i]; i++) {
+ if (starts_with(buf, len, keys[i])) {
+ event->ch = 0;
+ event->key = 0xFFFF-i;
+ return strlen(keys[i]);
+ }
+ }
+ return 0;
+}
+
+static bool extract_event(struct tb_event *event, struct bytebuffer *inbuf, int inputmode)
+{
+ const char *buf = inbuf->buf;
+ const int len = inbuf->len;
+ if (len == 0)
+ return false;
+
+ if (buf[0] == '\033') {
+ int n = parse_escape_seq(event, buf, len);
+ if (n != 0) {
+ bool success = true;
+ if (n < 0) {
+ success = false;
+ n = -n;
+ }
+ bytebuffer_truncate(inbuf, n);
+ return success;
+ } else {
+ // it's not escape sequence, then it's ALT or ESC,
+ // check inputmode
+ if (inputmode&TB_INPUT_ESC) {
+ // if we're in escape mode, fill ESC event, pop
+ // buffer, return success
+ event->ch = 0;
+ event->key = TB_KEY_ESC;
+ event->mod = 0;
+ bytebuffer_truncate(inbuf, 1);
+ return true;
+ } else if (inputmode&TB_INPUT_ALT) {
+ // if we're in alt mode, set ALT modifier to
+ // event and redo parsing
+ event->mod = TB_MOD_ALT;
+ bytebuffer_truncate(inbuf, 1);
+ return extract_event(event, inbuf, inputmode);
+ }
+ assert(!"never got here");
+ }
+ }
+
+ // if we're here, this is not an escape sequence and not an alt sequence
+ // so, it's a FUNCTIONAL KEY or a UNICODE character
+
+ // first of all check if it's a functional key
+ if ((unsigned char)buf[0] <= TB_KEY_SPACE ||
+ (unsigned char)buf[0] == TB_KEY_BACKSPACE2)
+ {
+ // fill event, pop buffer, return success */
+ event->ch = 0;
+ event->key = (uint16_t)buf[0];
+ bytebuffer_truncate(inbuf, 1);
+ return true;
+ }
+
+ // feh... we got utf8 here
+
+ // check if there is all bytes
+ if (len >= tb_utf8_char_length(buf[0])) {
+ /* everything ok, fill event, pop buffer, return success */
+ tb_utf8_char_to_unicode(&event->ch, buf);
+ event->key = 0;
+ bytebuffer_truncate(inbuf, tb_utf8_char_length(buf[0]));
+ return true;
+ }
+
+ // event isn't recognized, perhaps there is not enough bytes in utf8
+ // sequence
+ return false;
+}
diff --git a/termbox/term.inl.c b/termbox/term.inl.c
@@ -0,0 +1,311 @@
+enum {
+ T_ENTER_CA,
+ T_EXIT_CA,
+ T_SHOW_CURSOR,
+ T_HIDE_CURSOR,
+ T_CLEAR_SCREEN,
+ T_SGR0,
+ T_UNDERLINE,
+ T_BOLD,
+ T_BLINK,
+ T_REVERSE,
+ T_ENTER_KEYPAD,
+ T_EXIT_KEYPAD,
+ T_ENTER_MOUSE,
+ T_EXIT_MOUSE,
+ T_FUNCS_NUM,
+};
+
+#define ENTER_MOUSE_SEQ "\x1b[?1000h\x1b[?1002h\x1b[?1015h\x1b[?1006h"
+#define EXIT_MOUSE_SEQ "\x1b[?1006l\x1b[?1015l\x1b[?1002l\x1b[?1000l"
+
+#define EUNSUPPORTED_TERM -1
+
+// rxvt-256color
+static const char *rxvt_256color_keys[] = {
+ "\033[11~","\033[12~","\033[13~","\033[14~","\033[15~","\033[17~","\033[18~","\033[19~","\033[20~","\033[21~","\033[23~","\033[24~","\033[2~","\033[3~","\033[7~","\033[8~","\033[5~","\033[6~","\033[A","\033[B","\033[D","\033[C", 0
+};
+static const char *rxvt_256color_funcs[] = {
+ "\0337\033[?47h", "\033[2J\033[?47l\0338", "\033[?25h", "\033[?25l", "\033[H\033[2J", "\033[m", "\033[4m", "\033[1m", "\033[5m", "\033[7m", "\033=", "\033>", ENTER_MOUSE_SEQ, EXIT_MOUSE_SEQ,
+};
+
+// Eterm
+static const char *eterm_keys[] = {
+ "\033[11~","\033[12~","\033[13~","\033[14~","\033[15~","\033[17~","\033[18~","\033[19~","\033[20~","\033[21~","\033[23~","\033[24~","\033[2~","\033[3~","\033[7~","\033[8~","\033[5~","\033[6~","\033[A","\033[B","\033[D","\033[C", 0
+};
+static const char *eterm_funcs[] = {
+ "\0337\033[?47h", "\033[2J\033[?47l\0338", "\033[?25h", "\033[?25l", "\033[H\033[2J", "\033[m", "\033[4m", "\033[1m", "\033[5m", "\033[7m", "", "", "", "",
+};
+
+// screen
+static const char *screen_keys[] = {
+ "\033OP","\033OQ","\033OR","\033OS","\033[15~","\033[17~","\033[18~","\033[19~","\033[20~","\033[21~","\033[23~","\033[24~","\033[2~","\033[3~","\033[1~","\033[4~","\033[5~","\033[6~","\033OA","\033OB","\033OD","\033OC", 0
+};
+static const char *screen_funcs[] = {
+ "\033[?1049h", "\033[?1049l", "\033[34h\033[?25h", "\033[?25l", "\033[H\033[J", "\033[m", "\033[4m", "\033[1m", "\033[5m", "\033[7m", "\033[?1h\033=", "\033[?1l\033>", ENTER_MOUSE_SEQ, EXIT_MOUSE_SEQ,
+};
+
+// rxvt-unicode
+static const char *rxvt_unicode_keys[] = {
+ "\033[11~","\033[12~","\033[13~","\033[14~","\033[15~","\033[17~","\033[18~","\033[19~","\033[20~","\033[21~","\033[23~","\033[24~","\033[2~","\033[3~","\033[7~","\033[8~","\033[5~","\033[6~","\033[A","\033[B","\033[D","\033[C", 0
+};
+static const char *rxvt_unicode_funcs[] = {
+ "\033[?1049h", "\033[r\033[?1049l", "\033[?25h", "\033[?25l", "\033[H\033[2J", "\033[m\033(B", "\033[4m", "\033[1m", "\033[5m", "\033[7m", "\033=", "\033>", ENTER_MOUSE_SEQ, EXIT_MOUSE_SEQ,
+};
+
+// linux
+static const char *linux_keys[] = {
+ "\033[[A","\033[[B","\033[[C","\033[[D","\033[[E","\033[17~","\033[18~","\033[19~","\033[20~","\033[21~","\033[23~","\033[24~","\033[2~","\033[3~","\033[1~","\033[4~","\033[5~","\033[6~","\033[A","\033[B","\033[D","\033[C", 0
+};
+static const char *linux_funcs[] = {
+ "", "", "\033[?25h\033[?0c", "\033[?25l\033[?1c", "\033[H\033[J", "\033[0;10m", "\033[4m", "\033[1m", "\033[5m", "\033[7m", "", "", "", "",
+};
+
+// xterm
+static const char *xterm_keys[] = {
+ "\033OP","\033OQ","\033OR","\033OS","\033[15~","\033[17~","\033[18~","\033[19~","\033[20~","\033[21~","\033[23~","\033[24~","\033[2~","\033[3~","\033OH","\033OF","\033[5~","\033[6~","\033OA","\033OB","\033OD","\033OC", 0
+};
+static const char *xterm_funcs[] = {
+ "\033[?1049h", "\033[?1049l", "\033[?12l\033[?25h", "\033[?25l", "\033[H\033[2J", "\033(B\033[m", "\033[4m", "\033[1m", "\033[5m", "\033[7m", "\033[?1h\033=", "\033[?1l\033>", ENTER_MOUSE_SEQ, EXIT_MOUSE_SEQ,
+};
+
+static struct term {
+ const char *name;
+ const char **keys;
+ const char **funcs;
+} terms[] = {
+ {"rxvt-256color", rxvt_256color_keys, rxvt_256color_funcs},
+ {"Eterm", eterm_keys, eterm_funcs},
+ {"screen", screen_keys, screen_funcs},
+ {"rxvt-unicode", rxvt_unicode_keys, rxvt_unicode_funcs},
+ {"linux", linux_keys, linux_funcs},
+ {"xterm", xterm_keys, xterm_funcs},
+ {0, 0, 0},
+};
+
+static bool init_from_terminfo = false;
+static const char **keys;
+static const char **funcs;
+
+static int try_compatible(const char *term, const char *name,
+ const char **tkeys, const char **tfuncs)
+{
+ if (strstr(term, name)) {
+ keys = tkeys;
+ funcs = tfuncs;
+ return 0;
+ }
+
+ return EUNSUPPORTED_TERM;
+}
+
+static int init_term_builtin(void)
+{
+ int i;
+ const char *term = getenv("TERM");
+
+ if (term) {
+ for (i = 0; terms[i].name; i++) {
+ if (!strcmp(terms[i].name, term)) {
+ keys = terms[i].keys;
+ funcs = terms[i].funcs;
+ return 0;
+ }
+ }
+
+ /* let's do some heuristic, maybe it's a compatible terminal */
+ if (try_compatible(term, "xterm", xterm_keys, xterm_funcs) == 0)
+ return 0;
+ if (try_compatible(term, "rxvt", rxvt_unicode_keys, rxvt_unicode_funcs) == 0)
+ return 0;
+ if (try_compatible(term, "linux", linux_keys, linux_funcs) == 0)
+ return 0;
+ if (try_compatible(term, "Eterm", eterm_keys, eterm_funcs) == 0)
+ return 0;
+ if (try_compatible(term, "screen", screen_keys, screen_funcs) == 0)
+ return 0;
+ if (try_compatible(term, "tmux", screen_keys, screen_funcs) == 0)
+ return 0;
+ /* let's assume that 'cygwin' is xterm compatible */
+ if (try_compatible(term, "cygwin", xterm_keys, xterm_funcs) == 0)
+ return 0;
+ }
+
+ return EUNSUPPORTED_TERM;
+}
+
+//----------------------------------------------------------------------
+// terminfo
+//----------------------------------------------------------------------
+
+static char *read_file(const char *file) {
+ FILE *f = fopen(file, "rb");
+ if (!f)
+ return 0;
+
+ struct stat st;
+ if (fstat(fileno(f), &st) != 0) {
+ fclose(f);
+ return 0;
+ }
+
+ char *data = malloc(st.st_size);
+ if (!data) {
+ fclose(f);
+ return 0;
+ }
+
+ if (fread(data, 1, st.st_size, f) != (size_t)st.st_size) {
+ fclose(f);
+ free(data);
+ return 0;
+ }
+
+ fclose(f);
+ return data;
+}
+
+static char *terminfo_try_path(const char *path, const char *term) {
+ char tmp[4096];
+ snprintf(tmp, sizeof(tmp), "%s/%c/%s", path, term[0], term);
+ tmp[sizeof(tmp)-1] = '\0';
+ char *data = read_file(tmp);
+ if (data) {
+ return data;
+ }
+
+ // fallback to darwin specific dirs structure
+ snprintf(tmp, sizeof(tmp), "%s/%x/%s", path, term[0], term);
+ tmp[sizeof(tmp)-1] = '\0';
+ return read_file(tmp);
+}
+
+static char *load_terminfo(void) {
+ char tmp[4096];
+ const char *term = getenv("TERM");
+ if (!term) {
+ return 0;
+ }
+
+ // if TERMINFO is set, no other directory should be searched
+ const char *terminfo = getenv("TERMINFO");
+ if (terminfo) {
+ return terminfo_try_path(terminfo, term);
+ }
+
+ // next, consider ~/.terminfo
+ const char *home = getenv("HOME");
+ if (home) {
+ snprintf(tmp, sizeof(tmp), "%s/.terminfo", home);
+ tmp[sizeof(tmp)-1] = '\0';
+ char *data = terminfo_try_path(tmp, term);
+ if (data)
+ return data;
+ }
+
+ // next, TERMINFO_DIRS
+ const char *dirs = getenv("TERMINFO_DIRS");
+ if (dirs) {
+ snprintf(tmp, sizeof(tmp), "%s", dirs);
+ tmp[sizeof(tmp)-1] = '\0';
+ char *dir = strtok(tmp, ":");
+ while (dir) {
+ const char *cdir = dir;
+ if (strcmp(cdir, "") == 0) {
+ cdir = "/usr/share/terminfo";
+ }
+ char *data = terminfo_try_path(cdir, term);
+ if (data)
+ return data;
+ dir = strtok(0, ":");
+ }
+ }
+
+ // fallback to /usr/share/terminfo
+ return terminfo_try_path("/usr/share/terminfo", term);
+}
+
+#define TI_MAGIC 0432
+#define TI_ALT_MAGIC 542
+#define TI_HEADER_LENGTH 12
+#define TB_KEYS_NUM 22
+
+static const char *terminfo_copy_string(char *data, int str, int table) {
+ const int16_t off = *(int16_t*)(data + str);
+ const char *src = data + table + off;
+ int len = strlen(src);
+ char *dst = malloc(len+1);
+ strcpy(dst, src);
+ return dst;
+}
+
+static const int16_t ti_funcs[] = {
+ 28, 40, 16, 13, 5, 39, 36, 27, 26, 34, 89, 88,
+};
+
+static const int16_t ti_keys[] = {
+ 66, 68 /* apparently not a typo; 67 is F10 for whatever reason */, 69,
+ 70, 71, 72, 73, 74, 75, 67, 216, 217, 77, 59, 76, 164, 82, 81, 87, 61,
+ 79, 83,
+};
+
+static int init_term(void) {
+ int i;
+ char *data = load_terminfo();
+ if (!data) {
+ init_from_terminfo = false;
+ return init_term_builtin();
+ }
+
+ int16_t *header = (int16_t*)data;
+
+ const int number_sec_len = header[0] == TI_ALT_MAGIC ? 4 : 2;
+
+ if ((header[1] + header[2]) % 2) {
+ // old quirk to align everything on word boundaries
+ header[2] += 1;
+ }
+
+ const int str_offset = TI_HEADER_LENGTH +
+ header[1] + header[2] + number_sec_len * header[3];
+ const int table_offset = str_offset + 2 * header[4];
+
+ keys = malloc(sizeof(const char*) * (TB_KEYS_NUM+1));
+ for (i = 0; i < TB_KEYS_NUM; i++) {
+ keys[i] = terminfo_copy_string(data,
+ str_offset + 2 * ti_keys[i], table_offset);
+ }
+ keys[TB_KEYS_NUM] = 0;
+
+ funcs = malloc(sizeof(const char*) * T_FUNCS_NUM);
+ // the last two entries are reserved for mouse. because the table offset is
+ // not there, the two entries have to fill in manually
+ for (i = 0; i < T_FUNCS_NUM-2; i++) {
+ funcs[i] = terminfo_copy_string(data,
+ str_offset + 2 * ti_funcs[i], table_offset);
+ }
+
+ funcs[T_FUNCS_NUM-2] = ENTER_MOUSE_SEQ;
+ funcs[T_FUNCS_NUM-1] = EXIT_MOUSE_SEQ;
+
+ init_from_terminfo = true;
+ free(data);
+ return 0;
+}
+
+static void shutdown_term(void) {
+ if (init_from_terminfo) {
+ int i;
+ for (i = 0; i < TB_KEYS_NUM; i++) {
+ free((void*)keys[i]);
+ }
+ // the last two entries are reserved for mouse. because the table offset
+ // is not there, the two entries have to fill in manually and do not
+ // need to be freed.
+ for (i = 0; i < T_FUNCS_NUM-2; i++) {
+ free((void*)funcs[i]);
+ }
+ free(keys);
+ free(funcs);
+ }
+}
diff --git a/termbox/termbox.c b/termbox/termbox.c
@@ -0,0 +1,678 @@
+#include <assert.h>
+#include <stdlib.h>
+#include <string.h>
+#include <errno.h>
+#include <fcntl.h>
+#include <signal.h>
+#include <stdio.h>
+#include <stdbool.h>
+#include <sys/select.h>
+#include <sys/ioctl.h>
+#include <sys/time.h>
+#include <sys/stat.h>
+#include <termios.h>
+#include <unistd.h>
+#include <wchar.h>
+
+#include "termbox.h"
+
+#include "bytebuffer.inl.c"
+#include "term.inl.c"
+#include "input.inl.c"
+
+struct cellbuf {
+ int width;
+ int height;
+ struct tb_cell *cells;
+};
+
+#define CELL(buf, x, y) (buf)->cells[(y) * (buf)->width + (x)]
+#define IS_CURSOR_HIDDEN(cx, cy) (cx == -1 || cy == -1)
+#define LAST_COORD_INIT -1
+
+static struct termios orig_tios;
+
+static struct cellbuf back_buffer;
+static struct cellbuf front_buffer;
+static struct bytebuffer output_buffer;
+static struct bytebuffer input_buffer;
+
+static int termw = -1;
+static int termh = -1;
+
+static int inputmode = TB_INPUT_ESC;
+static int outputmode = TB_OUTPUT_NORMAL;
+
+static int inout;
+static int winch_fds[2];
+
+static int lastx = LAST_COORD_INIT;
+static int lasty = LAST_COORD_INIT;
+static int cursor_x = -1;
+static int cursor_y = -1;
+
+static uint16_t background = TB_DEFAULT;
+static uint16_t foreground = TB_DEFAULT;
+
+static void write_cursor(int x, int y);
+static void write_sgr(uint16_t fg, uint16_t bg);
+
+static void cellbuf_init(struct cellbuf *buf, int width, int height);
+static void cellbuf_resize(struct cellbuf *buf, int width, int height);
+static void cellbuf_clear(struct cellbuf *buf);
+static void cellbuf_free(struct cellbuf *buf);
+
+static void update_size(void);
+static void update_term_size(void);
+static void send_attr(uint16_t fg, uint16_t bg);
+static void send_char(int x, int y, uint32_t c);
+static void send_clear(void);
+static void sigwinch_handler(int xxx);
+static int wait_fill_event(struct tb_event *event, struct timeval *timeout);
+
+/* may happen in a different thread */
+static volatile int buffer_size_change_request;
+
+/* -------------------------------------------------------- */
+
+int tb_init_fd(int inout_)
+{
+ inout = inout_;
+ if (inout == -1) {
+ return TB_EFAILED_TO_OPEN_TTY;
+ }
+
+ if (init_term() < 0) {
+ close(inout);
+ return TB_EUNSUPPORTED_TERMINAL;
+ }
+
+ if (pipe(winch_fds) < 0) {
+ close(inout);
+ return TB_EPIPE_TRAP_ERROR;
+ }
+
+ struct sigaction sa;
+ memset(&sa, 0, sizeof(sa));
+ sa.sa_handler = sigwinch_handler;
+ sa.sa_flags = 0;
+ sigaction(SIGWINCH, &sa, 0);
+
+ tcgetattr(inout, &orig_tios);
+
+ struct termios tios;
+ memcpy(&tios, &orig_tios, sizeof(tios));
+
+ tios.c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP
+ | INLCR | IGNCR | ICRNL | IXON);
+ tios.c_oflag &= ~OPOST;
+ tios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
+ tios.c_cflag &= ~(CSIZE | PARENB);
+ tios.c_cflag |= CS8;
+ tios.c_cc[VMIN] = 0;
+ tios.c_cc[VTIME] = 0;
+ tcsetattr(inout, TCSAFLUSH, &tios);
+
+ bytebuffer_init(&input_buffer, 128);
+ bytebuffer_init(&output_buffer, 32 * 1024);
+
+ bytebuffer_puts(&output_buffer, funcs[T_ENTER_CA]);
+ bytebuffer_puts(&output_buffer, funcs[T_ENTER_KEYPAD]);
+ bytebuffer_puts(&output_buffer, funcs[T_HIDE_CURSOR]);
+ send_clear();
+
+ update_term_size();
+ cellbuf_init(&back_buffer, termw, termh);
+ cellbuf_init(&front_buffer, termw, termh);
+ cellbuf_clear(&back_buffer);
+ cellbuf_clear(&front_buffer);
+
+ return 0;
+}
+
+int tb_init_file(const char* name){
+ return tb_init_fd(open(name, O_RDWR));
+}
+
+int tb_init(void)
+{
+ return tb_init_file("/dev/tty");
+}
+
+void tb_shutdown(void)
+{
+ if (termw == -1) {
+ fputs("tb_shutdown() should not be called twice.", stderr);
+ abort();
+ }
+
+ bytebuffer_puts(&output_buffer, funcs[T_SHOW_CURSOR]);
+ bytebuffer_puts(&output_buffer, funcs[T_SGR0]);
+ bytebuffer_puts(&output_buffer, funcs[T_CLEAR_SCREEN]);
+ bytebuffer_puts(&output_buffer, funcs[T_EXIT_CA]);
+ bytebuffer_puts(&output_buffer, funcs[T_EXIT_KEYPAD]);
+ bytebuffer_puts(&output_buffer, funcs[T_EXIT_MOUSE]);
+ bytebuffer_flush(&output_buffer, inout);
+ tcsetattr(inout, TCSAFLUSH, &orig_tios);
+
+ shutdown_term();
+ close(inout);
+ close(winch_fds[0]);
+ close(winch_fds[1]);
+
+ cellbuf_free(&back_buffer);
+ cellbuf_free(&front_buffer);
+ bytebuffer_free(&output_buffer);
+ bytebuffer_free(&input_buffer);
+ termw = termh = -1;
+}
+
+void tb_present(void)
+{
+ int x,y,w,i;
+ struct tb_cell *back, *front;
+
+ /* invalidate cursor position */
+ lastx = LAST_COORD_INIT;
+ lasty = LAST_COORD_INIT;
+
+ if (buffer_size_change_request) {
+ update_size();
+ buffer_size_change_request = 0;
+ }
+
+ for (y = 0; y < front_buffer.height; ++y) {
+ for (x = 0; x < front_buffer.width; ) {
+ back = &CELL(&back_buffer, x, y);
+ front = &CELL(&front_buffer, x, y);
+ w = wcwidth(back->ch);
+ if (w < 1) w = 1;
+ if (memcmp(back, front, sizeof(struct tb_cell)) == 0) {
+ x += w;
+ continue;
+ }
+ memcpy(front, back, sizeof(struct tb_cell));
+ send_attr(back->fg, back->bg);
+ if (w > 1 && x >= front_buffer.width - (w - 1)) {
+ // Not enough room for wide ch, so send spaces
+ for (i = x; i < front_buffer.width; ++i) {
+ send_char(i, y, ' ');
+ }
+ } else {
+ send_char(x, y, back->ch);
+ for (i = 1; i < w; ++i) {
+ front = &CELL(&front_buffer, x + i, y);
+ front->ch = 0;
+ front->fg = back->fg;
+ front->bg = back->bg;
+ }
+ }
+ x += w;
+ }
+ }
+ if (!IS_CURSOR_HIDDEN(cursor_x, cursor_y))
+ write_cursor(cursor_x, cursor_y);
+ bytebuffer_flush(&output_buffer, inout);
+}
+
+void tb_set_cursor(int cx, int cy)
+{
+ if (IS_CURSOR_HIDDEN(cursor_x, cursor_y) && !IS_CURSOR_HIDDEN(cx, cy))
+ bytebuffer_puts(&output_buffer, funcs[T_SHOW_CURSOR]);
+
+ if (!IS_CURSOR_HIDDEN(cursor_x, cursor_y) && IS_CURSOR_HIDDEN(cx, cy))
+ bytebuffer_puts(&output_buffer, funcs[T_HIDE_CURSOR]);
+
+ cursor_x = cx;
+ cursor_y = cy;
+ if (!IS_CURSOR_HIDDEN(cursor_x, cursor_y))
+ write_cursor(cursor_x, cursor_y);
+}
+
+void tb_put_cell(int x, int y, const struct tb_cell *cell)
+{
+ if ((unsigned)x >= (unsigned)back_buffer.width)
+ return;
+ if ((unsigned)y >= (unsigned)back_buffer.height)
+ return;
+ CELL(&back_buffer, x, y) = *cell;
+}
+
+void tb_change_cell(int x, int y, uint32_t ch, uint16_t fg, uint16_t bg)
+{
+ struct tb_cell c = {ch, fg, bg};
+ tb_put_cell(x, y, &c);
+}
+
+void tb_blit(int x, int y, int w, int h, const struct tb_cell *cells)
+{
+ if (x + w < 0 || x >= back_buffer.width)
+ return;
+ if (y + h < 0 || y >= back_buffer.height)
+ return;
+ int xo = 0, yo = 0, ww = w, hh = h;
+ if (x < 0) {
+ xo = -x;
+ ww -= xo;
+ x = 0;
+ }
+ if (y < 0) {
+ yo = -y;
+ hh -= yo;
+ y = 0;
+ }
+ if (ww > back_buffer.width - x)
+ ww = back_buffer.width - x;
+ if (hh > back_buffer.height - y)
+ hh = back_buffer.height - y;
+
+ int sy;
+ struct tb_cell *dst = &CELL(&back_buffer, x, y);
+ const struct tb_cell *src = cells + yo * w + xo;
+ size_t size = sizeof(struct tb_cell) * ww;
+
+ for (sy = 0; sy < hh; ++sy) {
+ memcpy(dst, src, size);
+ dst += back_buffer.width;
+ src += w;
+ }
+}
+
+struct tb_cell *tb_cell_buffer(void)
+{
+ return back_buffer.cells;
+}
+
+int tb_poll_event(struct tb_event *event)
+{
+ return wait_fill_event(event, 0);
+}
+
+int tb_peek_event(struct tb_event *event, int timeout)
+{
+ struct timeval tv;
+ tv.tv_sec = timeout / 1000;
+ tv.tv_usec = (timeout - (tv.tv_sec * 1000)) * 1000;
+ return wait_fill_event(event, &tv);
+}
+
+int tb_width(void)
+{
+ return termw;
+}
+
+int tb_height(void)
+{
+ return termh;
+}
+
+void tb_clear(void)
+{
+ if (buffer_size_change_request) {
+ update_size();
+ buffer_size_change_request = 0;
+ }
+ cellbuf_clear(&back_buffer);
+}
+
+int tb_select_input_mode(int mode)
+{
+ if (mode) {
+ if ((mode & (TB_INPUT_ESC | TB_INPUT_ALT)) == 0)
+ mode |= TB_INPUT_ESC;
+
+ /* technically termbox can handle that, but let's be nice and show here
+ what mode is actually used */
+ if ((mode & (TB_INPUT_ESC | TB_INPUT_ALT)) == (TB_INPUT_ESC | TB_INPUT_ALT))
+ mode &= ~TB_INPUT_ALT;
+
+ inputmode = mode;
+ if (mode&TB_INPUT_MOUSE) {
+ bytebuffer_puts(&output_buffer, funcs[T_ENTER_MOUSE]);
+ bytebuffer_flush(&output_buffer, inout);
+ } else {
+ bytebuffer_puts(&output_buffer, funcs[T_EXIT_MOUSE]);
+ bytebuffer_flush(&output_buffer, inout);
+ }
+ }
+ return inputmode;
+}
+
+int tb_select_output_mode(int mode)
+{
+ if (mode)
+ outputmode = mode;
+ return outputmode;
+}
+
+void tb_set_clear_attributes(uint16_t fg, uint16_t bg)
+{
+ foreground = fg;
+ background = bg;
+}
+
+/* -------------------------------------------------------- */
+
+static int convertnum(uint32_t num, char* buf) {
+ int i, l = 0;
+ int ch;
+ do {
+ buf[l++] = '0' + (num % 10);
+ num /= 10;
+ } while (num);
+ for(i = 0; i < l / 2; i++) {
+ ch = buf[i];
+ buf[i] = buf[l - 1 - i];
+ buf[l - 1 - i] = ch;
+ }
+ return l;
+}
+
+#define WRITE_LITERAL(X) bytebuffer_append(&output_buffer, (X), sizeof(X)-1)
+#define WRITE_INT(X) bytebuffer_append(&output_buffer, buf, convertnum((X), buf))
+
+static void write_cursor(int x, int y) {
+ char buf[32];
+ WRITE_LITERAL("\033[");
+ WRITE_INT(y+1);
+ WRITE_LITERAL(";");
+ WRITE_INT(x+1);
+ WRITE_LITERAL("H");
+}
+
+static void write_sgr(uint16_t fg, uint16_t bg) {
+ char buf[32];
+
+ if (fg == TB_DEFAULT && bg == TB_DEFAULT)
+ return;
+
+ switch (outputmode) {
+ case TB_OUTPUT_256:
+ case TB_OUTPUT_216:
+ case TB_OUTPUT_GRAYSCALE:
+ WRITE_LITERAL("\033[");
+ if (fg != TB_DEFAULT) {
+ WRITE_LITERAL("38;5;");
+ WRITE_INT(fg);
+ if (bg != TB_DEFAULT) {
+ WRITE_LITERAL(";");
+ }
+ }
+ if (bg != TB_DEFAULT) {
+ WRITE_LITERAL("48;5;");
+ WRITE_INT(bg);
+ }
+ WRITE_LITERAL("m");
+ break;
+ case TB_OUTPUT_NORMAL:
+ default:
+ WRITE_LITERAL("\033[");
+ if (fg != TB_DEFAULT) {
+ WRITE_LITERAL("3");
+ WRITE_INT(fg - 1);
+ if (bg != TB_DEFAULT) {
+ WRITE_LITERAL(";");
+ }
+ }
+ if (bg != TB_DEFAULT) {
+ WRITE_LITERAL("4");
+ WRITE_INT(bg - 1);
+ }
+ WRITE_LITERAL("m");
+ break;
+ }
+}
+
+static void cellbuf_init(struct cellbuf *buf, int width, int height)
+{
+ buf->cells = (struct tb_cell*)malloc(sizeof(struct tb_cell) * width * height);
+ assert(buf->cells);
+ buf->width = width;
+ buf->height = height;
+}
+
+static void cellbuf_resize(struct cellbuf *buf, int width, int height)
+{
+ if (buf->width == width && buf->height == height)
+ return;
+
+ int oldw = buf->width;
+ int oldh = buf->height;
+ struct tb_cell *oldcells = buf->cells;
+
+ cellbuf_init(buf, width, height);
+ cellbuf_clear(buf);
+
+ int minw = (width < oldw) ? width : oldw;
+ int minh = (height < oldh) ? height : oldh;
+ int i;
+
+ for (i = 0; i < minh; ++i) {
+ struct tb_cell *csrc = oldcells + (i * oldw);
+ struct tb_cell *cdst = buf->cells + (i * width);
+ memcpy(cdst, csrc, sizeof(struct tb_cell) * minw);
+ }
+
+ free(oldcells);
+}
+
+static void cellbuf_clear(struct cellbuf *buf)
+{
+ int i;
+ int ncells = buf->width * buf->height;
+
+ for (i = 0; i < ncells; ++i) {
+ buf->cells[i].ch = ' ';
+ buf->cells[i].fg = foreground;
+ buf->cells[i].bg = background;
+ }
+}
+
+static void cellbuf_free(struct cellbuf *buf)
+{
+ free(buf->cells);
+}
+
+static void get_term_size(int *w, int *h)
+{
+ struct winsize sz;
+ memset(&sz, 0, sizeof(sz));
+
+ ioctl(inout, TIOCGWINSZ, &sz);
+
+ if (w) *w = sz.ws_col;
+ if (h) *h = sz.ws_row;
+}
+
+static void update_term_size(void)
+{
+ struct winsize sz;
+ memset(&sz, 0, sizeof(sz));
+
+ ioctl(inout, TIOCGWINSZ, &sz);
+
+ termw = sz.ws_col;
+ termh = sz.ws_row;
+}
+
+static void send_attr(uint16_t fg, uint16_t bg)
+{
+#define LAST_ATTR_INIT 0xFFFF
+ static uint16_t lastfg = LAST_ATTR_INIT, lastbg = LAST_ATTR_INIT;
+ if (fg != lastfg || bg != lastbg) {
+ bytebuffer_puts(&output_buffer, funcs[T_SGR0]);
+
+ uint16_t fgcol;
+ uint16_t bgcol;
+
+ switch (outputmode) {
+ case TB_OUTPUT_256:
+ fgcol = fg & 0xFF;
+ bgcol = bg & 0xFF;
+ break;
+
+ case TB_OUTPUT_216:
+ fgcol = fg & 0xFF; if (fgcol > 215) fgcol = 7;
+ bgcol = bg & 0xFF; if (bgcol > 215) bgcol = 0;
+ fgcol += 0x10;
+ bgcol += 0x10;
+ break;
+
+ case TB_OUTPUT_GRAYSCALE:
+ fgcol = fg & 0xFF; if (fgcol > 23) fgcol = 23;
+ bgcol = bg & 0xFF; if (bgcol > 23) bgcol = 0;
+ fgcol += 0xe8;
+ bgcol += 0xe8;
+ break;
+
+ case TB_OUTPUT_NORMAL:
+ default:
+ fgcol = fg & 0x0F;
+ bgcol = bg & 0x0F;
+ }
+
+ if (fg & TB_BOLD)
+ bytebuffer_puts(&output_buffer, funcs[T_BOLD]);
+ if (bg & TB_BOLD)
+ bytebuffer_puts(&output_buffer, funcs[T_BLINK]);
+ if (fg & TB_UNDERLINE)
+ bytebuffer_puts(&output_buffer, funcs[T_UNDERLINE]);
+ if ((fg & TB_REVERSE) || (bg & TB_REVERSE))
+ bytebuffer_puts(&output_buffer, funcs[T_REVERSE]);
+
+ write_sgr(fgcol, bgcol);
+
+ lastfg = fg;
+ lastbg = bg;
+ }
+}
+
+static void send_char(int x, int y, uint32_t c)
+{
+ char buf[7];
+ int bw = tb_utf8_unicode_to_char(buf, c);
+ if (x-1 != lastx || y != lasty)
+ write_cursor(x, y);
+ lastx = x; lasty = y;
+ if(!c) buf[0] = ' '; // replace 0 with whitespace
+ bytebuffer_append(&output_buffer, buf, bw);
+}
+
+static void send_clear(void)
+{
+ send_attr(foreground, background);
+ bytebuffer_puts(&output_buffer, funcs[T_CLEAR_SCREEN]);
+ if (!IS_CURSOR_HIDDEN(cursor_x, cursor_y))
+ write_cursor(cursor_x, cursor_y);
+ bytebuffer_flush(&output_buffer, inout);
+
+ /* we need to invalidate cursor position too and these two vars are
+ * used only for simple cursor positioning optimization, cursor
+ * actually may be in the correct place, but we simply discard
+ * optimization once and it gives us simple solution for the case when
+ * cursor moved */
+ lastx = LAST_COORD_INIT;
+ lasty = LAST_COORD_INIT;
+}
+
+static void sigwinch_handler(int xxx)
+{
+ (void) xxx;
+ const int zzz = 1;
+ write(winch_fds[1], &zzz, sizeof(int));
+}
+
+static void update_size(void)
+{
+ update_term_size();
+ cellbuf_resize(&back_buffer, termw, termh);
+ cellbuf_resize(&front_buffer, termw, termh);
+ cellbuf_clear(&front_buffer);
+ send_clear();
+}
+
+static int read_up_to(int n) {
+ assert(n > 0);
+ const int prevlen = input_buffer.len;
+ bytebuffer_resize(&input_buffer, prevlen + n);
+
+ int read_n = 0;
+ while (read_n <= n) {
+ ssize_t r = 0;
+ if (read_n < n) {
+ r = read(inout, input_buffer.buf + prevlen + read_n, n - read_n);
+ }
+#ifdef __CYGWIN__
+ // While linux man for tty says when VMIN == 0 && VTIME == 0, read
+ // should return 0 when there is nothing to read, cygwin's read returns
+ // -1. Not sure why and if it's correct to ignore it, but let's pretend
+ // it's zero.
+ if (r < 0) r = 0;
+#endif
+ if (r < 0) {
+ // EAGAIN / EWOULDBLOCK shouldn't occur here
+ assert(errno != EAGAIN && errno != EWOULDBLOCK);
+ return -1;
+ } else if (r > 0) {
+ read_n += r;
+ } else {
+ bytebuffer_resize(&input_buffer, prevlen + read_n);
+ return read_n;
+ }
+ }
+ assert(!"unreachable");
+ return 0;
+}
+
+static int wait_fill_event(struct tb_event *event, struct timeval *timeout)
+{
+ // ;-)
+#define ENOUGH_DATA_FOR_PARSING 64
+ fd_set events;
+ memset(event, 0, sizeof(struct tb_event));
+
+ // try to extract event from input buffer, return on success
+ event->type = TB_EVENT_KEY;
+ if (extract_event(event, &input_buffer, inputmode))
+ return event->type;
+
+ // it looks like input buffer is incomplete, let's try the short path,
+ // but first make sure there is enough space
+ int n = read_up_to(ENOUGH_DATA_FOR_PARSING);
+ if (n < 0)
+ return -1;
+ if (n > 0 && extract_event(event, &input_buffer, inputmode))
+ return event->type;
+
+ // n == 0, or not enough data, let's go to select
+ while (1) {
+ FD_ZERO(&events);
+ FD_SET(inout, &events);
+ FD_SET(winch_fds[0], &events);
+ int maxfd = (winch_fds[0] > inout) ? winch_fds[0] : inout;
+ int result = select(maxfd+1, &events, 0, 0, timeout);
+ if (!result)
+ return 0;
+
+ if (FD_ISSET(inout, &events)) {
+ event->type = TB_EVENT_KEY;
+ n = read_up_to(ENOUGH_DATA_FOR_PARSING);
+ if (n < 0)
+ return -1;
+
+ if (n == 0)
+ continue;
+
+ if (extract_event(event, &input_buffer, inputmode))
+ return event->type;
+ }
+ if (FD_ISSET(winch_fds[0], &events)) {
+ event->type = TB_EVENT_RESIZE;
+ int zzz = 0;
+ read(winch_fds[0], &zzz, sizeof(int));
+ buffer_size_change_request = 1;
+ get_term_size(&event->w, &event->h);
+ return TB_EVENT_RESIZE;
+ }
+ }
+}
diff --git a/termbox/termbox.h b/termbox/termbox.h
@@ -0,0 +1,320 @@
+#pragma once
+
+#include <stdint.h>
+
+/* for shared objects */
+#if __GNUC__ >= 4
+ #define SO_IMPORT __attribute__((visibility("default")))
+#else
+ #define SO_IMPORT
+#endif
+
+#ifdef __cplusplus
+extern "C" {
+#endif
+
+/* Key constants. See also struct tb_event's key field.
+ *
+ * These are a safe subset of terminfo keys, which exist on all popular
+ * terminals. Termbox uses only them to stay truly portable.
+ */
+#define TB_KEY_F1 (0xFFFF-0)
+#define TB_KEY_F2 (0xFFFF-1)
+#define TB_KEY_F3 (0xFFFF-2)
+#define TB_KEY_F4 (0xFFFF-3)
+#define TB_KEY_F5 (0xFFFF-4)
+#define TB_KEY_F6 (0xFFFF-5)
+#define TB_KEY_F7 (0xFFFF-6)
+#define TB_KEY_F8 (0xFFFF-7)
+#define TB_KEY_F9 (0xFFFF-8)
+#define TB_KEY_F10 (0xFFFF-9)
+#define TB_KEY_F11 (0xFFFF-10)
+#define TB_KEY_F12 (0xFFFF-11)
+#define TB_KEY_INSERT (0xFFFF-12)
+#define TB_KEY_DELETE (0xFFFF-13)
+#define TB_KEY_HOME (0xFFFF-14)
+#define TB_KEY_END (0xFFFF-15)
+#define TB_KEY_PGUP (0xFFFF-16)
+#define TB_KEY_PGDN (0xFFFF-17)
+#define TB_KEY_ARROW_UP (0xFFFF-18)
+#define TB_KEY_ARROW_DOWN (0xFFFF-19)
+#define TB_KEY_ARROW_LEFT (0xFFFF-20)
+#define TB_KEY_ARROW_RIGHT (0xFFFF-21)
+#define TB_KEY_MOUSE_LEFT (0xFFFF-22)
+#define TB_KEY_MOUSE_RIGHT (0xFFFF-23)
+#define TB_KEY_MOUSE_MIDDLE (0xFFFF-24)
+#define TB_KEY_MOUSE_RELEASE (0xFFFF-25)
+#define TB_KEY_MOUSE_WHEEL_UP (0xFFFF-26)
+#define TB_KEY_MOUSE_WHEEL_DOWN (0xFFFF-27)
+
+/* These are all ASCII code points below SPACE character and a BACKSPACE key. */
+#define TB_KEY_CTRL_TILDE 0x00
+#define TB_KEY_CTRL_2 0x00 /* clash with 'CTRL_TILDE' */
+#define TB_KEY_CTRL_A 0x01
+#define TB_KEY_CTRL_B 0x02
+#define TB_KEY_CTRL_C 0x03
+#define TB_KEY_CTRL_D 0x04
+#define TB_KEY_CTRL_E 0x05
+#define TB_KEY_CTRL_F 0x06
+#define TB_KEY_CTRL_G 0x07
+#define TB_KEY_BACKSPACE 0x08
+#define TB_KEY_CTRL_H 0x08 /* clash with 'CTRL_BACKSPACE' */
+#define TB_KEY_TAB 0x09
+#define TB_KEY_CTRL_I 0x09 /* clash with 'TAB' */
+#define TB_KEY_CTRL_J 0x0A
+#define TB_KEY_CTRL_K 0x0B
+#define TB_KEY_CTRL_L 0x0C
+#define TB_KEY_ENTER 0x0D
+#define TB_KEY_CTRL_M 0x0D /* clash with 'ENTER' */
+#define TB_KEY_CTRL_N 0x0E
+#define TB_KEY_CTRL_O 0x0F
+#define TB_KEY_CTRL_P 0x10
+#define TB_KEY_CTRL_Q 0x11
+#define TB_KEY_CTRL_R 0x12
+#define TB_KEY_CTRL_S 0x13
+#define TB_KEY_CTRL_T 0x14
+#define TB_KEY_CTRL_U 0x15
+#define TB_KEY_CTRL_V 0x16
+#define TB_KEY_CTRL_W 0x17
+#define TB_KEY_CTRL_X 0x18
+#define TB_KEY_CTRL_Y 0x19
+#define TB_KEY_CTRL_Z 0x1A
+#define TB_KEY_ESC 0x1B
+#define TB_KEY_CTRL_LSQ_BRACKET 0x1B /* clash with 'ESC' */
+#define TB_KEY_CTRL_3 0x1B /* clash with 'ESC' */
+#define TB_KEY_CTRL_4 0x1C
+#define TB_KEY_CTRL_BACKSLASH 0x1C /* clash with 'CTRL_4' */
+#define TB_KEY_CTRL_5 0x1D
+#define TB_KEY_CTRL_RSQ_BRACKET 0x1D /* clash with 'CTRL_5' */
+#define TB_KEY_CTRL_6 0x1E
+#define TB_KEY_CTRL_7 0x1F
+#define TB_KEY_CTRL_SLASH 0x1F /* clash with 'CTRL_7' */
+#define TB_KEY_CTRL_UNDERSCORE 0x1F /* clash with 'CTRL_7' */
+#define TB_KEY_SPACE 0x20
+#define TB_KEY_BACKSPACE2 0x7F
+#define TB_KEY_CTRL_8 0x7F /* clash with 'BACKSPACE2' */
+
+/* These are non-existing ones.
+ *
+ * #define TB_KEY_CTRL_1 clash with '1'
+ * #define TB_KEY_CTRL_9 clash with '9'
+ * #define TB_KEY_CTRL_0 clash with '0'
+ */
+
+/*
+ * Alt modifier constant, see tb_event.mod field and tb_select_input_mode function.
+ * Mouse-motion modifier
+ */
+#define TB_MOD_ALT 0x01
+#define TB_MOD_MOTION 0x02
+
+/* Colors (see struct tb_cell's fg and bg fields). */
+#define TB_DEFAULT 0x00
+#define TB_BLACK 0x01
+#define TB_RED 0x02
+#define TB_GREEN 0x03
+#define TB_YELLOW 0x04
+#define TB_BLUE 0x05
+#define TB_MAGENTA 0x06
+#define TB_CYAN 0x07
+#define TB_WHITE 0x08
+
+/* Attributes, it is possible to use multiple attributes by combining them
+ * using bitwise OR ('|'). Although, colors cannot be combined. But you can
+ * combine attributes and a single color. See also struct tb_cell's fg and bg
+ * fields.
+ */
+#define TB_BOLD 0x0100
+#define TB_UNDERLINE 0x0200
+#define TB_REVERSE 0x0400
+
+/* A cell, single conceptual entity on the terminal screen. The terminal screen
+ * is basically a 2d array of cells. It has the following fields:
+ * - 'ch' is a unicode character
+ * - 'fg' foreground color and attributes
+ * - 'bg' background color and attributes
+ */
+struct tb_cell {
+ uint32_t ch;
+ uint16_t fg;
+ uint16_t bg;
+};
+
+#define TB_EVENT_KEY 1
+#define TB_EVENT_RESIZE 2
+#define TB_EVENT_MOUSE 3
+
+/* An event, single interaction from the user. The 'mod' and 'ch' fields are
+ * valid if 'type' is TB_EVENT_KEY. The 'w' and 'h' fields are valid if 'type'
+ * is TB_EVENT_RESIZE. The 'x' and 'y' fields are valid if 'type' is
+ * TB_EVENT_MOUSE. The 'key' field is valid if 'type' is either TB_EVENT_KEY
+ * or TB_EVENT_MOUSE. The fields 'key' and 'ch' are mutually exclusive; only
+ * one of them can be non-zero at a time.
+ */
+struct tb_event {
+ uint8_t type;
+ uint8_t mod; /* modifiers to either 'key' or 'ch' below */
+ uint16_t key; /* one of the TB_KEY_* constants */
+ uint32_t ch; /* unicode character */
+ int32_t w;
+ int32_t h;
+ int32_t x;
+ int32_t y;
+};
+
+/* Error codes returned by tb_init(). All of them are self-explanatory, except
+ * the pipe trap error. Termbox uses unix pipes in order to deliver a message
+ * from a signal handler (SIGWINCH) to the main event reading loop. Honestly in
+ * most cases you should just check the returned code as < 0.
+ */
+#define TB_EUNSUPPORTED_TERMINAL -1
+#define TB_EFAILED_TO_OPEN_TTY -2
+#define TB_EPIPE_TRAP_ERROR -3
+
+/* Initializes the termbox library. This function should be called before any
+ * other functions. Function tb_init is same as tb_init_file("/dev/tty").
+ * After successful initialization, the library must be
+ * finalized using the tb_shutdown() function.
+ */
+SO_IMPORT int tb_init(void);
+SO_IMPORT int tb_init_file(const char* name);
+SO_IMPORT int tb_init_fd(int inout);
+SO_IMPORT void tb_shutdown(void);
+
+/* Returns the size of the internal back buffer (which is the same as
+ * terminal's window size in characters). The internal buffer can be resized
+ * after tb_clear() or tb_present() function calls. Both dimensions have an
+ * unspecified negative value when called before tb_init() or after
+ * tb_shutdown().
+ */
+SO_IMPORT int tb_width(void);
+SO_IMPORT int tb_height(void);
+
+/* Clears the internal back buffer using TB_DEFAULT color or the
+ * color/attributes set by tb_set_clear_attributes() function.
+ */
+SO_IMPORT void tb_clear(void);
+SO_IMPORT void tb_set_clear_attributes(uint16_t fg, uint16_t bg);
+
+/* Synchronizes the internal back buffer with the terminal. */
+SO_IMPORT void tb_present(void);
+
+#define TB_HIDE_CURSOR -1
+
+/* Sets the position of the cursor. Upper-left character is (0, 0). If you pass
+ * TB_HIDE_CURSOR as both coordinates, then the cursor will be hidden. Cursor
+ * is hidden by default.
+ */
+SO_IMPORT void tb_set_cursor(int cx, int cy);
+
+/* Changes cell's parameters in the internal back buffer at the specified
+ * position.
+ */
+SO_IMPORT void tb_put_cell(int x, int y, const struct tb_cell *cell);
+SO_IMPORT void tb_change_cell(int x, int y, uint32_t ch, uint16_t fg, uint16_t bg);
+
+/* Copies the buffer from 'cells' at the specified position, assuming the
+ * buffer is a two-dimensional array of size ('w' x 'h'), represented as a
+ * one-dimensional buffer containing lines of cells starting from the top.
+ *
+ * (DEPRECATED: use tb_cell_buffer() instead and copy memory on your own)
+ */
+SO_IMPORT void tb_blit(int x, int y, int w, int h, const struct tb_cell *cells);
+
+/* Returns a pointer to internal cell back buffer. You can get its dimensions
+ * using tb_width() and tb_height() functions. The pointer stays valid as long
+ * as no tb_clear() and tb_present() calls are made. The buffer is
+ * one-dimensional buffer containing lines of cells starting from the top.
+ */
+SO_IMPORT struct tb_cell *tb_cell_buffer(void);
+
+#define TB_INPUT_CURRENT 0 /* 000 */
+#define TB_INPUT_ESC 1 /* 001 */
+#define TB_INPUT_ALT 2 /* 010 */
+#define TB_INPUT_MOUSE 4 /* 100 */
+
+/* Sets the termbox input mode. Termbox has two input modes:
+ * 1. Esc input mode.
+ * When ESC sequence is in the buffer and it doesn't match any known
+ * ESC sequence => ESC means TB_KEY_ESC.
+ * 2. Alt input mode.
+ * When ESC sequence is in the buffer and it doesn't match any known
+ * sequence => ESC enables TB_MOD_ALT modifier for the next keyboard event.
+ *
+ * You can also apply TB_INPUT_MOUSE via bitwise OR operation to either of the
+ * modes (e.g. TB_INPUT_ESC | TB_INPUT_MOUSE). If none of the main two modes
+ * were set, but the mouse mode was, TB_INPUT_ESC mode is used. If for some
+ * reason you've decided to use (TB_INPUT_ESC | TB_INPUT_ALT) combination, it
+ * will behave as if only TB_INPUT_ESC was selected.
+ *
+ * If 'mode' is TB_INPUT_CURRENT, it returns the current input mode.
+ *
+ * Default termbox input mode is TB_INPUT_ESC.
+ */
+SO_IMPORT int tb_select_input_mode(int mode);
+
+#define TB_OUTPUT_CURRENT 0
+#define TB_OUTPUT_NORMAL 1
+#define TB_OUTPUT_256 2
+#define TB_OUTPUT_216 3
+#define TB_OUTPUT_GRAYSCALE 4
+
+/* Sets the termbox output mode. Termbox has three output options:
+ * 1. TB_OUTPUT_NORMAL => [1..8]
+ * This mode provides 8 different colors:
+ * black, red, green, yellow, blue, magenta, cyan, white
+ * Shortcut: TB_BLACK, TB_RED, ...
+ * Attributes: TB_BOLD, TB_UNDERLINE, TB_REVERSE
+ *
+ * Example usage:
+ * tb_change_cell(x, y, '@', TB_BLACK | TB_BOLD, TB_RED);
+ *
+ * 2. TB_OUTPUT_256 => [0..256]
+ * In this mode you can leverage the 256 terminal mode:
+ * 0x00 - 0x07: the 8 colors as in TB_OUTPUT_NORMAL
+ * 0x08 - 0x0f: TB_* | TB_BOLD
+ * 0x10 - 0xe7: 216 different colors
+ * 0xe8 - 0xff: 24 different shades of grey
+ *
+ * Example usage:
+ * tb_change_cell(x, y, '@', 184, 240);
+ * tb_change_cell(x, y, '@', 0xb8, 0xf0);
+ *
+ * 3. TB_OUTPUT_216 => [0..216]
+ * This mode supports the 3rd range of the 256 mode only.
+ * But you don't need to provide an offset.
+ *
+ * 4. TB_OUTPUT_GRAYSCALE => [0..23]
+ * This mode supports the 4th range of the 256 mode only.
+ * But you dont need to provide an offset.
+ *
+ * Execute build/src/demo/output to see its impact on your terminal.
+ *
+ * If 'mode' is TB_OUTPUT_CURRENT, it returns the current output mode.
+ *
+ * Default termbox output mode is TB_OUTPUT_NORMAL.
+ */
+SO_IMPORT int tb_select_output_mode(int mode);
+
+/* Wait for an event up to 'timeout' milliseconds and fill the 'event'
+ * structure with it, when the event is available. Returns the type of the
+ * event (one of TB_EVENT_* constants) or -1 if there was an error or 0 in case
+ * there were no event during 'timeout' period.
+ */
+SO_IMPORT int tb_peek_event(struct tb_event *event, int timeout);
+
+/* Wait for an event forever and fill the 'event' structure with it, when the
+ * event is available. Returns the type of the event (one of TB_EVENT_*
+ * constants) or -1 if there was an error.
+ */
+SO_IMPORT int tb_poll_event(struct tb_event *event);
+
+/* Utility utf8 functions. */
+#define TB_EOF -1
+SO_IMPORT int tb_utf8_char_length(char c);
+SO_IMPORT int tb_utf8_char_to_unicode(uint32_t *out, const char *c);
+SO_IMPORT int tb_utf8_unicode_to_char(char *out, uint32_t c);
+
+#ifdef __cplusplus
+}
+#endif
diff --git a/termbox/utf8.c b/termbox/utf8.c
@@ -0,0 +1,79 @@
+#include "termbox.h"
+
+static const unsigned char utf8_length[256] = {
+ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
+ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
+ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
+ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
+ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
+ 1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,
+ 2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,2,
+ 3,3,3,3,3,3,3,3,3,3,3,3,3,3,3,3,4,4,4,4,4,4,4,4,5,5,5,5,6,6,1,1
+};
+
+static const unsigned char utf8_mask[6] = {
+ 0x7F,
+ 0x1F,
+ 0x0F,
+ 0x07,
+ 0x03,
+ 0x01
+};
+
+int tb_utf8_char_length(char c)
+{
+ return utf8_length[(unsigned char)c];
+}
+
+int tb_utf8_char_to_unicode(uint32_t *out, const char *c)
+{
+ if (*c == 0)
+ return TB_EOF;
+
+ int i;
+ unsigned char len = tb_utf8_char_length(*c);
+ unsigned char mask = utf8_mask[len-1];
+ uint32_t result = c[0] & mask;
+ for (i = 1; i < len; ++i) {
+ result <<= 6;
+ result |= c[i] & 0x3f;
+ }
+
+ *out = result;
+ return (int)len;
+}
+
+int tb_utf8_unicode_to_char(char *out, uint32_t c)
+{
+ int len = 0;
+ int first;
+ int i;
+
+ if (c < 0x80) {
+ first = 0;
+ len = 1;
+ } else if (c < 0x800) {
+ first = 0xc0;
+ len = 2;
+ } else if (c < 0x10000) {
+ first = 0xe0;
+ len = 3;
+ } else if (c < 0x200000) {
+ first = 0xf0;
+ len = 4;
+ } else if (c < 0x4000000) {
+ first = 0xf8;
+ len = 5;
+ } else {
+ first = 0xfc;
+ len = 6;
+ }
+
+ for (i = len - 1; i > 0; --i) {
+ out[i] = (c & 0x3f) | 0x80;
+ c >>= 6;
+ }
+ out[0] = c | first;
+
+ return len;
+}