From 70bcfde0212b4009467f8551c197743b70355b99 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?P=C5=99emysl=20Janouch?=
Date: Sat, 18 Oct 2014 20:34:53 +0200
Subject: [PATCH] Add a demo for drawing in curses
---
CMakeLists.txt | 5 +-
demo-draw.c | 230 +++++++++++++++++++++++++++++++++++++++++++++++++
2 files changed, 234 insertions(+), 1 deletion(-)
create mode 100644 demo-draw.c
diff --git a/CMakeLists.txt b/CMakeLists.txt
index 29c81e1..c68e5b5 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -74,10 +74,13 @@ set_target_properties (termo-static PROPERTIES
add_executable (demo-async EXCLUDE_FROM_ALL demo-async.c)
target_link_libraries (demo-async termo-static ${lib_libraries})
+add_executable (demo-draw EXCLUDE_FROM_ALL demo-draw.c)
+target_link_libraries (demo-draw termo-static ${lib_libraries})
+
add_executable (demo EXCLUDE_FROM_ALL demo.c)
target_link_libraries (demo termo-static ${lib_libraries})
-set (demos demo demo-async)
+set (demos demo demo-async demo-draw)
if (glib_FOUND)
include_directories (${glib_INCLUDE_DIRS})
add_executable (demo-glib EXCLUDE_FROM_ALL demo-glib.c)
diff --git a/demo-draw.c b/demo-draw.c
new file mode 100644
index 0000000..85d81fc
--- /dev/null
+++ b/demo-draw.c
@@ -0,0 +1,230 @@
+// might need this for sigset_t
+#define _XOPEN_SOURCE 600
+
+#include
+#include
+#include
+#include
+#include
+#include
+
+#include
+#include
+
+#include
+#include "termo.h"
+
+typedef struct app_data app_data_t;
+struct app_data
+{
+ termo_t *tk;
+
+ // Current attributes for the left mouse button
+ int current_attrs_left;
+ // Current attributes for the right mouse button
+ int current_attrs_right;
+};
+
+static int g_winch_pipe[2];
+
+static void
+display (const char *format, ...)
+{
+ va_list ap;
+
+ mvwhline (stdscr, 0, 0, A_REVERSE, COLS);
+ attron (A_REVERSE);
+
+ va_start (ap, format);
+ vw_printw (stdscr, format, ap);
+ va_end (ap);
+
+ attroff (A_REVERSE);
+ refresh ();
+}
+
+static void
+init_palette (app_data_t *app)
+{
+ start_color ();
+
+ // Also does init_pair (0, -1, -1);
+ use_default_colors ();
+ // Duplicate it for convenience.
+ init_pair (9, -1, -1);
+
+ // Add the basic 8 colors to the default pair. Once normally, once
+ // inverted to workaround VTE's inability to set a bright background.
+ for (int i = 0; i < 8; i++)
+ {
+ init_pair (1 + i, COLOR_WHITE, COLOR_BLACK + i);
+ init_pair (10 + i, COLOR_BLACK + i, COLOR_WHITE);
+ }
+
+ // This usually creates a solid black or white.
+ app->current_attrs_left =
+ app->current_attrs_right = COLOR_PAIR (9) | A_REVERSE | A_BOLD;
+}
+
+static void
+redraw (void)
+{
+ int i;
+
+ mvwhline (stdscr, 1, 0, A_REVERSE, COLS);
+ mvwhline (stdscr, 2, 0, A_REVERSE, COLS);
+
+ for (i = 0; i < COLS; i++)
+ {
+ int pair = (float) i / COLS * 9;
+ mvaddch (1, i, ' ' | COLOR_PAIR (pair));
+ mvaddch (2, i, ' ' | COLOR_PAIR (pair + 9) | A_REVERSE | A_BOLD);
+ }
+
+ display ("Choose a color from the palette and draw. "
+ "Press Escape or ^C to quit.");
+ refresh ();
+}
+
+static bool
+on_key (app_data_t *app, termo_key_t *key)
+{
+ if (key->type == TERMO_TYPE_KEYSYM
+ && key->code.sym == TERMO_SYM_ESCAPE)
+ return false;
+
+ if (key->type == TERMO_TYPE_KEY
+ && (key->modifiers & TERMO_KEYMOD_CTRL)
+ && (key->code.codepoint == 'C' || key->code.codepoint == 'c'))
+ return false;
+
+ if (key->type != TERMO_TYPE_MOUSE)
+ return true;
+
+ int line, col, button;
+ termo_mouse_event_t event;
+
+ termo_interpret_mouse (app->tk, key, &event, &button, &line, &col);
+ if (event != TERMO_MOUSE_PRESS && event != TERMO_MOUSE_DRAG)
+ return true;
+
+ int *attrs;
+ if (button == 1)
+ attrs = &app->current_attrs_left;
+ else if (button == 3)
+ attrs = &app->current_attrs_right;
+ else
+ return true;
+
+ chtype ch = mvwinch (stdscr, line, col);
+ if (line >= 3)
+ {
+ // Paste the attributes where the user clicked.
+ addch (' ' | *attrs);
+ refresh ();
+ }
+ else if (line > 0)
+ // Copy attributes from the pallete.
+ *attrs = ch & (A_COLOR | A_ATTRIBUTES);
+
+ return true;
+}
+
+static void
+winch_handler (int signum)
+{
+ (void) signum;
+ write (g_winch_pipe[1], "x", 1);
+}
+
+int
+main (int argc, char *argv[])
+{
+ (void) argc;
+ (void) argv;
+
+ TERMO_CHECK_VERSION;
+ setlocale (LC_CTYPE, "");
+
+ struct sigaction act;
+ act.sa_handler = winch_handler;
+ act.sa_flags = SA_RESTART;
+ sigemptyset (&act.sa_mask);
+
+ // Set up a self-pipe so that we can actually poll on SIGWINCH
+ if (sigaction (SIGWINCH, &act, NULL) || pipe (g_winch_pipe))
+ {
+ fprintf (stderr, "Cannot set up signal handler\n");
+ exit (EXIT_FAILURE);
+ }
+
+ termo_t *tk = termo_new (STDIN_FILENO, NULL, 0);
+ if (!tk)
+ {
+ fprintf (stderr, "Cannot allocate termo instance\n");
+ exit (EXIT_FAILURE);
+ }
+
+ termo_set_mouse_proto (tk, termo_guess_mouse_proto (tk));
+ termo_set_mouse_tracking_mode (tk, TERMO_MOUSE_TRACKING_DRAG);
+
+ // Set up curses for our drawing needs
+ if (!initscr () || nonl () == ERR || curs_set (0) == ERR)
+ {
+ fprintf (stderr, "Cannot initialize curses\n");
+ exit (EXIT_FAILURE);
+ }
+
+ app_data_t app;
+ memset (&app, 0, sizeof app);
+ app.tk = tk;
+
+ init_palette (&app);
+ redraw ();
+
+ termo_result_t ret;
+ termo_key_t key;
+
+ // We listen for mouse/key input and terminal resize events
+ struct pollfd fds[2] =
+ {
+ { .fd = STDIN_FILENO, .events = POLLIN },
+ { .fd = g_winch_pipe[0], .events = POLLIN },
+ };
+
+ // Run a simple event loop with poll()
+ int nextwait = -1;
+ bool running = true;
+ while (running)
+ {
+ if (!poll (fds, 2, nextwait))
+ if (termo_getkey_force (tk, &key) == TERMO_RES_KEY)
+ running &= on_key (&app, &key);
+
+ if (fds[1].revents & (POLLIN | POLLHUP | POLLERR))
+ {
+ char x;
+ read (fds[1].fd, &x, 1);
+
+ // The "official" simple and flicker-prone method of resizing
+ // the internal buffers of curses
+ endwin ();
+ refresh ();
+
+ redraw ();
+ }
+ if (fds[0].revents & (POLLIN | POLLHUP | POLLERR))
+ termo_advisereadable (tk);
+
+ while ((ret = termo_getkey (tk, &key)) == TERMO_RES_KEY)
+ running &= on_key (&app, &key);
+
+ nextwait = -1;
+ if (ret == TERMO_RES_AGAIN)
+ nextwait = termo_get_waittime (tk);
+ }
+
+ endwin ();
+ termo_destroy (tk);
+}
+