Added shuffle functionality, cleaned up and improved column drawing, and

did a ton of cleanup of compiler warnings.
This commit is contained in:
Bruce Hill 2019-05-24 20:49:24 -07:00
parent 53bee94a9f
commit e5ac52b1d5
3 changed files with 202 additions and 154 deletions

View File

@ -1,6 +1,6 @@
PREFIX=
CC=cc
CFLAGS=-O0 -std=gnu99 -Wall
CFLAGS=-O0 -std=gnu99 -Wall -Wpedantic -Weverything -Wno-missing-field-initializers -Wno-padded -Wno-missing-noreturn -Wno-sign-conversion -Wno-cast-qual -Wno-sign-compare -Wno-shorten-64-to-32
LIBS=
NAME=bb
G=

342
bb.c
View File

@ -28,17 +28,17 @@
#define MAX(a,b) ((a) < (b) ? (b) : (a))
#define MIN(a,b) ((a) > (b) ? (b) : (a))
#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) != NULL)
#define alt_screen() writez(termfd, "\e[?1049h")
#define default_screen() writez(termfd, "\e[?1049l")
#define hide_cursor() writez(termfd, "\e[?25l");
#define show_cursor() writez(termfd, "\e[?25h");
#define alt_screen() writez(termfd, "\033[?1049h")
#define default_screen() writez(termfd, "\033[?1049l")
#define hide_cursor() writez(termfd, "\033[?25l");
#define show_cursor() writez(termfd, "\033[?25h");
#define queue_select(state, name) do {\
if ((state)->to_select) free(state->to_select); \
(state)->to_select = memcheck(strdup(name)); \
char *__name = (name); \
(state)->to_select = realloc((state)->to_select, strlen(__name)+1); \
strcpy((state)->to_select, __name); \
} while (0)
#define err(...) do { \
@ -50,6 +50,7 @@
cleanup_and_exit(1); \
} while (0)
extern binding_t bindings[];
static struct termios orig_termios;
static int termfd = 0;
static int width, height;
@ -64,6 +65,7 @@ typedef enum {
SORT_MTIME = 'm',
SORT_CTIME = 'c',
SORT_ATIME = 'a',
SORT_RANDOM = 'r',
RSORT_NAME = 'N',
RSORT_SIZE = 'S',
@ -71,12 +73,13 @@ typedef enum {
RSORT_MTIME = 'M',
RSORT_CTIME = 'C',
RSORT_ATIME = 'A',
RSORT_RANDOM = 'R',
} sortmethod_t;
// This bit toggles 'A' (0) vs 'a' (1)
#define SORT_DESCENDING 32
#define IS_REVERSED(method) (!((method) & SORT_DESCENDING))
#define COLUMN(method) ((method) & ~SORT_DESCENDING)
#define DESCENDING(method) ((method) | SORT_DESCENDING)
typedef struct entry_s {
struct entry_s *next, **atme;
@ -84,9 +87,9 @@ typedef struct entry_s {
ino_t d_ino;
__uint16_t d_reclen;
__uint8_t d_type;
__uint8_t d_namlen;
__uint16_t d_namlen;
char *d_name;
char d_fullname[0];
char d_fullname[1];
} entry_t;
typedef struct {
@ -101,8 +104,9 @@ typedef struct {
char sort;
} bb_state_t;
static void update_term_size(int _)
static void update_term_size(int sig)
{
(void)sig;
struct winsize sz = {0};
ioctl(termfd, TIOCGWINSZ, &sz);
width = sz.ws_col;
@ -134,17 +138,18 @@ static void init_term()
update_term_size(0);
signal(SIGWINCH, update_term_size);
// Initiate mouse tracking:
writez(termfd, "\e[?1000h\e[?1002h\e[?1015h\e[?1006h");
writez(termfd, "\033[?1000h\033[?1002h\033[?1015h\033[?1006h");
}
static void cleanup_and_exit(int sig)
static void cleanup_and_exit(int sig)
{
(void)sig;
if (termfd) {
tcsetattr(termfd, TCSAFLUSH, &orig_termios);
close(termfd);
termfd = 0;
}
printf("\e[?1000l\e[?1002l\e[?1015l\e[?1006l\e[?1049l\e[?25h\n"); // Restore default screen and show cursor
printf("\033[?1000l\033[?1002l\033[?1015l\033[?1006l\033[?1049l\033[?25h\n"); // Restore default screen and show cursor
fflush(stdout);
unlink(cmdfilename);
exit(1);
@ -155,7 +160,7 @@ static void close_term()
signal(SIGWINCH, SIG_IGN);
// Disable mouse tracking:
writez(termfd, "\e[?1000l\e[?1002l\e[?1015l\e[?1006l");
writez(termfd, "\033[?1000l\033[?1002l\033[?1015l\033[?1006l");
tcsetattr(termfd, TCSAFLUSH, &orig_termios);
close(termfd);
@ -176,7 +181,7 @@ static int run_cmd_on_selection(bb_state_t *state, const char *cmd)
if ((child = fork()) == 0) {
signal(SIGINT, SIG_DFL);
// TODO: is there a max number of args? Should this be batched?
char **const args = memcheck(calloc(MAX(1, state->nselected) + 5, sizeof(char*)));
char **args = memcheck(calloc(MAX(1, state->nselected) + 5, sizeof(char*)));
int i = 0;
args[i++] = "sh";
args[i++] = "-c";
@ -217,7 +222,7 @@ static int run_cmd_on_selection(bb_state_t *state, const char *cmd)
static void term_move(int x, int y)
{
static char buf[32] = {0};
int len = snprintf(buf, sizeof(buf), "\e[%d;%dH", y+1, x+1);
int len = snprintf(buf, sizeof(buf), "\033[%d;%dH", y+1, x+1);
if (len > 0)
write(termfd, buf, len);
}
@ -230,7 +235,7 @@ static int write_escaped(int fd, const char *str, size_t n, const char *reset_co
char buf[5];
int ret = 0;
int backlog = 0;
for (int i = 0; i < n; i++) {
for (size_t i = 0; i < n; i++) {
int escapelen = 0;
if (str[i] <= '\x1b' && escapes[(int)str[i]] != ' ')
escapelen = sprintf(buf, "\\%c", escapes[(int)str[i]]);
@ -247,7 +252,7 @@ static int write_escaped(int fd, const char *str, size_t n, const char *reset_co
write(fd, &str[i-backlog], backlog);
backlog = 0;
}
writez(fd, "\e[31m");
writez(fd, "\033[31m");
write(fd, buf, escapelen);
writez(fd, reset_color);
}
@ -268,10 +273,10 @@ static void render(bb_state_t *state, int lazy)
if (lazy) {
// Use terminal scrolling:
if (lastscroll > state->scroll) {
int n = sprintf(buf, "\e[3;%dr\e[%dT\e[1;%dr", height-1, lastscroll - state->scroll, height);
int n = sprintf(buf, "\033[3;%dr\033[%dT\033[1;%dr", height-1, lastscroll - state->scroll, height);
write(termfd, buf, n);
} else if (lastscroll < state->scroll) {
int n = sprintf(buf, "\e[3;%dr\e[%dS\e[1;%dr", height-1, state->scroll - lastscroll, height);
int n = sprintf(buf, "\033[3;%dr\033[%dS\033[1;%dr", height-1, state->scroll - lastscroll, height);
write(termfd, buf, n);
}
}
@ -279,13 +284,13 @@ static void render(bb_state_t *state, int lazy)
for (char *col = state->columns; *col; ++col) {
switch (*col) {
case 's':
namewidth -= sizewidth + 2;
namewidth -= sizewidth + 3;
break;
case 'm': case 'c': case 'a':
namewidth -= datewidth + 2;
namewidth -= datewidth + 3;
break;
case 'p':
namewidth -= permwidth + 2;
namewidth -= permwidth + 3;
break;
}
}
@ -293,60 +298,62 @@ static void render(bb_state_t *state, int lazy)
if (!lazy) {
// Path
term_move(0,0);
writez(termfd, "\e[0;1;30;47m ");
write_escaped(termfd, state->path, strlen(state->path), "\e[0;1;30;47m");
writez(termfd, "\e[K\e[0m");
writez(termfd, "\033[0;2;37m ");
write_escaped(termfd, state->path, strlen(state->path), "\033[0;2;37m");
writez(termfd, "\033[K\033[0m");
// Columns
term_move(0,1);
writez(termfd, " ");
writez(termfd, " \033[0;44;30m");
for (char *col = state->columns; *col; ++col) {
if (col != state->columns) writez(termfd, "");
writez(termfd, COLUMN(state->sort) == *col ? (IS_REVERSED(state->sort) ? "" : "") : " ");
const char *colname;
int colwidth = 0;
switch (*col) {
case 's':
for (int i = writez(termfd, " Size"); i < sizewidth-1; i++)
write(termfd, " ", 1);
colname = " Size"; colwidth = sizewidth;
break;
case 'p':
for (int i = writez(termfd, "Per"); i < permwidth-1; i++)
write(termfd, " ", 1);
colname = "Per"; colwidth = permwidth;
break;
case 'm':
for (int i = writez(termfd, "Modified"); i < datewidth-1; i++)
write(termfd, " ", 1);
colname = " Modified"; colwidth = datewidth;
break;
case 'a':
for (int i = writez(termfd, "Accessed"); i < datewidth-1; i++)
write(termfd, " ", 1);
colname = " Accessed"; colwidth = datewidth;
break;
case 'c':
for (int i = writez(termfd, "Created"); i < datewidth-1; i++)
write(termfd, " ", 1);
colname = " Created"; colwidth = datewidth;
break;
case 'n':
for (int i = writez(termfd, "Name"); i < namewidth; i++)
write(termfd, " ", 1);
colname = "Name"; colwidth = namewidth;
break;
default:
continue;
}
if (col != state->columns) writez(termfd, "");
writez(termfd, DESCENDING(state->sort) == *col ? (IS_REVERSED(state->sort) ? "" : "") : " ");
for (ssize_t i = writez(termfd, colname); i < colwidth-1; i++)
write(termfd, " ", 1);
}
writez(termfd, "\033[0m");
}
if (state->nselected > 0) {
int len = snprintf(buf, sizeof(buf), "%lu selected ", state->nselected);
if (strlen(state->path) + 1 + len < width) {
term_move(width-len, 0);
writez(termfd, "\e[0;1;30;47m");
writez(termfd, "\033[0;1;30;47m");
write(termfd, buf, len);
writez(termfd, "\e[0m");
writez(termfd, "\033[0m");
}
}
entry_t **files = state->files;
static const char *NORMAL_COLOR = "\e[0m";
static const char *CURSOR_COLOR = "\e[0;30;43m";
static const char *LINKDIR_COLOR = "\e[0;36m";
static const char *DIR_COLOR = "\e[0;34m";
static const char *LINK_COLOR = "\e[0;33m";
static const char *NORMAL_COLOR = "\033[0m";
static const char *CURSOR_COLOR = "\033[0;30;43m";
static const char *LINKDIR_COLOR = "\033[0;36m";
static const char *DIR_COLOR = "\033[0;34m";
static const char *LINK_COLOR = "\033[0;33m";
for (int i = state->scroll; i < state->scroll + height - 3; i++) {
if (lazy) {
if (i == state->cursor || i == lastcursor)
@ -360,7 +367,7 @@ static void render(bb_state_t *state, int lazy)
int y = i - state->scroll + 2;
term_move(0, y);
if (i >= state->nfiles) {
writez(termfd, "\e[K");
writez(termfd, "\033[K");
continue;
}
@ -370,7 +377,7 @@ static void render(bb_state_t *state, int lazy)
{ // Selection box:
if (IS_SELECTED(entry))
writez(termfd, "\e[43m \e[0m");
writez(termfd, "\033[41m \033[0m");
else
writez(termfd, " ");
}
@ -389,7 +396,7 @@ static void render(bb_state_t *state, int lazy)
writez(termfd, color);
for (char *col = state->columns; *col; ++col) {
if (col != state->columns) writez(termfd, "");
if (col != state->columns) writez(termfd, " ");
switch (*col) {
case 's': {
int j = 0;
@ -428,7 +435,8 @@ static void render(bb_state_t *state, int lazy)
break;
case 'n': {
int wrote = write_escaped(termfd, entry->d_name, entry->d_namlen, color);
ssize_t wrote = write(termfd, " ", 1);
wrote += write_escaped(termfd, entry->d_name, entry->d_namlen, color);
if (entry->d_isdir) {
writez(termfd, "/");
++wrote;
@ -439,7 +447,7 @@ static void render(bb_state_t *state, int lazy)
ssize_t pathlen;
if ((pathlen = readlink(entry->d_name, linkpath, sizeof(linkpath))) < 0)
err("readlink() failed");
writez(termfd, "\e[2m -> ");
writez(termfd, "\033[2m -> ");
wrote += 4;
wrote += write_escaped(termfd, linkpath, pathlen, color);
if (entry->d_isdir) {
@ -447,19 +455,19 @@ static void render(bb_state_t *state, int lazy)
++wrote;
}
}
for (int i = wrote; i < namewidth-1; i++)
while (wrote++ < namewidth - 1)
write(termfd, " ", 1);
break;
}
}
}
writez(termfd, " \e[0m\e[K"); // Reset color and attributes
writez(termfd, " \033[0m\033[K"); // Reset color and attributes
}
static const char *help = "Press '?' to see key bindings ";
term_move(0, height - 1);
writez(termfd, "\e[K");
term_move(MAX(0, width - strlen(help)), height - 1);
writez(termfd, "\033[K");
term_move(MAX(0, width - (int)strlen(help)), height - 1);
writez(termfd, help);
lastcursor = state->cursor;
lastscroll = state->scroll;
@ -469,17 +477,18 @@ static int compare_files(void *r, const void *v1, const void *v2)
{
char sort = *((char *)r);
int sign = IS_REVERSED(sort) ? -1 : 1;
sort = DESCENDING(sort);
const entry_t *f1 = *((const entry_t**)v1), *f2 = *((const entry_t**)v2);
int diff = -(f1->d_isdir - f2->d_isdir);
if (diff) return -diff; // Always sort dirs before files
if (sort == SORT_NONE) return 0;
if (COLUMN(sort) == SORT_NAME) {
if (sort == SORT_NONE || sort == SORT_RANDOM) return 0;
if (sort == SORT_NAME) {
const char *p1 = f1->d_name, *p2 = f2->d_name;
while (*p1 && *p2) {
char c1 = *p1, c2 = *p2;
if ('A' <= c1 && 'Z' <= c1) c1 = c1 - 'A' + 'a';
if ('A' <= c2 && 'Z' <= c2) c2 = c2 - 'A' + 'a';
int diff = (c1 - c2);
diff = (c1 - c2);
if ('0' <= c1 && c1 <= '9' && '0' <= c2 && c2 <= '9') {
long n1 = strtol(p1, (char**)&p1, 10);
long n2 = strtol(p2, (char**)&p2, 10);
@ -488,7 +497,7 @@ static int compare_files(void *r, const void *v1, const void *v2)
} else if (diff) {
return diff*sign;
} else {
++p1, ++p2;
++p1; ++p2;
}
}
return (*p1 - *p2)*sign;
@ -496,26 +505,26 @@ static int compare_files(void *r, const void *v1, const void *v2)
struct stat info1, info2;
lstat(f1->d_fullname, &info1);
lstat(f2->d_fullname, &info2);
switch (COLUMN(sort)) {
switch (sort) {
case SORT_PERM:
return -((info1.st_mode & 0x3FF) - (info2.st_mode & 0x3FF))*sign;
case SORT_SIZE:
return -(info1.st_size - info2.st_size)*sign;
return info1.st_size > info2.st_size ? -sign : sign;
case SORT_MTIME:
if (info1.st_mtimespec.tv_sec == info2.st_mtimespec.tv_sec)
return -(info1.st_mtimespec.tv_nsec - info2.st_mtimespec.tv_nsec)*sign;
return info1.st_mtimespec.tv_nsec > info2.st_mtimespec.tv_nsec ? -sign : sign;
else
return -(info1.st_mtimespec.tv_sec - info2.st_mtimespec.tv_sec)*sign;
return info1.st_mtimespec.tv_sec > info2.st_mtimespec.tv_sec ? -sign : sign;
case SORT_CTIME:
if (info1.st_ctimespec.tv_sec == info2.st_ctimespec.tv_sec)
return -(info1.st_ctimespec.tv_nsec - info2.st_ctimespec.tv_nsec)*sign;
return info1.st_ctimespec.tv_nsec > info2.st_ctimespec.tv_nsec ? -sign : sign;
else
return -(info1.st_ctimespec.tv_sec - info2.st_ctimespec.tv_sec)*sign;
return info1.st_ctimespec.tv_sec > info2.st_ctimespec.tv_sec ? -sign : sign;
case SORT_ATIME:
if (info1.st_atimespec.tv_sec == info2.st_atimespec.tv_sec)
return -(info1.st_atimespec.tv_nsec - info2.st_atimespec.tv_nsec)*sign;
return info1.st_atimespec.tv_nsec > info2.st_atimespec.tv_nsec ? -sign : sign;
else
return -(info1.st_atimespec.tv_sec - info2.st_atimespec.tv_sec)*sign;
return info1.st_atimespec.tv_sec > info2.st_atimespec.tv_sec ? -sign : sign;
}
return 0;
}
@ -579,7 +588,7 @@ 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;
e->next = NULL; e->atme = NULL;
--state->nselected;
return 1;
}
@ -646,7 +655,7 @@ static void populate_files(bb_state_t *state)
entry->d_isdir = S_ISDIR(statbuf.st_mode);
}
entry->d_namlen = dp->d_namlen;
entry->next = NULL, entry->atme = NULL;
entry->next = NULL; entry->atme = NULL;
if (state->nfiles >= filecap) {
filecap += 100;
@ -676,14 +685,37 @@ static int explore(bb_state_t *state)
sort_files:
qsort_r(&state->files[1], state->nfiles-1, sizeof(entry_t*), &state->sort, compare_files);
if (DESCENDING(state->sort) == SORT_RANDOM) {
entry_t **files = &state->files[1];
size_t ndirs = 0, nents = state->nfiles - 1;
for (int i = 0; i < nents; i++) {
if (state->files[i]->d_isdir) ++ndirs;
else break;
}
for (size_t i = 0; i < ndirs - 1; i++) {
//size_t j = i + rand() / (RAND_MAX / (ndirs - i) + 1);
size_t j = i + rand() / (RAND_MAX / (ndirs - 1 - i));
entry_t *tmp = files[j];
files[j] = files[i];
files[i] = tmp;
}
for (size_t i = ndirs; i < nents - 1; i++) {
//size_t j = i + rand() / (RAND_MAX / (nents - i) + 1);
size_t j = i + rand() / (RAND_MAX / (nents - 1 - i));
entry_t *tmp = files[j];
files[j] = files[i];
files[i] = tmp;
}
}
// Put the cursor on the first *real* file if one exists
if (state->cursor == 0 && state->nfiles > 1)
++state->cursor;
if (state->to_select) {
char *sel = state->to_select;
for (int i = 0; i < state->nfiles; i++) {
if (strcmp(state->to_select, state->files[i]->d_name) == 0) {
if (strcmp(sel, sel[0] == '/' ? state->files[i]->d_fullname : state->files[i]->d_name) == 0) {
state->cursor = i;
if (state->nfiles > height - 4)
state->scroll = MAX(0, i - MIN(SCROLLOFF, (height-4)/2));
@ -705,18 +737,19 @@ static int explore(bb_state_t *state)
next_input:
if (width != lastwidth || height != lastheight) {
scrolloff = MIN(SCROLLOFF, (height-4)/2);
lastwidth = width, lastheight = height;
lastwidth = width; lastheight = height;
lazy = 0;
goto redraw;
}
scan_cmdfile:
{
#define cleanup_cmd() do { if (cmdfile) fclose(cmdfile); if (cmd) free(cmd); } while (0)
FILE *cmdfile = fopen(cmdfilename, "r");
if (!cmdfile)
if (!cmdfile) {
if (!lazy) goto redraw;
goto get_user_input;
}
if (cmdpos)
fseek(cmdfile, cmdpos, SEEK_SET);
@ -737,34 +770,37 @@ static int explore(bb_state_t *state)
goto refresh;
case 'q': // quit
cleanup_cmd();
return 0;
goto quit;
case 's': // sort:, select:, scroll:, spread:
switch (cmd[1]) {
case 'o': // sort:
state->sort = *value;
queue_select(state, state->files[state->cursor]->d_name);
cleanup_cmd();
goto sort_files;
case 'c': { // scroll:
//int oldscroll = state->scroll;
int isabs = value[0] != '-' && value[0] != '+';
long delta = strtol(value, &value, 10);
if (*value == '%') delta = (delta * height)/100;
//int fudge = state->cursor - clamped(state->cursor, state->scroll + scrolloff, state->scroll + (height-4) - scrolloff);
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));
switch (*value) {
case 'n': case 'N': case 's': case 'S':
case 'p': case 'P': case 'm': case 'M':
case 'c': case 'C': case 'a': case 'A':
case 'r': case 'R':
state->sort = *value;
queue_select(state, state->files[state->cursor]->d_name);
cleanup_cmd();
goto sort_files;
}
state->cursor = clamped(state->cursor, state->scroll, state->scroll + (height-4));
/*
//if (!isabs && abs(state->scroll - oldscroll) == abs(delta)) {
state->cursor = clamped(state->cursor, state->scroll + scrolloff, state->scroll + (height-4) - scrolloff);
if (fudge && fudge < 0 != delta < 0)
state->cursor += fudge;
*/
goto next_cmd;
case 'c': { // scroll:
int isdelta = value[0] == '+' || value[0] == '-';
long n = strtol(value, &value, 10);
if (*value == '%')
n = (n * (value[1] == 'n' ? state->nfiles : height)) / 100;
if (isdelta) {
state->cursor += n;
if (state->nfiles > height-4)
state->scroll += n;
} else {
state->cursor = (int)n;
if (state->nfiles > height-4)
state->scroll = (int)n;
}
if (state->nfiles > height-4)
state->scroll = clamped(state->scroll, 0, state->nfiles-1 - (height-4));
state->cursor = clamped(state->cursor, 0, state->nfiles-1);
break;
}
@ -783,30 +819,43 @@ static int explore(bb_state_t *state)
}
break;
}
case 'c': { // cd:
char *rpbuf = realpath(value, NULL);
if (!rpbuf) continue;
if (strcmp(rpbuf, state->path) == 0) {
free(rpbuf);
continue;
case 'c':
switch (cmd[1]) {
case 'd': { // cd:
char *rpbuf = realpath(value, NULL);
if (!rpbuf) continue;
if (strcmp(rpbuf, state->path) == 0) {
free(rpbuf);
continue;
}
if (chdir(rpbuf) == 0) {
if (strcmp(value, "..") == 0)
queue_select(state, state->path);
strcpy(state->path, rpbuf);
free(rpbuf);
cleanup_cmd();
goto refresh;
} else {
free(rpbuf);
}
}
case 'o': // cols:
for (char *col = value, *dst = state->columns;
*col && dst < &state->columns[sizeof(state->columns)-1]; col++) {
*(dst++) = DESCENDING(*col);
*dst = '\0';
}
}
if (chdir(rpbuf) == 0) {
strcpy(state->path, rpbuf);
free(rpbuf);
cleanup_cmd();
goto refresh;
} else {
free(rpbuf);
}
}
case 't': // toggle:
case 't': { // toggle:
lazy = 0;
entry_t *e = find_file(state, value);
if (e) {
if (IS_SELECTED(e)) deselect_file(state, e);
else select_file(state, e);
}
case 'd': // deselect:
break;
}
case 'd': { // deselect:
lazy = 0;
if (strcmp(value, "*") == 0) {
clear_selection(state);
@ -814,6 +863,7 @@ static int explore(bb_state_t *state)
entry_t *e = find_file(state, value);
if (e) select_file(state, e);
}
}
case 'g': { // goto:
for (int i = 0; i < state->nfiles; i++) {
if (strcmp(value[0] == '/' ?
@ -836,14 +886,15 @@ static int explore(bb_state_t *state)
case 'm': { // move:
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;
int isdelta = value[0] == '-' || value[0] == '+';
long n = strtol(value, &value, 10);
if (*value == '%')
n = (n * (value[1] == 'n' ? state->nfiles : height)) / 100;
if (isdelta) state->cursor += n;
else state->cursor = n;
state->cursor = clamped(state->cursor, 0, state->nfiles-1);
delta = state->cursor - oldcur;
int delta = state->cursor - oldcur;
if (state->nfiles > height-4) {
if (delta > 0) {
@ -905,7 +956,7 @@ static int explore(bb_state_t *state)
case 'n': x += namewidth; break;
}
if (x >= mouse_x) {
if (COLUMN(state->sort) == *col)
if (DESCENDING(state->sort) == *col)
state->sort ^= SORT_DESCENDING;
else
state->sort = *col;
@ -952,32 +1003,32 @@ static int explore(bb_state_t *state)
case KEY_CTRL_H: {
term_move(0,height-1);
writez(termfd, "\e[K\e[33;1mPress any key...\e[0m");
writez(termfd, "\033[K\033[33;1mPress any key...\033[0m");
while ((key = term_getkey(termfd, &mouse_x, &mouse_y, 1000)) == -1)
;
term_move(0,height-1);
writez(termfd, "\e[K\e[1m<\e[33m");
writez(termfd, "\033[K\033[1m<\033[33m");
const char *name = keyname(key);
char buf[32] = {key};
char buf[32] = {(char)key};
if (name) writez(termfd, name);
else if (' ' <= key && key <= '~')
write(termfd, buf, 1);
else {
sprintf(buf, "\e[31m\\x%02X", key);
sprintf(buf, "\033[31m\\x%02X", key);
writez(termfd, buf);
}
writez(termfd, "\e[0;1m> is bound to: \e[34;1m");
writez(termfd, "\033[0;1m> is bound to: \033[34;1m");
for (int i = 0; bindings[i].keys[0] > 0; i++) {
for (int j = 0; bindings[i].keys[j]; j++) {
if (key == bindings[i].keys[j]) {
writez(termfd, bindings[i].description);
writez(termfd, "\e[0m");
writez(termfd, "\033[0m");
goto next_input;
}
}
}
writez(termfd, "--- nothing ---\e[0m");
writez(termfd, "--- nothing ---\033[0m");
goto next_input;
}
@ -1002,7 +1053,7 @@ static int explore(bb_state_t *state)
if (cmdpos != 0)
err("Command file still open");
term_move(0, height-1);
//writez(termfd, "\e[K");
//writez(termfd, "\033[K");
if (binding->flags & NORMAL_TERM) {
default_screen();
show_cursor();
@ -1019,6 +1070,7 @@ static int explore(bb_state_t *state)
}
goto next_input;
quit:
return 0;
}
@ -1026,12 +1078,12 @@ static void print_bindings(int verbose)
{
struct winsize sz = {0};
ioctl(STDOUT_FILENO, TIOCGWINSZ, &sz);
int width = sz.ws_col;
if (width == 0) width = 80;
int _width = sz.ws_col;
if (_width == 0) _width = 80;
char buf[1024];
char *kb = "Key Bindings";
printf("\n\e[33;1;4m\e[%dG%s\e[0m\n\n", (width-(int)strlen(kb))/2, kb);
printf("\n\033[33;1;4m\033[%dG%s\033[0m\n\n", (_width-(int)strlen(kb))/2, kb);
for (int i = 0; bindings[i].keys[0]; i++) {
char *p = buf;
for (int j = 0; bindings[i].keys[j]; j++) {
@ -1043,17 +1095,17 @@ static void print_bindings(int verbose)
else if (' ' <= key && key <= '~')
p += sprintf(p, "%c", (char)key);
else
p += sprintf(p, "\e[31m\\x%02X", key);
p += sprintf(p, "\033[31m\\x%02X", key);
}
*p = '\0';
printf("\e[1m\e[%dG%s\e[0m", width/2 - 1 - (int)strlen(buf), buf);
printf("\e[0m\e[%dG\e[34;1m%s\e[0m", width/2 + 1, bindings[i].description);
printf("\033[1m\033[%dG%s\033[0m", _width/2 - 1 - (int)strlen(buf), buf);
printf("\033[0m\033[%dG\033[34;1m%s\033[0m", _width/2 + 1, bindings[i].description);
if (verbose) {
printf("\n\e[%dG\e[0;32m", MAX(1, (width - (int)strlen(bindings[i].command))/2));
printf("\n\033[%dG\033[0;32m", MAX(1, (_width - (int)strlen(bindings[i].command))/2));
fflush(stdout);
write_escaped(STDOUT_FILENO, bindings[i].command, strlen(bindings[i].command), "\e[0;32m");
write_escaped(STDOUT_FILENO, bindings[i].command, strlen(bindings[i].command), "\033[0;32m");
}
printf("\e[0m\n");
printf("\033[0m\n");
}
printf("\n");
}
@ -1112,14 +1164,6 @@ int main(int argc, char *argv[])
printf("Usage: bb [-h/--help] [-s] [-b] [-0] [path]\n");
return 1;
}
if (strcmp(argv[i], "--columns") == 0 && i + 1 < argc) {
strncpy(state.columns, argv[++i], sizeof(state.columns));
continue;
}
if (strcmp(argv[i], "--sort") == 0 && i + 1 < argc) {
state.sort = *argv[++i];
continue;
}
if (argv[i][0] == '-' && argv[i][1] == '-') {
if (argv[i][2] == '\0') break;
continue;

12
keys.h
View File

@ -2,8 +2,8 @@
* Definitions of some key character constants
*/
#ifndef _KEYS_DEFINED
#define _KEYS_DEFINED
#ifndef FILE__KEYS_H
#define FILE__KEYS_H
#include <poll.h>
@ -87,6 +87,9 @@
#define ESC_DELAY 10
int term_getkey(int fd, int *mouse_x, int *mouse_y, int timeout);
const char *keyname(int key);
static inline int nextchar(int fd, int timeout)
{
char c;
@ -174,7 +177,8 @@ int term_getkey(int fd, int *mouse_x, int *mouse_y, int timeout)
y = y * 10 + (buf - '0');
if (buf != 'm' && buf != 'M') return -1;
*mouse_x = x - 1, *mouse_y = y - 1;
*mouse_x = x - 1;
*mouse_y = y - 1;
if (buf == 'm')
return KEY_MOUSE_RELEASE;
@ -205,7 +209,7 @@ int term_getkey(int fd, int *mouse_x, int *mouse_y, int timeout)
return -1;
}
const char *keyname(key)
const char *keyname(int key)
{
// TODO: currently only the keys I'm using are named
switch (key) {