Working version of running (nearly) everything on bb IPC commands

This commit is contained in:
Bruce Hill 2019-05-23 00:57:25 -07:00
parent 566ee2f272
commit e080490d51
3 changed files with 349 additions and 309 deletions

5
bb.1
View File

@ -8,6 +8,7 @@ bb \- A bitty browser for command line file management
[\fI-d\fR]
[\fI-s\fR]
[\fI-0\fR]
[\fI-c commands... \fR]
[\fIdirectory\fR]
.SH DESCRIPTION
\fBbb\fR is a tiny TUI console application for browsing and managing files.
@ -22,6 +23,10 @@ Print the selected files on exit.
If printing the selected files on exit, use NULL-terminated strings instead of
newline-separated.
.B \-c commands...
From within \fBbb\fR, running \fBbb -c ...\fR will execute an internal bb
command for modifying bb's state.
.B directory
Open to this directory.

577
bb.c
View File

@ -25,10 +25,11 @@
#define MAX(a,b) ((a) < (b) ? (b) : (a))
#define MIN(a,b) ((a) > (b) ? (b) : (a))
#define MAX_PATH 4096
#define startswith(str, start) (strncmp(str, start, strlen(start)) == 0)
#define writez(fd, str) write(fd, str, strlen(str))
#define IS_SELECTED(p) ((p)->atme)
#define KEY_DELAY -1
#define KEY_DELAY 50
static struct termios orig_termios;
static int termfd;
@ -39,7 +40,7 @@ typedef enum {
SORT_ALPHA = 0,
SORT_SIZE,
SORT_BITS,
SORT_DATE
SORT_TIME
} sortmethod_t;
typedef struct entry_s {
@ -63,7 +64,12 @@ typedef struct {
sortmethod_t sortmethod;
} bb_state_t;
static void err(const char *msg, ...);
#define err(...) do { \
close_term(); \
fprintf(stderr, __VA_ARGS__); \
_err(); \
} while (0)
static void _err();
static void update_term_size(void)
{
@ -73,6 +79,13 @@ static void update_term_size(void)
height = sz.ws_row;
}
static inline int clamped(int x, int low, int high)
{
if (x < low) return low;
if (x > high) return high;
return x;
}
static void do_nothing(int _)
{
// Used as SIGINT handler
@ -112,28 +125,34 @@ static void close_term()
writez(termfd, "\e[?1049l");
// Show cursor:
writez(termfd, "\e[?25h");
// Restore scrollable region
char buf[16];
sprintf(buf, "\e[1;%dr", height);
writez(termfd, buf);
tcsetattr(termfd, TCSAFLUSH, &orig_termios);
close(termfd);
}
static void err(const char *msg, ...)
static void _err()
{
close_term();
va_list args;
va_start(args, msg);
int len = fprintf(stderr, msg, args);
va_end(args);
if (errno)
fprintf(stderr, "\n%s", strerror(errno));
fprintf(stderr, "\n");
exit(1);
}
static char bb_tmpfile[] = "/tmp/bb.XXXXXX";
static int run_cmd_on_selection(bb_state_t *state, const char *cmd)
{
pid_t child;
sig_t old_handler = signal(SIGINT, do_nothing);
strcpy(bb_tmpfile, "/tmp/bb.XXXXXX");
if (!mktemp(bb_tmpfile))
err("Could not create temp file");
if ((child = fork()) == 0) {
// TODO: is there a max number of args? Should this be batched?
char **const args = calloc(MAX(1, state->nselected) + 5, sizeof(char*));
@ -148,12 +167,15 @@ static int run_cmd_on_selection(bb_state_t *state, const char *cmd)
}
args[i] = NULL;
char bb_depth_str[64] = {0}, bb_ipc_str[64] = {0};
{ // Set environment variable to track shell nesting
char *depthstr = getenv("BB_DEPTH");
int depth = depthstr ? atoi(depthstr) : 0;
char buf[64] = {0};
snprintf(buf, sizeof(buf), "BB_DEPTH=%d", depth + 1);
putenv(buf);
snprintf(bb_depth_str, sizeof(bb_depth_str), "%d", depth + 1);
setenv("BB_DEPTH", bb_depth_str, 1);
setenv("BBCMD", bb_tmpfile, 1);
setenv("BBCURSOR", state->files[state->cursor]->d_name, 1);
setenv("BBFULLCURSOR", state->files[state->cursor]->d_fullname, 1);
}
execvp("sh", args);
@ -163,53 +185,13 @@ static int run_cmd_on_selection(bb_state_t *state, const char *cmd)
if (child == -1)
err("Failed to fork");
int status;
waitpid(child, &status, 0);
signal(SIGINT, old_handler);
return status;
}
static pid_t run_cmd(int *child_out, int *child_in, const char *cmd, ...)
{
int child_outfds[2], child_infds[2];
pid_t child;
if (child_out)
pipe(child_outfds);
if (child_in)
pipe(child_infds);
if ((child = fork())) {
if (child == -1)
err("Failed to fork");
if (child_out) {
*child_out = child_outfds[0];
close(child_outfds[1]);
}
if (child_in) {
*child_in = child_infds[1];
close(child_infds[0]);
}
} else {
if (child_out) {
dup2(child_outfds[1], STDOUT_FILENO);
close(child_outfds[0]);
}
if (child_in) {
dup2(child_infds[0], STDIN_FILENO);
close(child_infds[1]);
}
char *formatted_cmd;
va_list args;
va_start(args, cmd);
int len = vasprintf(&formatted_cmd, cmd, args);
va_end(args);
if (formatted_cmd)
execlp("sh", "sh", "-c", formatted_cmd);
err("Failed to execute command %d: '%s'", len, formatted_cmd);
}
return child;
}
static void term_move(int x, int y)
{
static char buf[32] = {0};
@ -263,7 +245,7 @@ static void render(bb_state_t *state, int lazy)
{ // Column labels
char buf[] = "\e[32m Size Date Perm Name\e[0m";
buf[8] = state->sortmethod == SORT_SIZE ? (state->sort_reverse ? '-' : '+') : ' ';
buf[21] = state->sortmethod == SORT_DATE ? (state->sort_reverse ? '-' : '+') : ' ';
buf[21] = state->sortmethod == SORT_TIME ? (state->sort_reverse ? '-' : '+') : ' ';
buf[36] = state->sortmethod == SORT_BITS ? (state->sort_reverse ? '-' : '+') : ' ';
buf[42] = state->sortmethod == SORT_ALPHA ? (state->sort_reverse ? '-' : '+') : ' ';
writez(termfd, buf);
@ -460,6 +442,16 @@ static int compare_date(void *r, const void *v1, const void *v2)
return -(info1.st_mtimespec.tv_sec - info2.st_mtimespec.tv_sec)*sign;
}
static entry_t *find_file(bb_state_t *state, const char *name)
{
for (int i = 0; i < state->nfiles; i++) {
entry_t *e = state->files[i];
if (strcmp(name[0] == '/' ? e->d_fullname : e->d_name, name) == 0)
return e;
}
return NULL;
}
static void write_selection(int fd, entry_t *firstselected, char sep)
{
while (firstselected) {
@ -491,21 +483,46 @@ static void clear_selection(bb_state_t *state)
state->nselected = 0;
}
static int select_file(bb_state_t *state, entry_t *e)
{
if (IS_SELECTED(e)) return 0;
if (strcmp(e->d_name, "..") == 0) return 0;
if (state->firstselected)
state->firstselected->atme = &e->next;
e->next = state->firstselected;
e->atme = &state->firstselected;
state->firstselected = e;
++state->nselected;
return 1;
}
static int deselect_file(bb_state_t *state, entry_t *e)
{
if (!IS_SELECTED(e)) return 0;
if (e->next) e->next->atme = e->atme;
*(e->atme) = e->next;
e->next = NULL, e->atme = NULL;
--state->nselected;
return 1;
}
static void explore(char *path, int print_dir, int print_selection, char sep)
{
char *tmp = path;
char *original_path = calloc(strlen(tmp) + 1, 1);
if (!original_path) err("allocation failure");
strcpy(original_path, path);
path = calloc(strlen(tmp) + 1, 1);
char realpath_buf[MAX_PATH+1];
path = strdup(path);
if (!path) err("allocation failure");
strcpy(path, tmp);
tmp = NULL;
char *original_path = strdup(path);
if (!original_path) err("allocation failure");
char to_select[MAX_PATH+1] = {0};
bb_state_t state = {0};
memset(&state, 0, sizeof(bb_state_t));
tail_call:
if (!realpath(path, realpath_buf))
err("realpath failed on %s", path);
path = realloc(path, strlen(realpath_buf));
strcpy(path, realpath_buf);
if (state.files) {
for (int i = 0; i < state.nfiles; i++) {
@ -525,7 +542,7 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
size_t hashsize = 2 * state.nselected;
entry_t **selecthash = calloc(hashsize, sizeof(entry_t*));
if (!selecthash)
err("Failed to allocate %d spaces for selecthash", hashsize);
err("Failed to allocate %ld spaces for selecthash", hashsize);
for (entry_t *p = state.firstselected; p; p = p->next) {
int probe = ((int)p->d_ino) % hashsize;
while (selecthash[probe])
@ -596,7 +613,7 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
sort_files:
cmp = compare_alpha;
if (state.sortmethod == SORT_SIZE) cmp = compare_size;
if (state.sortmethod == SORT_DATE) cmp = compare_date;
if (state.sortmethod == SORT_TIME) cmp = compare_date;
if (state.sortmethod == SORT_BITS) cmp = compare_perm;
qsort_r(&state.files[1], state.nfiles-1, sizeof(entry_t*), &state.sort_reverse, cmp);
@ -644,20 +661,24 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
} else if (mouse_y >= 2 && state.scroll + (mouse_y - 2) < state.nfiles) {
int clicked = state.scroll + (mouse_y - 2);
if (dt_ms > 200) {
lazy = 1;
// Single click
if (mouse_x == 0) {
// Toggle
picked = clicked;
goto toggle;
if (IS_SELECTED(state.files[clicked]))
deselect_file(&state, state.files[clicked]);
else
select_file(&state, state.files[clicked]);
goto redraw;
} else {
state.cursor = clicked;
lazy = 1;
goto redraw;
}
} else {
// Double click
picked = clicked;
goto open_file;
// TODO: hacky
state.cursor = clicked;
key = '\n';
goto user_bindings;
}
}
break;
@ -676,9 +697,6 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
init_term();
goto redraw;
case 'q': case 'Q':
goto done;
case KEY_MOUSE_WHEEL_DOWN:
if (state.cursor >= state.nfiles - 1)
goto skip_redraw;
@ -697,146 +715,28 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
--state.scroll;
goto redraw;
case KEY_CTRL_D: case KEY_PGDN:
if (state.cursor == state.nfiles - 1)
goto skip_redraw;
lazy = 1;
state.cursor = MIN(state.nfiles - 1, state.cursor + (height - 4) / 2);
if (state.nfiles <= height - 4)
goto redraw;
state.scroll += (height - 4)/2;
if (state.scroll > state.nfiles - (height - 4) - 1)
state.scroll = state.nfiles - (height - 4) - 1;
goto redraw;
case KEY_CTRL_U: case KEY_PGUP:
state.cursor = MAX(0, state.cursor - (height - 4) / 2);
if (state.nfiles <= height - 4)
goto redraw;
lazy = 1;
state.scroll -= (height - 4)/2;
if (state.scroll < 0)
state.scroll = 0;
goto redraw;
case 'g': case KEY_HOME:
lazy = 1;
state.cursor = 0;
state.scroll = 0;
goto redraw;
case 'G': case KEY_END:
lazy = 1;
state.cursor = state.nfiles - 1;
if (state.nfiles > height - 4)
state.scroll = state.nfiles - (height - 4) - 1;
goto redraw;
case ' ':
picked = state.cursor;
toggle:
lazy = 1;
if (strcmp(state.files[picked]->d_name, "..") == 0)
goto skip_redraw;
if (IS_SELECTED(state.files[picked])) {
toggle_off:;
entry_t *e = state.files[picked];
if (e->next) e->next->atme = e->atme;
*(e->atme) = e->next;
e->next = NULL, e->atme = NULL;
--state.nselected;
} else {
toggle_on:;
entry_t *e = state.files[picked];
if (state.firstselected)
state.firstselected->atme = &e->next;
e->next = state.firstselected;
e->atme = &state.firstselected;
state.firstselected = e;
++state.nselected;
}
goto redraw;
case KEY_CTRL_A:
for (int i = 0; i < state.nfiles; i++) {
entry_t *e = state.files[i];
if (e->atme) continue;
if (strcmp(e->d_name, "..") == 0)
continue;
if (state.firstselected)
state.firstselected->atme = &e->next;
e->next = state.firstselected;
e->atme = &state.firstselected;
state.firstselected = e;
++state.nselected;
}
goto redraw;
case KEY_ESC:
clear_selection(&state);
goto redraw;
case KEY_F5: case KEY_CTRL_R:
strcpy(to_select, state.files[state.cursor]->d_name);
goto tail_call;
case 'j': case KEY_ARROW_DOWN:
if (state.cursor >= state.nfiles - 1)
goto skip_redraw;
lazy = 1;
++state.cursor;
if (state.cursor > state.scroll + (height - 4) - 1 - scrolloff
&& state.scroll < state.nfiles - (height - 4) - 1)
++state.scroll;
goto redraw;
case 'k': case KEY_ARROW_UP:
if (state.cursor <= 0)
goto skip_redraw;
lazy = 1;
--state.cursor;
if (state.cursor < state.scroll + scrolloff && state.scroll > 0)
--state.scroll;
goto redraw;
case 'J':
if (state.cursor < state.nfiles - 1) {
if (IS_SELECTED(state.files[state.cursor])) {
picked = ++state.cursor;
if (!IS_SELECTED(state.files[picked]))
goto toggle_on;
} else {
picked = ++state.cursor;
if (IS_SELECTED(state.files[picked]))
goto toggle_off;
}
lazy = 1;
if (IS_SELECTED(state.files[state.cursor]))
select_file(&state, state.files[++state.cursor]);
else
deselect_file(&state, state.files[++state.cursor]);
goto redraw;
}
goto skip_redraw;
case 'K':
if (state.cursor > 0) {
if (IS_SELECTED(state.files[state.cursor])) {
picked = --state.cursor;
if (!IS_SELECTED(state.files[picked]))
goto toggle_on;
} else {
picked = --state.cursor;
if (IS_SELECTED(state.files[picked]))
goto toggle_off;
}
lazy = 1;
if (IS_SELECTED(state.files[state.cursor]))
select_file(&state, state.files[--state.cursor]);
else
deselect_file(&state, state.files[--state.cursor]);
goto redraw;
}
goto skip_redraw;
case 'h': case KEY_ARROW_LEFT:
// TODO: slightly hacky, depends on ".." being at 0 (currently always true)
picked = 0;
goto open_file;
case 's':
// Change sorting method:
term_move(0, height-1);
@ -875,10 +775,10 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
case 't': case 'T':
sort_date:
if (state.sortmethod == SORT_DATE)
if (state.sortmethod == SORT_TIME)
state.sort_reverse ^= 1;
else {
state.sortmethod = SORT_DATE;
state.sortmethod = SORT_TIME;
state.sort_reverse = 0;
}
break;
@ -896,85 +796,12 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
case '.':
state.showhidden ^= 1;
strcpy(to_select, state.files[state.cursor]->d_name);
goto tail_call;
case 'l': case '\r': case KEY_ARROW_RIGHT:
picked = state.cursor;
open_file: {
if (state.files[picked]->d_isdir) {
if (strcmp(state.files[picked]->d_name, "..") == 0) {
char *p = strrchr(state.path, '/');
if (p) strcpy(to_select, p+1);
else to_select[0] = '\0';
} else to_select[0] = '\0';
char tmp[MAX_PATH+1];
if (!realpath(state.files[picked]->d_fullname, tmp))
err("realpath failed");
free(path);
path = calloc(strlen(tmp) + 1, sizeof(char));
strcpy(path, tmp);
goto tail_call;
} else {
char *name = state.files[picked]->d_name;
close_term();
pid_t child = run_cmd(NULL, NULL,
#ifdef __APPLE__
"if file -bI %s | grep '^text/' >/dev/null; then $EDITOR %s; else open %s; fi",
#else
"if file -bi %s | grep '^text/' >/dev/null; then $EDITOR %s; else xdg-open %s; fi",
#endif
name, name, name);
waitpid(child, NULL, 0);
init_term();
goto redraw;
}
break;
}
case 'f': case '/': {
close_term();
int fd;
if (state.showhidden)
run_cmd(&fd, NULL, "find | cut -d/ -f2- | " PROG_FUZZY);
else
run_cmd(&fd, NULL, "find -not -name '\\.*' -not -path '*/\\.*' | cut -d/ -f2- | " PROG_FUZZY);
int len = 0, space = MAX_PATH, consumed;
while ((consumed = read(fd, &to_select[len], space)) > 0) {
if (consumed < 0) err("Error reading selection");
to_select[len + consumed] = '\0';
char *nl = strchr(&to_select[len], '\n');
if (nl) {
*nl = '\0';
break;
}
len += consumed;
space -= consumed;
}
close(fd);
init_term();
if (to_select[0] == '\0')
goto redraw;
char *pathend = strrchr(to_select, '/');
if (pathend) {
char *newpath = calloc(strlen(path) + 1 + (pathend - to_select), 1);
strcpy(newpath, path);
strcat(newpath, "/");
strncat(newpath, to_select, pathend - to_select);
free(path);
path = newpath;
strcpy(to_select, pathend + 1);
}
goto tail_call;
}
case '?': {
close_term();
int fd;
pid_t child = run_cmd(NULL, &fd, "less -r");
char tmpname[] = "/tmp/bb_help.XXXXXX";
int fd = mkstemp(tmpname);
writez(fd, "\n \e[33;1mKey Bindings:\e[0m\n");
for (int i = 0; bindings[i].key; i++) {
if (bindings[i].key > 0) continue;
@ -990,17 +817,27 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
writez(fd, "\n \e[33;1mScript Key Bindings:\e[0m\n");
for (int i = 0; bindings[i].key; i++) {
if (bindings[i].key <= 0) continue;
writez(fd, "\e[1m");
char buf[] = " X \e[0m";
*strchr(buf, 'X') = bindings[i].key;
writez(fd, buf);
writez(fd, bindings[i].command);
writez(fd, "\e[1m ");
char buf[16];
buf[0] = bindings[i].key;
if (' ' <= buf[0] && buf[0] <= '~')
write(fd, buf, 1);
else {
sprintf(buf, "\e[31m%04X", bindings[i].key);
writez(fd, buf);
}
writez(fd, " \e[0m\t");
write_escaped(fd, bindings[i].command, strlen(bindings[i].command), "\e[0m");
writez(fd, "\e[0m\n");
}
writez(fd, "\n");
close(fd);
waitpid(child, NULL, 0);
close_term();
char cmd[256];
sprintf(cmd, "less -r %s", tmpname);
system(cmd);
unlink(tmpname);
init_term();
goto redraw;
}
@ -1010,36 +847,182 @@ static void explore(char *path, int print_dir, int print_selection, char sep)
default:
// Search user-defined key bindings from config.h:
user_bindings:
for (int i = 0; bindings[i].key > 0; i++) {
if (key == bindings[i].key) {
term_move(0, height-1);
writez(termfd, "\e[K");
//writez(termfd, "\e[K");
struct termios cur_tios;
if (!(bindings[i].flags & ONSCREEN)) {
if (bindings[i].flags & NORMAL_TERM) {
close_term();
} else {
{ // Restore scrolling region
char buf[16];
sprintf(buf, "\e[1;%dr", height);
writez(termfd, buf);
}
/*
tcgetattr(termfd, &cur_tios);
tcsetattr(termfd, TCSAFLUSH, &orig_termios);
writez(termfd, "\e[?25h"); // Show cursor
*/
}
run_cmd_on_selection(&state, bindings[i].command);
if (!(bindings[i].flags & ONSCREEN)) {
if (bindings[i].flags & NORMAL_TERM) {
lazy = 0;
init_term();
} else {
lazy = 1;
/*
tcsetattr(termfd, TCSAFLUSH, &cur_tios);
writez(termfd, "\e[?25l"); // Hide cursor
*/
{ // Restore scrolling region
char buf[16];
sprintf(buf, "\e[3;%dr", height-1);
writez(termfd, buf);
}
}
if (bindings[i].flags & CLEAR_SELECTION)
clear_selection(&state);
// Scan for IPC requests
int needs_full_refresh = 0;
char newsort[3] = "";
FILE *tmpfile;
if ((tmpfile = fopen(bb_tmpfile, "r"))) {
char *line = NULL;
size_t capacity = 0;
ssize_t len;
while ((len = getline(&line, &capacity, tmpfile)) >= 0) {
if (len > 0 && line[len-1] == '\n') line[--len] = '\0';
if (strcmp(line, "refresh") == 0) {
needs_full_refresh = 1;
} else if (strcmp(line, "quit") == 0) {
goto done;
} else if (startswith(line, "sort:") && len >= 7) {
newsort[0] = line[strlen("sort:")];
newsort[1] = line[strlen("sort:") + 1];
} else if (startswith(line, "cd:")) {
free(path);
path = calloc(strlen(line+strlen("cd:")) + 1, sizeof(char));
strcpy(path, line+strlen("cd:"));
goto tail_call;
} else if (startswith(line, "toggle:")) {
lazy = 0;
entry_t *e = find_file(&state, line + strlen("select:"));
if (e) {
if (IS_SELECTED(e)) deselect_file(&state, e);
else select_file(&state, e);
}
} else if (startswith(line, "select:")) {
char *name = line + strlen("select:");
lazy = 0;
if (strcmp(name, "*") == 0) {
for (int i = 0; i < state.nfiles; i++)
select_file(&state, state.files[i]);
} else {
entry_t *e = find_file(&state, name);
if (e) select_file(&state, e);
}
} else if (startswith(line, "deselect:")) {
char *name = line + strlen("deselect:");
lazy = 0;
if (strcmp(name, "*") == 0) {
clear_selection(&state);
} else {
entry_t *e = find_file(&state, name);
if (e) select_file(&state, e);
}
} else if (startswith(line, "cursor:")) {
char *name = line + strlen("cursor:");
for (int i = 0; i < state.nfiles; i++) {
if (strcmp(name[0] == '/' ?
state.files[i]->d_fullname : state.files[i]->d_name,
name) == 0) {
state.cursor = i;
goto found_it;
}
}
free(path);
char *lastslash = strrchr(name, '/');
if (!lastslash) goto found_it;
size_t len = lastslash - name;
path = calloc(len + 1, sizeof(char));
memcpy(path, name, len);
strcpy(to_select, lastslash+1);
goto tail_call;
found_it:;
} else if (startswith(line, "move:")) {
char *value = line + strlen("move:");
int oldcur = state.cursor;
int isabs = value[0] != '-' && value[0] != '+';
long delta = strtol(value, &value, 10);
if (*value == '%') delta = (delta * height)/100;
if (isabs) state.cursor = delta;
else state.cursor += delta;
if (bindings[i].flags & REFRESH) {
state.cursor = clamped(state.cursor, 0, state.nfiles-1);
delta = state.cursor - oldcur;
if (state.nfiles > height-4) {
if (delta > 0) {
if (state.cursor >= state.scroll + (height-4) - scrolloff)
state.scroll += delta;
} else if (delta < 0) {
if (state.cursor <= state.scroll + scrolloff)
state.scroll += delta;
}
//int target = clamped(state.scroll, state.cursor - (height-4) + scrolloff, state.cursor - scrolloff);
//state.scroll += (delta > 0 ? 1 : -1)*MIN(abs(target-state.scroll), abs((int)delta));
//state.scroll = target;
state.scroll = clamped(state.scroll, state.cursor - (height-4) + 1, state.cursor);
state.scroll = clamped(state.scroll, 0, state.nfiles-1 - (height-4));
}
} else if (startswith(line, "scroll:")) {
char *value = line + strlen("scroll:");
int oldscroll = state.scroll;
int isabs = value[0] != '-' && value[0] != '+';
long delta = strtol(value, &value, 10);
if (*value == '%') delta = (delta * height)/100;
if (state.nfiles > height-4) {
if (isabs) state.scroll = delta;
else state.scroll += delta;
state.scroll = clamped(state.scroll, 0, state.nfiles-1 - (height-4));
}
state.cursor = clamped(state.cursor + delta, 0, state.nfiles-1);
}
}
if (line) free(line);
fclose(tmpfile);
unlink(bb_tmpfile);
}
sortmethod_t oldmethod = state.sortmethod;
int oldreverse = state.sort_reverse;
if (newsort[0] == '+')
state.sort_reverse = 0;
else if (newsort[0] == '+')
state.sort_reverse = 1;
switch (newsort[1]) {
case 'a': state.sortmethod = SORT_ALPHA; break;
case 's': state.sortmethod = SORT_SIZE; break;
case 't': state.sortmethod = SORT_TIME; break;
case 'p': state.sortmethod = SORT_BITS; break;
default: break;
}
if (needs_full_refresh) {
strcpy(to_select, state.files[state.cursor]->d_name);
goto tail_call;
}
if (state.sortmethod != oldmethod || oldreverse != state.sort_reverse)
goto sort_files;
goto redraw;
}
}
@ -1058,7 +1041,6 @@ done:
int main(int argc, char *argv[])
{
char _realpath[MAX_PATH+1];
char *path = ".";
char sep = '\n';
int print_dir = 0, print_selection = 0;
@ -1069,6 +1051,17 @@ int main(int argc, char *argv[])
printf("Usage: bb [-h/--help] [-s] [-b] [-0] [path]\n");
return 0;
}
if (strcmp(argv[i], "-c") == 0) {
char *bb_cmdfile = getenv("BBCMD");
if (!bb_cmdfile)
err("Can only execute bb commands from inside bb");
FILE *f = fopen(bb_cmdfile, "w");
for (i = i+1; i < argc; i++) {
fprintf(f, "%s\n", argv[i]);
}
fclose(f);
return 0;
}
if (argv[i][0] == '-' && argv[i][1] == '-')
continue;
if (argv[i][0] == '-') {
@ -1094,8 +1087,6 @@ int main(int argc, char *argv[])
}
}
init_term();
if (!realpath(path, _realpath))
err("realpath failed");
explore(_realpath, print_dir, print_selection, sep);
explore(path, print_dir, print_selection, sep);
return 0;
}

View File

@ -8,9 +8,7 @@
#define AND_PAUSE " && read -n1 -p '\n\e[2m...press any key to continue...\e[0m\e[?25l'"
#define SCROLLOFF 5
#define REFRESH (1<<0)
#define CLEAR_SELECTION (1<<1)
#define ONSCREEN (1<<2)
#define NORMAL_TERM (1<<0)
struct {
int key;
@ -22,19 +20,65 @@ struct {
// Please note that these are sh scripts, not bash scripts, so bash-isms
// won't work unless you make your script use `bash -c "<your script>"`
////////////////////////////////////////////////////////////////////////
{'e', "$EDITOR \"$@\""},
{'L', PIPE_SELECTION_TO "less"},
{'D', "rm -rf \"$@\"", CLEAR_SELECTION | REFRESH | ONSCREEN},
{'d', "rm -rfi \"$@\"", CLEAR_SELECTION | REFRESH | ONSCREEN},
{'m', "mv -i \"$@\" .", CLEAR_SELECTION | REFRESH | ONSCREEN},
{'c', "cp -i \"$@\" .", CLEAR_SELECTION | REFRESH | ONSCREEN},
{'C', "for f; do cp \"$f\" \"$f.copy\"; done", REFRESH | ONSCREEN},
{'n', "read -p '\e[33;1mNew file:\e[0m ' name && touch \"$name\"", ONSCREEN | REFRESH},
{'N', "read -p '\e[33;1mNew dir:\e[0m ' name && mkdir \"$name\"", ONSCREEN | REFRESH},
{'|', "read -p \"\e[33;1m>\e[0m \" cmd && " PIPE_SELECTION_TO "$SHELL -c \"$cmd\"" AND_PAUSE, REFRESH},
{'>', "$SHELL", REFRESH},
{'r', "for f; do read -p \"Rename $f: \" renamed && mv \"$f\" \"$renamed\"; done",
REFRESH | CLEAR_SELECTION | ONSCREEN},
{'e', "$EDITOR \"$@\"", NORMAL_TERM},
{'L', PIPE_SELECTION_TO "less", NORMAL_TERM},
{'D', "rm -rf \"$@\"; bb -c 'deselect:*' refresh"},
{'d', "rm -rfi \"$@\"; bb -c 'deselect:*' refresh"},
{'m', "mv -i \"$@\" .; bb -c 'deselect:*' refresh"},
{'c', "cp -i \"$@\" .; bb -c refresh"},
{'C', "for f; do cp \"$f\" \"$f.copy\"; done; bb -c refresh"},
{'n', "read -p '\e[33;1mNew file:\e[0m \e[K' name && touch \"$name\"; bb -c refresh"},
{'N', "read -p '\e[33;1mNew dir:\e[0m \e[K' name && mkdir \"$name\"; bb -c refresh"},
{'|', "read -p '\e[33;1m|>\e[0m \e[K' cmd && " PIPE_SELECTION_TO "$SHELL -c \"$cmd\"" AND_PAUSE},
{':', "read -p '\e[33;1m:>\e[0m \e[K' cmd && $SHELL -c \"$cmd\" -- \"$@\"" AND_PAUSE},
{'>', "$SHELL", NORMAL_TERM},
{'r', "for f; do read -p \"Rename $f: \e[K\" renamed && mv \"$f\" \"$renamed\"; done;"
" bb -c 'deselect:*' refresh"},
{'h', "bb -c \"cd:..\""},
{KEY_ARROW_LEFT, "bb -c 'cd:..'"},
{'j', "bb -c 'move:+1'"},
{KEY_ARROW_DOWN, "bb -c 'move:+1'"},
{'k', "bb -c 'move:-1'"},
{KEY_ARROW_UP, "bb -c 'move:-1'"},
{'l', "bb -c \"cd:$BBFULLCURSOR\""},
{KEY_ARROW_RIGHT, "bb -c \"cd:$BBFULLCURSOR\""},
#ifdef __APPLE__
{'\r', "if test -x \"$BBCURSOR\"; then \"$BBCURSOR\"; "
"elif test -d \"$BBCURSOR\"; then bb -c \"cd:$BBFULLCURSOR\"; "
"elif file -bI \"$BBCURSOR\" | grep '^text/' >/dev/null; then $EDITOR \"$BBCURSOR\"; "
"else open \"$BBCURSOR\"; fi"},
#else
{'\r', "if test -x \"$BBCURSOR\"; then \"$BBCURSOR\"; "
"elif test -d \"$BBCURSOR\"; then bb -c \"cd:$BBFULLCURSOR\"; "
"elif file -bi \"$BBCURSOR\" | grep '^text/' >/dev/null; then $EDITOR \"$BBCURSOR\"; "
"else xdg-open \"$BBCURSOR\"; fi"},
#endif
{' ', "bb -c \"toggle:$BBCURSOR\""},
//{-1, "J\t\e[0;34mMove selection state down\e[0m"},
//{-1, "K\t\e[0;34mMove selection state up\e[0m"},
{'q', "bb -c quit"},
{'Q', "bb -c quit"},
{'g', "bb -c move:0"},
{KEY_HOME, "bb -c move:0"},
{'G', "bb -c move:9999999999"},
{KEY_END, "bb -c move:999999999"},
{'f', "bb -c \"cursor:`fzf`\"", NORMAL_TERM},
{'/', "bb -c \"cursor:`ls -a|fzf`\"", NORMAL_TERM},
{KEY_ESC, "bb -c 'deselect:*'"},
{KEY_F5, "bb -c refresh"},
{KEY_CTRL_R, "bb -c refresh"},
{KEY_CTRL_A, "bb -c 'select:*'"},
//{-1, "Ctrl-C\t\e[0;34mAbort and exit\e[0m"},
{KEY_PGDN, "bb -c 'scroll:+100%'"},
{KEY_CTRL_D, "bb -c 'scroll:+50%'"},
{KEY_PGUP, "bb -c 'scroll:-100%'"},
{KEY_CTRL_U, "bb -c 'scroll:-50%'"},
//{-1, "Ctrl-Z\t\e[0;34mSuspend\e[0m"},
// Hard-coded behaviors (these are just placeholders for the help):
{-1, "?\t\e[0;34mOpen help menu\e[0m"},