Proper history functionality via cd:- and cd:+
This commit is contained in:
parent
2c44abc2c4
commit
6a1522645e
57
bb.c
57
bb.c
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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
|
||||||
|
8
types.h
8
types.h
@ -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;
|
||||||
|
|
||||||
|
5
utils.h
5
utils.h
@ -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))
|
||||||
|
Loading…
Reference in New Issue
Block a user