From ea19304c4f80479b49cd226b8592c99b07bae573 Mon Sep 17 00:00:00 2001 From: Bruce Hill Date: Sat, 18 Apr 2020 21:18:23 -0700 Subject: [PATCH] Added Lua bindings, some documentation, and moved things around a bit. --- C/Makefile | 21 ++++++ C/btui.h | 1 + test.c => C/test.c | 10 ++- Makefile | 23 ++++++- README.md | 59 +++++++++++++++++ btui.h | 11 ++-- lua/Makefile | 30 +++++++++ lua/btui.h | 1 + lua/lbtui.c | 160 +++++++++++++++++++++++++++++++++++++++++++++ lua/test.lua | 22 +++++++ 10 files changed, 328 insertions(+), 10 deletions(-) create mode 100644 C/Makefile create mode 120000 C/btui.h rename test.c => C/test.c (79%) create mode 100644 lua/Makefile create mode 120000 lua/btui.h create mode 100644 lua/lbtui.c create mode 100644 lua/test.lua diff --git a/C/Makefile b/C/Makefile new file mode 100644 index 0000000..3e4adce --- /dev/null +++ b/C/Makefile @@ -0,0 +1,21 @@ +PREFIX= +CC ?= gcc +O ?= -O2 +CFLAGS=-std=c99 -D_XOPEN_SOURCE=500 -D_GNU_SOURCE -D_POSIX_C_SOURCE=200809L +CWARN=-Wall -Wpedantic -Wextra -Wno-unknown-pragmas -Wno-missing-field-initializers\ + -Wno-padded -Wsign-conversion -Wno-missing-noreturn -Wno-cast-qual -Wtype-limits +#CFLAGS += -fsanitize=address -fno-omit-frame-pointer +G= + +all: + +clean: + rm -f test + +ctest: test.c ../btui.h + $(CC) $(CFLAGS) $(CWARN) $(G) $(O) $< -o $@ + +test: ctest + ./ctest + +.PHONY: all, clean, test diff --git a/C/btui.h b/C/btui.h new file mode 120000 index 0000000..6c720e8 --- /dev/null +++ b/C/btui.h @@ -0,0 +1 @@ +../btui.h \ No newline at end of file diff --git a/test.c b/C/test.c similarity index 79% rename from test.c rename to C/test.c index 9e00191..1396b43 100644 --- a/test.c +++ b/C/test.c @@ -13,7 +13,9 @@ int main(void) btui_printf(bt, "Update %d, size = %dx%d", i++, bt->width, bt->height); btui_flush(bt); - int key = btui_getkey(bt, NULL, NULL); + int mouse_x = -1, mouse_y = -1; + int key = btui_getkey(bt, &mouse_x, &mouse_y); + btui_clear(bt); switch (key) { case 'q': case KEY_CTRL_C: done = 1; break; case -1: break; @@ -24,9 +26,13 @@ int main(void) btui_scroll(bt, 1, bt->height-1, -1); break; default: { + if (mouse_x != -1) { + x = mouse_x; + y = mouse_y; + } char buf[256] = {0}; btui_keyname(key, buf); - btui_move_cursor(bt, x, y++); + btui_move_cursor(bt, x, y); //btui_set_attributes(bt, BTUI_FG_YELLOW | BTUI_BOLD); btui_set_fg_hex(bt, 0xacff40); btui_printf(bt, "Pressed: %s", buf); diff --git a/Makefile b/Makefile index 053f98b..cc0e065 100644 --- a/Makefile +++ b/Makefile @@ -7,12 +7,29 @@ CWARN=-Wall -Wpedantic -Wextra -Wno-unknown-pragmas -Wno-missing-field-initializ #CFLAGS += -fsanitize=address -fno-omit-frame-pointer G= -all: test +all: checksyntax + +checksyntax: btui.h + $(CC) $(CFLAGS) $(CWARN) $(G) $(O) -fsyntax-only $< clean: - rm -f test + @cd lua; make clean + @cd C; make clean + @cd python; make clean %: %.c btui.h $(CC) $(CFLAGS) $(CWARN) $(G) $(O) $< -o $@ -.PHONY: all, clean +c: + @cd C; make + +testc: + @cd C; make test + +lua: + @cd lua; make + +testlua: + @cd lua; make test + +.PHONY: all, checksyntax, clean, c, testc, lua, testlua diff --git a/README.md b/README.md index 9b8ff9b..9d7731b 100644 --- a/README.md +++ b/README.md @@ -3,3 +3,62 @@ BTUI is a minimal, embeddable single header file alternative to bloatware like ncurses. BTUI aims to be under 1/1,000th the size of ncurses, while also providing a more usable and modern API. + +## Language Bindings + +BTUI comes with bindings for C and Lua, with plans to add Python bindings. + +### C API + +BTUI has the following C function definitions, as well as definitions for some +constants, including terminal escape values and keycodes. + + btui_t* btui_enable(void); + void btui_disable(btui_t *bt); + int btui_getkey(btui_t *bt, int *mouse_x, int *mouse_y); + int btui_move_cursor(btui_t *bt, int x, int y); + char *btui_keyname(int key, char *buf); + int btui_keynamed(const char *name); + int btui_set_attributes(btui_t *bt, unsigned long attrs); + int btui_set_fg_rgb(btui_t *bt, unsigned char r, unsigned char g, unsigned char b); + int btui_set_bg_rgb(btui_t *bt, unsigned char r, unsigned char g, unsigned char b); + int btui_set_fg_hex(btui_t *bt, int hex); + int btui_set_bg_hex(btui_t *bt, int hex); + #define btui_printf(bt, ...) fprintf((bt)->out, __VA_ARGS__) + #define btui_puts(bt, s) fputs(s, (bt)->out) + #define btui_flush(bt) fflush((bt)->out) + #define btui_clear(bt) fputs("\033[2J", (bt)->out) + #define btui_clear_below(bt) fputs("\033[J", (bt)->out) + #define btui_clear_above(bt) fputs("\033[1J", (bt)->out) + #define btui_clear_eol(bt) fputs("\033[K", (bt)->out) + #define btui_clear_line(bt) fputs("\033[2K", (bt)->out) + +### Lua API + +The Lua library returns a function that takes one argument: a function that will +be called with a `BTUI` object, which can be used to do TUI actions. Errors will +be propagated out of the function, but the terminal will be cleaned up nicely +before the error is printed. Here's a simple example program: + + local btui = require("btui") + + btui(function(bt) + local key = nil + local x, y = 0, 0 + while key ~= "q" and key ~= "Ctrl-c" do + bt:clear() + bt:move(x, y) + bt:print("Pressed: ", key) + if key == "e" then error("ERR MESSAGE") end + local s = ("Size: (%dx%d)"):format(bt:width(), bt:height()) + bt:move(bt:width()-#s, bt:height()-1) + bt:print(s) + + local mouse_x, mouse_y + key, mouse_x, mouse_y = bt:getkey() + if mouse_x then x, y = mouse_x, mouse_y end + end + if key == "Ctrl-c" then + error("Interrupt received!") + end + end) diff --git a/btui.h b/btui.h index 21b65a0..41eca23 100644 --- a/btui.h +++ b/btui.h @@ -24,8 +24,6 @@ typedef struct { int size_changed; } btui_t; -static btui_t current_bt; - btui_t* btui_enable(void); void btui_disable(btui_t *bt); int btui_getkey(btui_t *bt, int *mouse_x, int *mouse_y); @@ -38,6 +36,7 @@ int btui_set_bg_rgb(btui_t *bt, unsigned char r, unsigned char g, unsigned char int btui_set_fg_hex(btui_t *bt, int hex); int btui_set_bg_hex(btui_t *bt, int hex); #define btui_printf(bt, ...) fprintf((bt)->out, __VA_ARGS__) +#define btui_puts(bt, s) fputs(s, (bt)->out) #define btui_flush(bt) fflush((bt)->out) #define btui_clear(bt) fputs("\033[2J", (bt)->out) #define btui_clear_below(bt) fputs("\033[J", (bt)->out) @@ -45,6 +44,8 @@ int btui_set_bg_hex(btui_t *bt, int hex); #define btui_clear_eol(bt) fputs("\033[K", (bt)->out) #define btui_clear_line(bt) fputs("\033[2K", (bt)->out) +static btui_t current_bt; + // Terminal escape sequences: #define T_WRAP "7" #define T_SHOW_CURSOR "25" @@ -129,9 +130,9 @@ static keyname_t key_names[] = { {KEY_ARROW_UP, "Up"}, {KEY_ARROW_DOWN, "Down"}, {KEY_ARROW_LEFT, "Left"}, {KEY_ARROW_RIGHT, "Right"}, {MOUSE_LEFT_PRESS, "Left press"}, {MOUSE_RIGHT_PRESS, "Right press"}, {MOUSE_MIDDLE_PRESS, "Middle press"}, {MOUSE_LEFT_DRAG, "Left drag"}, {MOUSE_RIGHT_DRAG, "Right drag"}, {MOUSE_MIDDLE_DRAG, "Middle drag"}, - {MOUSE_LEFT_RELEASE, "Left click"}, {MOUSE_RIGHT_RELEASE, "Right click"}, {MOUSE_MIDDLE_RELEASE, "Middle click"}, - {MOUSE_LEFT_RELEASE, "Left up"}, {MOUSE_RIGHT_RELEASE, "Right up"}, {MOUSE_MIDDLE_RELEASE, "Middle up"}, {MOUSE_LEFT_RELEASE, "Left release"}, {MOUSE_RIGHT_RELEASE, "Right release"}, {MOUSE_MIDDLE_RELEASE, "Middle release"}, + {MOUSE_LEFT_RELEASE, "Left up"}, {MOUSE_RIGHT_RELEASE, "Right up"}, {MOUSE_MIDDLE_RELEASE, "Middle up"}, + {MOUSE_LEFT_RELEASE, "Left click"}, {MOUSE_RIGHT_RELEASE, "Right click"}, {MOUSE_MIDDLE_RELEASE, "Middle click"}, {MOUSE_LEFT_DOUBLE, "Double left click"}, {MOUSE_RIGHT_DOUBLE, "Double right click"}, {MOUSE_MIDDLE_DOUBLE, "Double middle click"}, {MOUSE_WHEEL_RELEASE, "Mouse wheel up"}, {MOUSE_WHEEL_PRESS, "Mouse wheel down"}, {KEY_TAB, "Tab"}, {KEY_ENTER, "Enter"}, {KEY_ENTER, "Return"}, @@ -386,7 +387,7 @@ static const struct termios normal_termios = { .c_cc[VTIME] = 0, }; -const struct termios tui_termios = { +static const struct termios tui_termios = { .c_iflag = 0, .c_oflag = ONLCR | NL0 | CR0 | TAB0 | BS0 | VT0 | FF0, .c_lflag = ECHOE | ECHOK | ECHOCTL | ECHOKE, diff --git a/lua/Makefile b/lua/Makefile new file mode 100644 index 0000000..d648527 --- /dev/null +++ b/lua/Makefile @@ -0,0 +1,30 @@ +PREFIX= +CC ?= gcc +O ?= -O2 +CFLAGS=-std=c99 -D_XOPEN_SOURCE=500 -D_GNU_SOURCE -D_POSIX_C_SOURCE=200809L +CWARN=-Wall -Wpedantic -Wextra -Wno-unknown-pragmas -Wno-missing-field-initializers\ + -Wno-padded -Wsign-conversion -Wno-missing-noreturn -Wno-cast-qual -Wtype-limits +#CFLAGS += -fsanitize=address -fno-omit-frame-pointer +G= +LUA_DIR=/usr/local +LUA_INC=$(LUA_DIR)/include +LUA_BIN=$(LUA_DIR)/bin +LUA= lua +#LUA_SHARED_FLAGS=-bundle -undefined dynamic_lookup +LUA_SHARED_FLAGS=-shared -fPIC + +all: btui.so + +clean: + rm -f lbtui.o btui.so + +test: btui.so test.lua + $(LUA) test.lua + +btui.so: lbtui.o ../btui.h + $(CC) $(CFLAGS) $(CWARN) $(G) $(O) $(LUA_SHARED_FLAGS) -I$(LUA_INC) -o $@ $< + +lbtui.o: lbtui.c + $(CC) $(CFLAGS) $(CWARN) $(G) $(O) -fPIC -c -o $@ $< + +.PHONY: all, clean, testlua, test diff --git a/lua/btui.h b/lua/btui.h new file mode 120000 index 0000000..6c720e8 --- /dev/null +++ b/lua/btui.h @@ -0,0 +1 @@ +../btui.h \ No newline at end of file diff --git a/lua/lbtui.c b/lua/lbtui.c new file mode 100644 index 0000000..01292a6 --- /dev/null +++ b/lua/lbtui.c @@ -0,0 +1,160 @@ +/* +* lbtui.c +* A Lua library binding for btui, Bruce's Text User Interface library. +*/ + +#include "lua.h" +#include "lauxlib.h" +#include "btui.h" + +// The C API changed from 5.1 to 5.2, so these shims help the code compile on >=5.2 +#if LUA_VERSION_NUM >= 502 +#define lua_objlen(L, i) lua_rawlen(L, i) +#define luaL_register(L, _, R) luaL_setfuncs(L, R, 0) +#endif +// Lua 5.3 introduced lua_isinteger, fall back to lua_isnumber +#if LUA_VERSION_NUM < 503 +#define lua_isinteger(L, i) lua_isnumber(L, i) +#endif + +const int BTUI_METATABLE; + +static int Lbtui_enable(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + *bt = btui_enable(); + btui_move_cursor(*bt, 0, 0); + btui_flush(*bt); + return 0; +} + +static int Lbtui_disable(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + if (*bt == NULL) luaL_error(L, "BTUI object not initialized"); + btui_disable(*bt); + return 0; +} + +static int Lbtui_getkey(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + if (*bt == NULL) luaL_error(L, "BTUI object not initialized"); + int mouse_x = -1, mouse_y = -1; + int key = btui_getkey(*bt, &mouse_x, &mouse_y); + if (key == -1) return 0; + char buf[256] = {0}; + btui_keyname(key, buf); + lua_pushstring(L, buf); + if (mouse_x != -1 || mouse_y != -1) { + lua_pushinteger(L, mouse_x); + lua_pushinteger(L, mouse_y); + return 3; + } + return 1; +} + +static int Lbtui_print(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + int n = lua_gettop(L); + for (int i = 2; i <= n; i++) { + btui_puts(*bt, luaL_tolstring(L, i, NULL)); + lua_pop(L, 1); + } + btui_flush(*bt); + return 0; +} + +static int Lbtui_clear(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + btui_clear(*bt); + btui_flush(*bt); + return 0; +} + +static int Lbtui_move(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + if (lua_gettop(L) < 3) luaL_error(L, "Expected x and y values"); + int isnum; + int x = lua_tointegerx(L, 2, &isnum); + if (!isnum) luaL_error(L, "Expected integer x value"); + int y = lua_tointegerx(L, 3, &isnum); + if (!isnum) luaL_error(L, "Expected integer y value"); + btui_move_cursor(*bt, x, y); + btui_flush(*bt); + return 0; +} + +static int Lbtui_width(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + lua_pushinteger(L, (*bt)->width); + return 1; +} + +static int Lbtui_height(lua_State *L) +{ + btui_t **bt = (btui_t**)lua_touserdata(L, 1); + if (bt == NULL) luaL_error(L, "Not a BTUI object"); + lua_pushinteger(L, (*bt)->height); + return 1; +} + +static int Lbtui_wrap(lua_State *L) +{ + if (lua_gettop(L) < 1) luaL_error(L, "expected a callable object"); + btui_t **bt = (btui_t**)lua_newuserdata(L, sizeof(btui_t*)); + lua_pushlightuserdata(L, (void*)&BTUI_METATABLE); + lua_gettable(L, LUA_REGISTRYINDEX); + lua_setmetatable(L, -2); + *bt = btui_enable(); + btui_move_cursor(*bt, 0, 0); + int status = lua_pcall(L, 1, 0, 0); + btui_disable(*bt); + if (status != LUA_OK) + lua_error(L); + return 0; +} + +static int Lbtui_tostring(lua_State *L) +{ + lua_pushliteral(L, ""); + return 1; +} + +static const luaL_Reg Rclass_metamethods[] = +{ + { "__tostring", Lbtui_tostring}, + { "enable", Lbtui_enable}, + { "disable", Lbtui_disable}, + { "getkey", Lbtui_getkey}, + { "print", Lbtui_print}, + { "clear", Lbtui_clear}, + { "move", Lbtui_move}, + { "width", Lbtui_width}, + { "height", Lbtui_height}, + { NULL, NULL} +}; + +LUALIB_API int luaopen_btui(lua_State *L) +{ + lua_pushlightuserdata(L, (void*)&BTUI_METATABLE); + lua_createtable(L, 0, 16); + luaL_register(L, NULL, Rclass_metamethods); + lua_pushvalue(L, -1); + lua_setfield(L, -2, "__index"); + lua_settable(L, LUA_REGISTRYINDEX); + + lua_pushcfunction(L, Lbtui_wrap); + return 1; +} diff --git a/lua/test.lua b/lua/test.lua new file mode 100644 index 0000000..81590ae --- /dev/null +++ b/lua/test.lua @@ -0,0 +1,22 @@ +local btui = require("btui") + +btui(function(bt) + local key = nil + local x, y = 0, 0 + while key ~= "q" and key ~= "Ctrl-c" do + bt:clear() + bt:move(x, y) + bt:print("Pressed: ", key) + if key == "e" then error("ERR MESSAGE") end + local s = ("Size: (%dx%d)"):format(bt:width(), bt:height()) + bt:move(bt:width()-#s, bt:height()-1) + bt:print(s) + + local mouse_x, mouse_y + key, mouse_x, mouse_y = bt:getkey() + if mouse_x then x, y = mouse_x, mouse_y end + end + if key == "Ctrl-c" then + error("Interrupt received!") + end +end)