Added a --read0 option and fixed up the documentation.

This commit is contained in:
Bruce Hill 2019-11-01 14:43:06 +01:00
parent 00653d0b11
commit fedc1751fe
3 changed files with 48 additions and 24 deletions

View File

@ -32,5 +32,7 @@ Here's a simple program to move a file from the current directory:
`$XDG_DATA/ask/<name>.hist` (`~/.local/share/ask/<name>.hist` by default) for `$XDG_DATA/ask/<name>.hist` (`~/.local/share/ask/<name>.hist` by default) for
use with up/down arrow keys. Maximum of 1000 entries are stored per log file. use with up/down arrow keys. Maximum of 1000 entries are stored per log file.
For the full set of command line options, run `man ./ask.1`.
## License ## License
`ask` is released under the MIT License. See LICENSE for details. `ask` is released under the MIT License. See LICENSE for details.

34
ask.1
View File

@ -5,16 +5,23 @@
ask \- A tiny command line tool for getting user input ask \- A tiny command line tool for getting user input
.SH SYNOPSIS .SH SYNOPSIS
.B ask .B ask
[\fI-hPqvyn\fR] [\fI-Q\fR|\fI--quickpick\fR]
[\fI-q\fR |\fI--query=initial\fR] [\fI-P\fR|\fI--password\fR]
[[\fI-p\fR |\fI--prompt=\fR]\fIprompt \fR [\fI-0\fR|\fI--read0\fR]
[\fIoptions...\fR]] [\fI-y\fR|\fI--yes\fR]
[\fI-n\fR|\fI--no\fR]
[(\fI-H \fR|\fI--history=\fR) name]
[\fI-h\fR|\fI--help\fR]
[\fI-v\fR|\fI--version\fR]
[(\fI-q\fR |\fI--query=\fR) initial]
[(\fI-p\fR |\fI--prompt=\fR) prompt]
[\fIoptions...\fR]
.SH DESCRIPTION .SH DESCRIPTION
\fBask\fR is a tiny console application that displays a prompt, gets user input \fBask\fR is a tiny console application that displays a prompt, gets user input
(with line editing and fuzzy finding functionality), and prints the result to (with line editing and fuzzy finding functionality), and prints the result to
standard output standard output
.SH OPTIONS .SH OPTIONS
.B \-q .B \-Q
.B \--quickpick .B \--quickpick
When used with fuzzy finding, as soon as exactly one match is found, exit and When used with fuzzy finding, as soon as exactly one match is found, exit and
print it. print it.
@ -23,6 +30,10 @@ print it.
.B \--password .B \--password
Use password mode, which does not print user input as it's being typed. Use password mode, which does not print user input as it's being typed.
.B \-0
.B \--read0
Read input delineated by NULL bytes instead of newlines.
.B \-v .B \-v
.B \--version .B \--version
Print \fBask\fR's version and exit. Print \fBask\fR's version and exit.
@ -31,16 +42,21 @@ Print \fBask\fR's version and exit.
.B \--help .B \--help
Print \fBask\fR's usage and exit. Print \fBask\fR's usage and exit.
.B \-q .B \-q <query>
.B \--query= .B \--query=<query>
If given, pre-populate the user input with this value. If given, pre-populate the user input with this value.
.B \-p .B \-p <prompt>
.B \--prompt= .B \--prompt=<prompt>
If provided, display the given prompt in bold. If the \fI-p\fR and If provided, display the given prompt in bold. If the \fI-p\fR and
\fI--prompt=\fR flags are not used, the first positional argument is used as \fI--prompt=\fR flags are not used, the first positional argument is used as
the prompt, or \fB"> "\fR if there are no positional arguments. the prompt, or \fB"> "\fR if there are no positional arguments.
.B \-H <file>
.B \--history=<file>
Use the given file as a history file. With a history file, you can browse
previously selected values with the up/down arrow keys.
.B \-y .B \-y
.B \--yes .B \--yes
Quickpick between "y" and "n" with "[Y/n]" appended to the prompt, exiting with Quickpick between "y" and "n" with "[Y/n]" appended to the prompt, exiting with

36
ask.c
View File

@ -2,11 +2,12 @@
* Copyright 2019 Bruce Hill * Copyright 2019 Bruce Hill
* Released under the MIT License (see LICENSE for details) * Released under the MIT License (see LICENSE for details)
* Usage: ask [-Q|--quickpick] [-y|--yes] [-n|--no] [-P|--password] [[-H |--history=]name] * Usage: ask [-Q|--quickpick] [-y|--yes] [-n|--no] [-P|--password] [[-H |--history=]name]
* [-v|--version] [-h|--help] [-q |--query=initial] [[-p |--prompt=]prompt [options...]] * [-v|--version] [-h|--help] [-0|--read0] [-q |--query=initial] [[-p |--prompt=]prompt [options...]]
* --password: password mode * --password: password mode
* --quickpick: quickpick mode (exit when only one option remains) * --quickpick: quickpick mode (exit when only one option remains)
* --version: print version and exit * --version: print version and exit
* --help: print usage and exit * --help: print usage and exit
* --read0: read input delimited by NULL bytes instead of newlines
* --query: initial value to pre-populate user input * --query: initial value to pre-populate user input
* --prompt: use the given prompt (displayed in bold) * --prompt: use the given prompt (displayed in bold)
* --yes: append " [Y/n]" to the prompt, use quickpick mode, * --yes: append " [Y/n]" to the prompt, use quickpick mode,
@ -28,7 +29,7 @@
#include "bterm.h" #include "bterm.h"
#define ASK_VERSION "0.3" #define ASK_VERSION "0.4"
#define LOWERCASE(c) ('A' <= (c) && (c) <= 'Z' ? ((c) + 'a' - 'A') : (c)) #define LOWERCASE(c) ('A' <= (c) && (c) <= 'Z' ? ((c) + 'a' - 'A') : (c))
#define EQ(a, b) (case_sensitive ? (a) == (b) : LOWERCASE(a) == LOWERCASE(b)) #define EQ(a, b) (case_sensitive ? (a) == (b) : LOWERCASE(a) == LOWERCASE(b))
#define PASSWORD "-\\|/" #define PASSWORD "-\\|/"
@ -373,20 +374,8 @@ int main(int argc, char *argv[])
int yes = 0, no = 0; int yes = 0, no = 0;
char *prompt = NULL, *query = NULL, *histname = NULL; char *prompt = NULL, *query = NULL, *histname = NULL;
char **opts = NULL; char **opts = NULL;
char delim = '\n';
size_t linescap = 0, linecap = 0; size_t linescap = 0, linecap = 0;
int nopts = 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)nopts >= linescap)
opts = memcheck(realloc(opts, (linescap += 100)*sizeof(char*)));
if (!line[0]) continue;
if (line[strlen(line)-1] == '\n')
line[strlen(line)-1] = '\0';
opts[nopts++] = memcheck(strdup(line));
}
}
int a; int a;
for (a = 1; a < argc; a++) { for (a = 1; a < argc; a++) {
if (strcmp(argv[a], "-H") == 0) { if (strcmp(argv[a], "-H") == 0) {
@ -399,6 +388,7 @@ int main(int argc, char *argv[])
case 'Q': quickpick = 1; break; case 'Q': quickpick = 1; break;
case 'h': goto help_flag; case 'h': goto help_flag;
case 'v': goto version_flag; case 'v': goto version_flag;
case '0': delim = '\0'; break;
case 'y': yes = 1; quickpick = 1; break; case 'y': yes = 1; quickpick = 1; break;
case 'n': no = 1; quickpick = 1; break; case 'n': no = 1; quickpick = 1; break;
} }
@ -407,6 +397,8 @@ int main(int argc, char *argv[])
prompt = argv[++a]; prompt = argv[++a];
} else if (strcmp(argv[a], "-q") == 0) { } else if (strcmp(argv[a], "-q") == 0) {
query = argv[++a]; query = argv[++a];
} else if (strncmp(argv[a], "--read0", strlen("--read0")) == 0) {
delim = '\0';
} else if (strncmp(argv[a], "--prompt=", strlen("--prompt=")) == 0) { } else if (strncmp(argv[a], "--prompt=", strlen("--prompt=")) == 0) {
prompt = &argv[a][strlen("--prompt=")]; prompt = &argv[a][strlen("--prompt=")];
} else if (strncmp(argv[a], "--query=", strlen("--query=")) == 0) { } else if (strncmp(argv[a], "--query=", strlen("--query=")) == 0) {
@ -437,6 +429,20 @@ int main(int argc, char *argv[])
} }
if (!prompt && a < argc) prompt = argv[a++]; if (!prompt && a < argc) prompt = argv[a++];
int nopts = 0;
char *line = NULL;
struct pollfd pfd = {STDIN_FILENO, POLLIN, 0};
if (poll(&pfd, 1, 50) > 0) {
while ((getdelim(&line, &linecap, delim, stdin)) >= 0) {
if ((size_t)nopts >= linescap)
opts = memcheck(realloc(opts, (linescap += 100)*sizeof(char*)));
if (!line[0]) continue;
if (line[strlen(line)-1] == '\n')
line[strlen(line)-1] = '\0';
opts[nopts++] = memcheck(strdup(line));
}
}
while (a < argc) { while (a < argc) {
if ((size_t)nopts >= linescap) if ((size_t)nopts >= linescap)
opts = memcheck(realloc(opts, (linescap += 100)*sizeof(char*))); opts = memcheck(realloc(opts, (linescap += 100)*sizeof(char*)));