2019-05-20 19:28:47 -07:00
|
|
|
/*
|
2019-05-22 01:56:39 -07:00
|
|
|
* Bitty Browser (bb)
|
2019-05-20 19:28:47 -07:00
|
|
|
* Copyright 2019 Bruce Hill
|
|
|
|
* Released under the MIT license
|
2019-09-30 17:06:27 -07:00
|
|
|
*
|
|
|
|
* This file contains the main source code of `bb`.
|
2019-05-20 19:28:47 -07:00
|
|
|
*/
|
2019-05-31 00:12:21 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
#include "bb.h"
|
2019-07-12 16:19:31 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
// Variables used within this file to track global state
|
2019-05-27 14:12:47 -07:00
|
|
|
static struct termios orig_termios, bb_termios;
|
|
|
|
static FILE *tty_out = NULL, *tty_in = NULL;
|
2019-11-01 09:19:25 -07:00
|
|
|
static struct winsize winsize = {0};
|
2019-05-25 04:30:51 -07:00
|
|
|
static char *cmdfilename = NULL;
|
2019-11-11 10:09:07 -08:00
|
|
|
static proc_t *running_procs = NULL;
|
2019-11-01 06:50:44 -07:00
|
|
|
static int dirty = 1;
|
2019-05-25 04:30:51 -07:00
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
2019-10-13 21:20:00 -07:00
|
|
|
* Use bb to browse the filesystem.
|
2019-05-28 22:29:22 -07:00
|
|
|
*/
|
2019-11-11 10:48:12 -08:00
|
|
|
void bb_browse(bb_t *bb, const char *initial_path)
|
2019-05-23 00:57:25 -07:00
|
|
|
{
|
2019-09-30 17:06:27 -07:00
|
|
|
static long cmdpos = 0;
|
|
|
|
int check_cmds = 1;
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 1;
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-11-11 10:48:12 -08:00
|
|
|
if (populate_files(bb, initial_path))
|
|
|
|
err("Could not find initial path: \"%s\"", initial_path);
|
|
|
|
run_script(bb, runstartup);
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
goto force_check_cmds;
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-10-13 18:56:09 -07:00
|
|
|
while (!bb->should_quit) {
|
|
|
|
redraw:
|
|
|
|
render(bb);
|
|
|
|
|
|
|
|
next_input:
|
2019-11-01 06:50:44 -07:00
|
|
|
if (dirty) goto redraw;
|
2019-05-30 00:33:51 -07:00
|
|
|
|
2019-10-13 18:56:09 -07:00
|
|
|
if (check_cmds) {
|
|
|
|
FILE *cmdfile;
|
|
|
|
force_check_cmds:
|
|
|
|
cmdfile = fopen(cmdfilename, "r");
|
|
|
|
if (!cmdfile) {
|
2019-10-27 14:58:23 -07:00
|
|
|
cmdpos = 0;
|
2019-11-01 06:50:44 -07:00
|
|
|
if (dirty) goto redraw;
|
2019-10-13 18:56:09 -07:00
|
|
|
goto get_keyboard_input;
|
|
|
|
}
|
2019-09-30 15:46:24 -07:00
|
|
|
|
2019-10-13 18:56:09 -07:00
|
|
|
if (cmdpos)
|
|
|
|
fseek(cmdfile, cmdpos, SEEK_SET);
|
|
|
|
|
|
|
|
char *cmd = NULL;
|
|
|
|
size_t space = 0;
|
|
|
|
while (getdelim(&cmd, &space, '\0', cmdfile) >= 0) {
|
|
|
|
cmdpos = ftell(cmdfile);
|
|
|
|
if (!cmd[0]) continue;
|
|
|
|
run_bbcmd(bb, cmd);
|
|
|
|
if (bb->should_quit) {
|
|
|
|
free(cmd);
|
|
|
|
fclose(cmdfile);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(cmd);
|
|
|
|
fclose(cmdfile);
|
|
|
|
unlink(cmdfilename);
|
|
|
|
cmdpos = 0;
|
|
|
|
check_cmds = 0;
|
|
|
|
goto redraw;
|
2019-05-22 14:33:14 -07:00
|
|
|
}
|
|
|
|
|
2019-10-13 18:56:09 -07:00
|
|
|
int key, mouse_x, mouse_y;
|
|
|
|
get_keyboard_input:
|
|
|
|
key = bgetkey(tty_in, &mouse_x, &mouse_y);
|
|
|
|
if (key == -1) goto next_input;
|
|
|
|
static char bbmousecol[2] = {0, 0};
|
|
|
|
static char bbclicked[PATH_MAX];
|
|
|
|
if (mouse_x != -1 && mouse_y != -1) {
|
|
|
|
bbmousecol[0] = '\0';
|
|
|
|
// Get bb column:
|
|
|
|
for (int col = 0, x = 0; bb->columns[col]; col++, x++) {
|
|
|
|
x += columns[(int)bb->columns[col]].width;
|
|
|
|
if (x >= mouse_x) {
|
|
|
|
bbmousecol[0] = bb->columns[col];
|
|
|
|
break;
|
|
|
|
}
|
2019-10-03 13:26:31 -07:00
|
|
|
}
|
2019-10-13 18:56:09 -07:00
|
|
|
if (mouse_y == 1) {
|
|
|
|
strcpy(bbclicked, "<column label>");
|
2019-11-01 09:19:25 -07:00
|
|
|
} else if (2 <= mouse_y && mouse_y <= winsize.ws_row - 2
|
2019-10-13 18:56:09 -07:00
|
|
|
&& bb->scroll + (mouse_y - 2) <= bb->nfiles - 1) {
|
|
|
|
strcpy(bbclicked, bb->files[bb->scroll + (mouse_y - 2)]->fullname);
|
|
|
|
} else {
|
|
|
|
bbclicked[0] = '\0';
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
2019-10-13 18:56:09 -07:00
|
|
|
setenv("BBMOUSECOL", bbmousecol, 1);
|
|
|
|
setenv("BBCLICKED", bbclicked, 1);
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
2019-10-13 18:56:09 -07:00
|
|
|
// Search user-defined key bindings
|
|
|
|
binding_t *binding = NULL;
|
|
|
|
for (int i = 0; bindings[i].script && i < sizeof(bindings)/sizeof(bindings[0]); i++) {
|
|
|
|
if (key == bindings[i].key) {
|
|
|
|
binding = &bindings[i];
|
2019-10-12 16:04:19 -07:00
|
|
|
break;
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
}
|
2019-10-13 18:56:09 -07:00
|
|
|
if (!binding)
|
|
|
|
goto next_input;
|
|
|
|
|
|
|
|
if (cmdpos != 0)
|
|
|
|
err("Command file still open");
|
|
|
|
if (is_simple_bbcmd(binding->script)) {
|
|
|
|
run_bbcmd(bb, binding->script);
|
2019-10-12 16:04:19 -07:00
|
|
|
} else {
|
2019-11-01 09:19:25 -07:00
|
|
|
move_cursor(tty_out, 0, winsize.ws_row-1);
|
2019-11-01 06:50:44 -07:00
|
|
|
fputs("\033[K", tty_out);
|
2019-10-13 18:56:09 -07:00
|
|
|
restore_term(&default_termios);
|
|
|
|
run_script(bb, binding->script);
|
|
|
|
init_term();
|
|
|
|
check_cmds = 1;
|
2019-10-12 16:04:19 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2019-10-27 14:58:23 -07:00
|
|
|
* Clean up the terminal before going to the default signal handling behavior.
|
2019-09-30 17:06:27 -07:00
|
|
|
*/
|
2019-10-27 14:58:23 -07:00
|
|
|
void cleanup_and_raise(int sig)
|
2019-09-30 17:06:27 -07:00
|
|
|
{
|
|
|
|
cleanup();
|
2019-10-27 14:58:23 -07:00
|
|
|
int childsig = (sig == SIGTSTP || sig == SIGSTOP) ? sig : SIGHUP;
|
|
|
|
for (proc_t *p = running_procs; p; p = p->running.next)
|
|
|
|
kill(p->pid, childsig);
|
2019-09-30 17:06:27 -07:00
|
|
|
raise(sig);
|
2019-10-27 14:58:23 -07:00
|
|
|
// This code will only ever be run if sig is SIGTSTP/SIGSTOP, otherwise, raise() won't return:
|
|
|
|
init_term();
|
|
|
|
struct sigaction sa = {.sa_handler = &cleanup_and_raise, .sa_flags = SA_NODEFER | SA_RESETHAND};
|
|
|
|
sigaction(sig, &sa, NULL);
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2019-10-27 14:58:23 -07:00
|
|
|
* Reset the screen and delete the cmdfile
|
2019-09-30 17:06:27 -07:00
|
|
|
*/
|
|
|
|
void cleanup(void)
|
|
|
|
{
|
|
|
|
if (cmdfilename) {
|
|
|
|
unlink(cmdfilename);
|
|
|
|
free(cmdfilename);
|
|
|
|
cmdfilename = NULL;
|
|
|
|
}
|
2019-11-01 08:46:12 -07:00
|
|
|
if (tty_out) {
|
|
|
|
fputs(T_OFF(T_ALT_SCREEN) T_ON(T_SHOW_CURSOR), tty_out);
|
|
|
|
fflush(tty_out);
|
|
|
|
tcsetattr(fileno(tty_out), TCSANOW, &orig_termios);
|
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Returns the color of a file listing, given its mode.
|
|
|
|
*/
|
|
|
|
const char* color_of(mode_t mode)
|
|
|
|
{
|
|
|
|
if (S_ISDIR(mode)) return DIR_COLOR;
|
|
|
|
else if (S_ISLNK(mode)) return LINK_COLOR;
|
|
|
|
else if (mode & (S_IXUSR | S_IXGRP | S_IXOTH))
|
|
|
|
return EXECUTABLE_COLOR;
|
|
|
|
else return NORMAL_COLOR;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Used for sorting, this function compares files according to the sorting-related options,
|
|
|
|
* like bb->sort
|
|
|
|
*/
|
|
|
|
#ifdef __APPLE__
|
|
|
|
int compare_files(void *v, const void *v1, const void *v2)
|
|
|
|
#else
|
|
|
|
int compare_files(const void *v1, const void *v2, void *v)
|
|
|
|
#endif
|
|
|
|
{
|
|
|
|
#define COMPARE(a, b) if ((a) != (b)) { return sign*((a) < (b) ? 1 : -1); }
|
|
|
|
#define COMPARE_TIME(t1, t2) COMPARE((t1).tv_sec, (t2).tv_sec) COMPARE((t1).tv_nsec, (t2).tv_nsec)
|
|
|
|
bb_t *bb = (bb_t*)v;
|
|
|
|
const entry_t *e1 = *((const entry_t**)v1), *e2 = *((const entry_t**)v2);
|
|
|
|
|
|
|
|
int sign = 1;
|
|
|
|
if (!bb->interleave_dirs) {
|
|
|
|
COMPARE(E_ISDIR(e1), E_ISDIR(e2));
|
|
|
|
}
|
|
|
|
|
|
|
|
for (char *sort = bb->sort + 1; *sort; sort += 2) {
|
|
|
|
sign = sort[-1] == '-' ? -1 : 1;
|
|
|
|
switch (*sort) {
|
|
|
|
case COL_SELECTED: COMPARE(IS_SELECTED(e1), IS_SELECTED(e2)); break;
|
|
|
|
case COL_NAME: {
|
|
|
|
/* This sorting method is not identical to strverscmp(). Notably, bb's sort
|
|
|
|
* will order: [0, 1, 9, 00, 01, 09, 10, 000, 010] instead of strverscmp()'s
|
|
|
|
* order: [000, 00, 01, 010, 09, 0, 1, 9, 10]. I believe bb's sort is consistent
|
|
|
|
* with how people want their files grouped: all files padded to n digits
|
|
|
|
* will be grouped together, and files with the same padding will be sorted
|
|
|
|
* ordinally. This version also does case-insensitivity by lowercasing words,
|
|
|
|
* so the following characters come before all letters: [\]^_`
|
|
|
|
*/
|
|
|
|
const char *n1 = e1->name, *n2 = e2->name;
|
|
|
|
while (*n1 && *n2) {
|
|
|
|
char c1 = LOWERCASE(*n1), c2 = LOWERCASE(*n2);
|
|
|
|
if ('0' <= c1 && c1 <= '9' && '0' <= c2 && c2 <= '9') {
|
|
|
|
long i1 = strtol(n1, (char**)&n1, 10);
|
|
|
|
long i2 = strtol(n2, (char**)&n2, 10);
|
|
|
|
// Shorter numbers always go before longer. In practice, I assume
|
|
|
|
// filenames padded to the same number of digits should be grouped
|
|
|
|
// together, instead of
|
|
|
|
// [1.png, 0001.png, 2.png, 0002.png, 3.png], it makes more sense to have:
|
|
|
|
// [1.png, 2.png, 3.png, 0001.png, 0002.png]
|
|
|
|
COMPARE((n2 - e2->name), (n1 - e1->name));
|
|
|
|
COMPARE(i2, i1);
|
|
|
|
} else {
|
|
|
|
COMPARE(c2, c1);
|
|
|
|
++n1; ++n2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
COMPARE(LOWERCASE(*n2), LOWERCASE(*n1));
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case COL_PERM: COMPARE((e1->info.st_mode & 0x3FF), (e2->info.st_mode & 0x3FF)); break;
|
|
|
|
case COL_SIZE: COMPARE(e1->info.st_size, e2->info.st_size); break;
|
|
|
|
case COL_MTIME: COMPARE_TIME(mtime(e1->info), mtime(e2->info)); break;
|
|
|
|
case COL_CTIME: COMPARE_TIME(ctime(e1->info), ctime(e2->info)); break;
|
|
|
|
case COL_ATIME: COMPARE_TIME(atime(e1->info), atime(e2->info)); break;
|
|
|
|
case COL_RANDOM: COMPARE(e1->shufflepos, e2->shufflepos); break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
#undef COMPARE
|
|
|
|
#undef COMPARE_TIME
|
|
|
|
}
|
2019-05-22 14:33:14 -07:00
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Print a string, but replacing bytes like '\n' with a red-colored "\n".
|
|
|
|
* The color argument is what color to put back after the red.
|
|
|
|
* Returns the number of bytes that were escaped.
|
|
|
|
*/
|
2019-05-28 21:36:42 -07:00
|
|
|
int fputs_escaped(FILE *f, const char *str, const char *color)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-22 19:05:56 -07:00
|
|
|
static const char *escapes = " abtnvfr e";
|
2019-05-27 21:14:34 -07:00
|
|
|
int escaped = 0;
|
2019-05-25 20:55:59 -07:00
|
|
|
for (const char *c = str; *c; ++c) {
|
2019-05-27 21:14:34 -07:00
|
|
|
if (*c > 0 && *c <= '\x1b' && escapes[(int)*c] != ' ') { // "\n", etc.
|
2019-05-27 14:12:47 -07:00
|
|
|
fprintf(f, "\033[31m\\%c%s", escapes[(int)*c], color);
|
2019-05-27 21:14:34 -07:00
|
|
|
++escaped;
|
|
|
|
} else if (*c >= 0 && !(' ' <= *c && *c <= '~')) { // "\x02", etc.
|
2019-05-27 14:12:47 -07:00
|
|
|
fprintf(f, "\033[31m\\x%02X%s", *c, color);
|
2019-05-27 21:14:34 -07:00
|
|
|
++escaped;
|
|
|
|
} else {
|
2019-05-27 14:12:47 -07:00
|
|
|
fputc(*c, f);
|
2019-05-27 21:14:34 -07:00
|
|
|
}
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
2019-05-27 21:14:34 -07:00
|
|
|
return escaped;
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
2019-09-30 17:06:27 -07:00
|
|
|
* Initialize the terminal files for /dev/tty and set up some desired
|
|
|
|
* attributes like passing Ctrl-c as a key instead of interrupting
|
2019-05-28 22:29:22 -07:00
|
|
|
*/
|
2019-09-30 17:06:27 -07:00
|
|
|
void init_term(void)
|
2019-05-28 21:36:42 -07:00
|
|
|
{
|
2019-10-27 14:58:23 -07:00
|
|
|
if (tcsetattr(fileno(tty_out), TCSANOW, &bb_termios) == -1)
|
2019-09-30 17:06:27 -07:00
|
|
|
err("Couldn't tcsetattr");
|
|
|
|
update_term_size(0);
|
|
|
|
// Initiate mouse tracking and disable text wrapping:
|
|
|
|
fputs(T_ENTER_BBMODE, tty_out);
|
2019-10-27 14:58:23 -07:00
|
|
|
fflush(tty_out);
|
2019-05-28 21:36:42 -07:00
|
|
|
}
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
/*
|
|
|
|
* Return whether or not 's' is a simple bb command that doesn't need
|
|
|
|
* a full shell instance (e.g. "bb +cd:.." or "bb +move:+1").
|
2019-06-15 19:07:00 -07:00
|
|
|
*/
|
2019-09-30 17:06:27 -07:00
|
|
|
static int is_simple_bbcmd(const char *s)
|
2019-05-29 17:26:18 -07:00
|
|
|
{
|
2019-09-30 17:06:27 -07:00
|
|
|
if (!s) return 0;
|
|
|
|
while (*s == ' ') ++s;
|
|
|
|
if (s[0] != '+' && strncmp(s, "bb +", 4) != 0)
|
|
|
|
return 0;
|
|
|
|
const char *special = ";$&<>|\n*?\\\"'";
|
|
|
|
for (const char *p = special; *p; ++p) {
|
|
|
|
if (strchr(s, *p))
|
|
|
|
return 0;
|
2019-05-29 17:26:18 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
return 1;
|
2019-05-29 17:26:18 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
2019-09-30 17:06:27 -07:00
|
|
|
* Load a file's info into an entry_t and return it (if found).
|
|
|
|
* The returned entry must be free()ed by the caller.
|
|
|
|
* Warning: this does not deduplicate entries, and it's best if there aren't
|
|
|
|
* duplicate entries hanging around.
|
2019-05-28 22:29:22 -07:00
|
|
|
*/
|
2019-09-30 17:06:27 -07:00
|
|
|
entry_t* load_entry(bb_t *bb, const char *path, int clear_dots)
|
2019-05-22 19:05:56 -07:00
|
|
|
{
|
2019-09-30 17:06:27 -07:00
|
|
|
struct stat linkedstat, filestat;
|
|
|
|
if (!path || !path[0]) return NULL;
|
|
|
|
if (lstat(path, &filestat) == -1) return NULL;
|
2019-11-01 09:08:56 -07:00
|
|
|
char pbuf[PATH_MAX] = {0};
|
2019-11-01 08:46:39 -07:00
|
|
|
char *slash = strrchr(path, '/');
|
|
|
|
if (slash) {
|
|
|
|
strncpy(pbuf, path, (slash - path));
|
|
|
|
normalize_path(bb->path, pbuf, pbuf);
|
|
|
|
strcat(pbuf, slash);
|
|
|
|
} else {
|
|
|
|
strcpy(pbuf, bb->path);
|
|
|
|
strcat(pbuf, path);
|
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
if (pbuf[strlen(pbuf)-1] == '/' && pbuf[1])
|
|
|
|
pbuf[strlen(pbuf)-1] = '\0';
|
2019-05-23 01:37:41 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
// Check for pre-existing:
|
|
|
|
for (entry_t *e = bb->hash[(int)filestat.st_ino & HASH_MASK]; e; e = e->hash.next) {
|
|
|
|
if (e->info.st_ino == filestat.st_ino && e->info.st_dev == filestat.st_dev
|
|
|
|
// Need to check filename in case of hard links
|
|
|
|
&& strcmp(pbuf, e->fullname) == 0)
|
|
|
|
return e;
|
2019-05-23 01:37:41 -07:00
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
ssize_t linkpathlen = -1;
|
|
|
|
char linkbuf[PATH_MAX];
|
|
|
|
if (S_ISLNK(filestat.st_mode)) {
|
|
|
|
linkpathlen = readlink(pbuf, linkbuf, sizeof(linkbuf));
|
|
|
|
if (linkpathlen < 0) err("Couldn't read link: '%s'", pbuf);
|
|
|
|
linkbuf[linkpathlen] = '\0';
|
|
|
|
while (linkpathlen > 0 && linkbuf[linkpathlen-1] == '/') linkbuf[--linkpathlen] = '\0';
|
|
|
|
if (stat(pbuf, &linkedstat) == -1) memset(&linkedstat, 0, sizeof(linkedstat));
|
|
|
|
}
|
|
|
|
size_t pathlen = strlen(pbuf);
|
|
|
|
size_t entry_size = sizeof(entry_t) + (pathlen + 1) + (size_t)(linkpathlen + 1);
|
|
|
|
entry_t *entry = memcheck(calloc(entry_size, 1));
|
|
|
|
char *end = stpcpy(entry->fullname, pbuf);
|
|
|
|
if (linkpathlen >= 0)
|
|
|
|
entry->linkname = strcpy(end + 1, linkbuf);
|
|
|
|
if (strcmp(entry->fullname, "/") == 0) {
|
|
|
|
entry->name = entry->fullname;
|
|
|
|
} else {
|
|
|
|
entry->name = strrchr(entry->fullname, '/');
|
|
|
|
if (!entry->name) err("No slash found in '%s' from '%s'", entry->fullname, path);
|
|
|
|
++entry->name;
|
|
|
|
}
|
|
|
|
if (S_ISLNK(filestat.st_mode))
|
|
|
|
entry->linkedmode = linkedstat.st_mode;
|
|
|
|
entry->info = filestat;
|
2019-10-27 14:58:23 -07:00
|
|
|
LL_PREPEND(bb->hash[(int)filestat.st_ino & HASH_MASK], entry, hash);
|
2019-09-30 17:06:27 -07:00
|
|
|
entry->index = -1;
|
|
|
|
bb->hash[(int)filestat.st_ino & HASH_MASK] = entry;
|
|
|
|
return entry;
|
|
|
|
}
|
2019-05-22 19:05:56 -07:00
|
|
|
|
2019-10-13 18:33:46 -07:00
|
|
|
/*
|
|
|
|
* Return whether a string matches a command
|
2019-10-13 19:34:16 -07:00
|
|
|
* e.g. matches_cmd("sel:x", "select:") == 1, matches_cmd("q", "quit") == 1
|
2019-10-13 18:33:46 -07:00
|
|
|
*/
|
|
|
|
static inline int matches_cmd(const char *str, const char *cmd)
|
|
|
|
{
|
2019-10-13 19:34:16 -07:00
|
|
|
if ((strchr(cmd, ':') == NULL) != (strchr(str, ':') == NULL))
|
|
|
|
return 0;
|
|
|
|
while (*str == *cmd && *cmd && *cmd != ':') ++str, ++cmd;
|
2019-10-13 18:33:46 -07:00
|
|
|
return *str == '\0' || *str == ':';
|
|
|
|
}
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
/*
|
|
|
|
* Memory allocation failures are unrecoverable in bb, so this wrapper just
|
|
|
|
* prints an error message and exits if that happens.
|
|
|
|
*/
|
|
|
|
void* memcheck(void *p)
|
|
|
|
{
|
|
|
|
if (!p) err("Allocation failure");
|
|
|
|
return p;
|
|
|
|
}
|
2019-05-31 17:44:18 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
/*
|
|
|
|
* Prepend `root` to relative paths, replace "~" with $HOME.
|
|
|
|
* The normalized path is stored in `normalized`.
|
|
|
|
*/
|
2019-11-06 08:21:44 -08:00
|
|
|
char *normalize_path(const char *root, const char *path, char *normalized)
|
2019-09-30 17:06:27 -07:00
|
|
|
{
|
2019-11-01 08:46:39 -07:00
|
|
|
char pbuf[PATH_MAX] = {0};
|
2019-09-30 17:06:27 -07:00
|
|
|
if (path[0] == '~' && (path[1] == '\0' || path[1] == '/')) {
|
|
|
|
char *home;
|
2019-11-06 08:21:44 -08:00
|
|
|
if (!(home = getenv("HOME"))) return NULL;
|
2019-11-01 08:46:39 -07:00
|
|
|
strcpy(pbuf, home);
|
2019-09-30 17:06:27 -07:00
|
|
|
++path;
|
2019-11-01 08:46:39 -07:00
|
|
|
} else if (path[0] != '/') {
|
|
|
|
strcpy(pbuf, root);
|
|
|
|
if (root[strlen(root)-1] != '/') strcat(pbuf, "/");
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
2019-11-01 08:46:39 -07:00
|
|
|
strcat(pbuf, path);
|
2019-11-06 08:21:44 -08:00
|
|
|
if (realpath(pbuf, normalized) == NULL) {
|
2019-11-04 08:25:25 -08:00
|
|
|
strcpy(normalized, pbuf); // TODO: normalize better?
|
2019-11-06 08:21:44 -08:00
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
return normalized;
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
2019-05-22 19:05:56 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
/*
|
|
|
|
* Remove all the files currently stored in bb->files and if `bb->path` is
|
|
|
|
* non-NULL, update `bb` with a listing of the files in `path`
|
|
|
|
*/
|
2019-10-13 21:20:00 -07:00
|
|
|
int populate_files(bb_t *bb, const char *path)
|
2019-09-30 17:06:27 -07:00
|
|
|
{
|
2019-10-13 21:20:00 -07:00
|
|
|
int samedir = path && strcmp(bb->path, path) == 0;
|
|
|
|
int old_scroll = bb->scroll;
|
|
|
|
char old_selected[PATH_MAX] = "";
|
|
|
|
if (samedir && bb->nfiles > 0) strcpy(old_selected, bb->files[bb->cursor]->fullname);
|
|
|
|
|
2019-11-01 09:08:56 -07:00
|
|
|
char pbuf[PATH_MAX] = {0}, prev[PATH_MAX] = {0};
|
2019-10-13 21:20:00 -07:00
|
|
|
strcpy(prev, bb->path);
|
|
|
|
if (path == NULL) {
|
|
|
|
pbuf[0] = '\0';
|
|
|
|
} else if (strcmp(path, "<selection>") == 0) {
|
|
|
|
strcpy(pbuf, path);
|
|
|
|
} else if (strcmp(path, "..") == 0 && strcmp(bb->path, "<selection>") == 0) {
|
|
|
|
if (!bb->prev_path[0]) return -1;
|
|
|
|
strcpy(pbuf, bb->prev_path);
|
2019-11-06 08:21:44 -08:00
|
|
|
if (chdir(pbuf)) {
|
|
|
|
warn("Could not cd to: \"%s\"", pbuf);
|
|
|
|
return -1;
|
|
|
|
}
|
2019-10-13 21:20:00 -07:00
|
|
|
} else {
|
2019-11-06 08:21:44 -08:00
|
|
|
if (!normalize_path(bb->path, path, pbuf))
|
|
|
|
warn("Could not normalize path: \"%s\"", path);
|
2019-10-13 21:20:00 -07:00
|
|
|
if (pbuf[strlen(pbuf)-1] != '/')
|
|
|
|
strcat(pbuf, "/");
|
2019-11-06 08:21:44 -08:00
|
|
|
if (chdir(pbuf)) {
|
|
|
|
warn("Could not cd to: \"%s\"", pbuf);
|
|
|
|
return -1;
|
|
|
|
}
|
2019-10-13 21:20:00 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(bb->path, "<selection>") != 0) {
|
|
|
|
strcpy(bb->prev_path, prev);
|
|
|
|
setenv("BBPREVPATH", bb->prev_path, 1);
|
|
|
|
}
|
|
|
|
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 1;
|
2019-10-13 21:20:00 -07:00
|
|
|
strcpy(bb->path, pbuf);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
|
|
|
// Clear old files (if any)
|
|
|
|
if (bb->files) {
|
|
|
|
for (int i = 0; i < bb->nfiles; i++) {
|
|
|
|
bb->files[i]->index = -1;
|
|
|
|
try_free_entry(bb->files[i]);
|
|
|
|
bb->files[i] = NULL;
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
free(bb->files);
|
|
|
|
bb->files = NULL;
|
|
|
|
}
|
|
|
|
bb->nfiles = 0;
|
|
|
|
bb->cursor = 0;
|
|
|
|
bb->scroll = 0;
|
2019-05-29 17:26:18 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
if (!bb->path[0])
|
2019-10-13 21:20:00 -07:00
|
|
|
return 0;
|
2019-05-29 17:26:18 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
size_t space = 0;
|
|
|
|
if (strcmp(bb->path, "<selection>") == 0) {
|
2019-11-08 06:28:01 -08:00
|
|
|
for (entry_t *e = bb->selected; e; e = e->selected.next) {
|
2019-09-30 17:06:27 -07:00
|
|
|
if ((size_t)bb->nfiles + 1 > space)
|
|
|
|
bb->files = memcheck(realloc(bb->files, (space += 100)*sizeof(void*)));
|
|
|
|
e->index = bb->nfiles;
|
|
|
|
bb->files[bb->nfiles++] = e;
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
} else {
|
|
|
|
DIR *dir = opendir(bb->path);
|
|
|
|
if (!dir)
|
|
|
|
err("Couldn't open dir: %s", bb->path);
|
2019-05-22 19:05:56 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
for (struct dirent *dp; (dp = readdir(dir)) != NULL; ) {
|
|
|
|
if (dp->d_name[0] == '.') {
|
|
|
|
if (dp->d_name[1] == '.' && dp->d_name[2] == '\0') {
|
|
|
|
if (!bb->show_dotdot || strcmp(bb->path, "/") == 0) continue;
|
|
|
|
} else if (dp->d_name[1] == '\0') {
|
|
|
|
if (!bb->show_dot) continue;
|
|
|
|
} else if (!bb->show_dotfiles) continue;
|
2019-05-26 16:01:56 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
if ((size_t)bb->nfiles + 1 > space)
|
|
|
|
bb->files = memcheck(realloc(bb->files, (space += 100)*sizeof(void*)));
|
|
|
|
// Don't normalize path so we can get "." and ".."
|
|
|
|
entry_t *entry = load_entry(bb, dp->d_name, 0);
|
|
|
|
if (!entry) err("Failed to load entry: '%s'", dp->d_name);
|
|
|
|
entry->index = bb->nfiles;
|
|
|
|
bb->files[bb->nfiles++] = entry;
|
|
|
|
}
|
|
|
|
closedir(dir);
|
|
|
|
}
|
2019-05-29 17:26:18 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
for (int i = 0; i < bb->nfiles; i++) {
|
|
|
|
int j = rand() / (RAND_MAX / (i + 1)); // This is not optimal, but doesn't need to be
|
|
|
|
bb->files[i]->shufflepos = bb->files[j]->shufflepos;
|
|
|
|
bb->files[j]->shufflepos = i;
|
|
|
|
}
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
sort_files(bb);
|
|
|
|
if (samedir) {
|
|
|
|
set_scroll(bb, old_scroll);
|
2019-10-13 21:20:00 -07:00
|
|
|
if (old_selected[0]) {
|
|
|
|
entry_t *e = load_entry(bb, old_selected, 0);
|
|
|
|
if (e) set_cursor(bb, e->index);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
entry_t *p = load_entry(bb, prev, 0);
|
|
|
|
if (p) {
|
|
|
|
if (IS_VIEWED(p)) set_cursor(bb, p->index);
|
|
|
|
else try_free_entry(p);
|
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
2019-10-13 21:20:00 -07:00
|
|
|
return 0;
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
2019-09-30 15:46:24 -07:00
|
|
|
|
|
|
|
/*
|
2019-09-30 17:06:27 -07:00
|
|
|
* Print the current key bindings
|
2019-05-28 22:29:22 -07:00
|
|
|
*/
|
2019-09-30 17:06:27 -07:00
|
|
|
void print_bindings(int fd)
|
2019-05-25 04:30:51 -07:00
|
|
|
{
|
2019-09-30 17:06:27 -07:00
|
|
|
char buf[1000], buf2[1024];
|
2019-10-12 16:04:19 -07:00
|
|
|
for (int i = 0; bindings[i].script && i < sizeof(bindings)/sizeof(bindings[0]); i++) {
|
2019-09-30 17:06:27 -07:00
|
|
|
if (bindings[i].key == -1) {
|
|
|
|
const char *label = bindings[i].description;
|
2019-11-01 09:19:25 -07:00
|
|
|
sprintf(buf, "\n\033[33;1;4m\033[%dG%s\033[0m\n", (winsize.ws_col-(int)strlen(label))/2, label);
|
2019-09-30 17:06:27 -07:00
|
|
|
write(fd, buf, strlen(buf));
|
|
|
|
continue;
|
2019-05-28 23:51:20 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
int to_skip = -1;
|
|
|
|
char *p = buf;
|
2019-10-12 16:04:19 -07:00
|
|
|
for (int j = i; bindings[j].script && strcmp(bindings[j].description, bindings[i].description) == 0; j++) {
|
2019-09-30 17:06:27 -07:00
|
|
|
if (j > i) p = stpcpy(p, ", ");
|
|
|
|
++to_skip;
|
|
|
|
int key = bindings[j].key;
|
2019-10-12 16:04:19 -07:00
|
|
|
p = bkeyname(key, p);
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
*p = '\0';
|
2019-11-01 09:19:25 -07:00
|
|
|
sprintf(buf2, "\033[1m\033[%dG%s\033[0m", winsize.ws_col/2 - 1 - (int)strlen(buf), buf);
|
2019-09-30 17:06:27 -07:00
|
|
|
write(fd, buf2, strlen(buf2));
|
2019-11-01 09:19:25 -07:00
|
|
|
sprintf(buf2, "\033[1m\033[%dG\033[34m%s\033[0m", winsize.ws_col/2 + 1,
|
2019-09-30 17:06:27 -07:00
|
|
|
bindings[i].description);
|
|
|
|
write(fd, buf2, strlen(buf2));
|
|
|
|
write(fd, "\033[0m\n", strlen("\033[0m\n"));
|
|
|
|
i += to_skip;
|
2019-05-31 17:44:18 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
write(fd, "\n", 1);
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Run a bb internal command (e.g. "+refresh") and return an indicator of what
|
|
|
|
* needs to happen next.
|
|
|
|
*/
|
2019-10-13 19:34:16 -07:00
|
|
|
void run_bbcmd(bb_t *bb, const char *cmd)
|
2019-05-25 20:55:59 -07:00
|
|
|
{
|
2019-09-30 15:46:24 -07:00
|
|
|
if (cmd[0] == '+') ++cmd;
|
|
|
|
else if (strncmp(cmd, "bb +", 4) == 0) cmd = &cmd[4];
|
|
|
|
const char *value = strchr(cmd, ':');
|
2019-05-25 20:55:59 -07:00
|
|
|
if (value) ++value;
|
2019-05-30 00:33:51 -07:00
|
|
|
#define set_bool(target) do { if (!value) { target = !target; } else { target = value[0] == '1'; } } while (0)
|
2019-11-01 06:53:50 -07:00
|
|
|
if (matches_cmd(cmd, ".")) { // +.
|
2019-10-13 18:33:46 -07:00
|
|
|
set_bool(bb->show_dot);
|
2019-10-13 21:20:00 -07:00
|
|
|
populate_files(bb, bb->path);
|
2019-11-01 06:53:50 -07:00
|
|
|
} else if (matches_cmd(cmd, "..")) { // +..
|
|
|
|
set_bool(bb->show_dotdot);
|
|
|
|
populate_files(bb, bb->path);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "bind:")) { // +bind:<keys>:<script>
|
2019-10-13 18:33:46 -07:00
|
|
|
char *value_copy = memcheck(strdup(value));
|
|
|
|
char *keys = trim(value_copy);
|
2019-10-13 19:34:16 -07:00
|
|
|
if (!keys[0]) { free(value_copy); return; }
|
2019-10-13 18:33:46 -07:00
|
|
|
char *script = strchr(keys+1, ':');
|
2019-10-13 19:34:16 -07:00
|
|
|
if (!script) {
|
|
|
|
free(value_copy);
|
2019-10-30 05:43:49 -07:00
|
|
|
warn("No script provided.");
|
|
|
|
return;
|
2019-10-13 19:34:16 -07:00
|
|
|
}
|
2019-10-13 18:33:46 -07:00
|
|
|
*script = '\0';
|
|
|
|
script = trim(script + 1);
|
|
|
|
char *description;
|
|
|
|
if (script[0] == '#') {
|
|
|
|
description = trim(strsep(&script, "\n") + 1);
|
|
|
|
if (!script) script = "";
|
|
|
|
else script = trim(script);
|
|
|
|
} else description = script;
|
|
|
|
for (char *key; (key = strsep(&keys, ",")); ) {
|
|
|
|
int is_section = strcmp(key, "Section") == 0;
|
|
|
|
int keyval = bkeywithname(key);
|
|
|
|
if (keyval == -1 && !is_section) continue;
|
|
|
|
for (int i = 0; i < sizeof(bindings)/sizeof(bindings[0]); i++) {
|
|
|
|
if (bindings[i].script && (bindings[i].key != keyval || is_section))
|
|
|
|
continue;
|
|
|
|
binding_t binding = {keyval, memcheck(strdup(script)),
|
|
|
|
memcheck(strdup(description))};
|
|
|
|
if (bindings[i].key == keyval) {
|
|
|
|
free(bindings[i].description);
|
|
|
|
free(bindings[i].script);
|
|
|
|
for (; i + 1 < sizeof(bindings)/sizeof(bindings[0]) && bindings[i+1].key; i++)
|
|
|
|
bindings[i] = bindings[i+1];
|
2019-05-29 18:00:49 -07:00
|
|
|
}
|
2019-10-13 18:33:46 -07:00
|
|
|
bindings[i] = binding;
|
|
|
|
break;
|
2019-05-27 19:51:04 -07:00
|
|
|
}
|
2019-05-29 18:00:49 -07:00
|
|
|
}
|
2019-10-13 18:33:46 -07:00
|
|
|
free(value_copy);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "cd:")) { // +cd:
|
2019-11-01 06:50:44 -07:00
|
|
|
if (populate_files(bb, value))
|
2019-10-30 05:43:49 -07:00
|
|
|
warn("Could not open directory: \"%s\"", value);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "columns:")) { // +columns:
|
2019-10-13 18:33:46 -07:00
|
|
|
strncpy(bb->columns, value, MAX_COLS);
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 1;
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "deselect:")) { // +deselect
|
2019-10-13 18:33:46 -07:00
|
|
|
char pbuf[PATH_MAX];
|
2019-11-01 08:46:39 -07:00
|
|
|
normalize_path(bb->path, value, pbuf);
|
2019-10-13 18:33:46 -07:00
|
|
|
entry_t *e = load_entry(bb, pbuf, 0);
|
|
|
|
if (e) {
|
|
|
|
set_selected(bb, e, 0);
|
2019-10-13 19:34:16 -07:00
|
|
|
return;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-10-13 18:33:46 -07:00
|
|
|
// Filename may no longer exist:
|
2019-11-08 06:28:01 -08:00
|
|
|
for (e = bb->selected; e; e = e->selected.next) {
|
2019-10-13 18:33:46 -07:00
|
|
|
if (strcmp(e->fullname, pbuf) == 0) {
|
|
|
|
set_selected(bb, e, 0);
|
|
|
|
break;
|
2019-07-12 16:19:31 -07:00
|
|
|
}
|
|
|
|
}
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "deselect")) { // +deselect
|
2019-11-08 06:28:01 -08:00
|
|
|
while (bb->selected)
|
|
|
|
set_selected(bb, bb->selected, 0);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "dotfiles:") || matches_cmd(cmd, "dotfiles")) { // +dotfiles:
|
2019-10-13 18:33:46 -07:00
|
|
|
set_bool(bb->show_dotfiles);
|
2019-10-13 21:20:00 -07:00
|
|
|
setenv("BBDOTFILES", bb->show_dotfiles ? "1" : "", 1);
|
|
|
|
populate_files(bb, bb->path);
|
2019-10-27 14:58:23 -07:00
|
|
|
} else if (matches_cmd(cmd, "fg:") || matches_cmd(cmd, "fg")) { // +fg:
|
2019-11-01 09:15:38 -07:00
|
|
|
int nprocs = 0;
|
|
|
|
for (proc_t *p = running_procs; p; p = p->running.next) ++nprocs;
|
2019-10-27 14:58:23 -07:00
|
|
|
int fg = value ? nprocs - (int)strtol(value, NULL, 10) : 0;
|
|
|
|
proc_t *child = NULL;
|
|
|
|
for (proc_t *p = running_procs; p && !child; p = p->running.next) {
|
|
|
|
if (fg-- == 0) child = p;
|
|
|
|
}
|
|
|
|
if (!child) return;
|
2019-11-01 09:19:25 -07:00
|
|
|
move_cursor(tty_out, 0, winsize.ws_row-1);
|
2019-10-27 14:58:23 -07:00
|
|
|
fputs("\033[K", tty_out);
|
|
|
|
restore_term(&default_termios);
|
|
|
|
kill(-(child->pid), SIGCONT);
|
|
|
|
wait_for_process(&child);
|
|
|
|
init_term();
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 1;
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "goto:")) { // +goto:
|
2019-10-13 18:33:46 -07:00
|
|
|
entry_t *e = load_entry(bb, value, 1);
|
2019-10-30 05:43:49 -07:00
|
|
|
if (!e) {
|
2019-11-04 08:25:25 -08:00
|
|
|
warn("Could not find file to go to: \"%s\".", value);
|
2019-10-30 05:43:49 -07:00
|
|
|
return;
|
|
|
|
}
|
2019-10-13 18:33:46 -07:00
|
|
|
if (IS_VIEWED(e)) {
|
|
|
|
set_cursor(bb, e->index);
|
2019-10-13 19:34:16 -07:00
|
|
|
return;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-10-13 18:33:46 -07:00
|
|
|
char pbuf[PATH_MAX];
|
|
|
|
strcpy(pbuf, e->fullname);
|
|
|
|
char *lastslash = strrchr(pbuf, '/');
|
2019-10-13 19:34:16 -07:00
|
|
|
if (!lastslash) err("No slash found in filename: %s", pbuf);
|
2019-10-13 18:33:46 -07:00
|
|
|
*lastslash = '\0'; // Split in two
|
2019-10-13 21:20:00 -07:00
|
|
|
populate_files(bb, pbuf);
|
2019-10-13 18:33:46 -07:00
|
|
|
if (IS_VIEWED(e))
|
|
|
|
set_cursor(bb, e->index);
|
|
|
|
else try_free_entry(e);
|
|
|
|
} else if (matches_cmd(cmd, "help")) { // +help
|
2019-11-11 10:25:15 -08:00
|
|
|
char filename[256] = "/tmp/bbhelp.XXXXXX";
|
|
|
|
int fd = mkostemp(filename, O_WRONLY);
|
2019-11-11 10:29:56 -08:00
|
|
|
if (fd == -1) err("Couldn't create temporary help file in /tmp/");
|
2019-11-11 10:25:15 -08:00
|
|
|
print_bindings(fd);
|
|
|
|
close(fd);
|
|
|
|
char script[512] = "less -rKX < ";
|
|
|
|
strcat(script, filename);
|
|
|
|
run_script(bb, script);
|
2019-11-11 10:29:56 -08:00
|
|
|
if (unlink(filename) == -1) err("Couldn't delete temporary help file: '%s'", filename);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "interleave:") || matches_cmd(cmd, "interleave")) { // +interleave
|
2019-10-13 18:33:46 -07:00
|
|
|
set_bool(bb->interleave_dirs);
|
|
|
|
sort_files(bb);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "move:")) { // +move:
|
2019-10-13 18:33:46 -07:00
|
|
|
int oldcur, isdelta, n;
|
|
|
|
move:
|
2019-10-13 19:34:16 -07:00
|
|
|
if (bb->nfiles == 0) return;
|
2019-10-13 18:33:46 -07:00
|
|
|
oldcur = bb->cursor;
|
|
|
|
isdelta = value[0] == '-' || value[0] == '+';
|
|
|
|
n = (int)strtol(value, (char**)&value, 10);
|
|
|
|
if (*value == '%')
|
2019-11-01 09:19:25 -07:00
|
|
|
n = (n * (value[1] == 'n' ? bb->nfiles : winsize.ws_row)) / 100;
|
2019-10-13 18:33:46 -07:00
|
|
|
if (isdelta) set_cursor(bb, bb->cursor + n);
|
|
|
|
else set_cursor(bb, n);
|
2019-10-13 19:34:16 -07:00
|
|
|
if (matches_cmd(cmd, "spread:")) { // +spread:
|
2019-10-13 18:33:46 -07:00
|
|
|
int sel = IS_SELECTED(bb->files[oldcur]);
|
|
|
|
for (int i = bb->cursor; i != oldcur; i += (oldcur > i ? 1 : -1))
|
|
|
|
set_selected(bb, bb->files[i], sel);
|
2019-10-02 13:14:28 -07:00
|
|
|
}
|
2019-10-13 18:33:46 -07:00
|
|
|
} else if (matches_cmd(cmd, "quit")) { // +quit
|
|
|
|
bb->should_quit = 1;
|
|
|
|
} else if (matches_cmd(cmd, "refresh")) { // +refresh
|
2019-10-13 21:20:00 -07:00
|
|
|
populate_files(bb, bb->path);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "scroll:")) { // +scroll:
|
2019-10-13 18:33:46 -07:00
|
|
|
// TODO: figure out the best version of this
|
|
|
|
int isdelta = value[0] == '+' || value[0] == '-';
|
|
|
|
int n = (int)strtol(value, (char**)&value, 10);
|
|
|
|
if (*value == '%')
|
2019-11-01 09:19:25 -07:00
|
|
|
n = (n * (value[1] == 'n' ? bb->nfiles : winsize.ws_row)) / 100;
|
2019-10-13 18:33:46 -07:00
|
|
|
if (isdelta)
|
|
|
|
set_scroll(bb, bb->scroll + n);
|
|
|
|
else
|
|
|
|
set_scroll(bb, n);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "select:") || matches_cmd(cmd, "select")) { // +select:
|
2019-11-08 06:28:01 -08:00
|
|
|
if (!value) {
|
|
|
|
for (int i = 0; i < bb->nfiles; i++)
|
|
|
|
set_selected(bb, bb->files[i], 1);
|
|
|
|
} else {
|
|
|
|
entry_t *e = load_entry(bb, value, 1);
|
|
|
|
if (e) set_selected(bb, e, 1);
|
|
|
|
}
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "sort:")) { // +sort:
|
2019-10-13 18:33:46 -07:00
|
|
|
set_sort(bb, value);
|
|
|
|
sort_files(bb);
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "spread:")) { // +spread:
|
2019-10-13 18:33:46 -07:00
|
|
|
goto move;
|
2019-10-13 19:34:16 -07:00
|
|
|
} else if (matches_cmd(cmd, "toggle:") || matches_cmd(cmd, "toggle")) { // +toggle
|
|
|
|
if (!value && !bb->nfiles) return;
|
2019-10-13 18:33:46 -07:00
|
|
|
if (!value) value = bb->files[bb->cursor]->fullname;
|
|
|
|
entry_t *e = load_entry(bb, value, 1);
|
2019-10-30 05:43:49 -07:00
|
|
|
if (!e) {
|
2019-11-04 08:25:25 -08:00
|
|
|
warn("Could not find file to toggle: \"%s\".", value);
|
2019-10-30 05:43:49 -07:00
|
|
|
return;
|
|
|
|
}
|
2019-10-13 19:34:16 -07:00
|
|
|
set_selected(bb, e, !IS_SELECTED(e));
|
2019-10-13 18:33:46 -07:00
|
|
|
} else {
|
2019-10-27 14:58:23 -07:00
|
|
|
warn("Invalid bb command: +%s.", cmd);
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Draw everything to the screen.
|
2019-11-01 06:50:44 -07:00
|
|
|
* If `dirty` is false, then use terminal scrolling to move the file listing
|
2019-09-30 17:06:27 -07:00
|
|
|
* around and only update the files that have changed.
|
|
|
|
*/
|
|
|
|
void render(bb_t *bb)
|
|
|
|
{
|
|
|
|
static int lastcursor = -1, lastscroll = -1;
|
|
|
|
char buf[64];
|
|
|
|
|
2019-11-01 06:50:44 -07:00
|
|
|
if (!dirty) {
|
2019-09-30 17:06:27 -07:00
|
|
|
// Use terminal scrolling:
|
|
|
|
if (lastscroll > bb->scroll) {
|
2019-11-01 09:19:25 -07:00
|
|
|
fprintf(tty_out, "\033[3;%dr\033[%dT\033[1;%dr", winsize.ws_row-1, lastscroll - bb->scroll, winsize.ws_row);
|
2019-09-30 17:06:27 -07:00
|
|
|
} else if (lastscroll < bb->scroll) {
|
2019-11-01 09:19:25 -07:00
|
|
|
fprintf(tty_out, "\033[3;%dr\033[%dS\033[1;%dr", winsize.ws_row-1, bb->scroll - lastscroll, winsize.ws_row);
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-01 06:50:44 -07:00
|
|
|
if (dirty) {
|
2019-09-30 17:06:27 -07:00
|
|
|
// Path
|
|
|
|
move_cursor(tty_out, 0, 0);
|
|
|
|
const char *color = TITLE_COLOR;
|
|
|
|
fputs(color, tty_out);
|
|
|
|
fputs_escaped(tty_out, bb->path, color);
|
|
|
|
fputs(" \033[K\033[0m", tty_out);
|
|
|
|
|
|
|
|
static const char *help = "Press '?' to see key bindings ";
|
2019-11-01 09:19:25 -07:00
|
|
|
move_cursor(tty_out, MAX(0, winsize.ws_col - (int)strlen(help)), 0);
|
2019-09-30 17:06:27 -07:00
|
|
|
fputs(help, tty_out);
|
|
|
|
fputs("\033[K\033[0m", tty_out);
|
|
|
|
|
|
|
|
// Columns
|
|
|
|
move_cursor(tty_out, 0, 1);
|
|
|
|
fputs("\033[0;44;30m\033[K", tty_out);
|
|
|
|
int x = 0;
|
|
|
|
for (int col = 0; bb->columns[col]; col++) {
|
|
|
|
const char *title = columns[(int)bb->columns[col]].name;
|
|
|
|
if (!title) title = "";
|
|
|
|
move_cursor(tty_out, x, 1);
|
|
|
|
if (col > 0) {
|
|
|
|
fputs("│\033[K", tty_out);
|
|
|
|
x += 1;
|
|
|
|
}
|
|
|
|
const char *indicator = " ";
|
|
|
|
if (bb->columns[col] == bb->sort[1])
|
|
|
|
indicator = bb->sort[0] == '-' ? RSORT_INDICATOR : SORT_INDICATOR;
|
|
|
|
move_cursor(tty_out, x, 1);
|
|
|
|
fputs(indicator, tty_out);
|
|
|
|
fputs(title, tty_out);
|
|
|
|
x += columns[(int)bb->columns[col]].width;
|
|
|
|
}
|
|
|
|
fputs(" \033[K\033[0m", tty_out);
|
|
|
|
}
|
|
|
|
|
|
|
|
entry_t **files = bb->files;
|
|
|
|
for (int i = bb->scroll; i < bb->scroll + ONSCREEN; i++) {
|
2019-11-01 06:50:44 -07:00
|
|
|
if (!dirty) {
|
2019-09-30 17:06:27 -07:00
|
|
|
if (i == bb->cursor || i == lastcursor)
|
|
|
|
goto do_render;
|
|
|
|
if (i < lastscroll || i >= lastscroll + ONSCREEN)
|
|
|
|
goto do_render;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
int y;
|
|
|
|
do_render:
|
|
|
|
y = i - bb->scroll + 2;
|
|
|
|
move_cursor(tty_out, 0, y);
|
|
|
|
|
|
|
|
if (i == bb->scroll && bb->nfiles == 0) {
|
|
|
|
const char *s = "...no files here...";
|
|
|
|
fprintf(tty_out, "\033[37;2m%s\033[0m\033[K\033[J", s);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (i >= bb->nfiles) {
|
|
|
|
fputs("\033[J", tty_out);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
entry_t *entry = files[i];
|
|
|
|
if (i == bb->cursor) fputs(CURSOR_COLOR, tty_out);
|
|
|
|
|
|
|
|
int use_fullname = strcmp(bb->path, "<selection>") == 0;
|
|
|
|
int x = 0;
|
|
|
|
for (int col = 0; bb->columns[col]; col++) {
|
|
|
|
fprintf(tty_out, "\033[%d;%dH\033[K", y+1, x+1);
|
|
|
|
if (col > 0) {
|
|
|
|
if (i == bb->cursor) fputs("│", tty_out);
|
|
|
|
else fputs("\033[37;2m│\033[22m", tty_out);
|
|
|
|
fputs(i == bb->cursor ? CURSOR_COLOR : "\033[0m", tty_out);
|
|
|
|
x += 1;
|
|
|
|
}
|
|
|
|
move_cursor(tty_out, x, y);
|
|
|
|
switch (bb->columns[col]) {
|
|
|
|
case COL_SELECTED:
|
|
|
|
fputs(IS_SELECTED(entry) ? SELECTED_INDICATOR : NOT_SELECTED_INDICATOR, tty_out);
|
|
|
|
fputs(i == bb->cursor ? CURSOR_COLOR : "\033[0m", tty_out);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case COL_RANDOM: {
|
|
|
|
double k = (double)entry->shufflepos/(double)bb->nfiles;
|
|
|
|
int color = (int)(k*232 + (1.-k)*255);
|
|
|
|
fprintf(tty_out, "\033[48;5;%dm \033[0m%s", color,
|
|
|
|
i == bb->cursor ? CURSOR_COLOR : "\033[0m");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case COL_SIZE: {
|
|
|
|
int j = 0;
|
|
|
|
const char* units = "BKMGTPEZY";
|
|
|
|
double bytes = (double)entry->info.st_size;
|
|
|
|
while (bytes > 1024) {
|
|
|
|
bytes /= 1024;
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
fprintf(tty_out, " %6.*f%c ", j > 0 ? 1 : 0, bytes, units[j]);
|
|
|
|
break;
|
2019-06-15 16:20:38 -07:00
|
|
|
}
|
2019-05-29 18:00:49 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
case COL_MTIME:
|
|
|
|
strftime(buf, sizeof(buf), " %I:%M%p %b %e %Y ", localtime(&(entry->info.st_mtime)));
|
|
|
|
fputs(buf, tty_out);
|
|
|
|
break;
|
2019-05-25 20:55:59 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
case COL_CTIME:
|
|
|
|
strftime(buf, sizeof(buf), " %I:%M%p %b %e %Y ", localtime(&(entry->info.st_ctime)));
|
|
|
|
fputs(buf, tty_out);
|
|
|
|
break;
|
2019-05-25 04:30:51 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
case COL_ATIME:
|
|
|
|
strftime(buf, sizeof(buf), " %I:%M%p %b %e %Y ", localtime(&(entry->info.st_atime)));
|
|
|
|
fputs(buf, tty_out);
|
|
|
|
break;
|
2019-05-31 01:17:23 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
case COL_PERM:
|
|
|
|
fprintf(tty_out, " %03o", entry->info.st_mode & 0777);
|
|
|
|
break;
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
case COL_NAME: {
|
|
|
|
char color[128];
|
|
|
|
strcpy(color, color_of(entry->info.st_mode));
|
|
|
|
if (i == bb->cursor) strcat(color, CURSOR_COLOR);
|
|
|
|
fputs(color, tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
char *name = use_fullname ? entry->fullname : entry->name;
|
|
|
|
if (entry->no_esc) fputs(name, tty_out);
|
|
|
|
else entry->no_esc |= !fputs_escaped(tty_out, name, color);
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
if (E_ISDIR(entry)) fputs("/", tty_out);
|
2019-05-24 16:55:36 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
if (entry->linkname) {
|
|
|
|
if (i != bb->cursor)
|
|
|
|
fputs("\033[37m", tty_out);
|
|
|
|
fputs("\033[2m -> \033[3m", tty_out);
|
|
|
|
strcpy(color, color_of(entry->linkedmode));
|
|
|
|
if (i == bb->cursor) strcat(color, CURSOR_COLOR);
|
|
|
|
strcat(color, "\033[3;2m");
|
|
|
|
fputs(color, tty_out);
|
|
|
|
if (entry->link_no_esc) fputs(entry->linkname, tty_out);
|
|
|
|
else entry->link_no_esc |= !fputs_escaped(tty_out, entry->linkname, color);
|
2019-05-26 13:41:09 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
if (S_ISDIR(entry->linkedmode))
|
|
|
|
fputs("/", tty_out);
|
2019-05-22 00:25:25 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
fputs("\033[22;23m", tty_out);
|
|
|
|
}
|
|
|
|
fputs(i == bb->cursor ? CURSOR_COLOR : "\033[0m", tty_out);
|
|
|
|
fputs("\033[K", tty_out);
|
2019-05-29 00:56:49 -07:00
|
|
|
break;
|
2019-05-23 04:44:48 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
default: break;
|
2019-05-29 00:56:49 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
x += columns[(int)bb->columns[col]].width;
|
|
|
|
}
|
|
|
|
fputs(" \033[K\033[0m", tty_out); // Reset color and attributes
|
|
|
|
}
|
|
|
|
|
2019-11-01 09:19:25 -07:00
|
|
|
move_cursor(tty_out, winsize.ws_col/2, winsize.ws_row - 1);
|
2019-10-27 14:58:23 -07:00
|
|
|
fputs("\033[0m\033[K", tty_out);
|
2019-11-01 09:19:25 -07:00
|
|
|
int x = winsize.ws_col;
|
2019-11-08 06:28:01 -08:00
|
|
|
if (bb->selected) { // Number of selected files
|
2019-09-30 17:06:27 -07:00
|
|
|
int n = 0;
|
2019-11-08 06:28:01 -08:00
|
|
|
for (entry_t *s = bb->selected; s; s = s->selected.next) ++n;
|
2019-10-27 14:58:23 -07:00
|
|
|
x -= 14;
|
2019-09-30 17:06:27 -07:00
|
|
|
for (int k = n; k; k /= 10) x--;
|
2019-11-01 09:19:25 -07:00
|
|
|
move_cursor(tty_out, MAX(0, x), winsize.ws_row - 1);
|
2019-09-30 17:06:27 -07:00
|
|
|
fprintf(tty_out, "\033[41;30m %d Selected \033[0m", n);
|
|
|
|
}
|
2019-11-01 09:15:38 -07:00
|
|
|
int nprocs = 0;
|
|
|
|
for (proc_t *p = running_procs; p; p = p->running.next) ++nprocs;
|
2019-10-27 14:58:23 -07:00
|
|
|
if (nprocs > 0) { // Number of suspended processes
|
|
|
|
x -= 13;
|
|
|
|
for (int k = nprocs; k; k /= 10) x--;
|
2019-11-01 09:19:25 -07:00
|
|
|
move_cursor(tty_out, MAX(0, x), winsize.ws_row - 1);
|
2019-10-27 14:58:23 -07:00
|
|
|
fprintf(tty_out, "\033[44;30m %d Suspended \033[0m", nprocs);
|
|
|
|
}
|
2019-11-01 09:19:25 -07:00
|
|
|
move_cursor(tty_out, winsize.ws_col/2, winsize.ws_row - 1);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
|
|
|
lastcursor = bb->cursor;
|
|
|
|
lastscroll = bb->scroll;
|
|
|
|
fflush(tty_out);
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 0;
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Close the /dev/tty terminals and restore some of the attributes.
|
|
|
|
*/
|
|
|
|
void restore_term(const struct termios *term)
|
|
|
|
{
|
2019-10-27 14:58:23 -07:00
|
|
|
tcsetattr(fileno(tty_out), TCSANOW, term);
|
|
|
|
fputs(T_LEAVE_BBMODE_PARTIAL, tty_out);
|
|
|
|
fflush(tty_out);
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Run a shell script with the selected files passed as sequential arguments to
|
|
|
|
* the script (or pass the cursor file if none are selected).
|
|
|
|
* Return the exit status of the script.
|
|
|
|
*/
|
|
|
|
int run_script(bb_t *bb, const char *cmd)
|
|
|
|
{
|
|
|
|
char *fullcmd = calloc(strlen(cmd) + strlen(bbcmdfn) + 1, sizeof(char));
|
|
|
|
strcpy(fullcmd, bbcmdfn);
|
|
|
|
strcat(fullcmd, cmd);
|
|
|
|
|
2019-10-27 14:58:23 -07:00
|
|
|
proc_t *proc = memcheck(calloc(1, sizeof(proc_t)));
|
|
|
|
if ((proc->pid = fork()) == 0) {
|
|
|
|
fclose(tty_out); tty_out = NULL;
|
|
|
|
fclose(tty_in); tty_in = NULL;
|
|
|
|
setpgid(0, 0);
|
2019-11-08 06:28:01 -08:00
|
|
|
char **args = memcheck(calloc(4 + bb->nselected + 1, sizeof(char*)));
|
|
|
|
int i = 0;
|
2019-09-30 17:06:27 -07:00
|
|
|
args[i++] = SH;
|
|
|
|
args[i++] = "-c";
|
|
|
|
args[i++] = fullcmd;
|
|
|
|
args[i++] = "--"; // ensure files like "-i" are not interpreted as flags for sh
|
2019-11-08 06:28:01 -08:00
|
|
|
// bb->selected is in most-recent order, so populate args in reverse to make sure
|
|
|
|
// that $1 is the first selected, etc.
|
|
|
|
i += bb->nselected;
|
|
|
|
for (entry_t *e = bb->selected; e; e = e->selected.next)
|
|
|
|
args[--i] = e->fullname;
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
setenv("BBCURSOR", bb->nfiles ? bb->files[bb->cursor]->fullname : "", 1);
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
int ttyout, ttyin;
|
|
|
|
ttyout = open("/dev/tty", O_RDWR);
|
|
|
|
ttyin = open("/dev/tty", O_RDONLY);
|
|
|
|
dup2(ttyout, STDOUT_FILENO);
|
|
|
|
dup2(ttyin, STDIN_FILENO);
|
|
|
|
execvp(SH, args);
|
|
|
|
err("Failed to execute command: '%s'", cmd);
|
|
|
|
return -1;
|
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-10-27 14:58:23 -07:00
|
|
|
if (proc->pid == -1)
|
2019-09-30 17:06:27 -07:00
|
|
|
err("Failed to fork");
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-10-27 14:58:23 -07:00
|
|
|
LL_PREPEND(running_procs, proc, running);
|
|
|
|
int status = wait_for_process(&proc);
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 1;
|
2019-09-30 17:06:27 -07:00
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set bb's file cursor to the given index (and adjust the scroll as necessary)
|
|
|
|
*/
|
|
|
|
void set_cursor(bb_t *bb, int newcur)
|
|
|
|
{
|
|
|
|
int oldcur = bb->cursor;
|
|
|
|
if (newcur > bb->nfiles - 1) newcur = bb->nfiles - 1;
|
|
|
|
if (newcur < 0) newcur = 0;
|
|
|
|
bb->cursor = newcur;
|
|
|
|
if (bb->nfiles <= ONSCREEN) {
|
|
|
|
bb->scroll = 0;
|
|
|
|
return;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
if (oldcur < bb->cursor) {
|
|
|
|
if (bb->scroll > bb->cursor)
|
|
|
|
bb->scroll = MAX(0, bb->cursor);
|
|
|
|
else if (bb->scroll < bb->cursor - ONSCREEN + 1 + SCROLLOFF)
|
|
|
|
bb->scroll = MIN(bb->nfiles - 1 - ONSCREEN + 1,
|
|
|
|
bb->scroll + (newcur - oldcur));
|
|
|
|
} else {
|
|
|
|
if (bb->scroll > bb->cursor - SCROLLOFF)
|
|
|
|
bb->scroll = MAX(0, bb->scroll + (newcur - oldcur));//bb->cursor - SCROLLOFF);
|
|
|
|
else if (bb->scroll < bb->cursor - ONSCREEN + 1)
|
|
|
|
bb->scroll = MIN(bb->cursor - ONSCREEN + 1,
|
|
|
|
bb->nfiles - 1 - ONSCREEN + 1);
|
2019-05-31 17:44:18 -07:00
|
|
|
}
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
2019-09-30 17:06:27 -07:00
|
|
|
* Set bb's scroll to the given index (and adjust the cursor as necessary)
|
2019-05-28 22:29:22 -07:00
|
|
|
*/
|
2019-09-30 17:06:27 -07:00
|
|
|
void set_scroll(bb_t *bb, int newscroll)
|
2019-05-23 03:40:04 -07:00
|
|
|
{
|
2019-09-30 17:06:27 -07:00
|
|
|
int delta = newscroll - bb->scroll;
|
|
|
|
if (bb->nfiles <= ONSCREEN) {
|
|
|
|
newscroll = 0;
|
|
|
|
} else {
|
|
|
|
if (newscroll > bb->nfiles - 1 - ONSCREEN + 1)
|
|
|
|
newscroll = bb->nfiles - 1 - ONSCREEN + 1;
|
|
|
|
if (newscroll < 0) newscroll = 0;
|
2019-05-23 03:40:04 -07:00
|
|
|
}
|
2019-09-30 17:06:27 -07:00
|
|
|
|
|
|
|
bb->scroll = newscroll;
|
|
|
|
bb->cursor += delta;
|
|
|
|
if (bb->cursor > bb->nfiles - 1) bb->cursor = bb->nfiles - 1;
|
|
|
|
if (bb->cursor < 0) bb->cursor = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Select or deselect a file.
|
|
|
|
*/
|
|
|
|
void set_selected(bb_t *bb, entry_t *e, int selected)
|
|
|
|
{
|
|
|
|
if (IS_SELECTED(e) == selected) return;
|
|
|
|
|
|
|
|
if (bb->nfiles > 0 && e != bb->files[bb->cursor])
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 1;
|
2019-09-30 17:06:27 -07:00
|
|
|
|
|
|
|
if (selected) {
|
2019-11-08 06:28:01 -08:00
|
|
|
LL_PREPEND(bb->selected, e, selected);
|
|
|
|
++bb->nselected;
|
2019-09-30 17:06:27 -07:00
|
|
|
} else {
|
2019-10-27 14:58:23 -07:00
|
|
|
LL_REMOVE(e, selected);
|
2019-09-30 17:06:27 -07:00
|
|
|
try_free_entry(e);
|
2019-11-08 06:28:01 -08:00
|
|
|
--bb->nselected;
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Set the sorting method used by bb to display files.
|
|
|
|
*/
|
|
|
|
void set_sort(bb_t *bb, const char *sort)
|
|
|
|
{
|
|
|
|
char sortbuf[strlen(sort)+1];
|
|
|
|
strcpy(sortbuf, sort);
|
|
|
|
for (char *s = sortbuf; s[0] && s[1]; s += 2) {
|
|
|
|
char *found;
|
|
|
|
if ((found = strchr(bb->sort, s[1]))) {
|
|
|
|
if (*s == '~')
|
|
|
|
*s = found[-1] == '+' && found == &bb->sort[1] ? '-' : '+';
|
|
|
|
memmove(found-1, found+1, strlen(found+1)+1);
|
|
|
|
} else if (*s == '~')
|
|
|
|
*s = '+';
|
|
|
|
}
|
|
|
|
size_t len = MIN(MAX_SORT, strlen(sort));
|
|
|
|
memmove(bb->sort + len, bb->sort, MAX_SORT+1 - len);
|
|
|
|
memmove(bb->sort, sortbuf, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the given entry is not viewed or selected, remove it from the
|
|
|
|
* hash, free it, and return 1.
|
|
|
|
*/
|
|
|
|
int try_free_entry(entry_t *e)
|
|
|
|
{
|
|
|
|
if (IS_SELECTED(e) || IS_VIEWED(e)) return 0;
|
2019-10-27 14:58:23 -07:00
|
|
|
LL_REMOVE(e, hash);
|
2019-09-30 17:06:27 -07:00
|
|
|
free(e);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Sort the files in bb according to bb's settings.
|
|
|
|
*/
|
|
|
|
void sort_files(bb_t *bb)
|
|
|
|
{
|
|
|
|
#ifdef __APPLE__
|
|
|
|
qsort_r(bb->files, (size_t)bb->nfiles, sizeof(entry_t*), bb, compare_files);
|
|
|
|
#else
|
|
|
|
qsort_r(bb->files, (size_t)bb->nfiles, sizeof(entry_t*), compare_files, bb);
|
|
|
|
#endif
|
|
|
|
for (int i = 0; i < bb->nfiles; i++)
|
|
|
|
bb->files[i]->index = i;
|
2019-11-01 06:50:44 -07:00
|
|
|
dirty = 1;
|
2019-09-30 17:06:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Trim trailing whitespace by inserting '\0' and return a pointer to after the
|
|
|
|
* first non-whitespace char
|
|
|
|
*/
|
|
|
|
static char *trim(char *s)
|
|
|
|
{
|
|
|
|
if (!s) return NULL;
|
|
|
|
while (*s == ' ' || *s == '\n') ++s;
|
|
|
|
char *end;
|
|
|
|
for (end = &s[strlen(s)-1]; end >= s && (*end == ' ' || *end == '\n'); end--)
|
|
|
|
*end = '\0';
|
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Hanlder for SIGWINCH events
|
|
|
|
*/
|
|
|
|
void update_term_size(int sig)
|
|
|
|
{
|
|
|
|
(void)sig;
|
2019-10-27 14:58:23 -07:00
|
|
|
if (tty_in) {
|
2019-11-01 09:19:25 -07:00
|
|
|
struct winsize oldsize = winsize;
|
|
|
|
ioctl(fileno(tty_in), TIOCGWINSZ, &winsize);
|
|
|
|
dirty |= (oldsize.ws_col != winsize.ws_col || oldsize.ws_row != winsize.ws_row);
|
2019-10-27 14:58:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Wait for a process to either suspend or exit and return the status.
|
|
|
|
*/
|
|
|
|
int wait_for_process(proc_t **proc)
|
|
|
|
{
|
|
|
|
signal(SIGTTOU, SIG_IGN);
|
|
|
|
tcsetpgrp(fileno(tty_out), (*proc)->pid);
|
|
|
|
int status;
|
|
|
|
while (waitpid((*proc)->pid, &status, WUNTRACED) < 0 && errno == EINTR) // Can happen, e.g. if process sends SIGTSTP
|
|
|
|
continue;
|
|
|
|
tcsetpgrp(fileno(tty_out), getpgid(0));
|
|
|
|
signal(SIGTTOU, SIG_DFL);
|
|
|
|
if (!WIFSTOPPED(status)) {
|
|
|
|
LL_REMOVE((*proc), running);
|
|
|
|
free(*proc);
|
|
|
|
*proc = NULL;
|
|
|
|
}
|
|
|
|
return status;
|
2019-05-23 03:40:04 -07:00
|
|
|
}
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2019-10-13 18:43:07 -07:00
|
|
|
char *initial_path = NULL, depthstr[16] = {0};
|
2019-05-22 01:39:15 -07:00
|
|
|
char sep = '\n';
|
2019-05-21 04:06:50 -07:00
|
|
|
int print_dir = 0, print_selection = 0;
|
2019-05-25 04:30:51 -07:00
|
|
|
|
2019-05-21 04:06:50 -07:00
|
|
|
for (int i = 1; i < argc; i++) {
|
2019-10-03 13:26:31 -07:00
|
|
|
// Commands are processed below, after flags have been parsed
|
2019-05-23 19:04:17 -07:00
|
|
|
if (argv[i][0] == '+') {
|
2019-06-12 15:11:29 -07:00
|
|
|
char *colon = strchr(argv[i], ':');
|
2019-10-03 13:26:31 -07:00
|
|
|
if (colon && !colon[1])
|
|
|
|
break;
|
|
|
|
} else if (strcmp(argv[i], "--") == 0) {
|
2019-05-23 19:04:17 -07:00
|
|
|
if (i + 1 < argc) initial_path = argv[i+1];
|
2019-10-03 13:26:31 -07:00
|
|
|
if (i + 2 < argc) {
|
|
|
|
printf("Extra arguments after %s\n%s", argv[i+1], usage_str);
|
|
|
|
return 1;
|
2019-06-12 15:11:29 -07:00
|
|
|
}
|
2019-10-03 13:26:31 -07:00
|
|
|
break;
|
|
|
|
} else if (strcmp(argv[i], "--help") == 0) {
|
|
|
|
help:
|
|
|
|
printf("%s%s", description_str, usage_str);
|
2019-05-25 23:31:03 -07:00
|
|
|
return 0;
|
2019-10-03 13:26:31 -07:00
|
|
|
} else if (strcmp(argv[i], "--version") == 0) {
|
2019-05-25 23:31:03 -07:00
|
|
|
version:
|
|
|
|
printf("bb " BB_VERSION "\n");
|
|
|
|
return 0;
|
2019-10-03 13:26:31 -07:00
|
|
|
} else if (argv[i][0] == '-' && argv[i][1] == '-') {
|
2019-05-23 05:42:33 -07:00
|
|
|
if (argv[i][2] == '\0') break;
|
2019-10-03 13:26:31 -07:00
|
|
|
printf("Unknown command line argument: %s\n%s", argv[i], usage_str);
|
|
|
|
return 1;
|
|
|
|
} else if (argv[i][0] == '-') {
|
2019-05-22 01:50:46 -07:00
|
|
|
for (char *c = &argv[i][1]; *c; c++) {
|
|
|
|
switch (*c) {
|
2019-10-03 13:26:31 -07:00
|
|
|
case 'h': goto help;
|
2019-05-25 23:31:03 -07:00
|
|
|
case 'v': goto version;
|
2019-10-03 13:26:31 -07:00
|
|
|
case 'd': print_dir = 1; break;
|
|
|
|
case '0': sep = '\0'; break;
|
|
|
|
case 's': print_selection = 1; break;
|
|
|
|
default: printf("Unknown command line argument: -%c\n%s", *c, usage_str);
|
|
|
|
return 1;
|
2019-05-22 01:50:46 -07:00
|
|
|
}
|
|
|
|
}
|
2019-10-03 13:26:31 -07:00
|
|
|
} else if (!initial_path) {
|
|
|
|
initial_path = argv[i];
|
|
|
|
} else {
|
|
|
|
printf("Unknown command line argument: %s\n%s", argv[i], usage_str);
|
|
|
|
return 1;
|
2019-05-22 01:50:46 -07:00
|
|
|
}
|
2019-05-21 04:06:50 -07:00
|
|
|
}
|
2019-10-03 13:26:31 -07:00
|
|
|
if (!initial_path) initial_path = ".";
|
2019-05-24 16:55:36 -07:00
|
|
|
|
2019-10-03 13:26:31 -07:00
|
|
|
cmdfilename = memcheck(strdup(CMDFILE_FORMAT));
|
|
|
|
int cmdfd;
|
|
|
|
if ((cmdfd = mkostemp(cmdfilename, O_APPEND)) == -1)
|
|
|
|
err("Couldn't create tmpfile: '%s'", CMDFILE_FORMAT);
|
2019-10-13 21:20:00 -07:00
|
|
|
|
2019-10-03 13:26:31 -07:00
|
|
|
// Set up environment variables
|
2019-10-13 21:20:00 -07:00
|
|
|
// Default values
|
|
|
|
setenv("SHELL", "bash", 0);
|
|
|
|
setenv("EDITOR", "nano", 0);
|
2019-10-03 13:26:31 -07:00
|
|
|
char *curdepth = getenv("BB_DEPTH");
|
|
|
|
int depth = curdepth ? atoi(curdepth) : 0;
|
|
|
|
sprintf(depthstr, "%d", depth + 1);
|
|
|
|
setenv("BB_DEPTH", depthstr, 1);
|
|
|
|
setenv("BBCMD", cmdfilename, 1);
|
2019-10-13 21:20:00 -07:00
|
|
|
setenv("BBSHELLFUNC", bbcmdfn, 1);
|
|
|
|
char full_initial_path[PATH_MAX];
|
|
|
|
getcwd(full_initial_path, PATH_MAX);
|
2019-11-01 08:46:39 -07:00
|
|
|
normalize_path(full_initial_path, initial_path, full_initial_path);
|
|
|
|
struct stat path_stat;
|
|
|
|
if (stat(full_initial_path, &path_stat) != 0)
|
2019-11-04 08:25:25 -08:00
|
|
|
err("Could not find initial path: \"%s\"", initial_path);
|
2019-11-01 08:46:39 -07:00
|
|
|
if (S_ISDIR(path_stat.st_mode)) {
|
|
|
|
if (strcmp(full_initial_path, "/") != 0) strcat(full_initial_path, "/");
|
|
|
|
} else {
|
|
|
|
write(cmdfd, "goto:", 5);
|
|
|
|
write(cmdfd, full_initial_path, strlen(full_initial_path) + 1); // Include null byte
|
|
|
|
char *slash = strrchr(full_initial_path, '/');
|
|
|
|
*slash = '\0';
|
|
|
|
}
|
2019-10-13 21:20:00 -07:00
|
|
|
setenv("BBINITIALPATH", full_initial_path, 1);
|
|
|
|
|
2019-10-27 14:58:23 -07:00
|
|
|
tty_in = fopen("/dev/tty", "r");
|
|
|
|
tty_out = fopen("/dev/tty", "w");
|
|
|
|
tcgetattr(fileno(tty_out), &orig_termios);
|
|
|
|
memcpy(&bb_termios, &orig_termios, sizeof(bb_termios));
|
|
|
|
cfmakeraw(&bb_termios);
|
|
|
|
bb_termios.c_cc[VMIN] = 0;
|
|
|
|
bb_termios.c_cc[VTIME] = 1;
|
2019-05-30 00:33:51 -07:00
|
|
|
atexit(cleanup);
|
2019-10-27 14:58:23 -07:00
|
|
|
struct sigaction sa_winch = {.sa_handler = &update_term_size};
|
|
|
|
sigaction(SIGWINCH, &sa_winch, NULL);
|
|
|
|
int signals[] = {SIGTERM, SIGINT, SIGXCPU, SIGXFSZ, SIGVTALRM, SIGPROF, SIGSEGV, SIGTSTP};
|
|
|
|
struct sigaction sa = {.sa_handler = &cleanup_and_raise, .sa_flags = SA_NODEFER | SA_RESETHAND};
|
2019-10-13 18:56:09 -07:00
|
|
|
for (int i = 0; i < sizeof(signals)/sizeof(signals[0]); i++)
|
2019-10-27 14:58:23 -07:00
|
|
|
sigaction(signals[i], &sa, NULL);
|
2019-05-24 16:55:36 -07:00
|
|
|
|
2019-10-13 21:20:00 -07:00
|
|
|
write(cmdfd, "\0", 1);
|
2019-10-03 13:26:31 -07:00
|
|
|
for (int i = 0; i < argc; i++) {
|
|
|
|
if (argv[i][0] == '+') {
|
2019-11-01 08:54:47 -07:00
|
|
|
char *cmd = argv[i];
|
|
|
|
char *colon = strchr(cmd, ':');
|
2019-10-03 13:26:31 -07:00
|
|
|
if (colon && !colon[1]) {
|
|
|
|
for (++i; i < argc; i++) {
|
2019-11-01 08:54:47 -07:00
|
|
|
write(cmdfd, cmd, strlen(cmd));
|
2019-10-03 13:26:31 -07:00
|
|
|
write(cmdfd, argv[i], strlen(argv[i])+1); // Include null byte
|
|
|
|
}
|
|
|
|
} else {
|
2019-11-01 08:54:47 -07:00
|
|
|
write(cmdfd, cmd, strlen(cmd)+1); // Include null byte
|
2019-10-03 13:26:31 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-10-27 14:58:23 -07:00
|
|
|
close(cmdfd); cmdfd = -1;
|
2019-10-03 13:26:31 -07:00
|
|
|
|
2019-10-27 14:58:23 -07:00
|
|
|
init_term();
|
2019-11-11 10:48:12 -08:00
|
|
|
bb_t bb = {
|
|
|
|
.columns = "*smpn",
|
|
|
|
.sort = "+n"
|
|
|
|
};
|
|
|
|
bb_browse(&bb, full_initial_path);
|
2019-10-27 14:58:23 -07:00
|
|
|
fputs(T_LEAVE_BBMODE, tty_out);
|
|
|
|
cleanup();
|
2019-10-13 18:56:09 -07:00
|
|
|
|
2019-11-11 10:48:12 -08:00
|
|
|
if (bb.selected && print_selection) {
|
|
|
|
for (entry_t *e = bb.selected; e; e = e->selected.next) {
|
2019-10-13 21:48:43 -07:00
|
|
|
write(STDOUT_FILENO, e->fullname, strlen(e->fullname));
|
2019-05-27 16:08:29 -07:00
|
|
|
write(STDOUT_FILENO, &sep, 1);
|
|
|
|
}
|
|
|
|
fflush(stdout);
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
2019-05-31 01:09:21 -07:00
|
|
|
if (print_dir)
|
2019-11-11 10:48:12 -08:00
|
|
|
printf("%s\n", bb.path);
|
2019-05-31 01:09:21 -07:00
|
|
|
|
2019-06-18 17:08:13 -07:00
|
|
|
// Cleanup:
|
2019-11-11 10:48:12 -08:00
|
|
|
populate_files(&bb, NULL);
|
|
|
|
while (bb.selected)
|
|
|
|
set_selected(&bb, bb.selected, 0);
|
2019-05-31 01:09:21 -07:00
|
|
|
if (cmdfilename) free(cmdfilename);
|
2019-05-25 04:30:51 -07:00
|
|
|
return 0;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-24 17:35:16 -07:00
|
|
|
|
|
|
|
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1
|