ask/ask.c

309 lines
9.5 KiB
C

/* ask - a simple command line asker
* Copyright 2019 Bruce Hill
* Released under the MIT License (see LICENSE for details)
* Usage: ask [-q|--quickpick] [-p|--password] [-v|--version] [-h|--help] [--initial=initial] [prompt [options...]]
* --password: password mode
* --quickpick: quickpick mode (exit when only one option remains)
* --version: print version and exit
* --help: print usage and exit
* --initial: initial value to pre-populate user input
*/
#include <poll.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/types.h>
#include <sys/uio.h>
#include <termios.h>
#include <unistd.h>
#include "bterm.h"
#define LOWERCASE(c) ('A' <= (c) && (c) <= 'Z' ? ((c) + 'a' - 'A') : (c))
#define EQ(a, b) (case_sensitive ? (a) == (b) : LOWERCASE(a) == LOWERCASE(b))
#define PASSWORD "-\\|/"
static int password = 0, quickpick = 0, case_sensitive = 0, to_skip = 0;
static inline void *memcheck(void *p)
{
if (!p) {
fprintf(stderr, "Memory allocation failure\n");
exit(1);
}
return p;
}
static int score(const char *str, const char *patt)
{
if (!patt[0]) return 0;
int score = 0;
const char *sp = str, *pp = patt;
while (*sp && !EQ(*sp, *pp)) ++sp;
while (*sp && *pp && EQ(*sp, *pp)) {
++score;
++sp;
++pp;
}
for (; *pp; ++sp, ++pp) {
while (*sp && !EQ(*sp, *pp)) ++sp;
if (!*sp) return 0;
}
if (*pp && !*sp) return 0;
return score;
}
static void draw_line(FILE *out, const char *line, const char *hl, const char *cur)
{
int dim = 0, backtrack = 0;
for (const char *pl = line, *phl = hl; *pl; pl++) {
if (phl >= cur) ++backtrack;
if (case_sensitive ? *pl == *phl : LOWERCASE(*pl) == LOWERCASE(*phl)) {
++phl;
if (dim) {
fputs("\033[22m", out);
dim = 0;
}
} else if (!dim) {
fputs("\033[2m", out);
dim = 1;
}
fputc(*pl, out);
}
if (backtrack)
fprintf(out, "\033[%dD", backtrack);
}
/*
* A basic fuzzy matcher and line inputter
*/
static char *get_input(FILE *in, FILE *out, const char *prompt, const char *initial, int nopts, char **opts)
{
size_t cap = initial ? strlen(initial) + 100 : 100;
char *buf = memcheck(calloc(cap, 1));
char *picked = NULL;
int b = 0, len = 0;
if (initial) {
if (!strcpy(buf, initial)) return NULL;
len = (int)strlen(initial);
b = len;
fputs(initial, out);
}
// Show cursor and set to blinking line:
while (1) {
fputs("\033[G\033[K\033[0m", out);
if (prompt) {
fputs("\033[1G\033[K\033[37;1m", out);
fputs(prompt, out);
fputs("\033[0m", out);
}
picked = NULL;
case_sensitive = 0;
for (const char *p = buf; *p; ++p)
case_sensitive |= ('A' <= *p && *p <= 'Z');
int ncandidates = 0;
for (int i = 0; i < nopts; i++)
ncandidates += score(opts[i], buf) > 0;
int bestscore = 0;
char *best = NULL;
int skipped = 0;
for (int i = 0; i < nopts; i++) {
int s = score(opts[i], buf);
if (s && skipped < to_skip) {
++skipped;
continue;
}
if (s > bestscore) {
best = opts[i];
bestscore = s;
}
}
picked = best;
if (quickpick && ncandidates == 1 && best)
goto finished;
if (password) {
if (best) fputs("\033[0;32m", out);
else if (nopts > 0) fputs("\033[0;31m", out);
else fputs("\033[0;2m", out);
fputc((PASSWORD)[strlen(buf) % strlen(PASSWORD)], out);
fputs("\033[0m", out);
} else {
if (best) {
draw_line(out, best, buf, &buf[b]);
} else {
if (nopts > 0)
fprintf(out, "\033[0;31m%s\033[0m", buf);
else
fprintf(out, "\033[0m%s\033[0m", buf);
if (b < (int)strlen(buf))
fprintf(out, "\033[%dD", (int)strlen(buf) - b);
}
}
fflush(out);
int key;
skip_redraw:
key = bgetkey(in, NULL, NULL, -1);
switch (key) {
case -1: case -2: case -3: goto skip_redraw;
case '\r':
// TODO: support backslash-enter
goto finished;
case KEY_CTRL_C: case KEY_ESC:
free(buf);
buf = NULL;
picked = NULL;
goto finished;
case KEY_CTRL_A:
if (b > 0) b = 0;
break;
case KEY_CTRL_E:
if (b < len) b = len;
break;
case KEY_CTRL_U: {
int to_clear = b;
if (to_clear) {
memmove(buf, buf+b, (size_t)(len-b));
buf[len -= b] = 0;
b = 0;
}
break;
}
case KEY_CTRL_K:
if (b < len)
buf[len = b] = 0;
break;
case KEY_CTRL_N:
if (ncandidates)
to_skip = (to_skip + 1) % ncandidates;
break;
case KEY_CTRL_P:
if (ncandidates)
to_skip = (to_skip + ncandidates - 1) % ncandidates;
break;
case KEY_BACKSPACE: case KEY_BACKSPACE2:
if (b > 0) {
--b;
memmove(buf+b, buf+b+1, (size_t)(len-b));
buf[--len] = 0;
}
break;
case KEY_DELETE: case KEY_CTRL_D:
if (b < len) {
memmove(buf+b, buf+b+1, (size_t)(len-b));
buf[--len] = 0;
}
break;
case KEY_ARROW_LEFT: case KEY_CTRL_B:
if (b > 0) --b;
break;
case KEY_ARROW_RIGHT: case KEY_CTRL_F:
if (b < len) ++b;
break;
default:
if (' ' <= key && key <= '~') {
if (len + 1 >= (int)cap)
buf = memcheck(realloc(buf, (cap += 100)));
if (b < len)
memmove(buf+b+1, buf+b, (size_t)(len-b+1));
buf[b++] = (char)key;
buf[++len] = 0;
}
break;
}
}
finished:
if (picked) picked = memcheck(strdup(picked));
else picked = buf;
return picked;
}
int main(int argc, char *argv[])
{
FILE *tty_in = fopen("/dev/tty", "r");
FILE *tty_out = fopen("/dev/tty", "w");
struct termios orig_termios, bb_termios;
tcgetattr(fileno(tty_out), &orig_termios);
cfmakeraw(&bb_termios);
if (tcsetattr(fileno(tty_out), TCSAFLUSH, &bb_termios) == -1)
return 1;
char *prompt = NULL, *initial = NULL;
char **lines = NULL;
size_t linescap = 0, linecap = 0;
int nlines = 0;
char *line = NULL;
struct pollfd pfd = {STDIN_FILENO, POLLIN, 0};
if (poll(&pfd, 1, 50) > 0) {
while ((getline(&line, &linecap, stdin)) >= 0) {
if ((size_t)nlines >= linescap)
lines = memcheck(realloc(lines, (linescap += 100)*sizeof(char*)));
if (!line[0]) continue;
if (line[strlen(line)-1] == '\n')
line[strlen(line)-1] = '\0';
lines[nlines++] = memcheck(strdup(line));
}
}
int a = 1;
while (a < argc) {
if (argv[a][0] == '-' && argv[a][1] != '-') {
for (char *p = &argv[a][1]; *p; p++) {
switch (*p) {
case 'p': password = 1; break;
case 'q': quickpick = 1; break;
case 'h': goto help_flag;
case 'v': goto version_flag;
}
}
} else if (strncmp(argv[a], "--initial=", strlen("--initial=")) == 0) {
initial = &argv[a][strlen("--initial=")];
} else if (strcmp(argv[a], "--password") == 0) {
password = 1;
} else if (strcmp(argv[a], "--quickpick") == 0) {
quickpick = 1;
} else if (strcmp(argv[a], "--help") == 0) {
help_flag:
printf("ask - A simple command line input tool.\n"
"Usage: ask [-q|--quickpick] [-p|--password] [-v|--version] [-h|--help] [--initial=initial] [prompt [options...]]\n");
return 0;
} else if (strcmp(argv[a], "--version") == 0) {
version_flag:
printf("ask v0.1\n");
return 0;
} else break;
++a;
}
if (!prompt && a < argc) prompt = argv[a++];
if (!prompt) prompt = "> ";
while (a < argc) {
if ((size_t)nlines >= linescap)
lines = memcheck(realloc(lines, (linescap += 100)*sizeof(char*)));
lines[nlines++] = argv[a++];
}
char *output = get_input(tty_in, tty_out, prompt, initial, nlines, lines);
fputs("\033[G\033[K\033[0m", tty_out);
fflush(tty_out);
tcsetattr(fileno(tty_out), TCSAFLUSH, &orig_termios);
fclose(tty_out);
tty_out = NULL;
fclose(tty_in);
tty_in = NULL;
if (!output) return 1;
puts(output);
free(output);
// This doesn't free the memory in lines, but it doesn't need to because
// the program is exiting
return 0;
}
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1