bb/bb.c

973 lines
33 KiB
C
Raw Normal View History

2019-05-20 19:28:47 -07:00
/*
* Bruce's Browser (bb)
* Copyright 2019 Bruce Hill
* Released under the MIT license
*/
#include <dirent.h>
#include <fcntl.h>
#include <signal.h>
2019-05-20 19:28:47 -07:00
#include <stdarg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <sys/dir.h>
#include <sys/errno.h>
#include <sys/ioctl.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <termios.h>
#include <time.h>
#include <unistd.h>
2019-05-21 20:06:53 -07:00
#include "config.h"
2019-05-20 19:28:47 -07:00
#include "keys.h"
#define MAX(a,b) ((a) < (b) ? (b) : (a))
#define MIN(a,b) ((a) > (b) ? (b) : (a))
#define MAX_PATH 4096
#define writez(fd, str) write(fd, str, strlen(str))
2019-05-21 03:59:30 -07:00
#define IS_SELECTED(p) ((p)->atme)
2019-05-20 19:28:47 -07:00
static struct termios orig_termios;
static int termfd;
static int width, height;
static int mouse_x, mouse_y;
typedef enum {
SORT_ALPHA = 0,
SORT_SIZE,
SORT_BITS,
SORT_DATE
} sortmethod_t;
typedef struct entry_s {
2019-05-21 03:59:30 -07:00
struct entry_s *next, **atme;
int visible : 1, d_isdir : 1;
ino_t d_ino;
__uint16_t d_reclen;
__uint8_t d_type;
__uint8_t d_namlen;
char *d_name;
char d_fullname[0];
} entry_t;
typedef struct {
char *path;
entry_t *firstselected, **files;
size_t nselected, nfiles;
int scroll, cursor;
struct timespec lastclick;
int showhidden, sort_reverse;
sortmethod_t sortmethod;
} bb_state_t;
static void err(const char *msg, ...);
2019-05-20 19:28:47 -07:00
static void update_term_size(void)
{
struct winsize sz = {0};
ioctl(termfd, TIOCGWINSZ, &sz);
width = sz.ws_col;
height = sz.ws_row;
}
static void do_nothing(int _)
{
// Used as SIGINT handler
}
2019-05-20 19:28:47 -07:00
static void init_term()
{
termfd = open("/dev/tty", O_RDWR);
tcgetattr(termfd, &orig_termios);
struct termios tios;
memcpy(&tios, &orig_termios, sizeof(tios));
tios.c_iflag &= ~(IGNBRK | BRKINT | PARMRK | ISTRIP
| INLCR | IGNCR | ICRNL | IXON);
tios.c_oflag &= ~OPOST;
tios.c_lflag &= ~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
tios.c_cflag &= ~(CSIZE | PARENB);
tios.c_cflag |= CS8;
tios.c_cc[VMIN] = 0;
tios.c_cc[VTIME] = 0;
tcsetattr(termfd, TCSAFLUSH, &tios);
// xterm-specific:
writez(termfd, "\e[?1049h");
2019-05-20 19:28:47 -07:00
update_term_size();
// Initiate mouse tracking:
writez(termfd, "\e[?1000h\e[?1002h\e[?1015h\e[?1006h");
// hide cursor
writez(termfd, "\e[?25l");
2019-05-20 19:28:47 -07:00
}
static void close_term()
{
// xterm-specific:
writez(termfd, "\e[?1049l");
// Show cursor:
writez(termfd, "\e[?25h");
2019-05-20 19:28:47 -07:00
tcsetattr(termfd, TCSAFLUSH, &orig_termios);
close(termfd);
}
static void err(const char *msg, ...)
{
close_term();
va_list args;
va_start(args, msg);
int len = fprintf(stderr, msg, args);
2019-05-20 19:28:47 -07:00
va_end(args);
if (errno)
fprintf(stderr, "\n%s", strerror(errno));
fprintf(stderr, "\n");
_exit(1);
}
static pid_t run_cmd(int *child_out, int *child_in, const char *cmd, ...)
2019-05-20 19:28:47 -07:00
{
int child_outfds[2], child_infds[2];
2019-05-20 19:28:47 -07:00
pid_t child;
if (child_out)
pipe(child_outfds);
if (child_in)
pipe(child_infds);
2019-05-20 19:28:47 -07:00
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]);
}
2019-05-20 19:28:47 -07:00
} 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]);
}
2019-05-20 19:28:47 -07:00
char *formatted_cmd;
va_list args;
va_start(args, cmd);
int len = vasprintf(&formatted_cmd, cmd, args);
2019-05-20 19:28:47 -07:00
va_end(args);
if (formatted_cmd)
execlp("sh", "sh", "-c", formatted_cmd);
err("Failed to execute command %d: '%s'", len, formatted_cmd);
2019-05-20 19:28:47 -07:00
_exit(0);
}
return child;
}
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);
if (len > 0)
write(termfd, buf, len);
}
static void render(bb_state_t *state)
2019-05-20 19:28:47 -07:00
{
writez(termfd, "\e[2J\e[0;1m"); // Clear, reset color + bold
term_move(0,0);
writez(termfd, state->path);
2019-05-21 03:59:30 -07:00
term_move(0,1);
{ // 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[36] = state->sortmethod == SORT_BITS ? (state->sort_reverse ? '-' : '+') : ' ';
buf[42] = state->sortmethod == SORT_ALPHA ? (state->sort_reverse ? '-' : '+') : ' ';
writez(termfd, buf);
}
2019-05-20 19:28:47 -07:00
entry_t **files = state->files;
for (int i = state->scroll; i < state->scroll + height - 3 && i < state->nfiles; i++) {
entry_t *entry = files[i];
2019-05-21 16:32:26 -07:00
struct stat info = {0};
lstat(entry->d_fullname, &info);
2019-05-20 19:28:47 -07:00
int x = 0;
2019-05-21 03:59:30 -07:00
int y = i - state->scroll + 2;
2019-05-20 19:28:47 -07:00
term_move(x, y);
2019-05-21 16:32:26 -07:00
{ // Selection box:
if (IS_SELECTED(entry))
writez(termfd, "\e[43m \e[0m");
else
writez(termfd, " ");
2019-05-20 19:28:47 -07:00
2019-05-21 16:32:26 -07:00
if (i == state->cursor)
writez(termfd, "\e[30;47m");
else if (entry->d_isdir && entry->d_type == DT_LNK)
writez(termfd, "\e[36m");
else if (entry->d_isdir)
writez(termfd, "\e[34m");
else if (entry->d_type == DT_LNK)
writez(termfd, "\e[33m");
}
2019-05-20 19:28:47 -07:00
2019-05-21 16:32:26 -07:00
{ // Filesize:
int j = 0;
const char* units = "BKMGTPEZY";
double bytes = (double)info.st_size;
while (bytes > 1024) {
bytes /= 1024;
j++;
}
char buf[16] = {0};
2019-05-21 03:59:30 -07:00
sprintf(buf, "%6.*f%c │", j > 0 ? 1 : 0, bytes, units[j]);
writez(termfd, buf);
2019-05-20 19:28:47 -07:00
}
2019-05-21 16:32:26 -07:00
{ // Date:
char buf[64];
strftime(buf, sizeof(buf), "%l:%M%p %b %e %Y", localtime(&(info.st_mtime)));
writez(termfd, buf);
2019-05-21 03:59:30 -07:00
//writez(termfd, " ");
writez(termfd, "");
}
2019-05-21 16:32:26 -07:00
{ // Permissions:
2019-05-21 03:59:30 -07:00
char buf[] = {
'0' + ((info.st_mode >> 6) & 7),
'0' + ((info.st_mode >> 3) & 7),
'0' + ((info.st_mode >> 0) & 7),
};
write(termfd, buf, 5);
writez(termfd, "");
}
2019-05-20 19:28:47 -07:00
2019-05-21 16:32:26 -07:00
{ // Name:
write(termfd, entry->d_name, entry->d_namlen);
2019-05-21 03:59:30 -07:00
if (entry->d_isdir)
writez(termfd, "/");
2019-05-21 16:32:26 -07:00
if (entry->d_type == DT_LNK) {
2019-05-21 17:13:28 -07:00
char linkpath[MAX_PATH+1] = {0};
2019-05-21 16:32:26 -07:00
ssize_t pathlen;
if ((pathlen = readlink(entry->d_name, linkpath, sizeof(linkpath))) < 0)
err("readlink() failed");
//writez(termfd, "\e[36m -> "); // Cyan FG
writez(termfd, " -> ");
write(termfd, linkpath, pathlen);
if (entry->d_isdir)
writez(termfd, "/");
}
2019-05-20 19:28:47 -07:00
}
2019-05-21 03:59:30 -07:00
writez(termfd, " \e[0m"); // Reset color and attributes
2019-05-20 19:28:47 -07:00
}
static const char *help = "Press '?' to see key bindings ";
2019-05-20 19:28:47 -07:00
char buf[32] = {0};
int len = snprintf(buf, sizeof(buf), " %lu selected", state->nselected);
if (strlen(help) + len < width + 1) {
term_move(0, height - 1);
write(termfd, buf, len);
}
term_move(MAX(0, width - strlen(help)), height - 1);
writez(termfd, help);
2019-05-20 19:28:47 -07:00
}
static int compare_alpha(void *r, const void *v1, const void *v2)
2019-05-20 19:28:47 -07:00
{
int sign = *((int *)r) ? -1 : 1;
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
const char *p1 = f1->d_name, *p2 = f2->d_name;
2019-05-20 19:28:47 -07:00
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);
if ('0' <= c1 && c1 <= '9' && '0' <= c2 && c2 <= '9') {
2019-05-20 19:28:47 -07:00
long n1 = strtol(p1, (char**)&p1, 10);
long n2 = strtol(p2, (char**)&p2, 10);
diff = ((p1 - f1->d_name) - (p2 - f2->d_name)) || (n1 - n2);
if (diff) return diff*sign;
2019-05-20 19:28:47 -07:00
} else if (diff) {
return diff*sign;
2019-05-20 19:28:47 -07:00
} else {
++p1, ++p2;
}
}
return (*p1 - *p2)*sign;
}
static int compare_perm(void *r, const void *v1, const void *v2)
{
int sign = *((int *)r) ? -1 : 1;
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
struct stat info1, info2;
lstat(f1->d_fullname, &info1);
lstat(f2->d_fullname, &info2);
return -((info1.st_mode & 0x3FF) - (info2.st_mode & 0x3FF))*sign;
}
static int compare_size(void *r, const void *v1, const void *v2)
{
int sign = *((int *)r) ? -1 : 1;
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
struct stat info1, info2;
lstat(f1->d_fullname, &info1);
lstat(f2->d_fullname, &info2);
return -(info1.st_size - info2.st_size)*sign;
}
static int compare_date(void *r, const void *v1, const void *v2)
{
int sign = *((int *)r) ? -1 : 1;
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
struct stat info1, info2;
lstat(f1->d_fullname, &info1);
lstat(f2->d_fullname, &info2);
if (info1.st_mtimespec.tv_sec == info2.st_mtimespec.tv_sec)
return -(info1.st_mtimespec.tv_nsec - info2.st_mtimespec.tv_nsec)*sign;
else
return -(info1.st_mtimespec.tv_sec - info2.st_mtimespec.tv_sec)*sign;
2019-05-20 19:28:47 -07:00
}
static void write_selection(int fd, entry_t *firstselected, char sep)
2019-05-20 19:28:47 -07:00
{
while (firstselected) {
const char *p = firstselected->d_fullname;
2019-05-20 19:28:47 -07:00
while (*p) {
const char *p2 = strchr(p, '\n');
if (!p2) p2 = p + strlen(p);
2019-05-20 19:28:47 -07:00
write(fd, p, p2 - p);
if (*p2 == '\n' && sep == '\n')
2019-05-20 19:28:47 -07:00
write(fd, "\\", 1);
p = p2;
}
write(fd, &sep, 1);
firstselected = firstselected->next;
2019-05-20 19:28:47 -07:00
}
}
2019-05-21 20:06:53 -07:00
static void clear_selection(bb_state_t *state)
{
entry_t **tofree = calloc(state->nselected, sizeof(entry_t*));
int i = 0;
for (entry_t *e = state->firstselected; e; e = e->next) {
if (!e->visible) tofree[i++] = e;
*e->atme = NULL;
e->atme = NULL;
}
while (i) free(tofree[--i]);
free(tofree);
state->nselected = 0;
}
static void explore(char *path, int print_dir, int print_selection, char sep)
2019-05-20 19:28:47 -07:00
{
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);
if (!path) err("allocation failure");
strcpy(path, tmp);
tmp = NULL;
2019-05-21 17:13:28 -07:00
char to_select[MAX_PATH+1] = {0};
bb_state_t state = {0};
memset(&state, 0, sizeof(bb_state_t));
tail_call:
if (state.files) {
for (int i = 0; i < state.nfiles; i++) {
entry_t *e = state.files[i];
e->visible = 0;
if (!IS_SELECTED(e))
free(e);
}
free(state.files);
state.files = NULL;
}
state.nfiles = 0;
state.path = path;
{ // Populate the file list:
// Hash inode -> entry_t with linear probing
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);
for (entry_t *p = state.firstselected; p; p = p->next) {
int probe = ((int)p->d_ino) % hashsize;
while (selecthash[probe])
probe = (probe + 1) % hashsize;
selecthash[probe] = p;
}
DIR *dir = opendir(state.path);
if (!dir)
err("Couldn't open dir: %s", state.path);
if (chdir(state.path) != 0)
err("Couldn't chdir into %s", state.path);
struct dirent *dp;
size_t pathlen = strlen(state.path);
size_t filecap = 0;
while ((dp = readdir(dir)) != NULL) {
if (dp->d_name[0] == '.' && dp->d_name[1] == '\0')
continue;
if (!state.showhidden && dp->d_name[0] == '.' && !(dp->d_name[1] == '.' && dp->d_name[2] == '\0'))
continue;
// Hashed lookup from selected:
if (state.nselected > 0) {
for (int probe = ((int)dp->d_ino) % hashsize; selecthash[probe]; probe = (probe + 1) % hashsize) {
if (selecthash[probe]->d_ino == dp->d_ino) {
selecthash[probe]->visible = 1;
state.files[state.nfiles++] = selecthash[probe];
goto next_file;
}
2019-05-20 19:28:47 -07:00
}
}
entry_t *entry = malloc(sizeof(entry_t) + pathlen + dp->d_namlen + 2);
strncpy(entry->d_fullname, state.path, pathlen);
entry->d_fullname[pathlen] = '/';
entry->d_name = &entry->d_fullname[pathlen + 1];
strncpy(entry->d_name, dp->d_name, dp->d_namlen + 1);
entry->d_ino = dp->d_ino;
entry->d_reclen = dp->d_reclen;
entry->d_type = dp->d_type;
entry->d_isdir = dp->d_type == DT_DIR;
entry->visible = 1;
if (!entry->d_isdir && entry->d_type == DT_LNK) {
struct stat statbuf;
if (stat(entry->d_fullname, &statbuf) == 0)
entry->d_isdir = S_ISDIR(statbuf.st_mode);
}
entry->d_namlen = dp->d_namlen;
entry->next = NULL, entry->atme = NULL;
if (state.nfiles >= filecap) {
filecap += 100;
state.files = realloc(state.files, filecap*sizeof(entry_t*));
if (!state.files) err("allocation failure");
}
state.files[state.nfiles++] = entry;
next_file:;
}
closedir(dir);
free(selecthash);
if (state.nfiles == 0) err("No files found (not even '..')");
2019-05-20 19:28:47 -07:00
}
state.cursor = 0;
state.scroll = 0;
clock_gettime(CLOCK_MONOTONIC, &state.lastclick);
int (*cmp)(void*, const void*, const void*);
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_BITS) cmp = compare_perm;
qsort_r(&state.files[1], state.nfiles-1, sizeof(entry_t*), &state.sort_reverse, cmp);
2019-05-20 19:28:47 -07:00
// Put the cursor on the first *real* file if one exists
if (state.nfiles > 1)
++state.cursor;
2019-05-20 19:28:47 -07:00
if (to_select[0]) {
for (int i = 0; i < state.nfiles; i++) {
if (strcmp(to_select, state.files[i]->d_name) == 0) {
state.cursor = i;
2019-05-21 17:13:28 -07:00
if (state.nfiles > height - 4)
state.scroll = MAX(0, i - MIN(SCROLLOFF, (height-4)/2));
2019-05-20 19:28:47 -07:00
break;
}
}
to_select[0] = '\0';
2019-05-20 19:28:47 -07:00
}
int picked, scrolloff;
while (1) {
redraw:
render(&state);
2019-05-20 19:28:47 -07:00
skip_redraw:
2019-05-21 03:59:30 -07:00
scrolloff = MIN(SCROLLOFF, (height-4)/2);
2019-05-21 20:06:53 -07:00
int key = term_getkey(termfd, &mouse_x, &mouse_y);
switch (key) {
2019-05-20 19:28:47 -07:00
case KEY_MOUSE_LEFT: {
struct timespec clicktime;
clock_gettime(CLOCK_MONOTONIC, &clicktime);
double dt_ms = 1e3*(double)(clicktime.tv_sec - state.lastclick.tv_sec);
dt_ms += 1e-6*(double)(clicktime.tv_nsec - state.lastclick.tv_nsec);
state.lastclick = clicktime;
if (mouse_y == 1) {
// Size Date Perm Name
if (mouse_x <= 8)
goto sort_size;
else if (mouse_x <= 30)
goto sort_date;
else if (mouse_x <= 35)
goto sort_perm;
else
goto sort_alpha;
} else if (mouse_y >= 2 && state.scroll + (mouse_y - 2) < state.nfiles) {
2019-05-21 03:59:30 -07:00
int clicked = state.scroll + (mouse_y - 2);
2019-05-20 19:28:47 -07:00
if (dt_ms > 200) {
// Single click
if (mouse_x == 0) {
// Toggle
picked = clicked;
goto toggle;
} else {
state.cursor = clicked;
goto redraw;
2019-05-20 19:28:47 -07:00
}
} else {
// Double click
picked = clicked;
goto open_file;
}
}
break;
2019-05-20 19:28:47 -07:00
}
case KEY_CTRL_C:
close_term();
if (print_dir)
printf("%s\n", original_path);
_exit(1);
return;
case KEY_CTRL_Z:
close_term();
raise(SIGTSTP);
init_term();
goto redraw;
case 'q': case 'Q':
2019-05-20 19:28:47 -07:00
goto done;
case KEY_MOUSE_WHEEL_DOWN:
if (state.cursor >= state.nfiles - 1)
goto skip_redraw;
++state.cursor;
if (state.nfiles <= height - 4)
goto redraw;
++state.scroll;
goto redraw;
case KEY_MOUSE_WHEEL_UP:
if (state.cursor <= 0)
goto skip_redraw;
--state.cursor;
if (state.nfiles <= height - 4)
goto redraw;
--state.scroll;
if (state.scroll < 0)
state.scroll = 0;
goto redraw;
case KEY_CTRL_D: case KEY_PGDN:
if (state.cursor == state.nfiles - 1)
goto skip_redraw;
2019-05-21 03:59:30 -07:00
state.cursor = MIN(state.nfiles - 1, state.cursor + (height - 4) / 2);
if (state.nfiles <= height - 4)
2019-05-20 19:28:47 -07:00
goto redraw;
2019-05-21 03:59:30 -07:00
state.scroll += (height - 4)/2;
if (state.scroll > state.nfiles - (height - 4) - 1)
state.scroll = state.nfiles - (height - 4) - 1;
2019-05-20 19:28:47 -07:00
goto redraw;
case KEY_CTRL_U: case KEY_PGUP:
2019-05-21 03:59:30 -07:00
state.cursor = MAX(0, state.cursor - (height - 4) / 2);
if (state.nfiles <= height - 4)
2019-05-20 19:28:47 -07:00
goto redraw;
2019-05-21 03:59:30 -07:00
state.scroll -= (height - 4)/2;
if (state.scroll < 0)
state.scroll = 0;
2019-05-20 19:28:47 -07:00
goto redraw;
case 'g': case KEY_HOME:
2019-05-21 03:59:30 -07:00
state.cursor = 0;
state.scroll = 0;
goto redraw;
case 'G': case KEY_END:
2019-05-21 03:59:30 -07:00
state.cursor = state.nfiles - 1;
if (state.nfiles > height - 4)
state.scroll = state.nfiles - (height - 4) - 1;
2019-05-21 03:59:30 -07:00
goto redraw;
case ' ':
picked = state.cursor;
2019-05-20 19:28:47 -07:00
toggle:
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];
2019-05-21 03:59:30 -07:00
if (e->next) e->next->atme = e->atme;
*(e->atme) = e->next;
e->next = NULL, e->atme = NULL;
--state.nselected;
2019-05-20 19:28:47 -07:00
} else {
toggle_on:;
entry_t *e = state.files[picked];
2019-05-21 03:59:30 -07:00
if (state.firstselected)
state.firstselected->atme = &e->next;
e->next = state.firstselected;
e->atme = &state.firstselected;
state.firstselected = e;
++state.nselected;
}
goto redraw;
2019-05-20 19:28:47 -07:00
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;
2019-05-21 20:06:53 -07:00
case KEY_ESC:
clear_selection(&state);
2019-05-20 19:28:47 -07:00
goto redraw;
case KEY_F5: case KEY_CTRL_R:
strcpy(to_select, state.files[state.cursor]->d_name);
goto tail_call;
2019-05-21 20:06:53 -07:00
case 'j': case KEY_ARROW_DOWN:
if (state.cursor >= state.nfiles - 1)
2019-05-20 19:28:47 -07:00
goto skip_redraw;
++state.cursor;
2019-05-21 03:59:30 -07:00
if (state.cursor > state.scroll + height - 4 - 1 - scrolloff && state.scroll < state.nfiles - (height - 4)) {
++state.scroll;
2019-05-20 19:28:47 -07:00
}
goto redraw;
2019-05-21 20:06:53 -07:00
case 'k': case KEY_ARROW_UP:
if (state.cursor <= 0)
2019-05-20 19:28:47 -07:00
goto skip_redraw;
--state.cursor;
if (state.cursor < state.scroll + scrolloff && state.scroll > 0) {
--state.scroll;
2019-05-20 19:28:47 -07:00
}
goto redraw;
2019-05-20 19:28:47 -07:00
case 'J':
if (state.cursor < state.nfiles - 1) {
if (IS_SELECTED(state.files[state.cursor])) {
picked = ++state.cursor;
2019-05-21 03:59:30 -07:00
if (!IS_SELECTED(state.files[picked]))
goto toggle_on;
} else {
picked = ++state.cursor;
2019-05-21 03:59:30 -07:00
if (IS_SELECTED(state.files[picked]))
goto toggle_off;
}
2019-05-21 03:59:30 -07:00
goto redraw;
2019-05-20 19:28:47 -07:00
}
goto skip_redraw;
2019-05-20 19:28:47 -07:00
case 'K':
if (state.cursor > 0) {
if (IS_SELECTED(state.files[state.cursor])) {
picked = --state.cursor;
2019-05-21 03:59:30 -07:00
if (!IS_SELECTED(state.files[picked]))
goto toggle_on;
} else {
picked = --state.cursor;
2019-05-21 03:59:30 -07:00
if (IS_SELECTED(state.files[picked]))
goto toggle_off;
}
2019-05-21 03:59:30 -07:00
goto redraw;
2019-05-20 19:28:47 -07:00
}
goto skip_redraw;
2019-05-21 20:06:53 -07:00
case 'h': case KEY_ARROW_LEFT:
2019-05-21 03:59:30 -07:00
picked = 0;
goto open_file;
case 's':
// Change sorting method:
term_move(0, height-1);
writez(termfd, "\e[K\e[1mSort by (a)lphabetic (s)ize (t)ime (p)ermissions:\e[0m \e[?25h");
try_sort_again:
switch (term_getkey(termfd, &mouse_x, &mouse_y)) {
case 'a': case 'A':
sort_alpha:
if (state.sortmethod == SORT_ALPHA)
state.sort_reverse ^= 1;
else {
state.sortmethod = SORT_ALPHA;
state.sort_reverse = 0;
}
break;
case 'p': case 'P':
sort_perm:
if (state.sortmethod == SORT_BITS)
state.sort_reverse ^= 1;
else {
state.sortmethod = SORT_BITS;
state.sort_reverse = 0;
}
break;
case 's': case 'S':
sort_size:
if (state.sortmethod == SORT_SIZE)
state.sort_reverse ^= 1;
else {
state.sortmethod = SORT_SIZE;
state.sort_reverse = 0;
}
break;
case 't': case 'T':
sort_date:
if (state.sortmethod == SORT_DATE)
state.sort_reverse ^= 1;
else {
state.sortmethod = SORT_DATE;
state.sort_reverse = 0;
}
break;
case -1:
goto try_sort_again;
default:
writez(termfd, "\e[?25l");
goto redraw;
}
// Hide cursor again
writez(termfd, "\e[?25l");
goto sort_files;
2019-05-21 03:59:30 -07:00
case '.':
state.showhidden ^= 1;
goto tail_call;
2019-05-21 20:06:53 -07:00
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';
2019-05-21 17:13:28 -07:00
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,
2019-05-20 19:28:47 -07:00
#ifdef __APPLE__
"if file -bI %s | grep '^text/' >/dev/null; then $EDITOR %s; else open %s; fi",
2019-05-20 19:28:47 -07:00
#else
"if file -bi %s | grep '^text/' >/dev/null; then $EDITOR %s; else xdg-open %s; fi",
2019-05-20 19:28:47 -07:00
#endif
name, name, name);
waitpid(child, NULL, 0);
init_term();
goto redraw;
2019-05-20 19:28:47 -07:00
}
break;
}
2019-05-21 17:13:28 -07:00
case 'f': case '/': {
close_term();
int fd;
if (state.showhidden)
2019-05-21 20:06:53 -07:00
run_cmd(&fd, NULL, "find | cut -d/ -f2- | " PROG_FUZZY);
2019-05-21 17:13:28 -07:00
else
2019-05-21 20:06:53 -07:00
run_cmd(&fd, NULL, "find -not -name '\\.*' -not -path '*/\\.*' | cut -d/ -f2- | " PROG_FUZZY);
2019-05-21 17:13:28 -07:00
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");
writez(fd, "\n \e[33;1mKey Bindings:\e[0m\n");
2019-05-21 20:06:53 -07:00
for (int i = 0; bindings[i].key; i++) {
if (bindings[i].key > 0) continue;
writez(fd, "\e[1m");
char *tab = strchr(bindings[i].command, '\t');
const char *spaces = " ";
write(fd, spaces, 16 - (tab - bindings[i].command));
write(fd, bindings[i].command, tab - bindings[i].command);
write(fd, " ", 1);
writez(fd, tab + 1);
writez(fd, "\e[0m\n");
}
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[0m\n");
}
writez(fd, "\n");
close(fd);
waitpid(child, NULL, 0);
init_term();
goto redraw;
}
case -1:
goto skip_redraw;
default:
// Search user-defined key bindings from config.h:
for (int i = 0; bindings[i].key > 0; i++) {
2019-05-21 20:06:53 -07:00
if (key == bindings[i].key) {
term_move(0, height-1);
struct termios cur_tios;
if (!(bindings[i].flags & ONSCREEN)) {
2019-05-21 20:06:53 -07:00
close_term();
} else {
tcgetattr(termfd, &cur_tios);
struct termios tios;
memcpy(&tios, &orig_termios, sizeof(tios));
tcsetattr(termfd, TCSAFLUSH, &tios);
// Show cursor
writez(termfd, "\e[?25h");
}
2019-05-21 20:06:53 -07:00
int scriptinfd;
2019-05-21 20:06:53 -07:00
pid_t child;
sig_t old_handler = signal(SIGINT, do_nothing);
child = run_cmd(NULL, &scriptinfd, bindings[i].command);
2019-05-21 20:06:53 -07:00
if (!(bindings[i].flags & NO_FILES)) {
char sep = (bindings[i].flags & NULL_SEP) ? '\0' : '\n';
2019-05-21 20:06:53 -07:00
if (state.nselected > 0) {
write_selection(scriptinfd, state.firstselected, sep);
2019-05-21 20:06:53 -07:00
} else if (strcmp(state.files[state.cursor]->d_name, "..") != 0) {
write(scriptinfd, state.files[state.cursor]->d_name, state.files[state.cursor]->d_namlen);
2019-05-21 20:06:53 -07:00
}
}
close(scriptinfd);
2019-05-21 20:06:53 -07:00
waitpid(child, NULL, 0);
signal(SIGINT, old_handler);
if (!(bindings[i].flags & ONSCREEN)) {
init_term();
} else {
tcsetattr(termfd, TCSAFLUSH, &cur_tios);
// hide cursor
writez(termfd, "\e[?25l");
}
2019-05-21 20:06:53 -07:00
if (bindings[i].flags & CLEAR_SELECTION)
clear_selection(&state);
if (bindings[i].flags & REFRESH) {
strcpy(to_select, state.files[state.cursor]->d_name);
2019-05-21 20:06:53 -07:00
goto tail_call;
}
2019-05-21 20:06:53 -07:00
goto redraw;
}
}
2019-05-20 19:28:47 -07:00
goto skip_redraw;
}
goto skip_redraw;
}
done:
close_term();
2019-05-21 04:06:50 -07:00
if (print_dir)
printf("%s\n", state.path);
if (print_selection)
write_selection(STDOUT_FILENO, state.firstselected, sep);
2019-05-20 19:28:47 -07:00
return;
}
int main(int argc, char *argv[])
{
2019-05-21 17:13:28 -07:00
char _realpath[MAX_PATH+1];
2019-05-21 04:06:50 -07:00
char *path = ".";
char sep = '\n';
2019-05-21 04:06:50 -07:00
int print_dir = 0, print_selection = 0;
for (int i = 1; i < argc; i++) {
if (strcmp(argv[i], "-d") == 0) {
print_dir = 1;
} else if (strcmp(argv[i], "-0") == 0) {
sep = '\0';
2019-05-21 04:06:50 -07:00
} else if (strcmp(argv[i], "-s") == 0) {
print_selection = 1;
} else if (path[0]) {
2019-05-21 04:06:50 -07:00
path = argv[i];
break;
}
}
init_term();
2019-05-21 04:06:50 -07:00
if (!realpath(path, _realpath))
2019-05-20 19:28:47 -07:00
err("realpath failed");
explore(_realpath, print_dir, print_selection, sep);
2019-05-20 19:28:47 -07:00
done:
return 0;
}