Proper history functionality via cd:- and cd:+

This commit is contained in:
Bruce Hill 2021-04-17 00:09:49 -07:00
parent 2c44abc2c4
commit 6a1522645e
4 changed files with 61 additions and 16 deletions

57
bb.c
View File

@ -30,7 +30,7 @@
#define BB_NAME "bb" #define BB_NAME "bb"
#endif #endif
#define BB_VERSION "0.29.0" #define BB_VERSION "0.30.0"
#define MAX_BINDINGS 1024 #define MAX_BINDINGS 1024
#define SCROLLOFF MIN(5, (winsize.ws_row-4)/2) #define SCROLLOFF MIN(5, (winsize.ws_row-4)/2)
@ -363,7 +363,7 @@ static entry_t* load_entry(bb_t *bb, const char *path)
for (entry_t *e = bb->hash[(int)filestat.st_ino & HASH_MASK]; e; e = e->hash.next) { 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 if (e->info.st_ino == filestat.st_ino && e->info.st_dev == filestat.st_dev
// Need to check filename in case of hard links // Need to check filename in case of hard links
&& strcmp(pbuf, e->fullname) == 0) && streq(pbuf, e->fullname))
return e; return e;
} }
@ -382,7 +382,7 @@ static entry_t* load_entry(bb_t *bb, const char *path)
char *end = stpcpy(entry->fullname, pbuf); char *end = stpcpy(entry->fullname, pbuf);
if (linkpathlen >= 0) if (linkpathlen >= 0)
entry->linkname = strcpy(end + 1, linkbuf); entry->linkname = strcpy(end + 1, linkbuf);
if (strcmp(entry->fullname, "/") == 0) { if (streq(entry->fullname, "/")) {
entry->name = entry->fullname; entry->name = entry->fullname;
} else { } else {
entry->name = strrchr(entry->fullname, '/') + 1; // Last path component entry->name = strrchr(entry->fullname, '/') + 1; // Last path component
@ -448,7 +448,21 @@ static char *normalize_path(const char *root, const char *path, char *normalized
// //
static int populate_files(bb_t *bb, const char *path) static int populate_files(bb_t *bb, const char *path)
{ {
int samedir = path && strcmp(bb->path, path) == 0; int clear_future_history = 0;
if (path == NULL)
;
else if (streq(path, "-")) {
if (bb->history->prev)
bb->history = bb->history->prev;
path = bb->history->path;
} else if (streq(path, "+")) {
if (bb->history->next)
bb->history = bb->history->next;
path = bb->history->path;
} else
clear_future_history = 1;
int samedir = path && streq(bb->path, path);
int old_scroll = bb->scroll; int old_scroll = bb->scroll;
int old_cursor = bb->cursor; int old_cursor = bb->cursor;
char old_selected[PATH_MAX] = ""; char old_selected[PATH_MAX] = "";
@ -467,8 +481,18 @@ static int populate_files(bb_t *bb, const char *path)
} }
} }
strcpy(bb->prev_path, prev); if (clear_future_history && !samedir) {
setenv("BBPREVPATH", bb->prev_path, 1); for (bb_history_t *next, *h = bb->history->next; h; h = next) {
next = h->next;
free(h);
}
bb_history_t *h = calloc(1, sizeof(bb_history_t));
strcpy(h->path, pbuf);
h->prev = bb->history;
bb->history->next = h;
bb->history = h;
}
bb->dirty = 1; bb->dirty = 1;
strcpy(bb->path, pbuf); strcpy(bb->path, pbuf);
@ -556,7 +580,7 @@ static void print_bindings(int fd)
} }
size_t shared = 0; size_t shared = 0;
char *p = buf; char *p = buf;
for (size_t j = i; bindings[j].script && strcmp(bindings[j].description, bindings[i].description) == 0; j++) { for (size_t j = i; bindings[j].script && streq(bindings[j].description, bindings[i].description); j++) {
if (j > i) p = stpcpy(p, ", "); if (j > i) p = stpcpy(p, ", ");
++shared; ++shared;
int key = bindings[j].key; int key = bindings[j].key;
@ -604,7 +628,7 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
else script = trim(script); else script = trim(script);
} else description = script; } else description = script;
for (char *key; (key = strsep(&keys, ",")); ) { for (char *key; (key = strsep(&keys, ",")); ) {
int is_section = strcmp(key, "Section") == 0; int is_section = streq(key, "Section");
int keyval = bkeywithname(key); int keyval = bkeywithname(key);
if (keyval == -1 && !is_section) continue; if (keyval == -1 && !is_section) continue;
for (size_t i = 0; i < sizeof(bindings)/sizeof(bindings[0]); i++) { for (size_t i = 0; i < sizeof(bindings)/sizeof(bindings[0]); i++) {
@ -648,7 +672,7 @@ static void run_bbcmd(bb_t *bb, const char *cmd)
} }
// Filename may no longer exist: // Filename may no longer exist:
for (e = bb->selected; e; e = e->selected.next) { for (e = bb->selected; e; e = e->selected.next) {
if (strcmp(e->fullname, pbuf) == 0) { if (streq(e->fullname, pbuf)) {
set_selected(bb, e, 0); set_selected(bb, e, 0);
return; return;
} }
@ -1033,7 +1057,7 @@ static int wait_for_process(proc_t **proc)
int main(int argc, char *argv[]) int main(int argc, char *argv[])
{ {
char *initial_path; char *initial_path;
if (argc >= 3 && strcmp(argv[argc-2], "--") == 0) { if (argc >= 3 && streq(argv[argc-2], "--")) {
initial_path = argv[argc-1]; initial_path = argv[argc-1];
argc -= 2; argc -= 2;
} else if (argc >= 2 && argv[argc-1][0] != '-' && argv[argc-1][0] != '+') { } else if (argc >= 2 && argv[argc-1][0] != '-' && argv[argc-1][0] != '+') {
@ -1051,11 +1075,11 @@ int main(int argc, char *argv[])
char *colon = strchr(argv[i], ':'); char *colon = strchr(argv[i], ':');
if (colon && !colon[1]) if (colon && !colon[1])
break; break;
} else if (strcmp(argv[i], "--help") == 0) { } else if (streq(argv[i], "--help")) {
help: help:
printf("%s%s", description_str, usage_str); printf("%s%s", description_str, usage_str);
return 0; return 0;
} else if (strcmp(argv[i], "--version") == 0) { } else if (streq(argv[i], "--version")) {
version: version:
printf(BB_NAME" "BB_VERSION"\n"); printf(BB_NAME" "BB_VERSION"\n");
return 0; return 0;
@ -1135,7 +1159,7 @@ int main(int argc, char *argv[])
if (stat(full_initial_path, &path_stat) != 0) if (stat(full_initial_path, &path_stat) != 0)
err("Could not find initial path: \"%s\"", initial_path); err("Could not find initial path: \"%s\"", initial_path);
if (S_ISDIR(path_stat.st_mode)) { if (S_ISDIR(path_stat.st_mode)) {
if (strcmp(full_initial_path, "/") != 0) strcat(full_initial_path, "/"); if (!streq(full_initial_path, "/")) strcat(full_initial_path, "/");
} else { } else {
write(cmdfd, "goto:", 5); write(cmdfd, "goto:", 5);
write(cmdfd, full_initial_path, strlen(full_initial_path) + 1); // Include null byte write(cmdfd, full_initial_path, strlen(full_initial_path) + 1); // Include null byte
@ -1174,9 +1198,12 @@ int main(int argc, char *argv[])
for (size_t i = 0; i < sizeof(signals)/sizeof(signals[0]); i++) for (size_t i = 0; i < sizeof(signals)/sizeof(signals[0]); i++)
sigaction(signals[i], &sa, NULL); sigaction(signals[i], &sa, NULL);
bb_history_t *history = calloc(1, sizeof(bb_history_t));
strcpy(history->path, full_initial_path);
bb_t bb = { bb_t bb = {
.columns = "*smpn", .columns = "*smpn",
.sort = "+n", .sort = "+n",
.history = history,
}; };
current_bb = &bb; current_bb = &bb;
set_globs(&bb, "*"); set_globs(&bb, "*");
@ -1200,6 +1227,10 @@ int main(int argc, char *argv[])
while (bb.selected) while (bb.selected)
set_selected(&bb, bb.selected, 0); set_selected(&bb, bb.selected, 0);
free(bb.globpats); free(bb.globpats);
for (bb_history_t *next; bb.history; bb.history = next) {
next = bb.history->next;
free(bb.history);
}
return 0; return 0;
} }

View File

@ -68,7 +68,10 @@ mark="$(readlink -f "$XDG_CONFIG_HOME"/bb/marks/"$mark")"
bbcmd cd:"$mark" bbcmd cd:"$mark"
## -,Backspace: Go to previous directory ## -,Backspace: Go to previous directory
[ "$BBPREVPATH" ] && bbcmd cd:"$BBPREVPATH" bbcmd cd:-
## +,=: Go to next directory
bbcmd cd:+
## ;: Show selected files ## ;: Show selected files
printf '%s\n' "$@" | less printf '%s\n' "$@" | less
@ -310,7 +313,7 @@ fi
## i: Toggle interleaving files and directories ## i: Toggle interleaving files and directories
bbcmd interleave bbcmd interleave
## F5,Ctrl-l: Refresh view ## F5,Ctrl-l,Ctrl-r: Refresh view
bbcmd refresh bbcmd refresh
## Ctrl-b: Bind a key to a script ## Ctrl-b: Bind a key to a script

View File

@ -53,13 +53,19 @@ typedef struct proc_s {
} running; } running;
} proc_t; } proc_t;
// History of paths
typedef struct bb_history_s {
char path[PATH_MAX];
struct bb_history_s *prev, *next;
} bb_history_t;
// Structure for bb program state: // Structure for bb program state:
typedef struct bb_s { typedef struct bb_s {
entry_t *hash[HASH_SIZE]; entry_t *hash[HASH_SIZE];
entry_t **files; entry_t **files;
entry_t *selected; entry_t *selected;
char path[PATH_MAX]; char path[PATH_MAX];
char prev_path[PATH_MAX]; bb_history_t *history;
int nfiles, nselected; int nfiles, nselected;
int scroll, cursor; int scroll, cursor;

View File

@ -10,6 +10,11 @@
#define FILE_UTILS__H #define FILE_UTILS__H
#include <stdio.h> #include <stdio.h>
#include <string.h>
#ifndef streq
#define streq(a,b) (strcmp(a,b)==0)
#endif
#define MAX(a,b) ((a) < (b) ? (b) : (a)) #define MAX(a,b) ((a) < (b) ? (b) : (a))
#define MIN(a,b) ((a) > (b) ? (b) : (a)) #define MIN(a,b) ((a) > (b) ? (b) : (a))