readr

readr
Log | Files | Refs

commit a72f5b5cdf14cb11627d1f3edaad84db78862ce4
Author: citbl <citbl@citbl.org>
Date:   Sun,  5 Oct 2025 21:31:23 +1000

init

Diffstat:
A.clang-format | 25+++++++++++++++++++++++++
A.clangd | 9+++++++++
A.gitignore | 5+++++
A.zed/debug.json | 18++++++++++++++++++
ATODO | 12++++++++++++
Aconfig.h | 14++++++++++++++
Adocs/readr.1 | 50++++++++++++++++++++++++++++++++++++++++++++++++++
Adocs/readr.html | 79+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Adocs/readr.scd | 43+++++++++++++++++++++++++++++++++++++++++++
Afeeds.c | 110+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Afeeds.h | 6++++++
Akeys.c | 51+++++++++++++++++++++++++++++++++++++++++++++++++++
Amakefile | 27+++++++++++++++++++++++++++
Areadr.c | 42++++++++++++++++++++++++++++++++++++++++++
Areadr.h | 34++++++++++++++++++++++++++++++++++
Arender.c | 64++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Atermbox2.h | 3780+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
Atui.c | 51+++++++++++++++++++++++++++++++++++++++++++++++++++
Atui.h | 9+++++++++
Autils.h | 296+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
20 files changed, 4725 insertions(+), 0 deletions(-)

diff --git a/.clang-format b/.clang-format @@ -0,0 +1,25 @@ +BasedOnStyle: Webkit +IndentWidth: 8 +ContinuationIndentWidth: 8 +UseTab: AlignWithSpaces +AlignTrailingComments: true +SpacesBeforeTrailingComments: 1 +KeepEmptyLinesAtTheStartOfBlocks: false +AllowShortBlocksOnASingleLine: true +AllowShortIfStatementsOnASingleLine: true +AllowShortCaseLabelsOnASingleLine: false +AllowShortEnumsOnASingleLine: true +AllowShortFunctionsOnASingleLine: false +AlignConsecutiveDeclarations: false +AlignConsecutiveAssignments: false +AlignConsecutiveMacros: false +SortIncludes: false + +IndentCaseLabels: false +ColumnLimit: 110 +PenaltyBreakBeforeFirstCallParameter: 1 +AlignAfterOpenBracket: DontAlign +BinPackArguments: false +BinPackParameters: false + +BreakAfterReturnType: TopLevelDefinitions diff --git a/.clangd b/.clangd @@ -0,0 +1,9 @@ +CompileFlags: + Add: [ + -x,c, + -Wall, + -Wextra, + "-std=c11", + -g, + "-I/opt/homebrew/include" + ] diff --git a/.gitignore b/.gitignore @@ -0,0 +1,5 @@ +readr +FEEDS +.DS_Store +*.dSYM/ + diff --git a/.zed/debug.json b/.zed/debug.json @@ -0,0 +1,18 @@ +// Project-local debug tasks +// +// For more documentation on how to configure debug tasks, +// see: https://zed.dev/docs/debugger +[ + { + "label": "Debug native binary", + "build": { + "command": "make", + "args": [""], + "cwd": "$ZED_WORKTREE_ROOT" + }, + "program": "$ZED_WORKTREE_ROOT/readr", + "args": [""], + "request": "launch", + "adapter": "CodeLLDB" + } +] diff --git a/TODO b/TODO @@ -0,0 +1,12 @@ +show more + show domains + show ending (html|pdf) +caching +instant display +background fetching +status bar +folders + indented feeds under one + unify duplicate urls +reload button +fetch content of webpage and read inline diff --git a/config.h b/config.h @@ -0,0 +1,14 @@ +#pragma once + +#define FEEDS "~/.config/readr/feeds" + +#define FEED_CAP 20 +#define POSTS_CAP 128 +#define FEEDS_CAP 32 +#define TITLE_CAP 128 + +#define TEXT_COLOR TB_WHITE +#define BACK_COLOR TB_DEFAULT +#define LINE_COLOR (TB_BLACK | TB_BRIGHT) +#define FEED_COLOR TB_CYAN +#define POST_COLOR TB_YELLOW diff --git a/docs/readr.1 b/docs/readr.1 @@ -0,0 +1,50 @@ +.\" Generated by scdoc 1.11.3 +.\" Complete documentation for this program is not available as a GNU info page +.ie \n(.g .ds Aq \(aq +.el .ds Aq ' +.nh +.ad l +.\" Begin generated content: +.TH "READR" "1" "2025-10-05" +.PP +.SH NAME +.PP +readr - small RSS viewer in terminal +.PP +.SH SYNOPSIS +.PP +readr +.PP +.SH DESCRIPTION +.PP +readr is a minimal RSS viewer for the terminal, it reads feeds sources from a file and displays the posts +.PP +.SH OPTIONS +.PP +see config.\&h, overwrite at will and recompile.\& +.PP +.SH REQUIREMENTS +.PP +\fBlibcurl\fR(1), \fBlibmrss\fR(1) +.PP +.SH EXAMPLES +.PP +create a file name ~/.\&config/readr/feeds +.PP +fill it with feeds, separated by a new line, e.\&g.\& +.PP +https://example.\&com/rss1 +https://example.\&com/rss2 +https://example.\&com/rss3 +.PP +launch readr +.PP +select feeds/posts, with arrows, and ENTER to open the post link.\& +.PP +.SH AUTHORS +.PP +citbl +.PP +.SH LICENSE +.PP +MIT diff --git a/docs/readr.html b/docs/readr.html @@ -0,0 +1,79 @@ +<!DOCTYPE html> +<html> +<!-- This is an automatically generated file. Do not edit. + Generated by scdoc 1.11.3 + Complete documentation for this program is not available as a GNU info page + --> +<head> + <meta charset="utf-8"/> + <meta name="viewport" content="width=device-width, initial-scale=1.0"/> + <style> + table.head, table.foot { width: 100%; } + td.head-rtitle, td.foot-os { text-align: right; } + td.head-vol { text-align: center; } + .Nd, .Bf, .Op { display: inline; } + .Pa, .Ad { font-style: italic; } + .Ms { font-weight: bold; } + .Bl-diag > dt { font-weight: bold; } + code.Nm, .Fl, .Cm, .Ic, code.In, .Fd, .Fn, .Cd { font-weight: bold; + font-family: inherit; } + </style> + <title>READR(1)</title> +<style>html{background: #333; color: #eee; max-width: 50rem; margin: 1rem auto; font-family: sans-serif;} a{color: white;}</style></head> +<body> +<table class="head"> + <tr> + <td class="head-ltitle">READR(1)</td> + <td class="head-vol">General Commands Manual</td> + <td class="head-rtitle">READR(1)</td> + </tr> +</table> +<div class="manual-text"> +<section class="Sh"> +<h1 class="Sh" id="NAME"><a class="permalink" href="#NAME">NAME</a></h1> +<p class="Pp">readr - small RSS viewer in terminal</p> +</section> +<section class="Sh"> +<h1 class="Sh" id="SYNOPSIS"><a class="permalink" href="#SYNOPSIS">SYNOPSIS</a></h1> +<p class="Pp">readr</p> +</section> +<section class="Sh"> +<h1 class="Sh" id="DESCRIPTION"><a class="permalink" href="#DESCRIPTION">DESCRIPTION</a></h1> +<p class="Pp">readr is a minimal RSS viewer for the terminal, it reads feeds + sources from a file and displays the posts</p> +</section> +<section class="Sh"> +<h1 class="Sh" id="OPTIONS"><a class="permalink" href="#OPTIONS">OPTIONS</a></h1> +<p class="Pp">see config.h, overwrite at will and recompile.</p> +</section> +<section class="Sh"> +<h1 class="Sh" id="REQUIREMENTS"><a class="permalink" href="#REQUIREMENTS">REQUIREMENTS</a></h1> +<p class="Pp"><b>libcurl</b>(1), <b>libmrss</b>(1)</p> +</section> +<section class="Sh"> +<h1 class="Sh" id="EXAMPLES"><a class="permalink" href="#EXAMPLES">EXAMPLES</a></h1> +<p class="Pp">create a file name ~/.config/readr/feeds</p> +<p class="Pp">fill it with feeds, separated by a new line, e.g.</p> +<p class="Pp">https://example.com/rss1 https://example.com/rss2 + https://example.com/rss3</p> +<p class="Pp">launch readr</p> +<p class="Pp">select feeds/posts, with arrows, and ENTER to open the post + link.</p> +</section> +<section class="Sh"> +<h1 class="Sh" id="AUTHORS"><a class="permalink" href="#AUTHORS">AUTHORS</a></h1> +<p class="Pp">citbl</p> +</section> +<section class="Sh"> +<h1 class="Sh" id="LICENSE"><a class="permalink" href="#LICENSE">LICENSE</a></h1> +<p class="Pp">MIT</p> +</section> +</div> +<table class="foot"> + <tr> + <td class="foot-date">2025-10-05</td> + <td class="foot-os"></td> + </tr> +</table> +</body> +</html> diff --git a/docs/readr.scd b/docs/readr.scd @@ -0,0 +1,43 @@ +READR(1) + +# NAME + +readr - small RSS viewer in terminal + +# SYNOPSIS + +readr + +# DESCRIPTION + +readr is a minimal RSS viewer for the terminal, it reads feeds sources from a file and displays the posts + +# OPTIONS + +see config.h, overwrite at will and recompile. + +# REQUIREMENTS + +*libcurl*(1), *libmrss*(1) + +# EXAMPLES + +create a file name ~/.config/readr/feeds + +fill it with feeds, separated by a new line, e.g. + +https://example.com/rss1 +https://example.com/rss2 +https://example.com/rss3 + +launch readr + +select feeds/posts, with arrows, and ENTER to open the post link. + +# AUTHORS + +citbl + +# LICENSE + +MIT diff --git a/feeds.c b/feeds.c @@ -0,0 +1,110 @@ +#include "feeds.h" + +#include <_string.h> +#include <mrss.h> +#include <stdlib.h> +#include <string.h> +#include <sys/param.h> + +#include "config.h" +#include "utils.h" + +void fetch_feed(feed_t*, char*); + +app_t +load_app(char* contents) +{ + app_t app = { + .feeds_cap = FEEDS_CAP, + .feeds = calloc(FEEDS_CAP, sizeof(char*)), + }; + + remove_all_chars(contents, '\r'); + char* line = strtok(contents, "\n"); + + while (line != NULL) { + feed_t* feed = (feed_t*)malloc(sizeof(feed_t)); + + if (feed == NULL) perror("could not alloc feed"); + + fetch_feed(feed, line); + + if (app.feeds_cap == app.feeds_len) { + app.feeds_cap *= 2; + app.feeds = realloc(app.feeds, app.feeds_cap); + } + app.feeds[app.feeds_len++] = feed; + line = strtok(NULL, "\n"); + } + + return app; +} + +void +fetch_feed(feed_t* feed, char* url) +{ + *feed = (feed_t) { .url = url, .posts_cap = POSTS_CAP }; + feed->posts = (post_t**)calloc(POSTS_CAP, sizeof(post_t*)); + + static mrss_t* rss = NULL; + mrss_error_t rc = mrss_parse_url(url, &rss); + + if (rc != MRSS_OK) { + fprintf(stderr, "parse feed failed: %s\n", mrss_strerror(rc)); + return; + } + + feed->title = (rss->title != NULL) ? strndup(rss->title, FEED_CAP) : "Unknown feed"; + + for (mrss_item_t* it = rss->item; it; it = it->next) { + char* title = (it->title && *it->title) ? it->title : ""; + const char* link = (it->link && *it->link) ? it->link : ""; + char* desc = (it->description && *it->description) ? it->description : ""; + const char* date = (it->pubDate && *it->pubDate) ? it->pubDate : ""; + const char* author = (it->author && *it->author) ? it->author : ""; + + remove_all_tags(desc); + + const size_t title_ = strlen(title), link_ = strlen(link), desc_ = strlen(desc), + date_ = strlen(date), author_ = strlen(author); + + size_t data_len = title_ + link_ + desc_ + date_ + author_ + 1; + + char* d = calloc(data_len + 1, sizeof(char)); + + if (d == NULL) { perror("could not alloc post data"); } + + strncat(d, title, title_); + strncat(d, link, link_); + strncat(d, desc, desc_); + strncat(d, date, date_); + strncat(d, author, author_); + d[data_len] = '\0'; + + post_t* post = (post_t*)malloc(sizeof(post_t)); + + if (post == NULL) { perror("could not alloc post"); } + + *post = (post_t) { + .data = d, + .data_len = data_len, + .title = (slice_t) { .start = 0, .len = title_ }, + .link = (slice_t) { .start = title_, .len = link_ }, + .date = (slice_t) { .start = title_ + link_, .len = date_ }, + .author = (slice_t) { .start = title_ + link_ + date_, .len = author_ }, + }; + + if (feed->posts_len == feed->posts_cap) { + feed->posts_cap *= 2; + feed->posts = realloc(feed->posts, feed->posts_cap); + } + feed->posts[feed->posts_len++] = post; + + // if (it->link && *it->link) { + // const char* domain = host_from_url(it->link); + // printf("%-60.60s \t%s \t%s\n", title, domain, it->link); + // } + } + + mrss_free(rss); +} diff --git a/feeds.h b/feeds.h @@ -0,0 +1,6 @@ +#pragma once + +#include "readr.h" + +app_t load_app(char* contents); +void fetch_feed(feed_t* feed, char* url); diff --git a/keys.c b/keys.c @@ -0,0 +1,51 @@ +#include <sys/param.h> +#include "readr.h" +#include "tui.h" +#include "utils.h" + +#define MAX_PANEL 1 + +void +handle_key(app_t* app, struct tb_event ev) +{ + switch (ev.key) { + case TB_KEY_ARROW_UP: + if (app->selected_panel == 0) { // change feed + app->selected_feed = MAX(app->selected_feed - 1, 0); + app->selected_post = 0; + } else if (app->selected_panel == 1) { // change post + app->selected_post = MAX(app->selected_post - 1, 0); + } + break; + + case TB_KEY_ARROW_DOWN: + if (app->selected_panel == 0) { // change feed + app->selected_feed = MIN(app->selected_feed + 1, app->feeds_len - 1); + app->selected_post = 0; + } else if (app->selected_panel == 1) { // change post + app->selected_post + = MIN(app->selected_post + 1, app->feeds[app->selected_feed]->posts_len - 1); + } + break; + + case TB_KEY_ARROW_LEFT: // change panel + app->selected_panel = MAX(app->selected_panel - 1, 0); + break; + + case TB_KEY_ARROW_RIGHT: // change panel + app->selected_panel = MIN(app->selected_panel + 1, MAX_PANEL); + break; + case TB_KEY_ENTER: { + if (app->selected_panel == 1) { + static char url[512] = { 0 }; + post_t* post = app->feeds[app->selected_feed]->posts[app->selected_post]; + size_t len = MIN(512, post->link.len); + strncpy(url, &post->data[post->link.start], len); + url[len] = '\0'; + open_url(url); + } + } break; + default: + break; + } +} diff --git a/makefile b/makefile @@ -0,0 +1,27 @@ +APP := readr +PKG := $(shell pkg-config --cflags --libs mrss) +CFLAGS := -std=c99 -g -Wall -Wextra -fsanitize=address -fsanitize=undefined +CREL := -std=c99 -Wall -Wextra -Wpedantic -Werror -O3 + +default: + cc **.c ${CFLAGS} ${PKG} -o ${APP} + +clean: + $(RM) readr + +release: clean + cc **.c ${CREL} ${PKG} -o ${APP} + +install: release man + mkdir -p /usr/local/bin + install -m755 readr /usr/local/bin/. + mkdir -p /usr/local/share/man/man1 + install -m644 docs/readr.1 /usr/local/share/man/man1/. + +manpage: + scdoc < docs/readr.scd > docs/readr.1 + +html: manpage + mandoc -T html docs/readr.1 > docs/readr.html.tmp + awk '{gsub(/<\/head>/,"<style>html{background: #333; color: #eee; max-width: 50rem; margin: 1rem auto; font-family: sans-serif;} a{color: white;}<\/style><\/head>")}1' docs/readr.html.tmp > docs/readr.html + $(RM) docs/readr.html.tmp diff --git a/readr.c b/readr.c @@ -0,0 +1,42 @@ +#include <mrss.h> +#include <stdlib.h> +#include <string.h> +#include <locale.h> + +#define UTILS_IMPL +#include "utils.h" + +#include "config.h" +#include "readr.h" +#include "feeds.h" +#include "tui.h" + +int +main(void) +{ + // for unicode blocks/lines, curvy quotes etc. + const char* ok = setlocale(LC_ALL, "C.UTF-8"); + + if (!ok) { + perror("setlocale"); + return 1; + } + + const char* path = expand_tilde(FEEDS); + if (!path) return 1; + + char* feeds_contents = read_file(path); + if (feeds_contents == NULL) { + printf("No '%s' file found. Create it and put one RSS feed URL per " + "line.\nAlternatively change 'config.h' with your desired feeds location " + "and recompile.\n", + FEEDS); + return 0; + } + + printf("Fetching feeds...\n\n%s\n...", feeds_contents); + + app_t app = load_app(feeds_contents); + + return present(&app); +} diff --git a/readr.h b/readr.h @@ -0,0 +1,34 @@ +#pragma once + +#include <stdio.h> +#include "termbox2.h" + +// +// data +// + +typedef struct { + size_t start, len; +} slice_t; + +typedef struct { + const char* data; + size_t data_len; + slice_t title, link, desc, date, author; +} post_t; + +typedef struct { + const char* url; + char* title; + post_t** posts; + int posts_len, posts_cap; + +} feed_t; + +typedef struct { + feed_t** feeds; + int feeds_len, feeds_cap; + int selected_panel; + int selected_feed; + int selected_post; +} app_t; diff --git a/render.c b/render.c @@ -0,0 +1,64 @@ +#include "termbox2.h" +#include "tui.h" +#include "config.h" +#include <sys/param.h> + +static void +draw_top_bar(void) +{ + const int width = tb_width(); + for (int i = 0; i < width; i++) { + tb_set_cell(i, 0, ' ', LINE_COLOR, LINE_COLOR); + } + + tb_print(1, 0, TEXT_COLOR, LINE_COLOR, "readr"); + tb_print(width - 10, 0, TEXT_COLOR, LINE_COLOR, "[q] quit"); +} + +static void +draw_background(void) +{ + const int height = tb_height(); + tb_set_clear_attrs(TEXT_COLOR, BACK_COLOR); + tb_clear(); + + for (int i = 0; i < height; i++) { + tb_set_cell(FEED_CAP + 5, i, 0x2595, LINE_COLOR, BACK_COLOR); /* vert line */ + } +} + +void +render(app_t* app) +{ + uintattr_t color; + + draw_background(); + + draw_top_bar(); + + int i; + for (i = 0; i < app->feeds_len; i++) { + const feed_t* feed = app->feeds[i]; + color = (app->selected_panel == 0 && i == app->selected_feed) ? (FEED_COLOR | TB_REVERSE) + : (FEED_COLOR); + tb_print(1, 4 + i, color, TB_DEFAULT, feed->title ? feed->title : "N/A"); + } + + char title[TITLE_CAP] = { 0 }; + feed_t* feed = app->feeds[app->selected_feed]; + + for (i = 0; i < feed->posts_len; i++) { + const post_t* post = feed->posts[i]; + size_t len = MIN(TITLE_CAP, post->title.len); + strncpy(title, &post->data[post->title.start], len); + title[len] = '\0'; + // printf("%s/%zu", title, post->title.len); + // exit(0); + color = (app->selected_panel == 1 && i == app->selected_post) ? (POST_COLOR | TB_REVERSE) + : (POST_COLOR); + // nonascii_replace(title, '\''); + tb_print(FEED_CAP + 10, 4 + i, color, TB_DEFAULT, title); + } + + tb_present(); +} diff --git a/termbox2.h b/termbox2.h @@ -0,0 +1,3780 @@ +/* +MIT License + +Copyright (c) 2010-2020 nsf <no.smile.face@gmail.com> + 2015-2024 Adam Saponara <as@php.net> + +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. +*/ + +#ifndef TERMBOX_H_INCL +#define TERMBOX_H_INCL + +#ifndef _XOPEN_SOURCE +#define _XOPEN_SOURCE +#endif + +#ifndef _DEFAULT_SOURCE +#define _DEFAULT_SOURCE +#endif + +#include <errno.h> +#include <fcntl.h> +#include <limits.h> +#include <signal.h> +#include <stdarg.h> +#include <stdint.h> +#include <stdio.h> +#include <stdlib.h> +#include <string.h> +#include <sys/ioctl.h> +#include <sys/select.h> +#include <sys/stat.h> +#include <sys/time.h> +#include <sys/types.h> +#include <termios.h> +#include <unistd.h> +#include <wchar.h> +#include <wctype.h> + +#ifdef PATH_MAX +#define TB_PATH_MAX PATH_MAX +#else +#define TB_PATH_MAX 4096 +#endif + +#ifdef __cplusplus +extern "C" { +#endif + +// __ffi_start + +#define TB_VERSION_STR "2.5.0" + +/* The following compile-time options are supported: + * + * TB_OPT_ATTR_W: Integer width of fg and bg attributes. Valid values + * (assuming system support) are 16, 32, and 64. (See + * uintattr_t). 32 or 64 enables output mode + * TB_OUTPUT_TRUECOLOR. 64 enables additional style + * attributes. (See tb_set_output_mode.) Larger values + * consume more memory in exchange for more features. + * Defaults to 16. + * + * TB_OPT_EGC: If set, enable extended grapheme cluster support + * (tb_extend_cell, tb_set_cell_ex). Consumes more memory. + * Defaults off. + * + * TB_OPT_PRINTF_BUF: Write buffer size for printf operations. Represents the + * largest string that can be sent in one call to tb_print* + * and tb_send* functions. Defaults to 4096. + * + * TB_OPT_READ_BUF: Read buffer size for tty reads. Defaults to 64. + * + * TB_OPT_TRUECOLOR: Deprecated. Sets TB_OPT_ATTR_W to 32 if not already set. + */ + +#if defined(TB_LIB_OPTS) || 0 // __tb_lib_opts +/* Ensure consistent compile-time options when using as a shared library */ +#undef TB_OPT_ATTR_W +#undef TB_OPT_EGC +#undef TB_OPT_PRINTF_BUF +#undef TB_OPT_READ_BUF +#define TB_OPT_ATTR_W 64 +#define TB_OPT_EGC +#endif + +/* Ensure sane `TB_OPT_ATTR_W` (16, 32, or 64) */ +#if defined TB_OPT_ATTR_W && TB_OPT_ATTR_W == 16 +#elif defined TB_OPT_ATTR_W && TB_OPT_ATTR_W == 32 +#elif defined TB_OPT_ATTR_W && TB_OPT_ATTR_W == 64 +#else +#undef TB_OPT_ATTR_W +#if defined TB_OPT_TRUECOLOR // Deprecated. Back-compat for old flag. +#define TB_OPT_ATTR_W 32 +#else +#define TB_OPT_ATTR_W 16 +#endif +#endif + +/* ASCII key constants (`tb_event.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' + +#define tb_key_i(i) 0xffff - (i) +/* Terminal-dependent key constants (`tb_event.key`) and terminfo caps */ +/* BEGIN codegen h */ +/* Produced by ./codegen.sh on Tue, 03 Sep 2024 04:17:47 +0000 */ +#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_BACK_TAB (0xffff - 22) +#define TB_KEY_MOUSE_LEFT (0xffff - 23) +#define TB_KEY_MOUSE_RIGHT (0xffff - 24) +#define TB_KEY_MOUSE_MIDDLE (0xffff - 25) +#define TB_KEY_MOUSE_RELEASE (0xffff - 26) +#define TB_KEY_MOUSE_WHEEL_UP (0xffff - 27) +#define TB_KEY_MOUSE_WHEEL_DOWN (0xffff - 28) + +#define TB_CAP_F1 0 +#define TB_CAP_F2 1 +#define TB_CAP_F3 2 +#define TB_CAP_F4 3 +#define TB_CAP_F5 4 +#define TB_CAP_F6 5 +#define TB_CAP_F7 6 +#define TB_CAP_F8 7 +#define TB_CAP_F9 8 +#define TB_CAP_F10 9 +#define TB_CAP_F11 10 +#define TB_CAP_F12 11 +#define TB_CAP_INSERT 12 +#define TB_CAP_DELETE 13 +#define TB_CAP_HOME 14 +#define TB_CAP_END 15 +#define TB_CAP_PGUP 16 +#define TB_CAP_PGDN 17 +#define TB_CAP_ARROW_UP 18 +#define TB_CAP_ARROW_DOWN 19 +#define TB_CAP_ARROW_LEFT 20 +#define TB_CAP_ARROW_RIGHT 21 +#define TB_CAP_BACK_TAB 22 +#define TB_CAP__COUNT_KEYS 23 +#define TB_CAP_ENTER_CA 23 +#define TB_CAP_EXIT_CA 24 +#define TB_CAP_SHOW_CURSOR 25 +#define TB_CAP_HIDE_CURSOR 26 +#define TB_CAP_CLEAR_SCREEN 27 +#define TB_CAP_SGR0 28 +#define TB_CAP_UNDERLINE 29 +#define TB_CAP_BOLD 30 +#define TB_CAP_BLINK 31 +#define TB_CAP_ITALIC 32 +#define TB_CAP_REVERSE 33 +#define TB_CAP_ENTER_KEYPAD 34 +#define TB_CAP_EXIT_KEYPAD 35 +#define TB_CAP_DIM 36 +#define TB_CAP_INVISIBLE 37 +#define TB_CAP__COUNT 38 +/* END codegen h */ + +/* Some hard-coded caps */ +#define TB_HARDCAP_ENTER_MOUSE "\x1b[?1000h\x1b[?1002h\x1b[?1015h\x1b[?1006h" +#define TB_HARDCAP_EXIT_MOUSE "\x1b[?1006l\x1b[?1015l\x1b[?1002l\x1b[?1000l" +#define TB_HARDCAP_STRIKEOUT "\x1b[9m" +#define TB_HARDCAP_UNDERLINE_2 "\x1b[21m" +#define TB_HARDCAP_OVERLINE "\x1b[53m" + +/* Colors (numeric) and attributes (bitwise) (`tb_cell.fg`, `tb_cell.bg`) */ +#define TB_DEFAULT 0x0000 +#define TB_BLACK 0x0001 +#define TB_RED 0x0002 +#define TB_GREEN 0x0003 +#define TB_YELLOW 0x0004 +#define TB_BLUE 0x0005 +#define TB_MAGENTA 0x0006 +#define TB_CYAN 0x0007 +#define TB_WHITE 0x0008 + +#if TB_OPT_ATTR_W == 16 +#define TB_BOLD 0x0100 +#define TB_UNDERLINE 0x0200 +#define TB_REVERSE 0x0400 +#define TB_ITALIC 0x0800 +#define TB_BLINK 0x1000 +#define TB_HI_BLACK 0x2000 +#define TB_BRIGHT 0x4000 +#define TB_DIM 0x8000 +#define TB_256_BLACK TB_HI_BLACK // `TB_256_BLACK` is deprecated +#else +// `TB_OPT_ATTR_W` is 32 or 64 +#define TB_BOLD 0x01000000 +#define TB_UNDERLINE 0x02000000 +#define TB_REVERSE 0x04000000 +#define TB_ITALIC 0x08000000 +#define TB_BLINK 0x10000000 +#define TB_HI_BLACK 0x20000000 +#define TB_BRIGHT 0x40000000 +#define TB_DIM 0x80000000 +#define TB_TRUECOLOR_BOLD TB_BOLD // `TB_TRUECOLOR_*` is deprecated +#define TB_TRUECOLOR_UNDERLINE TB_UNDERLINE +#define TB_TRUECOLOR_REVERSE TB_REVERSE +#define TB_TRUECOLOR_ITALIC TB_ITALIC +#define TB_TRUECOLOR_BLINK TB_BLINK +#define TB_TRUECOLOR_BLACK TB_HI_BLACK +#endif + +#if TB_OPT_ATTR_W == 64 +#define TB_STRIKEOUT 0x0000000100000000 +#define TB_UNDERLINE_2 0x0000000200000000 +#define TB_OVERLINE 0x0000000400000000 +#define TB_INVISIBLE 0x0000000800000000 +#endif + +/* Event types (`tb_event.type`) */ +#define TB_EVENT_KEY 1 +#define TB_EVENT_RESIZE 2 +#define TB_EVENT_MOUSE 3 + +/* Key modifiers (bitwise) (`tb_event.mod`) */ +#define TB_MOD_ALT 1 +#define TB_MOD_CTRL 2 +#define TB_MOD_SHIFT 4 +#define TB_MOD_MOTION 8 + +/* Input modes (bitwise) (`tb_set_input_mode`) */ +#define TB_INPUT_CURRENT 0 +#define TB_INPUT_ESC 1 +#define TB_INPUT_ALT 2 +#define TB_INPUT_MOUSE 4 + +/* Output modes (`tb_set_output_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 +#if TB_OPT_ATTR_W >= 32 +#define TB_OUTPUT_TRUECOLOR 5 +#endif + +/* Common function return values unless otherwise noted. + * + * Library behavior is undefined after receiving `TB_ERR_MEM`. Callers may + * attempt reinitializing by freeing memory, invoking `tb_shutdown`, then + * `tb_init`. + */ +#define TB_OK 0 +#define TB_ERR -1 +#define TB_ERR_NEED_MORE -2 +#define TB_ERR_INIT_ALREADY -3 +#define TB_ERR_INIT_OPEN -4 +#define TB_ERR_MEM -5 +#define TB_ERR_NO_EVENT -6 +#define TB_ERR_NO_TERM -7 +#define TB_ERR_NOT_INIT -8 +#define TB_ERR_OUT_OF_BOUNDS -9 +#define TB_ERR_READ -10 +#define TB_ERR_RESIZE_IOCTL -11 +#define TB_ERR_RESIZE_PIPE -12 +#define TB_ERR_RESIZE_SIGACTION -13 +#define TB_ERR_POLL -14 +#define TB_ERR_TCGETATTR -15 +#define TB_ERR_TCSETATTR -16 +#define TB_ERR_UNSUPPORTED_TERM -17 +#define TB_ERR_RESIZE_WRITE -18 +#define TB_ERR_RESIZE_POLL -19 +#define TB_ERR_RESIZE_READ -20 +#define TB_ERR_RESIZE_SSCANF -21 +#define TB_ERR_CAP_COLLISION -22 + +#define TB_ERR_SELECT TB_ERR_POLL +#define TB_ERR_RESIZE_SELECT TB_ERR_RESIZE_POLL + +/* Deprecated. Function types to be used with `tb_set_func`. */ +#define TB_FUNC_EXTRACT_PRE 0 +#define TB_FUNC_EXTRACT_POST 1 + +/* Define this to set the size of the buffer used in `tb_printf` + * and `tb_sendf` + */ +#ifndef TB_OPT_PRINTF_BUF +#define TB_OPT_PRINTF_BUF 4096 +#endif + +/* Define this to set the size of the read buffer used when reading + * from the tty + */ +#ifndef TB_OPT_READ_BUF +#define TB_OPT_READ_BUF 64 +#endif + +/* Define this for limited back compat with termbox v1 */ +#ifdef TB_OPT_V1_COMPAT +#define tb_change_cell tb_set_cell +#define tb_put_cell(x, y, c) tb_set_cell((x), (y), (c)->ch, (c)->fg, (c)->bg) +#define tb_set_clear_attributes tb_set_clear_attrs +#define tb_select_input_mode tb_set_input_mode +#define tb_select_output_mode tb_set_output_mode +#endif + +/* Define these to swap in a different allocator */ +#ifndef tb_malloc +#define tb_malloc malloc +#define tb_realloc realloc +#define tb_free free +#endif + +#if TB_OPT_ATTR_W == 64 +typedef uint64_t uintattr_t; +#elif TB_OPT_ATTR_W == 32 +typedef uint32_t uintattr_t; +#else // 16 +typedef uint16_t uintattr_t; +#endif + +/* A cell in a 2d grid representing the terminal screen. + * + * The terminal screen is represented as 2d array of cells. The structure is + * optimized for dealing with single-width (`wcwidth==1`) Unicode codepoints, + * however some support for grapheme clusters (e.g., combining diacritical + * marks) and wide codepoints (e.g., Hiragana) is provided through `ech`, + * `nech`, and `cech` via `tb_set_cell_ex`. `ech` is only valid when `nech>0`, + * otherwise `ch` is used. + * + * For non-single-width codepoints, given `N=wcwidth(ch)/wcswidth(ech)`: + * + * when `N==0`: termbox forces a single-width cell. Callers should avoid this + * if aiming to render text accurately. Callers may use + * `tb_set_cell_ex` or `tb_print*` to render `N==0` combining + * characters. + * + * when `N>1`: termbox zeroes out the following `N-1` cells and skips sending + * them to the tty. So, e.g., if the caller sets `x=0,y=0` to an + * `N==2` codepoint, the caller's next set should be at `x=2,y=0`. + * Anything set at `x=1,y=0` will be ignored. If there are not + * enough columns remaining on the line to render `N` width, spaces + * are sent instead. + * + * See `tb_present` for implementation. + */ +struct tb_cell { + uint32_t ch; // a Unicode codepoint + uintattr_t fg; // bitwise foreground attributes + uintattr_t bg; // bitwise background attributes +#ifdef TB_OPT_EGC + uint32_t* ech; // a grapheme cluster of Unicode codepoints, 0-terminated + size_t nech; // num elements in ech, 0 means use ch instead of ech + size_t cech; // num elements allocated for ech +#endif +}; + +/* An incoming event from the tty. + * + * Given the event type, the following fields are relevant: + * + * when `TB_EVENT_KEY`: `key` xor `ch` (one will be zero) and `mod`. Note + * there is overlap between `TB_MOD_CTRL` and + * `TB_KEY_CTRL_*`. `TB_MOD_CTRL` and `TB_MOD_SHIFT` are + * only set as modifiers to `TB_KEY_ARROW_*`. + * + * when `TB_EVENT_RESIZE`: `w` and `h` + * + * when `TB_EVENT_MOUSE`: `key` (`TB_KEY_MOUSE_*`), `x`, and `y` + */ +struct tb_event { + uint8_t type; // one of `TB_EVENT_*` constants + uint8_t mod; // bitwise `TB_MOD_*` constants + uint16_t key; // one of `TB_KEY_*` constants + uint32_t ch; // a Unicode codepoint + int32_t w; // resize width + int32_t h; // resize height + int32_t x; // mouse x + int32_t y; // mouse y +}; + +/* Initialize the termbox library. This function should be called before any + * other functions. `tb_init` is equivalent to `tb_init_file("/dev/tty")`. After + * successful initialization, the library must be finalized using `tb_shutdown`. + */ +int tb_init(void); +int tb_init_file(const char* path); +int tb_init_fd(int ttyfd); +int tb_init_rwfd(int rfd, int wfd); +int tb_shutdown(void); + +/* Return the size of the internal back buffer (which is the same as terminal's + * window size in rows and columns). The internal buffer can be resized after + * `tb_clear` or `tb_present` calls. Both dimensions have an unspecified + * negative value when called before `tb_init` or after `tb_shutdown`. + */ +int tb_width(void); +int tb_height(void); + +/* Clear the internal back buffer using `TB_DEFAULT` or the attributes set by + * `tb_set_clear_attrs`. + */ +int tb_clear(void); +int tb_set_clear_attrs(uintattr_t fg, uintattr_t bg); + +/* Synchronize the internal back buffer with the terminal by writing to tty. */ +int tb_present(void); + +/* Clear the internal front buffer effectively forcing a complete re-render of + * the back buffer to the tty. It is not necessary to call this under normal + * circumstances. */ +int tb_invalidate(void); + +/* Set the position of the cursor. Upper-left cell is (0, 0). */ +int tb_set_cursor(int cx, int cy); +int tb_hide_cursor(void); + +/* Set cell contents in the internal back buffer at the specified position. + * + * Use `tb_set_cell_ex` for rendering grapheme clusters (e.g., combining + * diacritical marks). + * + * Calling `tb_set_cell(x, y, ch, fg, bg)` is equivalent to + * `tb_set_cell_ex(x, y, &ch, 1, fg, bg)`. + * + * `tb_extend_cell` is a shortcut for appending 1 codepoint to `tb_cell.ech`. + * + * Non-printable (`iswprint(3)`) codepoints are replaced with `U+FFFD` at render + * time. + */ +int tb_set_cell(int x, int y, uint32_t ch, uintattr_t fg, uintattr_t bg); +int tb_set_cell_ex(int x, int y, uint32_t* ch, size_t nch, uintattr_t fg, uintattr_t bg); +int tb_extend_cell(int x, int y, uint32_t ch); + +/* Set the input mode. Termbox has two input modes: + * + * 1. `TB_INPUT_ESC` + * When escape (`\x1b`) is in the buffer and there's no match for an escape + * sequence, a key event for `TB_KEY_ESC` is returned. + * + * 2. `TB_INPUT_ALT` + * When escape (`\x1b`) is in the buffer and there's no match for an escape + * sequence, the next keyboard event is returned with a `TB_MOD_ALT` + * modifier. + * + * You can also apply `TB_INPUT_MOUSE` via bitwise OR operation to either of the + * modes (e.g., `TB_INPUT_ESC | TB_INPUT_MOUSE`) to receive `TB_EVENT_MOUSE` + * events. If none of the main two modes were set, but the mouse mode was, + * `TB_INPUT_ESC` is used. If for some reason you've decided to use + * `TB_INPUT_ESC | TB_INPUT_ALT`, it will behave as if only `TB_INPUT_ESC` was + * selected. + * + * If mode is `TB_INPUT_CURRENT`, return the current input mode. + * + * The default input mode is `TB_INPUT_ESC`. + */ +int tb_set_input_mode(int mode); + +/* Set the output mode. Termbox has multiple output modes: + * + * 1. `TB_OUTPUT_NORMAL` => [0..8] + * + * This mode provides 8 different colors: + * `TB_BLACK`, `TB_RED`, `TB_GREEN`, `TB_YELLOW`, + * `TB_BLUE`, `TB_MAGENTA`, `TB_CYAN`, `TB_WHITE` + * + * Plus `TB_DEFAULT` which skips sending a color code (i.e., uses the + * terminal's default color). + * + * Colors (including `TB_DEFAULT`) may be bitwise OR'd with attributes: + * `TB_BOLD`, `TB_UNDERLINE`, `TB_REVERSE`, `TB_ITALIC`, `TB_BLINK`, + * `TB_BRIGHT`, `TB_DIM` + * + * The following style attributes are also available if compiled with + * `TB_OPT_ATTR_W` set to 64: + * `TB_STRIKEOUT`, `TB_UNDERLINE_2`, `TB_OVERLINE`, `TB_INVISIBLE` + * + * As in all modes, the value 0 is interpreted as `TB_DEFAULT` for + * convenience. + * + * Some notes: `TB_REVERSE` and `TB_BRIGHT` can be applied as either `fg` or + * `bg` attributes for the same effect. The rest of the attributes apply to + * `fg` only and are ignored as `bg` attributes. + * + * Example usage: `tb_set_cell(x, y, '@', TB_BLACK | TB_BOLD, TB_RED)` + * + * 2. `TB_OUTPUT_256` => [0..255] + `TB_HI_BLACK` + * + * In this mode you get 256 distinct colors (plus default): + * 0x00 (1): `TB_DEFAULT` + * `TB_HI_BLACK` (1): `TB_BLACK` in `TB_OUTPUT_NORMAL` + * 0x01..0x07 (7): the next 7 colors as in `TB_OUTPUT_NORMAL` + * 0x08..0x0f (8): bright versions of the above + * 0x10..0xe7 (216): 216 different colors + * 0xe8..0xff (24): 24 different shades of gray + * + * All `TB_*` style attributes except `TB_BRIGHT` may be bitwise OR'd as in + * `TB_OUTPUT_NORMAL`. + * + * Note `TB_HI_BLACK` must be used for black, as 0x00 represents default. + * + * 3. `TB_OUTPUT_216` => [0..216] + * + * This mode supports the 216-color range of `TB_OUTPUT_256` only, but you + * don't need to provide an offset: + * 0x00 (1): `TB_DEFAULT` + * 0x01..0xd8 (216): 216 different colors + * + * 4. `TB_OUTPUT_GRAYSCALE` => [0..24] + * + * This mode supports the 24-color range of `TB_OUTPUT_256` only, but you + * don't need to provide an offset: + * 0x00 (1): `TB_DEFAULT` + * 0x01..0x18 (24): 24 different shades of gray + * + * 5. `TB_OUTPUT_TRUECOLOR` => [0x000000..0xffffff] + `TB_HI_BLACK` + * + * This mode provides 24-bit color on supported terminals. The format is + * 0xRRGGBB. + * + * All `TB_*` style attributes except `TB_BRIGHT` may be bitwise OR'd as in + * `TB_OUTPUT_NORMAL`. + * + * Note `TB_HI_BLACK` must be used for black, as 0x000000 represents default. + * + * To use the terminal default color (i.e., to not send an escape code), pass + * `TB_DEFAULT`. For convenience, the value 0 is interpreted as `TB_DEFAULT` in + * all modes. + * + * Note, cell attributes persist after switching output modes. Any translation + * between, for example, `TB_OUTPUT_NORMAL`'s `TB_RED` and + * `TB_OUTPUT_TRUECOLOR`'s 0xff0000 must be performed by the caller. Also note + * that cells previously rendered in one mode may persist unchanged until the + * front buffer is cleared (such as after a resize event) at which point it will + * be re-interpreted and flushed according to the current mode. Callers may + * invoke `tb_invalidate` if it is desirable to immediately re-interpret and + * flush the entire screen according to the current mode. + * + * Note, not all terminals support all output modes, especially beyond + * `TB_OUTPUT_NORMAL`. There is also no very reliable way to determine color + * support dynamically. If portability is desired, callers are recommended to + * use `TB_OUTPUT_NORMAL` or make output mode end-user configurable. The same + * advice applies to style attributes. + * + * If mode is `TB_OUTPUT_CURRENT`, return the current output mode. + * + * The default output mode is `TB_OUTPUT_NORMAL`. + */ +int tb_set_output_mode(int mode); + +/* Wait for an event up to `timeout_ms` milliseconds and populate `event` with + * it. If no event is available within the timeout period, `TB_ERR_NO_EVENT` + * is returned. On a resize event, the underlying `select(2)` call may be + * interrupted, yielding a return code of `TB_ERR_POLL`. In this case, you may + * check `errno` via `tb_last_errno`. If it's `EINTR`, you may elect to ignore + * that and call `tb_peek_event` again. + */ +int tb_peek_event(struct tb_event* event, int timeout_ms); + +/* Same as `tb_peek_event` except no timeout. */ +int tb_poll_event(struct tb_event* event); + +/* Internal termbox fds that can be used with `poll(2)`, `select(2)`, etc. + * externally. Callers must invoke `tb_poll_event` or `tb_peek_event` if + * fds become readable. */ +int tb_get_fds(int* ttyfd, int* resizefd); + +/* Print and printf functions. Specify param `out_w` to determine width of + * printed string. Strings are interpreted as UTF-8. + * + * Non-printable characters (`iswprint(3)`) and truncated UTF-8 byte sequences + * are replaced with U+FFFD. + * + * Newlines (`\n`) are supported with the caveat that `out_w` will return the + * width of the string as if it were on a single line. + * + * If the starting coordinate is out of bounds, `TB_ERR_OUT_OF_BOUNDS` is + * returned. If the starting coordinate is in bounds, but goes out of bounds, + * then the out-of-bounds portions of the string are ignored. + * + * For finer control, use `tb_set_cell`. + */ +int tb_print(int x, int y, uintattr_t fg, uintattr_t bg, const char* str); +int tb_printf(int x, int y, uintattr_t fg, uintattr_t bg, const char* fmt, ...); +int tb_print_ex(int x, int y, uintattr_t fg, uintattr_t bg, size_t* out_w, const char* str); +int tb_printf_ex(int x, int y, uintattr_t fg, uintattr_t bg, size_t* out_w, const char* fmt, ...); + +/* Send raw bytes to terminal. */ +int tb_send(const char* buf, size_t nbuf); +int tb_sendf(const char* fmt, ...); + +/* Deprecated. Set custom callbacks. `fn_type` is one of `TB_FUNC_*` constants, + * `fn` is a compatible function pointer, or NULL to clear. + * + * `TB_FUNC_EXTRACT_PRE`: + * If specified, invoke this function BEFORE termbox tries to extract any + * escape sequences from the input buffer. + * + * `TB_FUNC_EXTRACT_POST`: + * If specified, invoke this function AFTER termbox tries (and fails) to + * extract any escape sequences from the input buffer. + */ +int tb_set_func(int fn_type, int (*fn)(struct tb_event*, size_t*)); + +/* Return byte length of codepoint given first byte of UTF-8 sequence (1-6). */ +int tb_utf8_char_length(char c); + +/* Convert UTF-8 null-terminated byte sequence to UTF-32 codepoint. + * + * If `c` is an empty C string, return 0. `out` is left unchanged. + * + * If a null byte is encountered in the middle of the codepoint, return a + * negative number indicating how many bytes were processed. `out` is left + * unchanged. + * + * Otherwise, return byte length of codepoint (1-6). + */ +int tb_utf8_char_to_unicode(uint32_t* out, const char* c); + +/* Convert UTF-32 codepoint to UTF-8 null-terminated byte sequence. + * + * `out` must be char[7] or greater. Return byte length of codepoint (1-6). + */ +int tb_utf8_unicode_to_char(char* out, uint32_t c); + +/* Library utility functions */ +int tb_last_errno(void); +const char* tb_strerror(int err); +struct tb_cell* tb_cell_buffer(void); // Deprecated +int tb_has_truecolor(void); +int tb_has_egc(void); +int tb_attr_width(void); +const char* tb_version(void); + +/* Deprecation notice! + * + * The following will be removed in version 3.x (ABI version 3): + * + * TB_256_BLACK (use TB_HI_BLACK) + * TB_OPT_TRUECOLOR (use TB_OPT_ATTR_W) + * TB_TRUECOLOR_BOLD (use TB_BOLD) + * TB_TRUECOLOR_UNDERLINE (use TB_UNDERLINE) + * TB_TRUECOLOR_REVERSE (use TB_REVERSE) + * TB_TRUECOLOR_ITALIC (use TB_ITALICe) + * TB_TRUECOLOR_BLINK (use TB_BLINK) + * TB_TRUECOLOR_BLACK (use TB_HI_BLACK) + * tb_cell_buffer + * tb_set_func + * TB_FUNC_EXTRACT_PRE + * TB_FUNC_EXTRACT_POST + */ + +#ifdef __cplusplus +} +#endif + +#endif // TERMBOX_H_INCL + +#ifdef TB_IMPL + +#define if_err_return(rv, expr) \ + if (((rv) = (expr)) != TB_OK) return (rv) +#define if_err_break(rv, expr) \ + if (((rv) = (expr)) != TB_OK) break +#define if_ok_return(rv, expr) \ + if (((rv) = (expr)) == TB_OK) return (rv) +#define if_ok_or_need_more_return(rv, expr) \ + if (((rv) = (expr)) == TB_OK || (rv) == TB_ERR_NEED_MORE) return (rv) + +#define send_literal(rv, a) if_err_return((rv), bytebuf_nputs(&global.out, (a), sizeof(a) - 1)) + +#define send_num(rv, nbuf, n) \ + if_err_return((rv), bytebuf_nputs(&global.out, (nbuf), convert_num((n), (nbuf)))) + +#define snprintf_or_return(rv, str, sz, fmt, ...) \ + do { \ + (rv) = snprintf((str), (sz), (fmt), __VA_ARGS__); \ + if ((rv) < 0 || (rv) >= (int)(sz)) return TB_ERR; \ + } while (0) + +#define if_not_init_return() \ + if (!global.initialized) return TB_ERR_NOT_INIT + +struct bytebuf_t { + char* buf; + size_t len; + size_t cap; +}; + +struct cellbuf_t { + int width; + int height; + struct tb_cell* cells; +}; + +struct cap_trie_t { + char c; + struct cap_trie_t* children; + size_t nchildren; + int is_leaf; + uint16_t key; + uint8_t mod; +}; + +struct tb_global_t { + int ttyfd; + int rfd; + int wfd; + int ttyfd_open; + int resize_pipefd[2]; + int width; + int height; + int cursor_x; + int cursor_y; + int last_x; + int last_y; + uintattr_t fg; + uintattr_t bg; + uintattr_t last_fg; + uintattr_t last_bg; + int input_mode; + int output_mode; + char* terminfo; + size_t nterminfo; + const char* caps[TB_CAP__COUNT]; + struct cap_trie_t cap_trie; + struct bytebuf_t in; + struct bytebuf_t out; + struct cellbuf_t back; + struct cellbuf_t front; + struct termios orig_tios; + int has_orig_tios; + int last_errno; + int initialized; + int (*fn_extract_esc_pre)(struct tb_event*, size_t*); + int (*fn_extract_esc_post)(struct tb_event*, size_t*); + char errbuf[1024]; +}; + +static struct tb_global_t global = { 0 }; + +/* BEGIN codegen c */ +/* Produced by ./codegen.sh on Tue, 03 Sep 2024 04:17:48 +0000 */ + +static const int16_t terminfo_cap_indexes[] = { + 66, // kf1 (TB_CAP_F1) + 68, // kf2 (TB_CAP_F2) + 69, // kf3 (TB_CAP_F3) + 70, // kf4 (TB_CAP_F4) + 71, // kf5 (TB_CAP_F5) + 72, // kf6 (TB_CAP_F6) + 73, // kf7 (TB_CAP_F7) + 74, // kf8 (TB_CAP_F8) + 75, // kf9 (TB_CAP_F9) + 67, // kf10 (TB_CAP_F10) + 216, // kf11 (TB_CAP_F11) + 217, // kf12 (TB_CAP_F12) + 77, // kich1 (TB_CAP_INSERT) + 59, // kdch1 (TB_CAP_DELETE) + 76, // khome (TB_CAP_HOME) + 164, // kend (TB_CAP_END) + 82, // kpp (TB_CAP_PGUP) + 81, // knp (TB_CAP_PGDN) + 87, // kcuu1 (TB_CAP_ARROW_UP) + 61, // kcud1 (TB_CAP_ARROW_DOWN) + 79, // kcub1 (TB_CAP_ARROW_LEFT) + 83, // kcuf1 (TB_CAP_ARROW_RIGHT) + 148, // kcbt (TB_CAP_BACK_TAB) + 28, // smcup (TB_CAP_ENTER_CA) + 40, // rmcup (TB_CAP_EXIT_CA) + 16, // cnorm (TB_CAP_SHOW_CURSOR) + 13, // civis (TB_CAP_HIDE_CURSOR) + 5, // clear (TB_CAP_CLEAR_SCREEN) + 39, // sgr0 (TB_CAP_SGR0) + 36, // smul (TB_CAP_UNDERLINE) + 27, // bold (TB_CAP_BOLD) + 26, // blink (TB_CAP_BLINK) + 311, // sitm (TB_CAP_ITALIC) + 34, // rev (TB_CAP_REVERSE) + 89, // smkx (TB_CAP_ENTER_KEYPAD) + 88, // rmkx (TB_CAP_EXIT_KEYPAD) + 30, // dim (TB_CAP_DIM) + 32, // invis (TB_CAP_INVISIBLE) +}; + +// xterm +static const char* xterm_caps[] = { + "\033OP", // kf1 (TB_CAP_F1) + "\033OQ", // kf2 (TB_CAP_F2) + "\033OR", // kf3 (TB_CAP_F3) + "\033OS", // kf4 (TB_CAP_F4) + "\033[15~", // kf5 (TB_CAP_F5) + "\033[17~", // kf6 (TB_CAP_F6) + "\033[18~", // kf7 (TB_CAP_F7) + "\033[19~", // kf8 (TB_CAP_F8) + "\033[20~", // kf9 (TB_CAP_F9) + "\033[21~", // kf10 (TB_CAP_F10) + "\033[23~", // kf11 (TB_CAP_F11) + "\033[24~", // kf12 (TB_CAP_F12) + "\033[2~", // kich1 (TB_CAP_INSERT) + "\033[3~", // kdch1 (TB_CAP_DELETE) + "\033OH", // khome (TB_CAP_HOME) + "\033OF", // kend (TB_CAP_END) + "\033[5~", // kpp (TB_CAP_PGUP) + "\033[6~", // knp (TB_CAP_PGDN) + "\033OA", // kcuu1 (TB_CAP_ARROW_UP) + "\033OB", // kcud1 (TB_CAP_ARROW_DOWN) + "\033OD", // kcub1 (TB_CAP_ARROW_LEFT) + "\033OC", // kcuf1 (TB_CAP_ARROW_RIGHT) + "\033[Z", // kcbt (TB_CAP_BACK_TAB) + "\033[?1049h\033[22;0;0t", // smcup (TB_CAP_ENTER_CA) + "\033[?1049l\033[23;0;0t", // rmcup (TB_CAP_EXIT_CA) + "\033[?12l\033[?25h", // cnorm (TB_CAP_SHOW_CURSOR) + "\033[?25l", // civis (TB_CAP_HIDE_CURSOR) + "\033[H\033[2J", // clear (TB_CAP_CLEAR_SCREEN) + "\033(B\033[m", // sgr0 (TB_CAP_SGR0) + "\033[4m", // smul (TB_CAP_UNDERLINE) + "\033[1m", // bold (TB_CAP_BOLD) + "\033[5m", // blink (TB_CAP_BLINK) + "\033[3m", // sitm (TB_CAP_ITALIC) + "\033[7m", // rev (TB_CAP_REVERSE) + "\033[?1h\033=", // smkx (TB_CAP_ENTER_KEYPAD) + "\033[?1l\033>", // rmkx (TB_CAP_EXIT_KEYPAD) + "\033[2m", // dim (TB_CAP_DIM) + "\033[8m", // invis (TB_CAP_INVISIBLE) +}; + +// linux +static const char* linux_caps[] = { + "\033[[A", // kf1 (TB_CAP_F1) + "\033[[B", // kf2 (TB_CAP_F2) + "\033[[C", // kf3 (TB_CAP_F3) + "\033[[D", // kf4 (TB_CAP_F4) + "\033[[E", // kf5 (TB_CAP_F5) + "\033[17~", // kf6 (TB_CAP_F6) + "\033[18~", // kf7 (TB_CAP_F7) + "\033[19~", // kf8 (TB_CAP_F8) + "\033[20~", // kf9 (TB_CAP_F9) + "\033[21~", // kf10 (TB_CAP_F10) + "\033[23~", // kf11 (TB_CAP_F11) + "\033[24~", // kf12 (TB_CAP_F12) + "\033[2~", // kich1 (TB_CAP_INSERT) + "\033[3~", // kdch1 (TB_CAP_DELETE) + "\033[1~", // khome (TB_CAP_HOME) + "\033[4~", // kend (TB_CAP_END) + "\033[5~", // kpp (TB_CAP_PGUP) + "\033[6~", // knp (TB_CAP_PGDN) + "\033[A", // kcuu1 (TB_CAP_ARROW_UP) + "\033[B", // kcud1 (TB_CAP_ARROW_DOWN) + "\033[D", // kcub1 (TB_CAP_ARROW_LEFT) + "\033[C", // kcuf1 (TB_CAP_ARROW_RIGHT) + "\033\011", // kcbt (TB_CAP_BACK_TAB) + "", // smcup (TB_CAP_ENTER_CA) + "", // rmcup (TB_CAP_EXIT_CA) + "\033[?25h\033[?0c", // cnorm (TB_CAP_SHOW_CURSOR) + "\033[?25l\033[?1c", // civis (TB_CAP_HIDE_CURSOR) + "\033[H\033[J", // clear (TB_CAP_CLEAR_SCREEN) + "\033[m\017", // sgr0 (TB_CAP_SGR0) + "\033[4m", // smul (TB_CAP_UNDERLINE) + "\033[1m", // bold (TB_CAP_BOLD) + "\033[5m", // blink (TB_CAP_BLINK) + "", // sitm (TB_CAP_ITALIC) + "\033[7m", // rev (TB_CAP_REVERSE) + "", // smkx (TB_CAP_ENTER_KEYPAD) + "", // rmkx (TB_CAP_EXIT_KEYPAD) + "\033[2m", // dim (TB_CAP_DIM) + "", // invis (TB_CAP_INVISIBLE) +}; + +// screen +static const char* screen_caps[] = { + "\033OP", // kf1 (TB_CAP_F1) + "\033OQ", // kf2 (TB_CAP_F2) + "\033OR", // kf3 (TB_CAP_F3) + "\033OS", // kf4 (TB_CAP_F4) + "\033[15~", // kf5 (TB_CAP_F5) + "\033[17~", // kf6 (TB_CAP_F6) + "\033[18~", // kf7 (TB_CAP_F7) + "\033[19~", // kf8 (TB_CAP_F8) + "\033[20~", // kf9 (TB_CAP_F9) + "\033[21~", // kf10 (TB_CAP_F10) + "\033[23~", // kf11 (TB_CAP_F11) + "\033[24~", // kf12 (TB_CAP_F12) + "\033[2~", // kich1 (TB_CAP_INSERT) + "\033[3~", // kdch1 (TB_CAP_DELETE) + "\033[1~", // khome (TB_CAP_HOME) + "\033[4~", // kend (TB_CAP_END) + "\033[5~", // kpp (TB_CAP_PGUP) + "\033[6~", // knp (TB_CAP_PGDN) + "\033OA", // kcuu1 (TB_CAP_ARROW_UP) + "\033OB", // kcud1 (TB_CAP_ARROW_DOWN) + "\033OD", // kcub1 (TB_CAP_ARROW_LEFT) + "\033OC", // kcuf1 (TB_CAP_ARROW_RIGHT) + "\033[Z", // kcbt (TB_CAP_BACK_TAB) + "\033[?1049h", // smcup (TB_CAP_ENTER_CA) + "\033[?1049l", // rmcup (TB_CAP_EXIT_CA) + "\033[34h\033[?25h", // cnorm (TB_CAP_SHOW_CURSOR) + "\033[?25l", // civis (TB_CAP_HIDE_CURSOR) + "\033[H\033[J", // clear (TB_CAP_CLEAR_SCREEN) + "\033[m\017", // sgr0 (TB_CAP_SGR0) + "\033[4m", // smul (TB_CAP_UNDERLINE) + "\033[1m", // bold (TB_CAP_BOLD) + "\033[5m", // blink (TB_CAP_BLINK) + "", // sitm (TB_CAP_ITALIC) + "\033[7m", // rev (TB_CAP_REVERSE) + "\033[?1h\033=", // smkx (TB_CAP_ENTER_KEYPAD) + "\033[?1l\033>", // rmkx (TB_CAP_EXIT_KEYPAD) + "\033[2m", // dim (TB_CAP_DIM) + "", // invis (TB_CAP_INVISIBLE) +}; + +// rxvt-256color +static const char* rxvt_256color_caps[] = { + "\033[11~", // kf1 (TB_CAP_F1) + "\033[12~", // kf2 (TB_CAP_F2) + "\033[13~", // kf3 (TB_CAP_F3) + "\033[14~", // kf4 (TB_CAP_F4) + "\033[15~", // kf5 (TB_CAP_F5) + "\033[17~", // kf6 (TB_CAP_F6) + "\033[18~", // kf7 (TB_CAP_F7) + "\033[19~", // kf8 (TB_CAP_F8) + "\033[20~", // kf9 (TB_CAP_F9) + "\033[21~", // kf10 (TB_CAP_F10) + "\033[23~", // kf11 (TB_CAP_F11) + "\033[24~", // kf12 (TB_CAP_F12) + "\033[2~", // kich1 (TB_CAP_INSERT) + "\033[3~", // kdch1 (TB_CAP_DELETE) + "\033[7~", // khome (TB_CAP_HOME) + "\033[8~", // kend (TB_CAP_END) + "\033[5~", // kpp (TB_CAP_PGUP) + "\033[6~", // knp (TB_CAP_PGDN) + "\033[A", // kcuu1 (TB_CAP_ARROW_UP) + "\033[B", // kcud1 (TB_CAP_ARROW_DOWN) + "\033[D", // kcub1 (TB_CAP_ARROW_LEFT) + "\033[C", // kcuf1 (TB_CAP_ARROW_RIGHT) + "\033[Z", // kcbt (TB_CAP_BACK_TAB) + "\0337\033[?47h", // smcup (TB_CAP_ENTER_CA) + "\033[2J\033[?47l\0338", // rmcup (TB_CAP_EXIT_CA) + "\033[?25h", // cnorm (TB_CAP_SHOW_CURSOR) + "\033[?25l", // civis (TB_CAP_HIDE_CURSOR) + "\033[H\033[2J", // clear (TB_CAP_CLEAR_SCREEN) + "\033[m\017", // sgr0 (TB_CAP_SGR0) + "\033[4m", // smul (TB_CAP_UNDERLINE) + "\033[1m", // bold (TB_CAP_BOLD) + "\033[5m", // blink (TB_CAP_BLINK) + "", // sitm (TB_CAP_ITALIC) + "\033[7m", // rev (TB_CAP_REVERSE) + "\033=", // smkx (TB_CAP_ENTER_KEYPAD) + "\033>", // rmkx (TB_CAP_EXIT_KEYPAD) + "", // dim (TB_CAP_DIM) + "", // invis (TB_CAP_INVISIBLE) +}; + +// rxvt-unicode +static const char* rxvt_unicode_caps[] = { + "\033[11~", // kf1 (TB_CAP_F1) + "\033[12~", // kf2 (TB_CAP_F2) + "\033[13~", // kf3 (TB_CAP_F3) + "\033[14~", // kf4 (TB_CAP_F4) + "\033[15~", // kf5 (TB_CAP_F5) + "\033[17~", // kf6 (TB_CAP_F6) + "\033[18~", // kf7 (TB_CAP_F7) + "\033[19~", // kf8 (TB_CAP_F8) + "\033[20~", // kf9 (TB_CAP_F9) + "\033[21~", // kf10 (TB_CAP_F10) + "\033[23~", // kf11 (TB_CAP_F11) + "\033[24~", // kf12 (TB_CAP_F12) + "\033[2~", // kich1 (TB_CAP_INSERT) + "\033[3~", // kdch1 (TB_CAP_DELETE) + "\033[7~", // khome (TB_CAP_HOME) + "\033[8~", // kend (TB_CAP_END) + "\033[5~", // kpp (TB_CAP_PGUP) + "\033[6~", // knp (TB_CAP_PGDN) + "\033[A", // kcuu1 (TB_CAP_ARROW_UP) + "\033[B", // kcud1 (TB_CAP_ARROW_DOWN) + "\033[D", // kcub1 (TB_CAP_ARROW_LEFT) + "\033[C", // kcuf1 (TB_CAP_ARROW_RIGHT) + "\033[Z", // kcbt (TB_CAP_BACK_TAB) + "\033[?1049h", // smcup (TB_CAP_ENTER_CA) + "\033[r\033[?1049l", // rmcup (TB_CAP_EXIT_CA) + "\033[?12l\033[?25h", // cnorm (TB_CAP_SHOW_CURSOR) + "\033[?25l", // civis (TB_CAP_HIDE_CURSOR) + "\033[H\033[2J", // clear (TB_CAP_CLEAR_SCREEN) + "\033[m\033(B", // sgr0 (TB_CAP_SGR0) + "\033[4m", // smul (TB_CAP_UNDERLINE) + "\033[1m", // bold (TB_CAP_BOLD) + "\033[5m", // blink (TB_CAP_BLINK) + "\033[3m", // sitm (TB_CAP_ITALIC) + "\033[7m", // rev (TB_CAP_REVERSE) + "\033=", // smkx (TB_CAP_ENTER_KEYPAD) + "\033>", // rmkx (TB_CAP_EXIT_KEYPAD) + "", // dim (TB_CAP_DIM) + "", // invis (TB_CAP_INVISIBLE) +}; + +// Eterm +static const char* eterm_caps[] = { + "\033[11~", // kf1 (TB_CAP_F1) + "\033[12~", // kf2 (TB_CAP_F2) + "\033[13~", // kf3 (TB_CAP_F3) + "\033[14~", // kf4 (TB_CAP_F4) + "\033[15~", // kf5 (TB_CAP_F5) + "\033[17~", // kf6 (TB_CAP_F6) + "\033[18~", // kf7 (TB_CAP_F7) + "\033[19~", // kf8 (TB_CAP_F8) + "\033[20~", // kf9 (TB_CAP_F9) + "\033[21~", // kf10 (TB_CAP_F10) + "\033[23~", // kf11 (TB_CAP_F11) + "\033[24~", // kf12 (TB_CAP_F12) + "\033[2~", // kich1 (TB_CAP_INSERT) + "\033[3~", // kdch1 (TB_CAP_DELETE) + "\033[7~", // khome (TB_CAP_HOME) + "\033[8~", // kend (TB_CAP_END) + "\033[5~", // kpp (TB_CAP_PGUP) + "\033[6~", // knp (TB_CAP_PGDN) + "\033[A", // kcuu1 (TB_CAP_ARROW_UP) + "\033[B", // kcud1 (TB_CAP_ARROW_DOWN) + "\033[D", // kcub1 (TB_CAP_ARROW_LEFT) + "\033[C", // kcuf1 (TB_CAP_ARROW_RIGHT) + "", // kcbt (TB_CAP_BACK_TAB) + "\0337\033[?47h", // smcup (TB_CAP_ENTER_CA) + "\033[2J\033[?47l\0338", // rmcup (TB_CAP_EXIT_CA) + "\033[?25h", // cnorm (TB_CAP_SHOW_CURSOR) + "\033[?25l", // civis (TB_CAP_HIDE_CURSOR) + "\033[H\033[2J", // clear (TB_CAP_CLEAR_SCREEN) + "\033[m\017", // sgr0 (TB_CAP_SGR0) + "\033[4m", // smul (TB_CAP_UNDERLINE) + "\033[1m", // bold (TB_CAP_BOLD) + "\033[5m", // blink (TB_CAP_BLINK) + "", // sitm (TB_CAP_ITALIC) + "\033[7m", // rev (TB_CAP_REVERSE) + "", // smkx (TB_CAP_ENTER_KEYPAD) + "", // rmkx (TB_CAP_EXIT_KEYPAD) + "", // dim (TB_CAP_DIM) + "", // invis (TB_CAP_INVISIBLE) +}; + +static struct { + const char* name; + const char** caps; + const char* alias; +} builtin_terms[] = { + { "xterm", xterm_caps, "" }, + { "linux", linux_caps, "" }, + { "screen", screen_caps, "tmux" }, + { "rxvt-256color", rxvt_256color_caps, "" }, + { "rxvt-unicode", rxvt_unicode_caps, "rxvt" }, + { "Eterm", eterm_caps, "" }, + { NULL, NULL, NULL }, +}; + +/* END codegen c */ + +static struct { + const char* cap; + const uint16_t key; + const uint8_t mod; +} builtin_mod_caps[] = { + // xterm arrows + { "\x1b[1;2A", TB_KEY_ARROW_UP, TB_MOD_SHIFT }, + { "\x1b[1;3A", TB_KEY_ARROW_UP, TB_MOD_ALT }, + { "\x1b[1;4A", TB_KEY_ARROW_UP, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5A", TB_KEY_ARROW_UP, TB_MOD_CTRL }, + { "\x1b[1;6A", TB_KEY_ARROW_UP, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7A", TB_KEY_ARROW_UP, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8A", TB_KEY_ARROW_UP, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2B", TB_KEY_ARROW_DOWN, TB_MOD_SHIFT }, + { "\x1b[1;3B", TB_KEY_ARROW_DOWN, TB_MOD_ALT }, + { "\x1b[1;4B", TB_KEY_ARROW_DOWN, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5B", TB_KEY_ARROW_DOWN, TB_MOD_CTRL }, + { "\x1b[1;6B", TB_KEY_ARROW_DOWN, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7B", TB_KEY_ARROW_DOWN, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8B", TB_KEY_ARROW_DOWN, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2C", TB_KEY_ARROW_RIGHT, TB_MOD_SHIFT }, + { "\x1b[1;3C", TB_KEY_ARROW_RIGHT, TB_MOD_ALT }, + { "\x1b[1;4C", TB_KEY_ARROW_RIGHT, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5C", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL }, + { "\x1b[1;6C", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7C", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8C", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2D", TB_KEY_ARROW_LEFT, TB_MOD_SHIFT }, + { "\x1b[1;3D", TB_KEY_ARROW_LEFT, TB_MOD_ALT }, + { "\x1b[1;4D", TB_KEY_ARROW_LEFT, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5D", TB_KEY_ARROW_LEFT, TB_MOD_CTRL }, + { "\x1b[1;6D", TB_KEY_ARROW_LEFT, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7D", TB_KEY_ARROW_LEFT, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8D", TB_KEY_ARROW_LEFT, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + // xterm keys + { "\x1b[1;2H", TB_KEY_HOME, TB_MOD_SHIFT }, + { "\x1b[1;3H", TB_KEY_HOME, TB_MOD_ALT }, + { "\x1b[1;4H", TB_KEY_HOME, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5H", TB_KEY_HOME, TB_MOD_CTRL }, + { "\x1b[1;6H", TB_KEY_HOME, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7H", TB_KEY_HOME, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8H", TB_KEY_HOME, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2F", TB_KEY_END, TB_MOD_SHIFT }, + { "\x1b[1;3F", TB_KEY_END, TB_MOD_ALT }, + { "\x1b[1;4F", TB_KEY_END, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5F", TB_KEY_END, TB_MOD_CTRL }, + { "\x1b[1;6F", TB_KEY_END, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7F", TB_KEY_END, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8F", TB_KEY_END, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[2;2~", TB_KEY_INSERT, TB_MOD_SHIFT }, + { "\x1b[2;3~", TB_KEY_INSERT, TB_MOD_ALT }, + { "\x1b[2;4~", TB_KEY_INSERT, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[2;5~", TB_KEY_INSERT, TB_MOD_CTRL }, + { "\x1b[2;6~", TB_KEY_INSERT, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[2;7~", TB_KEY_INSERT, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[2;8~", TB_KEY_INSERT, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[3;2~", TB_KEY_DELETE, TB_MOD_SHIFT }, + { "\x1b[3;3~", TB_KEY_DELETE, TB_MOD_ALT }, + { "\x1b[3;4~", TB_KEY_DELETE, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[3;5~", TB_KEY_DELETE, TB_MOD_CTRL }, + { "\x1b[3;6~", TB_KEY_DELETE, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[3;7~", TB_KEY_DELETE, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[3;8~", TB_KEY_DELETE, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[5;2~", TB_KEY_PGUP, TB_MOD_SHIFT }, + { "\x1b[5;3~", TB_KEY_PGUP, TB_MOD_ALT }, + { "\x1b[5;4~", TB_KEY_PGUP, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[5;5~", TB_KEY_PGUP, TB_MOD_CTRL }, + { "\x1b[5;6~", TB_KEY_PGUP, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[5;7~", TB_KEY_PGUP, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[5;8~", TB_KEY_PGUP, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[6;2~", TB_KEY_PGDN, TB_MOD_SHIFT }, + { "\x1b[6;3~", TB_KEY_PGDN, TB_MOD_ALT }, + { "\x1b[6;4~", TB_KEY_PGDN, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[6;5~", TB_KEY_PGDN, TB_MOD_CTRL }, + { "\x1b[6;6~", TB_KEY_PGDN, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[6;7~", TB_KEY_PGDN, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[6;8~", TB_KEY_PGDN, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2P", TB_KEY_F1, TB_MOD_SHIFT }, + { "\x1b[1;3P", TB_KEY_F1, TB_MOD_ALT }, + { "\x1b[1;4P", TB_KEY_F1, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5P", TB_KEY_F1, TB_MOD_CTRL }, + { "\x1b[1;6P", TB_KEY_F1, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7P", TB_KEY_F1, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8P", TB_KEY_F1, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2Q", TB_KEY_F2, TB_MOD_SHIFT }, + { "\x1b[1;3Q", TB_KEY_F2, TB_MOD_ALT }, + { "\x1b[1;4Q", TB_KEY_F2, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5Q", TB_KEY_F2, TB_MOD_CTRL }, + { "\x1b[1;6Q", TB_KEY_F2, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7Q", TB_KEY_F2, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8Q", TB_KEY_F2, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2R", TB_KEY_F3, TB_MOD_SHIFT }, + { "\x1b[1;3R", TB_KEY_F3, TB_MOD_ALT }, + { "\x1b[1;4R", TB_KEY_F3, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5R", TB_KEY_F3, TB_MOD_CTRL }, + { "\x1b[1;6R", TB_KEY_F3, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7R", TB_KEY_F3, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8R", TB_KEY_F3, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[1;2S", TB_KEY_F4, TB_MOD_SHIFT }, + { "\x1b[1;3S", TB_KEY_F4, TB_MOD_ALT }, + { "\x1b[1;4S", TB_KEY_F4, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[1;5S", TB_KEY_F4, TB_MOD_CTRL }, + { "\x1b[1;6S", TB_KEY_F4, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[1;7S", TB_KEY_F4, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[1;8S", TB_KEY_F4, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[15;2~", TB_KEY_F5, TB_MOD_SHIFT }, + { "\x1b[15;3~", TB_KEY_F5, TB_MOD_ALT }, + { "\x1b[15;4~", TB_KEY_F5, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[15;5~", TB_KEY_F5, TB_MOD_CTRL }, + { "\x1b[15;6~", TB_KEY_F5, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[15;7~", TB_KEY_F5, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[15;8~", TB_KEY_F5, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[17;2~", TB_KEY_F6, TB_MOD_SHIFT }, + { "\x1b[17;3~", TB_KEY_F6, TB_MOD_ALT }, + { "\x1b[17;4~", TB_KEY_F6, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[17;5~", TB_KEY_F6, TB_MOD_CTRL }, + { "\x1b[17;6~", TB_KEY_F6, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[17;7~", TB_KEY_F6, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[17;8~", TB_KEY_F6, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[18;2~", TB_KEY_F7, TB_MOD_SHIFT }, + { "\x1b[18;3~", TB_KEY_F7, TB_MOD_ALT }, + { "\x1b[18;4~", TB_KEY_F7, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[18;5~", TB_KEY_F7, TB_MOD_CTRL }, + { "\x1b[18;6~", TB_KEY_F7, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[18;7~", TB_KEY_F7, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[18;8~", TB_KEY_F7, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[19;2~", TB_KEY_F8, TB_MOD_SHIFT }, + { "\x1b[19;3~", TB_KEY_F8, TB_MOD_ALT }, + { "\x1b[19;4~", TB_KEY_F8, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[19;5~", TB_KEY_F8, TB_MOD_CTRL }, + { "\x1b[19;6~", TB_KEY_F8, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[19;7~", TB_KEY_F8, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[19;8~", TB_KEY_F8, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[20;2~", TB_KEY_F9, TB_MOD_SHIFT }, + { "\x1b[20;3~", TB_KEY_F9, TB_MOD_ALT }, + { "\x1b[20;4~", TB_KEY_F9, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[20;5~", TB_KEY_F9, TB_MOD_CTRL }, + { "\x1b[20;6~", TB_KEY_F9, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[20;7~", TB_KEY_F9, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[20;8~", TB_KEY_F9, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[21;2~", TB_KEY_F10, TB_MOD_SHIFT }, + { "\x1b[21;3~", TB_KEY_F10, TB_MOD_ALT }, + { "\x1b[21;4~", TB_KEY_F10, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[21;5~", TB_KEY_F10, TB_MOD_CTRL }, + { "\x1b[21;6~", TB_KEY_F10, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[21;7~", TB_KEY_F10, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[21;8~", TB_KEY_F10, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[23;2~", TB_KEY_F11, TB_MOD_SHIFT }, + { "\x1b[23;3~", TB_KEY_F11, TB_MOD_ALT }, + { "\x1b[23;4~", TB_KEY_F11, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[23;5~", TB_KEY_F11, TB_MOD_CTRL }, + { "\x1b[23;6~", TB_KEY_F11, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[23;7~", TB_KEY_F11, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[23;8~", TB_KEY_F11, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b[24;2~", TB_KEY_F12, TB_MOD_SHIFT }, + { "\x1b[24;3~", TB_KEY_F12, TB_MOD_ALT }, + { "\x1b[24;4~", TB_KEY_F12, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[24;5~", TB_KEY_F12, TB_MOD_CTRL }, + { "\x1b[24;6~", TB_KEY_F12, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[24;7~", TB_KEY_F12, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b[24;8~", TB_KEY_F12, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + // rxvt arrows + { "\x1b[a", TB_KEY_ARROW_UP, TB_MOD_SHIFT }, + { "\x1b\x1b[A", TB_KEY_ARROW_UP, TB_MOD_ALT }, + { "\x1b\x1b[a", TB_KEY_ARROW_UP, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1bOa", TB_KEY_ARROW_UP, TB_MOD_CTRL }, + { "\x1b\x1bOa", TB_KEY_ARROW_UP, TB_MOD_CTRL | TB_MOD_ALT }, + + { "\x1b[b", TB_KEY_ARROW_DOWN, TB_MOD_SHIFT }, + { "\x1b\x1b[B", TB_KEY_ARROW_DOWN, TB_MOD_ALT }, + { "\x1b\x1b[b", TB_KEY_ARROW_DOWN, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1bOb", TB_KEY_ARROW_DOWN, TB_MOD_CTRL }, + { "\x1b\x1bOb", TB_KEY_ARROW_DOWN, TB_MOD_CTRL | TB_MOD_ALT }, + + { "\x1b[c", TB_KEY_ARROW_RIGHT, TB_MOD_SHIFT }, + { "\x1b\x1b[C", TB_KEY_ARROW_RIGHT, TB_MOD_ALT }, + { "\x1b\x1b[c", TB_KEY_ARROW_RIGHT, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1bOc", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL }, + { "\x1b\x1bOc", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL | TB_MOD_ALT }, + + { "\x1b[d", TB_KEY_ARROW_LEFT, TB_MOD_SHIFT }, + { "\x1b\x1b[D", TB_KEY_ARROW_LEFT, TB_MOD_ALT }, + { "\x1b\x1b[d", TB_KEY_ARROW_LEFT, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1bOd", TB_KEY_ARROW_LEFT, TB_MOD_CTRL }, + { "\x1b\x1bOd", TB_KEY_ARROW_LEFT, TB_MOD_CTRL | TB_MOD_ALT }, + + // rxvt keys + { "\x1b[7$", TB_KEY_HOME, TB_MOD_SHIFT }, + { "\x1b\x1b[7~", TB_KEY_HOME, TB_MOD_ALT }, + { "\x1b\x1b[7$", TB_KEY_HOME, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[7^", TB_KEY_HOME, TB_MOD_CTRL }, + { "\x1b[7@", TB_KEY_HOME, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b\x1b[7^", TB_KEY_HOME, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[7@", TB_KEY_HOME, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + + { "\x1b\x1b[8~", TB_KEY_END, TB_MOD_ALT }, + { "\x1b\x1b[8$", TB_KEY_END, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[8^", TB_KEY_END, TB_MOD_CTRL }, + { "\x1b\x1b[8^", TB_KEY_END, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[8@", TB_KEY_END, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[8@", TB_KEY_END, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[8$", TB_KEY_END, TB_MOD_SHIFT }, + + { "\x1b\x1b[2~", TB_KEY_INSERT, TB_MOD_ALT }, + { "\x1b\x1b[2$", TB_KEY_INSERT, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[2^", TB_KEY_INSERT, TB_MOD_CTRL }, + { "\x1b\x1b[2^", TB_KEY_INSERT, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[2@", TB_KEY_INSERT, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[2@", TB_KEY_INSERT, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[2$", TB_KEY_INSERT, TB_MOD_SHIFT }, + + { "\x1b\x1b[3~", TB_KEY_DELETE, TB_MOD_ALT }, + { "\x1b\x1b[3$", TB_KEY_DELETE, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[3^", TB_KEY_DELETE, TB_MOD_CTRL }, + { "\x1b\x1b[3^", TB_KEY_DELETE, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[3@", TB_KEY_DELETE, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[3@", TB_KEY_DELETE, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[3$", TB_KEY_DELETE, TB_MOD_SHIFT }, + + { "\x1b\x1b[5~", TB_KEY_PGUP, TB_MOD_ALT }, + { "\x1b\x1b[5$", TB_KEY_PGUP, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[5^", TB_KEY_PGUP, TB_MOD_CTRL }, + { "\x1b\x1b[5^", TB_KEY_PGUP, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[5@", TB_KEY_PGUP, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[5@", TB_KEY_PGUP, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[5$", TB_KEY_PGUP, TB_MOD_SHIFT }, + + { "\x1b\x1b[6~", TB_KEY_PGDN, TB_MOD_ALT }, + { "\x1b\x1b[6$", TB_KEY_PGDN, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[6^", TB_KEY_PGDN, TB_MOD_CTRL }, + { "\x1b\x1b[6^", TB_KEY_PGDN, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[6@", TB_KEY_PGDN, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[6@", TB_KEY_PGDN, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[6$", TB_KEY_PGDN, TB_MOD_SHIFT }, + + { "\x1b\x1b[11~", TB_KEY_F1, TB_MOD_ALT }, + { "\x1b\x1b[23~", TB_KEY_F1, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[11^", TB_KEY_F1, TB_MOD_CTRL }, + { "\x1b\x1b[11^", TB_KEY_F1, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[23^", TB_KEY_F1, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[23^", TB_KEY_F1, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[23~", TB_KEY_F1, TB_MOD_SHIFT }, + + { "\x1b\x1b[12~", TB_KEY_F2, TB_MOD_ALT }, + { "\x1b\x1b[24~", TB_KEY_F2, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[12^", TB_KEY_F2, TB_MOD_CTRL }, + { "\x1b\x1b[12^", TB_KEY_F2, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[24^", TB_KEY_F2, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[24^", TB_KEY_F2, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[24~", TB_KEY_F2, TB_MOD_SHIFT }, + + { "\x1b\x1b[13~", TB_KEY_F3, TB_MOD_ALT }, + { "\x1b\x1b[25~", TB_KEY_F3, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[13^", TB_KEY_F3, TB_MOD_CTRL }, + { "\x1b\x1b[13^", TB_KEY_F3, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[25^", TB_KEY_F3, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[25^", TB_KEY_F3, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[25~", TB_KEY_F3, TB_MOD_SHIFT }, + + { "\x1b\x1b[14~", TB_KEY_F4, TB_MOD_ALT }, + { "\x1b\x1b[26~", TB_KEY_F4, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[14^", TB_KEY_F4, TB_MOD_CTRL }, + { "\x1b\x1b[14^", TB_KEY_F4, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[26^", TB_KEY_F4, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[26^", TB_KEY_F4, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[26~", TB_KEY_F4, TB_MOD_SHIFT }, + + { "\x1b\x1b[15~", TB_KEY_F5, TB_MOD_ALT }, + { "\x1b\x1b[28~", TB_KEY_F5, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[15^", TB_KEY_F5, TB_MOD_CTRL }, + { "\x1b\x1b[15^", TB_KEY_F5, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[28^", TB_KEY_F5, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[28^", TB_KEY_F5, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[28~", TB_KEY_F5, TB_MOD_SHIFT }, + + { "\x1b\x1b[17~", TB_KEY_F6, TB_MOD_ALT }, + { "\x1b\x1b[29~", TB_KEY_F6, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[17^", TB_KEY_F6, TB_MOD_CTRL }, + { "\x1b\x1b[17^", TB_KEY_F6, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[29^", TB_KEY_F6, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[29^", TB_KEY_F6, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[29~", TB_KEY_F6, TB_MOD_SHIFT }, + + { "\x1b\x1b[18~", TB_KEY_F7, TB_MOD_ALT }, + { "\x1b\x1b[31~", TB_KEY_F7, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[18^", TB_KEY_F7, TB_MOD_CTRL }, + { "\x1b\x1b[18^", TB_KEY_F7, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[31^", TB_KEY_F7, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[31^", TB_KEY_F7, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[31~", TB_KEY_F7, TB_MOD_SHIFT }, + + { "\x1b\x1b[19~", TB_KEY_F8, TB_MOD_ALT }, + { "\x1b\x1b[32~", TB_KEY_F8, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[19^", TB_KEY_F8, TB_MOD_CTRL }, + { "\x1b\x1b[19^", TB_KEY_F8, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[32^", TB_KEY_F8, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[32^", TB_KEY_F8, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[32~", TB_KEY_F8, TB_MOD_SHIFT }, + + { "\x1b\x1b[20~", TB_KEY_F9, TB_MOD_ALT }, + { "\x1b\x1b[33~", TB_KEY_F9, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[20^", TB_KEY_F9, TB_MOD_CTRL }, + { "\x1b\x1b[20^", TB_KEY_F9, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[33^", TB_KEY_F9, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[33^", TB_KEY_F9, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[33~", TB_KEY_F9, TB_MOD_SHIFT }, + + { "\x1b\x1b[21~", TB_KEY_F10, TB_MOD_ALT }, + { "\x1b\x1b[34~", TB_KEY_F10, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[21^", TB_KEY_F10, TB_MOD_CTRL }, + { "\x1b\x1b[21^", TB_KEY_F10, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[34^", TB_KEY_F10, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[34^", TB_KEY_F10, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[34~", TB_KEY_F10, TB_MOD_SHIFT }, + + { "\x1b\x1b[23~", TB_KEY_F11, TB_MOD_ALT }, + { "\x1b\x1b[23$", TB_KEY_F11, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[23^", TB_KEY_F11, TB_MOD_CTRL }, + { "\x1b\x1b[23^", TB_KEY_F11, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[23@", TB_KEY_F11, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[23@", TB_KEY_F11, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[23$", TB_KEY_F11, TB_MOD_SHIFT }, + + { "\x1b\x1b[24~", TB_KEY_F12, TB_MOD_ALT }, + { "\x1b\x1b[24$", TB_KEY_F12, TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[24^", TB_KEY_F12, TB_MOD_CTRL }, + { "\x1b\x1b[24^", TB_KEY_F12, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1b\x1b[24@", TB_KEY_F12, TB_MOD_CTRL | TB_MOD_ALT | TB_MOD_SHIFT }, + { "\x1b[24@", TB_KEY_F12, TB_MOD_CTRL | TB_MOD_SHIFT }, + { "\x1b[24$", TB_KEY_F12, TB_MOD_SHIFT }, + + // linux console/putty arrows + { "\x1b[A", TB_KEY_ARROW_UP, TB_MOD_SHIFT }, + { "\x1b[B", TB_KEY_ARROW_DOWN, TB_MOD_SHIFT }, + { "\x1b[C", TB_KEY_ARROW_RIGHT, TB_MOD_SHIFT }, + { "\x1b[D", TB_KEY_ARROW_LEFT, TB_MOD_SHIFT }, + + // more putty arrows + { "\x1bOA", TB_KEY_ARROW_UP, TB_MOD_CTRL }, + { "\x1b\x1bOA", TB_KEY_ARROW_UP, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1bOB", TB_KEY_ARROW_DOWN, TB_MOD_CTRL }, + { "\x1b\x1bOB", TB_KEY_ARROW_DOWN, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1bOC", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL }, + { "\x1b\x1bOC", TB_KEY_ARROW_RIGHT, TB_MOD_CTRL | TB_MOD_ALT }, + { "\x1bOD", TB_KEY_ARROW_LEFT, TB_MOD_CTRL }, + { "\x1b\x1bOD", TB_KEY_ARROW_LEFT, TB_MOD_CTRL | TB_MOD_ALT }, + + { NULL, 0, 0 }, +}; + +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 }; + +static int tb_reset(void); +static int tb_printf_inner( + int x, int y, uintattr_t fg, uintattr_t bg, size_t* out_w, const char* fmt, va_list vl); +static int init_term_attrs(void); +static int init_term_caps(void); +static int init_cap_trie(void); +static int cap_trie_add(const char* cap, uint16_t key, uint8_t mod); +static int cap_trie_find(const char* buf, size_t nbuf, struct cap_trie_t** last, size_t* depth); +static int cap_trie_deinit(struct cap_trie_t* node); +static int init_resize_handler(void); +static int send_init_escape_codes(void); +static int send_clear(void); +static int update_term_size(void); +static int update_term_size_via_esc(void); +static int init_cellbuf(void); +static int tb_deinit(void); +static int load_terminfo(void); +static int load_terminfo_from_path(const char* path, const char* term); +static int read_terminfo_path(const char* path); +static int parse_terminfo_caps(void); +static int load_builtin_caps(void); +static const char* get_terminfo_string(int16_t str_offsets_pos, + int16_t str_offsets_len, + int16_t str_table_pos, + int16_t str_table_len, + int16_t str_index); +static int wait_event(struct tb_event* event, int timeout); +static int extract_event(struct tb_event* event); +static int extract_esc(struct tb_event* event); +static int extract_esc_user(struct tb_event* event, int is_post); +static int extract_esc_cap(struct tb_event* event); +static int extract_esc_mouse(struct tb_event* event); +static int resize_cellbufs(void); +static void handle_resize(int sig); +static int send_attr(uintattr_t fg, uintattr_t bg); +static int send_sgr(uint32_t fg, uint32_t bg, int fg_is_default, int bg_is_default); +static int send_cursor_if(int x, int y); +static int send_char(int x, int y, uint32_t ch); +static int send_cluster(int x, int y, uint32_t* ch, size_t nch); +static int convert_num(uint32_t num, char* buf); +static int cell_cmp(struct tb_cell* a, struct tb_cell* b); +static int cell_copy(struct tb_cell* dst, struct tb_cell* src); +static int cell_set(struct tb_cell* cell, uint32_t* ch, size_t nch, uintattr_t fg, uintattr_t bg); +static int cell_reserve_ech(struct tb_cell* cell, size_t n); +static int cell_free(struct tb_cell* cell); +static int cellbuf_init(struct cellbuf_t* c, int w, int h); +static int cellbuf_free(struct cellbuf_t* c); +static int cellbuf_clear(struct cellbuf_t* c); +static int cellbuf_get(struct cellbuf_t* c, int x, int y, struct tb_cell** out); +static int cellbuf_in_bounds(struct cellbuf_t* c, int x, int y); +static int cellbuf_resize(struct cellbuf_t* c, int w, int h); +static int bytebuf_puts(struct bytebuf_t* b, const char* str); +static int bytebuf_nputs(struct bytebuf_t* b, const char* str, size_t nstr); +static int bytebuf_shift(struct bytebuf_t* b, size_t n); +static int bytebuf_flush(struct bytebuf_t* b, int fd); +static int bytebuf_reserve(struct bytebuf_t* b, size_t sz); +static int bytebuf_free(struct bytebuf_t* b); + +int +tb_init(void) +{ + return tb_init_file("/dev/tty"); +} + +int +tb_init_file(const char* path) +{ + if (global.initialized) return TB_ERR_INIT_ALREADY; + int ttyfd = open(path, O_RDWR); + if (ttyfd < 0) { + global.last_errno = errno; + return TB_ERR_INIT_OPEN; + } + global.ttyfd_open = 1; + return tb_init_fd(ttyfd); +} + +int +tb_init_fd(int ttyfd) +{ + return tb_init_rwfd(ttyfd, ttyfd); +} + +int +tb_init_rwfd(int rfd, int wfd) +{ + int rv; + + tb_reset(); + global.ttyfd = rfd == wfd && isatty(rfd) ? rfd : -1; + global.rfd = rfd; + global.wfd = wfd; + + do { + if_err_break(rv, init_term_attrs()); + if_err_break(rv, init_term_caps()); + if_err_break(rv, init_cap_trie()); + if_err_break(rv, init_resize_handler()); + if_err_break(rv, send_init_escape_codes()); + if_err_break(rv, send_clear()); + if_err_break(rv, update_term_size()); + if_err_break(rv, init_cellbuf()); + global.initialized = 1; + } while (0); + + if (rv != TB_OK) { tb_deinit(); } + + return rv; +} + +int +tb_shutdown(void) +{ + if_not_init_return(); + tb_deinit(); + return TB_OK; +} + +int +tb_width(void) +{ + if_not_init_return(); + return global.width; +} + +int +tb_height(void) +{ + if_not_init_return(); + return global.height; +} + +int +tb_clear(void) +{ + if_not_init_return(); + return cellbuf_clear(&global.back); +} + +int +tb_set_clear_attrs(uintattr_t fg, uintattr_t bg) +{ + if_not_init_return(); + global.fg = fg; + global.bg = bg; + return TB_OK; +} + +int +tb_present(void) +{ + if_not_init_return(); + + int rv; + + // TODO: Assert global.back.(width,height) == global.front.(width,height) + + global.last_x = -1; + global.last_y = -1; + + int x, y, i; + for (y = 0; y < global.front.height; y++) { + for (x = 0; x < global.front.width;) { + struct tb_cell *back, *front; + if_err_return(rv, cellbuf_get(&global.back, x, y, &back)); + if_err_return(rv, cellbuf_get(&global.front, x, y, &front)); + + int w; + { +#ifdef TB_OPT_EGC + if (back->nech > 0) + w = wcswidth((wchar_t*)back->ech, back->nech); + else +#endif + // wcwidth simply returns -1 on overflow of wchar_t + w = wcwidth((wchar_t)back->ch); + } + if (w < 1) w = 1; + + if (cell_cmp(back, front) != 0) { + cell_copy(front, back); + + send_attr(back->fg, back->bg); + if (w > 1 && x >= global.front.width - (w - 1)) { + // Not enough room for wide char, send spaces + for (i = x; i < global.front.width; i++) { + send_char(i, y, ' '); + } + } else { + { +#ifdef TB_OPT_EGC + if (back->nech > 0) + send_cluster(x, y, back->ech, back->nech); + else +#endif + send_char(x, y, back->ch); + } + + // When wcwidth>1, we need to advance the cursor by more + // than 1, thereby skipping some cells. Set these skipped + // cells to an invalid codepoint in the front buffer, so + // that if this cell is later replaced by a wcwidth==1 char, + // we'll get a cell_cmp diff for the skipped cells and + // properly re-render. + for (i = 1; i < w; i++) { + struct tb_cell* front_wide; + uint32_t invalid = -1; + if_err_return(rv, + cellbuf_get(&global.front, x + i, y, &front_wide)); + if_err_return(rv, cell_set(front_wide, &invalid, 1, -1, -1)); + } + } + } + x += w; + } + } + + if_err_return(rv, send_cursor_if(global.cursor_x, global.cursor_y)); + if_err_return(rv, bytebuf_flush(&global.out, global.wfd)); + + return TB_OK; +} + +int +tb_invalidate(void) +{ + int rv; + if_not_init_return(); + if_err_return(rv, resize_cellbufs()); + return TB_OK; +} + +int +tb_set_cursor(int cx, int cy) +{ + if_not_init_return(); + int rv; + if (cx < 0) cx = 0; + if (cy < 0) cy = 0; + if (global.cursor_x == -1) { + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_SHOW_CURSOR])); + } + if_err_return(rv, send_cursor_if(cx, cy)); + global.cursor_x = cx; + global.cursor_y = cy; + return TB_OK; +} + +int +tb_hide_cursor(void) +{ + if_not_init_return(); + int rv; + if (global.cursor_x >= 0) { + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_HIDE_CURSOR])); + } + global.cursor_x = -1; + global.cursor_y = -1; + return TB_OK; +} + +int +tb_set_cell(int x, int y, uint32_t ch, uintattr_t fg, uintattr_t bg) +{ + return tb_set_cell_ex(x, y, &ch, 1, fg, bg); +} + +int +tb_set_cell_ex(int x, int y, uint32_t* ch, size_t nch, uintattr_t fg, uintattr_t bg) +{ + if_not_init_return(); + int rv; + struct tb_cell* cell; + if_err_return(rv, cellbuf_get(&global.back, x, y, &cell)); + if_err_return(rv, cell_set(cell, ch, nch, fg, bg)); + return TB_OK; +} + +int +tb_extend_cell(int x, int y, uint32_t ch) +{ + if_not_init_return(); +#ifdef TB_OPT_EGC + int rv; + struct tb_cell* cell; + size_t nech; + if_err_return(rv, cellbuf_get(&global.back, x, y, &cell)); + if (cell->nech > 0) { // append to ech + nech = cell->nech + 1; + if_err_return(rv, cell_reserve_ech(cell, nech + 1)); + cell->ech[nech - 1] = ch; + } else { // make new ech + nech = 2; + if_err_return(rv, cell_reserve_ech(cell, nech + 1)); + cell->ech[0] = cell->ch; + cell->ech[1] = ch; + } + cell->ech[nech] = '\0'; + cell->nech = nech; + return TB_OK; +#else + (void)x; + (void)y; + (void)ch; + return TB_ERR; +#endif +} + +int +tb_set_input_mode(int mode) +{ + if_not_init_return(); + if (mode == TB_INPUT_CURRENT) { return global.input_mode; } + + if ((mode & (TB_INPUT_ESC | TB_INPUT_ALT)) == 0) { mode |= TB_INPUT_ESC; } + + if ((mode & (TB_INPUT_ESC | TB_INPUT_ALT)) == (TB_INPUT_ESC | TB_INPUT_ALT)) { + mode &= ~TB_INPUT_ALT; + } + + if (mode & TB_INPUT_MOUSE) { + bytebuf_puts(&global.out, TB_HARDCAP_ENTER_MOUSE); + bytebuf_flush(&global.out, global.wfd); + } else { + bytebuf_puts(&global.out, TB_HARDCAP_EXIT_MOUSE); + bytebuf_flush(&global.out, global.wfd); + } + + global.input_mode = mode; + return TB_OK; +} + +int +tb_set_output_mode(int mode) +{ + if_not_init_return(); + switch (mode) { + case TB_OUTPUT_CURRENT: + return global.output_mode; + case TB_OUTPUT_NORMAL: + case TB_OUTPUT_256: + case TB_OUTPUT_216: + case TB_OUTPUT_GRAYSCALE: +#if TB_OPT_ATTR_W >= 32 + case TB_OUTPUT_TRUECOLOR: +#endif + global.last_fg = ~global.fg; + global.last_bg = ~global.bg; + global.output_mode = mode; + return TB_OK; + } + return TB_ERR; +} + +int +tb_peek_event(struct tb_event* event, int timeout_ms) +{ + if_not_init_return(); + return wait_event(event, timeout_ms); +} + +int +tb_poll_event(struct tb_event* event) +{ + if_not_init_return(); + return wait_event(event, -1); +} + +int +tb_get_fds(int* ttyfd, int* resizefd) +{ + if_not_init_return(); + + *ttyfd = global.rfd; + *resizefd = global.resize_pipefd[0]; + + return TB_OK; +} + +int +tb_print(int x, int y, uintattr_t fg, uintattr_t bg, const char* str) +{ + return tb_print_ex(x, y, fg, bg, NULL, str); +} + +int +tb_print_ex(int x, int y, uintattr_t fg, uintattr_t bg, size_t* out_w, const char* str) +{ + int rv, w, ix, x_prev; + uint32_t uni; + + if_not_init_return(); + + if (!cellbuf_in_bounds(&global.back, x, y)) { return TB_ERR_OUT_OF_BOUNDS; } + + ix = x; + x_prev = x; + if (out_w) *out_w = 0; + + while (*str) { + rv = tb_utf8_char_to_unicode(&uni, str); + + if (rv < 0) { + uni = 0xfffd; // replace invalid UTF-8 char with U+FFFD + str += rv * -1; + } else if (rv > 0) { + str += rv; + } else { + break; // shouldn't get here + } + + if (uni == '\n') { // TODO: \r, \t, \v, \f, etc? + x = ix; + x_prev = x; + y += 1; + continue; + } else if (!iswprint((wint_t)uni)) { + uni = 0xfffd; // replace non-printable with U+FFFD + } + + w = wcwidth((wchar_t)uni); + if (w < 0) { + return TB_ERR; // shouldn't happen if iswprint + } else if (w == 0) { // combining character + if (cellbuf_in_bounds(&global.back, x_prev, y)) { + if_err_return(rv, tb_extend_cell(x_prev, y, uni)); + } + } else { + if (cellbuf_in_bounds(&global.back, x, y)) { + if_err_return(rv, tb_set_cell(x, y, uni, fg, bg)); + } + x_prev = x; + x += w; + if (out_w) *out_w += w; + } + } + + return TB_OK; +} + +int +tb_printf(int x, int y, uintattr_t fg, uintattr_t bg, const char* fmt, ...) +{ + int rv; + va_list vl; + va_start(vl, fmt); + rv = tb_printf_inner(x, y, fg, bg, NULL, fmt, vl); + va_end(vl); + return rv; +} + +int +tb_printf_ex(int x, int y, uintattr_t fg, uintattr_t bg, size_t* out_w, const char* fmt, ...) +{ + int rv; + va_list vl; + va_start(vl, fmt); + rv = tb_printf_inner(x, y, fg, bg, out_w, fmt, vl); + va_end(vl); + return rv; +} + +int +tb_send(const char* buf, size_t nbuf) +{ + return bytebuf_nputs(&global.out, buf, nbuf); +} + +int +tb_sendf(const char* fmt, ...) +{ + int rv; + char buf[TB_OPT_PRINTF_BUF]; + va_list vl; + va_start(vl, fmt); + rv = vsnprintf(buf, sizeof(buf), fmt, vl); + va_end(vl); + if (rv < 0 || rv >= (int)sizeof(buf)) { return TB_ERR; } + return tb_send(buf, (size_t)rv); +} + +int +tb_set_func(int fn_type, int (*fn)(struct tb_event*, size_t*)) +{ + switch (fn_type) { + case TB_FUNC_EXTRACT_PRE: + global.fn_extract_esc_pre = fn; + return TB_OK; + case TB_FUNC_EXTRACT_POST: + global.fn_extract_esc_post = fn; + return TB_OK; + } + return TB_ERR; +} + +struct tb_cell* +tb_cell_buffer(void) +{ + if (!global.initialized) return NULL; + return global.back.cells; +} + +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 0; + + 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 && c[i] != '\0'; ++i) { + result <<= 6; + result |= c[i] & 0x3f; + } + + if (i != len) return i * -1; + + *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; + out[len] = '\0'; + + return len; +} + +int +tb_last_errno(void) +{ + return global.last_errno; +} + +const char* +tb_strerror(int err) +{ + switch (err) { + case TB_OK: + return "Success"; + case TB_ERR_NEED_MORE: + return "Not enough input"; + case TB_ERR_INIT_ALREADY: + return "Termbox initialized already"; + case TB_ERR_MEM: + return "Out of memory"; + case TB_ERR_NO_EVENT: + return "No event"; + case TB_ERR_NO_TERM: + return "No TERM in environment"; + case TB_ERR_NOT_INIT: + return "Termbox not initialized"; + case TB_ERR_OUT_OF_BOUNDS: + return "Out of bounds"; + case TB_ERR_UNSUPPORTED_TERM: + return "Unsupported terminal"; + case TB_ERR_CAP_COLLISION: + return "Termcaps collision"; + case TB_ERR_RESIZE_SSCANF: + return "Terminal width/height not received by sscanf() after " + "resize"; + case TB_ERR: + case TB_ERR_INIT_OPEN: + case TB_ERR_READ: + case TB_ERR_RESIZE_IOCTL: + case TB_ERR_RESIZE_PIPE: + case TB_ERR_RESIZE_SIGACTION: + case TB_ERR_POLL: + case TB_ERR_TCGETATTR: + case TB_ERR_TCSETATTR: + case TB_ERR_RESIZE_WRITE: + case TB_ERR_RESIZE_POLL: + case TB_ERR_RESIZE_READ: + default: + strerror_r(global.last_errno, global.errbuf, sizeof(global.errbuf)); + return (const char*)global.errbuf; + } +} + +int +tb_has_truecolor(void) +{ +#if TB_OPT_ATTR_W >= 32 + return 1; +#else + return 0; +#endif +} + +int +tb_has_egc(void) +{ +#ifdef TB_OPT_EGC + return 1; +#else + return 0; +#endif +} + +int +tb_attr_width(void) +{ + return TB_OPT_ATTR_W; +} + +const char* +tb_version(void) +{ + return TB_VERSION_STR; +} + +static int +tb_reset(void) +{ + int ttyfd_open = global.ttyfd_open; + memset(&global, 0, sizeof(global)); + global.ttyfd = -1; + global.rfd = -1; + global.wfd = -1; + global.ttyfd_open = ttyfd_open; + global.resize_pipefd[0] = -1; + global.resize_pipefd[1] = -1; + global.width = -1; + global.height = -1; + global.cursor_x = -1; + global.cursor_y = -1; + global.last_x = -1; + global.last_y = -1; + global.fg = TB_DEFAULT; + global.bg = TB_DEFAULT; + global.last_fg = ~global.fg; + global.last_bg = ~global.bg; + global.input_mode = TB_INPUT_ESC; + global.output_mode = TB_OUTPUT_NORMAL; + return TB_OK; +} + +static int +init_term_attrs(void) +{ + if (global.ttyfd < 0) { return TB_OK; } + + if (tcgetattr(global.ttyfd, &global.orig_tios) != 0) { + global.last_errno = errno; + return TB_ERR_TCGETATTR; + } + + struct termios tios; + memcpy(&tios, &global.orig_tios, sizeof(tios)); + global.has_orig_tios = 1; + + cfmakeraw(&tios); + tios.c_cc[VMIN] = 1; + tios.c_cc[VTIME] = 0; + + if (tcsetattr(global.ttyfd, TCSAFLUSH, &tios) != 0) { + global.last_errno = errno; + return TB_ERR_TCSETATTR; + } + + return TB_OK; +} + +int +tb_printf_inner(int x, int y, uintattr_t fg, uintattr_t bg, size_t* out_w, const char* fmt, va_list vl) +{ + int rv; + char buf[TB_OPT_PRINTF_BUF]; + rv = vsnprintf(buf, sizeof(buf), fmt, vl); + if (rv < 0 || rv >= (int)sizeof(buf)) { return TB_ERR; } + return tb_print_ex(x, y, fg, bg, out_w, buf); +} + +static int +init_term_caps(void) +{ + if (load_terminfo() == TB_OK) { return parse_terminfo_caps(); } + return load_builtin_caps(); +} + +static int +init_cap_trie(void) +{ + int rv, i; + + // Add caps from terminfo or built-in + // + // Collisions are expected as some terminfo entries have dupes. (For + // example, att605-pc collides on TB_CAP_F4 and TB_CAP_DELETE.) First cap + // in TB_CAP_* index order will win. + // + // TODO: Reorder TB_CAP_* so more critical caps come first. + for (i = 0; i < TB_CAP__COUNT_KEYS; i++) { + rv = cap_trie_add(global.caps[i], tb_key_i(i), 0); + if (rv != TB_OK && rv != TB_ERR_CAP_COLLISION) return rv; + } + + // Add built-in mod caps + // + // Collisions are OK here as well. This can happen if global.caps collides + // with builtin_mod_caps. It is desirable to give precedence to global.caps + // here. + for (i = 0; builtin_mod_caps[i].cap != NULL; i++) { + rv = cap_trie_add(builtin_mod_caps[i].cap, builtin_mod_caps[i].key, builtin_mod_caps[i].mod); + if (rv != TB_OK && rv != TB_ERR_CAP_COLLISION) return rv; + } + + return TB_OK; +} + +static int +cap_trie_add(const char* cap, uint16_t key, uint8_t mod) +{ + struct cap_trie_t *next, *node = &global.cap_trie; + size_t i, j; + + if (!cap || strlen(cap) <= 0) return TB_OK; // Nothing to do for empty caps + + for (i = 0; cap[i] != '\0'; i++) { + char c = cap[i]; + next = NULL; + + // Check if c is already a child of node + for (j = 0; j < node->nchildren; j++) { + if (node->children[j].c == c) { + next = &node->children[j]; + break; + } + } + if (!next) { + // We need to add a new child to node + node->nchildren += 1; + node->children = (struct cap_trie_t*)tb_realloc( + node->children, sizeof(*node) * node->nchildren); + if (!node->children) { return TB_ERR_MEM; } + next = &node->children[node->nchildren - 1]; + memset(next, 0, sizeof(*next)); + next->c = c; + } + + // Continue + node = next; + } + + if (node->is_leaf) { + // Already a leaf here + return TB_ERR_CAP_COLLISION; + } + + node->is_leaf = 1; + node->key = key; + node->mod = mod; + return TB_OK; +} + +static int +cap_trie_find(const char* buf, size_t nbuf, struct cap_trie_t** last, size_t* depth) +{ + struct cap_trie_t *next, *node = &global.cap_trie; + size_t i, j; + *last = node; + *depth = 0; + for (i = 0; i < nbuf; i++) { + char c = buf[i]; + next = NULL; + + // Find c in node.children + for (j = 0; j < node->nchildren; j++) { + if (node->children[j].c == c) { + next = &node->children[j]; + break; + } + } + if (!next) { + // Not found + return TB_OK; + } + node = next; + *last = node; + *depth += 1; + if (node->is_leaf && node->nchildren < 1) { break; } + } + return TB_OK; +} + +static int +cap_trie_deinit(struct cap_trie_t* node) +{ + size_t j; + for (j = 0; j < node->nchildren; j++) { + cap_trie_deinit(&node->children[j]); + } + if (node->children) { tb_free(node->children); } + memset(node, 0, sizeof(*node)); + return TB_OK; +} + +static int +init_resize_handler(void) +{ + if (pipe(global.resize_pipefd) != 0) { + global.last_errno = errno; + return TB_ERR_RESIZE_PIPE; + } + + struct sigaction sa; + memset(&sa, 0, sizeof(sa)); + sa.sa_handler = handle_resize; + if (sigaction(SIGWINCH, &sa, NULL) != 0) { + global.last_errno = errno; + return TB_ERR_RESIZE_SIGACTION; + } + + return TB_OK; +} + +static int +send_init_escape_codes(void) +{ + int rv; + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_ENTER_CA])); + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_ENTER_KEYPAD])); + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_HIDE_CURSOR])); + return TB_OK; +} + +static int +send_clear(void) +{ + int rv; + + if_err_return(rv, send_attr(global.fg, global.bg)); + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_CLEAR_SCREEN])); + + if_err_return(rv, send_cursor_if(global.cursor_x, global.cursor_y)); + if_err_return(rv, bytebuf_flush(&global.out, global.wfd)); + + global.last_x = -1; + global.last_y = -1; + + return TB_OK; +} + +static int +update_term_size(void) +{ + int rv, ioctl_errno; + + if (global.ttyfd < 0) { return TB_OK; } + + struct winsize sz; + memset(&sz, 0, sizeof(sz)); + + // Try ioctl TIOCGWINSZ + if (ioctl(global.ttyfd, TIOCGWINSZ, &sz) == 0) { + global.width = sz.ws_col; + global.height = sz.ws_row; + return TB_OK; + } + ioctl_errno = errno; + + // Try >cursor(9999,9999), >u7, <u6 + if_ok_return(rv, update_term_size_via_esc()); + + global.last_errno = ioctl_errno; + return TB_ERR_RESIZE_IOCTL; +} + +static int +update_term_size_via_esc(void) +{ +#ifndef TB_RESIZE_FALLBACK_MS +#define TB_RESIZE_FALLBACK_MS 1000 +#endif + + char move_and_report[] = "\x1b[9999;9999H\x1b[6n"; + ssize_t write_rv = write(global.wfd, move_and_report, strlen(move_and_report)); + if (write_rv != (ssize_t)strlen(move_and_report)) { return TB_ERR_RESIZE_WRITE; } + + fd_set fds; + FD_ZERO(&fds); + FD_SET(global.rfd, &fds); + + struct timeval timeout; + timeout.tv_sec = 0; + timeout.tv_usec = TB_RESIZE_FALLBACK_MS * 1000; + + int select_rv = select(global.rfd + 1, &fds, NULL, NULL, &timeout); + + if (select_rv != 1) { + global.last_errno = errno; + return TB_ERR_RESIZE_POLL; + } + + char buf[TB_OPT_READ_BUF]; + ssize_t read_rv = read(global.rfd, buf, sizeof(buf) - 1); + if (read_rv < 1) { + global.last_errno = errno; + return TB_ERR_RESIZE_READ; + } + buf[read_rv] = '\0'; + + int rw, rh; + if (sscanf(buf, "\x1b[%d;%dR", &rh, &rw) != 2) { return TB_ERR_RESIZE_SSCANF; } + + global.width = rw; + global.height = rh; + return TB_OK; +} + +static int +init_cellbuf(void) +{ + int rv; + if_err_return(rv, cellbuf_init(&global.back, global.width, global.height)); + if_err_return(rv, cellbuf_init(&global.front, global.width, global.height)); + if_err_return(rv, cellbuf_clear(&global.back)); + if_err_return(rv, cellbuf_clear(&global.front)); + return TB_OK; +} + +static int +tb_deinit(void) +{ + if (global.caps[0] != NULL && global.wfd >= 0) { + bytebuf_puts(&global.out, global.caps[TB_CAP_SHOW_CURSOR]); + bytebuf_puts(&global.out, global.caps[TB_CAP_SGR0]); + bytebuf_puts(&global.out, global.caps[TB_CAP_CLEAR_SCREEN]); + bytebuf_puts(&global.out, global.caps[TB_CAP_EXIT_CA]); + bytebuf_puts(&global.out, global.caps[TB_CAP_EXIT_KEYPAD]); + bytebuf_puts(&global.out, TB_HARDCAP_EXIT_MOUSE); + bytebuf_flush(&global.out, global.wfd); + } + if (global.ttyfd >= 0) { + if (global.has_orig_tios) { tcsetattr(global.ttyfd, TCSAFLUSH, &global.orig_tios); } + if (global.ttyfd_open) { + close(global.ttyfd); + global.ttyfd_open = 0; + } + } + + struct sigaction sa; + memset(&sa, 0, sizeof(sa)); + sa.sa_handler = SIG_DFL; + sigaction(SIGWINCH, &sa, NULL); + if (global.resize_pipefd[0] >= 0) close(global.resize_pipefd[0]); + if (global.resize_pipefd[1] >= 0) close(global.resize_pipefd[1]); + + cellbuf_free(&global.back); + cellbuf_free(&global.front); + bytebuf_free(&global.in); + bytebuf_free(&global.out); + + if (global.terminfo) tb_free(global.terminfo); + + cap_trie_deinit(&global.cap_trie); + + tb_reset(); + return TB_OK; +} + +static int +load_terminfo(void) +{ + int rv; + char tmp[TB_PATH_MAX]; + + // See terminfo(5) "Fetching Compiled Descriptions" for a description of + // this behavior. Some of these paths are compile-time ncurses options, so + // best guesses are used here. + const char* term = getenv("TERM"); + if (!term) { return TB_ERR; } + + // If TERMINFO is set, try that directory and stop + const char* terminfo = getenv("TERMINFO"); + if (terminfo) { return load_terminfo_from_path(terminfo, term); } + + // Next try ~/.terminfo + const char* home = getenv("HOME"); + if (home) { + snprintf_or_return(rv, tmp, sizeof(tmp), "%s/.terminfo", home); + if_ok_return(rv, load_terminfo_from_path(tmp, term)); + } + + // Next try TERMINFO_DIRS + // + // Note, empty entries are supposed to be interpretted as the "compiled-in + // default", which is of course system-dependent. Previously /etc/terminfo + // was used here. Let's skip empty entries altogether rather than give + // precedence to a guess, and check common paths after this loop. + const char* dirs = getenv("TERMINFO_DIRS"); + if (dirs) { + snprintf_or_return(rv, tmp, sizeof(tmp), "%s", dirs); + char* dir = strtok(tmp, ":"); + while (dir) { + const char* cdir = dir; + if (*cdir != '\0') { if_ok_return(rv, load_terminfo_from_path(cdir, term)); } + dir = strtok(NULL, ":"); + } + } + +#ifdef TB_TERMINFO_DIR + if_ok_return(rv, load_terminfo_from_path(TB_TERMINFO_DIR, term)); +#endif + if_ok_return(rv, load_terminfo_from_path("/usr/local/etc/terminfo", term)); + if_ok_return(rv, load_terminfo_from_path("/usr/local/share/terminfo", term)); + if_ok_return(rv, load_terminfo_from_path("/usr/local/lib/terminfo", term)); + if_ok_return(rv, load_terminfo_from_path("/etc/terminfo", term)); + if_ok_return(rv, load_terminfo_from_path("/usr/share/terminfo", term)); + if_ok_return(rv, load_terminfo_from_path("/usr/lib/terminfo", term)); + if_ok_return(rv, load_terminfo_from_path("/usr/share/lib/terminfo", term)); + if_ok_return(rv, load_terminfo_from_path("/lib/terminfo", term)); + + return TB_ERR; +} + +static int +load_terminfo_from_path(const char* path, const char* term) +{ + int rv; + char tmp[TB_PATH_MAX]; + + // Look for term at this terminfo location, e.g., <terminfo>/x/xterm + snprintf_or_return(rv, tmp, sizeof(tmp), "%s/%c/%s", path, term[0], term); + if_ok_return(rv, read_terminfo_path(tmp)); + +#ifdef __APPLE__ + // Try the Darwin equivalent path, e.g., <terminfo>/78/xterm + snprintf_or_return(rv, tmp, sizeof(tmp), "%s/%x/%s", path, term[0], term); + return read_terminfo_path(tmp); +#endif + + return TB_ERR; +} + +static int +read_terminfo_path(const char* path) +{ + FILE* fp = fopen(path, "rb"); + if (!fp) { return TB_ERR; } + + struct stat st; + if (fstat(fileno(fp), &st) != 0) { + fclose(fp); + return TB_ERR; + } + + size_t fsize = st.st_size; + char* data = (char*)tb_malloc(fsize); + if (!data) { + fclose(fp); + return TB_ERR; + } + + if (fread(data, 1, fsize, fp) != fsize) { + fclose(fp); + tb_free(data); + return TB_ERR; + } + + global.terminfo = data; + global.nterminfo = fsize; + + fclose(fp); + return TB_OK; +} + +static int +parse_terminfo_caps(void) +{ + // See term(5) "LEGACY STORAGE FORMAT" and "EXTENDED STORAGE FORMAT" for a + // description of this behavior. + + // Ensure there's at least a header's worth of data + if (global.nterminfo < 6) { return TB_ERR; } + + int16_t* header = (int16_t*)global.terminfo; + // header[0] the magic number (octal 0432 or 01036) + // header[1] the size, in bytes, of the names section + // header[2] the number of bytes in the boolean section + // header[3] the number of short integers in the numbers section + // header[4] the number of offsets (short integers) in the strings section + // header[5] the size, in bytes, of the string table + + // Legacy ints are 16-bit, extended ints are 32-bit + const int bytes_per_int = header[0] == 01036 ? 4 // 32-bit + : 2; // 16-bit + + // > Between the boolean section and the number section, a null byte will be + // > inserted, if necessary, to ensure that the number section begins on an + // > even byte + const int align_offset = (header[1] + header[2]) % 2 != 0 ? 1 : 0; + + const int pos_str_offsets = (6 * sizeof(int16_t)) // header (12 bytes) + + header[1] // length of names section + + header[2] // length of boolean section + + align_offset + (header[3] * bytes_per_int); // length of numbers section + + const int pos_str_table + = pos_str_offsets + (header[4] * sizeof(int16_t)); // length of string offsets table + + // Load caps + int i; + for (i = 0; i < TB_CAP__COUNT; i++) { + const char* cap = get_terminfo_string( + pos_str_offsets, header[4], pos_str_table, header[5], terminfo_cap_indexes[i]); + if (!cap) { + // Something is not right + return TB_ERR; + } + global.caps[i] = cap; + } + + return TB_OK; +} + +static int +load_builtin_caps(void) +{ + int i, j; + const char* term = getenv("TERM"); + + if (!term) { return TB_ERR_NO_TERM; } + + // Check for exact TERM match + for (i = 0; builtin_terms[i].name != NULL; i++) { + if (strcmp(term, builtin_terms[i].name) == 0) { + for (j = 0; j < TB_CAP__COUNT; j++) { + global.caps[j] = builtin_terms[i].caps[j]; + } + return TB_OK; + } + } + + // Check for partial TERM or alias match + for (i = 0; builtin_terms[i].name != NULL; i++) { + if (strstr(term, builtin_terms[i].name) != NULL + || (*(builtin_terms[i].alias) != '\0' + && strstr(term, builtin_terms[i].alias) != NULL)) { + for (j = 0; j < TB_CAP__COUNT; j++) { + global.caps[j] = builtin_terms[i].caps[j]; + } + return TB_OK; + } + } + + return TB_ERR_UNSUPPORTED_TERM; +} + +static const char* +get_terminfo_string(int16_t str_offsets_pos, + int16_t str_offsets_len, + int16_t str_table_pos, + int16_t str_table_len, + int16_t str_index) +{ + const int str_byte_index = (int)str_index * (int)sizeof(int16_t); + if (str_byte_index >= (int)str_offsets_len * (int)sizeof(int16_t)) { + // An offset beyond the table indicates absent + // See `convert_strings` in tinfo `read_entry.c` + return ""; + } + const int16_t* str_offset = (int16_t*)(global.terminfo + (int)str_offsets_pos + str_byte_index); + if ((char*)str_offset >= global.terminfo + global.nterminfo) { + // str_offset points beyond end of entry + // Truncated/corrupt terminfo entry? + return NULL; + } + if (*str_offset < 0 || *str_offset >= str_table_len) { + // A negative offset indicates absent + // An offset beyond the table indicates absent + // See `convert_strings` in tinfo `read_entry.c` + return ""; + } + if (((size_t)((int)str_table_pos + (int)*str_offset)) >= global.nterminfo) { + // string points beyond end of entry + // Truncated/corrupt terminfo entry? + return NULL; + } + return (const char*)(global.terminfo + (int)str_table_pos + (int)*str_offset); +} + +static int +wait_event(struct tb_event* event, int timeout) +{ + int rv; + char buf[TB_OPT_READ_BUF]; + + memset(event, 0, sizeof(*event)); + if_ok_return(rv, extract_event(event)); + + fd_set fds; + struct timeval tv; + tv.tv_sec = timeout / 1000; + tv.tv_usec = (timeout - (tv.tv_sec * 1000)) * 1000; + + do { + FD_ZERO(&fds); + FD_SET(global.rfd, &fds); + FD_SET(global.resize_pipefd[0], &fds); + + int maxfd = global.resize_pipefd[0] > global.rfd ? global.resize_pipefd[0] : global.rfd; + + int select_rv = select(maxfd + 1, &fds, NULL, NULL, (timeout < 0) ? NULL : &tv); + + if (select_rv < 0) { + // Let EINTR/EAGAIN bubble up + global.last_errno = errno; + return TB_ERR_POLL; + } else if (select_rv == 0) { + return TB_ERR_NO_EVENT; + } + + int tty_has_events = (FD_ISSET(global.rfd, &fds)); + int resize_has_events = (FD_ISSET(global.resize_pipefd[0], &fds)); + + if (tty_has_events) { + ssize_t read_rv = read(global.rfd, buf, sizeof(buf)); + if (read_rv < 0) { + global.last_errno = errno; + return TB_ERR_READ; + } else if (read_rv > 0) { + bytebuf_nputs(&global.in, buf, read_rv); + } + } + + if (resize_has_events) { + int ignore = 0; + read(global.resize_pipefd[0], &ignore, sizeof(ignore)); + // TODO: Harden against errors encountered mid-resize + if_err_return(rv, update_term_size()); + if_err_return(rv, resize_cellbufs()); + event->type = TB_EVENT_RESIZE; + event->w = global.width; + event->h = global.height; + return TB_OK; + } + + memset(event, 0, sizeof(*event)); + if_ok_return(rv, extract_event(event)); + } while (timeout == -1); + + return rv; +} + +static int +extract_event(struct tb_event* event) +{ + int rv; + struct bytebuf_t* in = &global.in; + + if (in->len == 0) { return TB_ERR; } + + if (in->buf[0] == '\x1b') { + // Escape sequence? + // In TB_INPUT_ESC, skip if the buffer is a single escape char + if (!((global.input_mode & TB_INPUT_ESC) && in->len == 1)) { + if_ok_or_need_more_return(rv, extract_esc(event)); + } + + // Escape key? + if (global.input_mode & TB_INPUT_ESC) { + event->type = TB_EVENT_KEY; + event->ch = 0; + event->key = TB_KEY_ESC; + event->mod = 0; + bytebuf_shift(in, 1); + return TB_OK; + } + + // Recurse for alt key + event->mod |= TB_MOD_ALT; + bytebuf_shift(in, 1); + return extract_event(event); + } + + // ASCII control key? + if ((uint16_t)in->buf[0] < TB_KEY_SPACE || in->buf[0] == TB_KEY_BACKSPACE2) { + event->type = TB_EVENT_KEY; + event->ch = 0; + event->key = (uint16_t)in->buf[0]; + event->mod |= TB_MOD_CTRL; + bytebuf_shift(in, 1); + return TB_OK; + } + + // UTF-8? + if (in->len >= (size_t)tb_utf8_char_length(in->buf[0])) { + event->type = TB_EVENT_KEY; + tb_utf8_char_to_unicode(&event->ch, in->buf); + event->key = 0; + bytebuf_shift(in, tb_utf8_char_length(in->buf[0])); + return TB_OK; + } + + // Need more input + return TB_ERR; +} + +static int +extract_esc(struct tb_event* event) +{ + int rv; + if_ok_or_need_more_return(rv, extract_esc_user(event, 0)); + if_ok_or_need_more_return(rv, extract_esc_cap(event)); + if_ok_or_need_more_return(rv, extract_esc_mouse(event)); + if_ok_or_need_more_return(rv, extract_esc_user(event, 1)); + return TB_ERR; +} + +static int +extract_esc_user(struct tb_event* event, int is_post) +{ + int rv; + size_t consumed = 0; + struct bytebuf_t* in = &global.in; + int (*fn)(struct tb_event*, size_t*); + + fn = is_post ? global.fn_extract_esc_post : global.fn_extract_esc_pre; + + if (!fn) { return TB_ERR; } + + rv = fn(event, &consumed); + if (rv == TB_OK) { bytebuf_shift(in, consumed); } + + if_ok_or_need_more_return(rv, rv); + return TB_ERR; +} + +static int +extract_esc_cap(struct tb_event* event) +{ + int rv; + struct bytebuf_t* in = &global.in; + struct cap_trie_t* node; + size_t depth; + + if_err_return(rv, cap_trie_find(in->buf, in->len, &node, &depth)); + if (node->is_leaf) { + // Found a leaf node + event->type = TB_EVENT_KEY; + event->ch = 0; + event->key = node->key; + event->mod = node->mod; + bytebuf_shift(in, depth); + return TB_OK; + } else if (node->nchildren > 0 && in->len <= depth) { + // Found a branch node (not enough input) + return TB_ERR_NEED_MORE; + } + + return TB_ERR; +} + +static int +extract_esc_mouse(struct tb_event* event) +{ + struct bytebuf_t* in = &global.in; + + enum { TYPE_VT200 = 0, TYPE_1006, TYPE_1015, TYPE_MAX }; + + const char* cmp[TYPE_MAX] = { // + // X10 mouse encoding, the simplest one + // \x1b [ M Cb Cx Cy + [TYPE_VT200] = "\x1b[M", + // xterm 1006 extended mode or urxvt 1015 extended mode + // xterm: \x1b [ < Cb ; Cx ; Cy (M or m) + [TYPE_1006] = "\x1b[<", + // urxvt: \x1b [ Cb ; Cx ; Cy M + [TYPE_1015] = "\x1b[" + }; + + int type = 0; + int ret = TB_ERR; + + // Unrolled at compile-time (probably) + for (; type < TYPE_MAX; type++) { + size_t size = strlen(cmp[type]); + + if (in->len >= size && (strncmp(cmp[type], in->buf, size)) == 0) { break; } + } + + if (type == TYPE_MAX) { + ret = TB_ERR; // No match + return ret; + } + + size_t buf_shift = 0; + + switch (type) { + case TYPE_VT200: + if (in->len >= 6) { + int b = in->buf[3] - 0x20; + int fail = 0; + + switch (b & 3) { + case 0: + event->key = ((b & 64) != 0) ? TB_KEY_MOUSE_WHEEL_UP : TB_KEY_MOUSE_LEFT; + break; + case 1: + event->key = ((b & 64) != 0) ? TB_KEY_MOUSE_WHEEL_DOWN : TB_KEY_MOUSE_MIDDLE; + break; + case 2: + event->key = TB_KEY_MOUSE_RIGHT; + break; + case 3: + event->key = TB_KEY_MOUSE_RELEASE; + break; + default: + ret = TB_ERR; + fail = 1; + break; + } + + if (!fail) { + if ((b & 32) != 0) { event->mod |= TB_MOD_MOTION; } + + // the coord is 1,1 for upper left + event->x = ((uint8_t)in->buf[4]) - 0x21; + event->y = ((uint8_t)in->buf[5]) - 0x21; + + ret = TB_OK; + } + + buf_shift = 6; + } + break; + case TYPE_1006: + // fallthrough + case TYPE_1015: { + size_t index_fail = (size_t)-1; + + enum { FIRST_M = 0, FIRST_SEMICOLON, LAST_SEMICOLON, FIRST_LAST_MAX }; + + size_t indices[FIRST_LAST_MAX] = { index_fail, index_fail, index_fail }; + int m_is_capital = 0; + + for (size_t i = 0; i < in->len; i++) { + if (in->buf[i] == ';') { + if (indices[FIRST_SEMICOLON] == index_fail) { + indices[FIRST_SEMICOLON] = i; + } else { + indices[LAST_SEMICOLON] = i; + } + } else if (indices[FIRST_M] == index_fail) { + if (in->buf[i] == 'm' || in->buf[i] == 'M') { + m_is_capital = (in->buf[i] == 'M'); + indices[FIRST_M] = i; + } + } + } + + if (indices[FIRST_M] == index_fail || indices[FIRST_SEMICOLON] == index_fail + || indices[LAST_SEMICOLON] == index_fail) { + ret = TB_ERR; + } else { + int start = (type == TYPE_1015 ? 2 : 3); + + unsigned n1 = strtoul(&in->buf[start], NULL, 10); + unsigned n2 = strtoul(&in->buf[indices[FIRST_SEMICOLON] + 1], NULL, 10); + unsigned n3 = strtoul(&in->buf[indices[LAST_SEMICOLON] + 1], NULL, 10); + + if (type == TYPE_1015) { n1 -= 0x20; } + + int fail = 0; + + switch (n1 & 3) { + case 0: + event->key = ((n1 & 64) != 0) ? TB_KEY_MOUSE_WHEEL_UP : TB_KEY_MOUSE_LEFT; + break; + case 1: + event->key = ((n1 & 64) != 0) ? TB_KEY_MOUSE_WHEEL_DOWN : TB_KEY_MOUSE_MIDDLE; + break; + case 2: + event->key = TB_KEY_MOUSE_RIGHT; + break; + case 3: + event->key = TB_KEY_MOUSE_RELEASE; + break; + default: + ret = TB_ERR; + fail = 1; + break; + } + + buf_shift = in->len; + + if (!fail) { + if (!m_is_capital) { + // on xterm mouse release is signaled by lowercase m + event->key = TB_KEY_MOUSE_RELEASE; + } + + if ((n1 & 32) != 0) { event->mod |= TB_MOD_MOTION; } + + event->x = ((uint8_t)n2) - 1; + event->y = ((uint8_t)n3) - 1; + + ret = TB_OK; + } + } + } break; + case TYPE_MAX: + ret = TB_ERR; + } + + if (buf_shift > 0) { bytebuf_shift(in, buf_shift); } + + if (ret == TB_OK) { event->type = TB_EVENT_MOUSE; } + + return ret; +} + +static int +resize_cellbufs(void) +{ + int rv; + if_err_return(rv, cellbuf_resize(&global.back, global.width, global.height)); + if_err_return(rv, cellbuf_resize(&global.front, global.width, global.height)); + if_err_return(rv, cellbuf_clear(&global.front)); + if_err_return(rv, send_clear()); + return TB_OK; +} + +static void +handle_resize(int sig) +{ + int errno_copy = errno; + write(global.resize_pipefd[1], &sig, sizeof(sig)); + errno = errno_copy; +} + +static int +send_attr(uintattr_t fg, uintattr_t bg) +{ + int rv; + + if (fg == global.last_fg && bg == global.last_bg) { return TB_OK; } + + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_SGR0])); + + uint32_t cfg, cbg; + switch (global.output_mode) { + default: + case TB_OUTPUT_NORMAL: + // The minus 1 below is because our colors are 1-indexed starting + // from black. Black is represented by a 30, 40, 90, or 100 for fg, + // bg, bright fg, or bright bg respectively. Red is 31, 41, 91, + // 101, etc. + cfg = (fg & TB_BRIGHT ? 90 : 30) + (fg & 0x0f) - 1; + cbg = (bg & TB_BRIGHT ? 100 : 40) + (bg & 0x0f) - 1; + break; + + case TB_OUTPUT_256: + cfg = fg & 0xff; + cbg = bg & 0xff; + if (fg & TB_HI_BLACK) cfg = 0; + if (bg & TB_HI_BLACK) cbg = 0; + break; + + case TB_OUTPUT_216: + cfg = fg & 0xff; + cbg = bg & 0xff; + if (cfg > 216) cfg = 216; + if (cbg > 216) cbg = 216; + cfg += 0x0f; + cbg += 0x0f; + break; + + case TB_OUTPUT_GRAYSCALE: + cfg = fg & 0xff; + cbg = bg & 0xff; + if (cfg > 24) cfg = 24; + if (cbg > 24) cbg = 24; + cfg += 0xe7; + cbg += 0xe7; + break; + +#if TB_OPT_ATTR_W >= 32 + case TB_OUTPUT_TRUECOLOR: + cfg = fg & 0xffffff; + cbg = bg & 0xffffff; + if (fg & TB_HI_BLACK) cfg = 0; + if (bg & TB_HI_BLACK) cbg = 0; + break; +#endif + } + + if (fg & TB_BOLD) if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_BOLD])); + + if (fg & TB_BLINK) if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_BLINK])); + + if (fg & TB_UNDERLINE) if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_UNDERLINE])); + + if (fg & TB_ITALIC) if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_ITALIC])); + + if (fg & TB_DIM) if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_DIM])); + +#if TB_OPT_ATTR_W == 64 + if (fg & TB_STRIKEOUT) if_err_return(rv, bytebuf_puts(&global.out, TB_HARDCAP_STRIKEOUT)); + + if (fg & TB_UNDERLINE_2) if_err_return(rv, bytebuf_puts(&global.out, TB_HARDCAP_UNDERLINE_2)); + + if (fg & TB_OVERLINE) if_err_return(rv, bytebuf_puts(&global.out, TB_HARDCAP_OVERLINE)); + + if (fg & TB_INVISIBLE) if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_INVISIBLE])); +#endif + + if ((fg & TB_REVERSE) || (bg & TB_REVERSE)) + if_err_return(rv, bytebuf_puts(&global.out, global.caps[TB_CAP_REVERSE])); + + int fg_is_default = (fg & 0xff) == 0; + int bg_is_default = (bg & 0xff) == 0; + if (global.output_mode == TB_OUTPUT_256) { + if (fg & TB_HI_BLACK) fg_is_default = 0; + if (bg & TB_HI_BLACK) bg_is_default = 0; + } +#if TB_OPT_ATTR_W >= 32 + if (global.output_mode == TB_OUTPUT_TRUECOLOR) { + fg_is_default = ((fg & 0xffffff) == 0) && ((fg & TB_HI_BLACK) == 0); + bg_is_default = ((bg & 0xffffff) == 0) && ((bg & TB_HI_BLACK) == 0); + } +#endif + + if_err_return(rv, send_sgr(cfg, cbg, fg_is_default, bg_is_default)); + + global.last_fg = fg; + global.last_bg = bg; + + return TB_OK; +} + +static int +send_sgr(uint32_t cfg, uint32_t cbg, int fg_is_default, int bg_is_default) +{ + int rv; + char nbuf[32]; + + if (fg_is_default && bg_is_default) { return TB_OK; } + + switch (global.output_mode) { + default: + case TB_OUTPUT_NORMAL: + send_literal(rv, "\x1b["); + if (!fg_is_default) { + send_num(rv, nbuf, cfg); + if (!bg_is_default) { send_literal(rv, ";"); } + } + if (!bg_is_default) { send_num(rv, nbuf, cbg); } + send_literal(rv, "m"); + break; + + case TB_OUTPUT_256: + case TB_OUTPUT_216: + case TB_OUTPUT_GRAYSCALE: + send_literal(rv, "\x1b["); + if (!fg_is_default) { + send_literal(rv, "38;5;"); + send_num(rv, nbuf, cfg); + if (!bg_is_default) { send_literal(rv, ";"); } + } + if (!bg_is_default) { + send_literal(rv, "48;5;"); + send_num(rv, nbuf, cbg); + } + send_literal(rv, "m"); + break; + +#if TB_OPT_ATTR_W >= 32 + case TB_OUTPUT_TRUECOLOR: + send_literal(rv, "\x1b["); + if (!fg_is_default) { + send_literal(rv, "38;2;"); + send_num(rv, nbuf, (cfg >> 16) & 0xff); + send_literal(rv, ";"); + send_num(rv, nbuf, (cfg >> 8) & 0xff); + send_literal(rv, ";"); + send_num(rv, nbuf, cfg & 0xff); + if (!bg_is_default) { send_literal(rv, ";"); } + } + if (!bg_is_default) { + send_literal(rv, "48;2;"); + send_num(rv, nbuf, (cbg >> 16) & 0xff); + send_literal(rv, ";"); + send_num(rv, nbuf, (cbg >> 8) & 0xff); + send_literal(rv, ";"); + send_num(rv, nbuf, cbg & 0xff); + } + send_literal(rv, "m"); + break; +#endif + } + return TB_OK; +} + +static int +send_cursor_if(int x, int y) +{ + int rv; + char nbuf[32]; + if (x < 0 || y < 0) { return TB_OK; } + send_literal(rv, "\x1b["); + send_num(rv, nbuf, y + 1); + send_literal(rv, ";"); + send_num(rv, nbuf, x + 1); + send_literal(rv, "H"); + return TB_OK; +} + +static int +send_char(int x, int y, uint32_t ch) +{ + return send_cluster(x, y, &ch, 1); +} + +static int +send_cluster(int x, int y, uint32_t* ch, size_t nch) +{ + int rv; + char chu8[8]; + + if (global.last_x != x - 1 || global.last_y != y) { if_err_return(rv, send_cursor_if(x, y)); } + global.last_x = x; + global.last_y = y; + + int i; + for (i = 0; i < (int)nch; i++) { + uint32_t ch32 = *(ch + i); + if (!iswprint((wint_t)ch32)) { + ch32 = 0xfffd; // replace non-printable codepoints with U+FFFD + } + int chu8_len = tb_utf8_unicode_to_char(chu8, ch32); + if_err_return(rv, bytebuf_nputs(&global.out, chu8, (size_t)chu8_len)); + } + + return TB_OK; +} + +static int +convert_num(uint32_t num, char* buf) +{ + int i, l = 0; + char ch; + do { + buf[l++] = (char)('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; +} + +static int +cell_cmp(struct tb_cell* a, struct tb_cell* b) +{ + if (a->ch != b->ch || a->fg != b->fg || a->bg != b->bg) { return 1; } +#ifdef TB_OPT_EGC + if (a->nech != b->nech) { + return 1; + } else if (a->nech > 0) { // a->nech == b->nech + return memcmp(a->ech, b->ech, a->nech); + } +#endif + return 0; +} + +static int +cell_copy(struct tb_cell* dst, struct tb_cell* src) +{ +#ifdef TB_OPT_EGC + if (src->nech > 0) { return cell_set(dst, src->ech, src->nech, src->fg, src->bg); } +#endif + return cell_set(dst, &src->ch, 1, src->fg, src->bg); +} + +static int +cell_set(struct tb_cell* cell, uint32_t* ch, size_t nch, uintattr_t fg, uintattr_t bg) +{ + cell->ch = ch ? *ch : 0; + cell->fg = fg; + cell->bg = bg; +#ifdef TB_OPT_EGC + if (nch <= 1) { + cell->nech = 0; + } else { + int rv; + if_err_return(rv, cell_reserve_ech(cell, nch + 1)); + memcpy(cell->ech, ch, sizeof(*ch) * nch); + cell->ech[nch] = '\0'; + cell->nech = nch; + } +#else + (void)nch; + (void)cell_reserve_ech; +#endif + return TB_OK; +} + +static int +cell_reserve_ech(struct tb_cell* cell, size_t n) +{ +#ifdef TB_OPT_EGC + if (cell->cech >= n) { return TB_OK; } + if (!(cell->ech = tb_realloc(cell->ech, n * sizeof(cell->ch)))) { return TB_ERR_MEM; } + cell->cech = n; + return TB_OK; +#else + (void)cell; + (void)n; + return TB_ERR; +#endif +} + +static int +cell_free(struct tb_cell* cell) +{ +#ifdef TB_OPT_EGC + if (cell->ech) { tb_free(cell->ech); } +#endif + memset(cell, 0, sizeof(*cell)); + return TB_OK; +} + +static int +cellbuf_init(struct cellbuf_t* c, int w, int h) +{ + c->cells = (struct tb_cell*)tb_malloc(sizeof(struct tb_cell) * w * h); + if (!c->cells) { return TB_ERR_MEM; } + memset(c->cells, 0, sizeof(struct tb_cell) * w * h); + c->width = w; + c->height = h; + return TB_OK; +} + +static int +cellbuf_free(struct cellbuf_t* c) +{ + if (c->cells) { + int i; + for (i = 0; i < c->width * c->height; i++) { + cell_free(&c->cells[i]); + } + tb_free(c->cells); + } + memset(c, 0, sizeof(*c)); + return TB_OK; +} + +static int +cellbuf_clear(struct cellbuf_t* c) +{ + int rv, i; + uint32_t space = (uint32_t)' '; + for (i = 0; i < c->width * c->height; i++) { + if_err_return(rv, cell_set(&c->cells[i], &space, 1, global.fg, global.bg)); + } + return TB_OK; +} + +static int +cellbuf_get(struct cellbuf_t* c, int x, int y, struct tb_cell** out) +{ + if (!cellbuf_in_bounds(c, x, y)) { + *out = NULL; + return TB_ERR_OUT_OF_BOUNDS; + } + *out = &c->cells[(y * c->width) + x]; + return TB_OK; +} + +static int +cellbuf_in_bounds(struct cellbuf_t* c, int x, int y) +{ + if (x < 0 || x >= c->width || y < 0 || y >= c->height) { return 0; } + return 1; +} + +static int +cellbuf_resize(struct cellbuf_t* c, int w, int h) +{ + int rv; + + int ow = c->width; + int oh = c->height; + + if (ow == w && oh == h) { return TB_OK; } + + w = w < 1 ? 1 : w; + h = h < 1 ? 1 : h; + + int minw = (w < ow) ? w : ow; + int minh = (h < oh) ? h : oh; + + struct tb_cell* prev = c->cells; + + if_err_return(rv, cellbuf_init(c, w, h)); + if_err_return(rv, cellbuf_clear(c)); + + int x, y; + for (x = 0; x < minw; x++) { + for (y = 0; y < minh; y++) { + struct tb_cell *src, *dst; + src = &prev[(y * ow) + x]; + if_err_return(rv, cellbuf_get(c, x, y, &dst)); + if_err_return(rv, cell_copy(dst, src)); + } + } + + tb_free(prev); + + return TB_OK; +} + +static int +bytebuf_puts(struct bytebuf_t* b, const char* str) +{ + if (!str || strlen(str) <= 0) return TB_OK; // Nothing to do for empty caps + return bytebuf_nputs(b, str, (size_t)strlen(str)); +} + +static int +bytebuf_nputs(struct bytebuf_t* b, const char* str, size_t nstr) +{ + int rv; + if_err_return(rv, bytebuf_reserve(b, b->len + nstr + 1)); + memcpy(b->buf + b->len, str, nstr); + b->len += nstr; + b->buf[b->len] = '\0'; + return TB_OK; +} + +static int +bytebuf_shift(struct bytebuf_t* b, size_t n) +{ + if (n > b->len) { n = b->len; } + size_t nmove = b->len - n; + memmove(b->buf, b->buf + n, nmove); + b->len -= n; + return TB_OK; +} + +static int +bytebuf_flush(struct bytebuf_t* b, int fd) +{ + if (b->len <= 0) { return TB_OK; } + ssize_t write_rv = write(fd, b->buf, b->len); + if (write_rv < 0 || (size_t)write_rv != b->len) { + // Note, errno will be 0 on partial write + global.last_errno = errno; + return TB_ERR; + } + b->len = 0; + return TB_OK; +} + +static int +bytebuf_reserve(struct bytebuf_t* b, size_t sz) +{ + if (b->cap >= sz) { return TB_OK; } + size_t newcap = b->cap > 0 ? b->cap : 1; + while (newcap < sz) { + newcap *= 2; + } + char* newbuf; + if (b->buf) { + newbuf = (char*)tb_realloc(b->buf, newcap); + } else { + newbuf = (char*)tb_malloc(newcap); + } + if (!newbuf) { return TB_ERR_MEM; } + b->buf = newbuf; + b->cap = newcap; + return TB_OK; +} + +static int +bytebuf_free(struct bytebuf_t* b) +{ + if (b->buf) { tb_free(b->buf); } + memset(b, 0, sizeof(*b)); + return TB_OK; +} + +#endif // TB_IMPL diff --git a/tui.c b/tui.c @@ -0,0 +1,51 @@ +#include "tui.h" + +#define TB_IMPL +#include "termbox2.h" + +int +present(app_t* app) +{ + int ret = tb_init(); + tb_hide_cursor(); + + if (ret) { + fprintf(stderr, "could not TUI\n"); + exit(1); + } + + render(app); + + struct tb_event ev; + int res; + + while (1) { + res = tb_poll_event(&ev); + + if (res == TB_OK) { + switch (ev.type) { + case (TB_EVENT_KEY): + if (ev.key == TB_KEY_CTRL_Q || ev.key == TB_KEY_F10 || ev.ch == 'q') goto RIP; + handle_key(app, ev); + break; + case (TB_EVENT_MOUSE): + // todo handle mouse events + break; + case (TB_EVENT_RESIZE): + break; + } + + render(app); + + } else if (res == TB_ERR_POLL && tb_last_errno() == EINTR) { + continue; + } else if (res != TB_ERR_NO_EVENT) { + fprintf(stderr, "(aborting) renderer error: %s\n", tb_strerror(res)); + return 1; + } + } + +RIP: + tb_shutdown(); + return 0; +} diff --git a/tui.h b/tui.h @@ -0,0 +1,9 @@ +#pragma once + +#include "readr.h" +#include "termbox2.h" + +int present(app_t*); + +void render(app_t*); +void handle_key(app_t*, struct tb_event); diff --git a/utils.h b/utils.h @@ -0,0 +1,296 @@ +#pragma once + +#include <stdio.h> +#include <stdlib.h> +#include <strings.h> +#include <stdbool.h> + +const char* get_home_dir(void); +const char* expand_tilde(const char*); +char* read_file(const char*); +char* host_from_url(const char*); +void remove_all_chars(char*, char); +void remove_all_tags(char*); +int open_url(const char*); +void nonascii_replace(char*, char); + +#ifdef UTILS_IMPL + +void +util_fill(int* dst, size_t n, int value) +{ + for (size_t i = 0; i < n; ++i) + dst[i] = value; +} + +#if defined(_WIN32) + +// to be tested on windows, and then have a shower + +const char* +get_home_dir(void) +{ + const char* p = getenv("USERPROFILE"); + if (p && *p) return p; + const char* drv = getenv("HOMEDRIVE"); + const char* pth = getenv("HOMEPATH"); + if (drv && pth) { + size_t n = strlen(drv) + strlen(pth) + 1; + char* buf = (char*)malloc(n); + if (!buf) return NULL; + snprintf(buf, n, "%s%s", drv, pth); + return buf; /* caller frees if not an env ptr */ + } + return NULL; +} +#else + +// macos/linux + +#include <pwd.h> +#include <unistd.h> + +const char* +get_home_dir(void) +{ + const char* p = getenv("HOME"); + if (p && *p) return p; + return NULL; +} + +#endif + +const char* +expand_tilde(const char* path) +{ + if (!path) return NULL; + const char* home = get_home_dir(); + if (!home) home = ""; /* degrade to empty if unknown */ + + /* Worst case: every char is '~' → expand to home each time */ + size_t hlen = strlen(home); + size_t out_cap = 1; /* for '\0' */ + for (const char* s = path; *s; ++s) + out_cap += (*s == '~') ? hlen : 1; + + char* out = (char*)malloc(out_cap); + if (!out) return NULL; + + char* w = out; + for (const char* s = path; *s; ++s) { + if (*s == '~') { + memcpy(w, home, hlen); + w += hlen; + } else { + *w++ = *s; + } + } + *w = '\0'; + +#if defined(_WIN32) + /* If get_home_dir() allocated (HOMEDRIVE+HOMEPATH path), free it. */ + const char* up = getenv("USERPROFILE"); + if (!(up && *up)) { free((void*)home); } +#endif + return out; +} + +char* +read_file(const char* file_path) +{ + FILE* fp = fopen(file_path, "rb"); + if (fp == NULL) { return NULL; } + + if (fseek(fp, 0, SEEK_END) != 0) { + fclose(fp); + printf("Failed to find the end of the file\n"); + return NULL; + } + + long file_size = ftell(fp); + + if (file_size < 0) { + fclose(fp); + printf("Failed to determine the file size\n"); + return NULL; + } + + rewind(fp); + + // check for overflow before casting + if ((unsigned long)file_size >= SIZE_MAX) { + fclose(fp); + printf("File too large to fit in memory\n"); + return NULL; + } + + char* contents = (char*)calloc(1, (size_t)file_size + 1); + if (contents == NULL) { + printf("Failed to allocate memory to read file\n"); + fclose(fp); + return NULL; + } + + size_t bytes_read = fread(contents, 1, (size_t)file_size, fp); + if (bytes_read != (size_t)file_size) { + free(contents); + fclose(fp); + printf("Failed to read the file in its entirety\n"); + return NULL; + } + + contents[file_size] = '\0'; + + fclose(fp); + return contents; +} + +char* +host_from_url(const char* url) +{ + const char *p = url, *start, *end; + + // skip scheme + const char* scheme = strstr(p, "://"); + if (scheme) + p = scheme + 3; + else if (p[0] == '/' && p[1] == '/') + p += 2; + + // IPv6 literal in brackets + if (*p == '[') { + start = p + 1; + end = strchr(start, ']'); + if (!end) return NULL; + return strndup(start, (size_t)(end - start)); + } + + start = p; + + // stop at / ? # : (drop path, query, fragment, and port) + end = strpbrk(start, "/?:#:"); + if (!end) end = start + strlen(start); + + // optional: strip leading "www." + if (end - start >= 4 && strncasecmp(start, "www.", 4) == 0) start += 4; + + return strndup(start, (size_t)(end - start)); +} + +void +remove_all_chars(char* str, char c) +{ + char *pr = str, *pw = str; + while (*pr) { + *pw = *pr++; + pw += (*pw != c); + } + *pw = '\0'; +} + +void +remove_all_tags(char* str) +{ + char *pr = str, *pw = str; + bool in_tag = false; + + while (*pr) { + if (*pr == '<') { + in_tag = true; + pr++; + continue; + } else if (*pr == '>') { + in_tag = false; + pr++; + continue; + } + if (!in_tag) { *pw++ = *pr; } + pr++; + } + *pw = '\0'; +} + +// +// open_url +// + +#define _POSIX_C_SOURCE 200809L + +#ifdef _WIN32 +#include <windows.h> +int +open_url(const char* url) +{ + HINSTANCE r = ShellExecuteA(NULL, "open", url, NULL, NULL, SW_SHOWNORMAL); + return ((uintptr_t)r > 32) ? 0 : -1; // ShellExecute error if <= 32 +} +#else +static void +shell_quote(const char* in, char* out, size_t outsz) +{ + // wrap in single quotes and escape embedded single quotes: ' -> '\'' + size_t n = 0; + if (n + 1 < outsz) out[n++] = '\''; + for (; *in && n + 4 < outsz; ++in) { + if (*in == '\'') { + memcpy(out + n, "'\\''", 4); + n += 4; + } else + out[n++] = *in; + } + if (n + 2 <= outsz) { + out[n++] = '\''; + out[n] = '\0'; + } else if (outsz) + out[outsz - 1] = '\0'; +} + +int +open_url(const char* url) +{ + char q[4096]; + char cmd[4600]; + shell_quote(url, q, sizeof q); +#ifdef __APPLE__ + snprintf(cmd, sizeof cmd, "open %s >/dev/null 2>&1 &", q); +#else + snprintf(cmd, sizeof cmd, "xdg-open %s >/dev/null 2>&1 &", q); +#endif + return system(cmd); // 0 on success from the shell +} +#endif + +// +// nonascii_replace +// + +void +nonascii_replace(char* s, char to) +{ + unsigned char *r = (unsigned char*)s, *w = (unsigned char*)s; + while (*r) { + if (*r < 0x80) { // ASCII byte + *w++ = *r++; + } else { // non-ASCII: skip UTF-8 seq, write one space + size_t need = 1; + if ((*r & 0xE0) == 0xC0) + need = 2; + else if ((*r & 0xF0) == 0xE0) + need = 3; + else if ((*r & 0xF8) == 0xF0) + need = 4; + + size_t i = 1; + if (need > 1) { + for (; i < need && r[i]; ++i) + if ((r[i] & 0xC0) != 0x80) break; + } + r += (i == need) ? need : 1; // tolerate invalid UTF-8 + + if (w == (unsigned char*)s || w[-1] != ' ') *w++ = to; + } + } + *w = '\0'; +} + +#endif