2019-05-20 19:28:47 -07:00
|
|
|
/*
|
2019-05-22 01:56:39 -07:00
|
|
|
* Bitty Browser (bb)
|
2019-05-20 19:28:47 -07:00
|
|
|
* Copyright 2019 Bruce Hill
|
|
|
|
* Released under the MIT license
|
|
|
|
*/
|
|
|
|
#include <dirent.h>
|
|
|
|
#include <fcntl.h>
|
2019-05-22 00:25:25 -07:00
|
|
|
#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
|
2019-05-23 00:57:25 -07:00
|
|
|
#define startswith(str, start) (strncmp(str, start, strlen(start)) == 0)
|
2019-05-20 19:28:47 -07:00
|
|
|
#define writez(fd, str) write(fd, str, strlen(str))
|
2019-05-23 01:37:41 -07:00
|
|
|
#define IS_SELECTED(p) (((p)->atme) != NULL)
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
#define KEY_DELAY 50
|
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;
|
|
|
|
|
2019-05-21 05:10:39 -07:00
|
|
|
typedef enum {
|
|
|
|
SORT_ALPHA = 0,
|
|
|
|
SORT_SIZE,
|
2019-05-23 02:13:13 -07:00
|
|
|
SORT_PERM,
|
2019-05-23 00:57:25 -07:00
|
|
|
SORT_TIME
|
2019-05-21 05:10:39 -07:00
|
|
|
} sortmethod_t;
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
typedef struct entry_s {
|
2019-05-21 03:59:30 -07:00
|
|
|
struct entry_s *next, **atme;
|
2019-05-21 05:10:39 -07:00
|
|
|
int visible : 1, d_isdir : 1;
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
2019-05-21 05:10:39 -07:00
|
|
|
int showhidden, sort_reverse;
|
|
|
|
sortmethod_t sortmethod;
|
2019-05-21 02:17:11 -07:00
|
|
|
} bb_state_t;
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
#define err(...) do { \
|
|
|
|
close_term(); \
|
|
|
|
fprintf(stderr, __VA_ARGS__); \
|
|
|
|
_err(); \
|
|
|
|
} while (0)
|
|
|
|
static void _err();
|
2019-05-21 02:17:11 -07:00
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
static inline int clamped(int x, int low, int high)
|
|
|
|
{
|
|
|
|
if (x < low) return low;
|
|
|
|
if (x > high) return high;
|
|
|
|
return x;
|
|
|
|
}
|
|
|
|
|
2019-05-22 00:25:25 -07:00
|
|
|
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:
|
2019-05-21 02:17:11 -07:00
|
|
|
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");
|
2019-05-21 02:17:11 -07:00
|
|
|
// hide cursor
|
|
|
|
writez(termfd, "\e[?25l");
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
static void close_term()
|
|
|
|
{
|
|
|
|
// xterm-specific:
|
2019-05-21 02:17:11 -07:00
|
|
|
writez(termfd, "\e[?1049l");
|
|
|
|
// Show cursor:
|
|
|
|
writez(termfd, "\e[?25h");
|
2019-05-23 01:37:41 -07:00
|
|
|
// Disable mouse tracking:
|
|
|
|
writez(termfd, "\e[?1000l\e[?1002l\e[?1015l\e[?1006l");
|
2019-05-23 00:57:25 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
tcsetattr(termfd, TCSAFLUSH, &orig_termios);
|
|
|
|
close(termfd);
|
|
|
|
}
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
static void _err()
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
|
|
|
if (errno)
|
|
|
|
fprintf(stderr, "\n%s", strerror(errno));
|
|
|
|
fprintf(stderr, "\n");
|
2019-05-22 15:33:18 -07:00
|
|
|
exit(1);
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
static char bb_tmpfile[] = "/tmp/bb.XXXXXX";
|
|
|
|
|
2019-05-22 14:33:14 -07:00
|
|
|
static int run_cmd_on_selection(bb_state_t *state, const char *cmd)
|
|
|
|
{
|
|
|
|
pid_t child;
|
|
|
|
sig_t old_handler = signal(SIGINT, do_nothing);
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
strcpy(bb_tmpfile, "/tmp/bb.XXXXXX");
|
|
|
|
if (!mktemp(bb_tmpfile))
|
|
|
|
err("Could not create temp file");
|
|
|
|
|
2019-05-22 14:33:14 -07:00
|
|
|
if ((child = fork()) == 0) {
|
|
|
|
// TODO: is there a max number of args? Should this be batched?
|
2019-05-22 14:56:28 -07:00
|
|
|
char **const args = calloc(MAX(1, state->nselected) + 5, sizeof(char*));
|
2019-05-22 14:33:14 -07:00
|
|
|
int i = 0;
|
|
|
|
args[i++] = "sh";
|
|
|
|
args[i++] = "-c";
|
|
|
|
args[i++] = (char*)cmd;
|
|
|
|
args[i++] = "--";
|
|
|
|
entry_t *first = state->firstselected ? state->firstselected : state->files[state->cursor];
|
|
|
|
for (entry_t *e = first; e; e = e->next) {
|
|
|
|
args[i++] = e->d_name;
|
|
|
|
}
|
|
|
|
args[i] = NULL;
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
char bb_depth_str[64] = {0}, bb_ipc_str[64] = {0};
|
2019-05-22 14:33:14 -07:00
|
|
|
{ // Set environment variable to track shell nesting
|
|
|
|
char *depthstr = getenv("BB_DEPTH");
|
|
|
|
int depth = depthstr ? atoi(depthstr) : 0;
|
2019-05-23 00:57:25 -07:00
|
|
|
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);
|
2019-05-22 14:33:14 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
execvp("sh", args);
|
|
|
|
err("Failed to execute command: '%s'", cmd);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (child == -1)
|
|
|
|
err("Failed to fork");
|
2019-05-23 00:57:25 -07:00
|
|
|
|
2019-05-22 14:33:14 -07:00
|
|
|
int status;
|
|
|
|
waitpid(child, &status, 0);
|
|
|
|
signal(SIGINT, old_handler);
|
|
|
|
return status;
|
|
|
|
}
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2019-05-22 19:05:56 -07:00
|
|
|
static int write_escaped(int fd, const char *str, size_t n, const char *reset_color)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-22 19:05:56 -07:00
|
|
|
// escapes['\n'] == 'n', etc.
|
|
|
|
static const char *escapes = " abtnvfr e";
|
|
|
|
char buf[5];
|
|
|
|
int ret = 0;
|
|
|
|
int backlog = 0;
|
|
|
|
for (int i = 0; i < n; i++) {
|
|
|
|
int escapelen = 0;
|
|
|
|
if (str[i] <= '\x1b' && escapes[str[i]] != ' ')
|
|
|
|
escapelen = sprintf(buf, "\\%c", escapes[str[i]]);
|
|
|
|
else if (!(' ' <= str[i] && str[i] <= '~'))
|
|
|
|
escapelen = sprintf(buf, "\\x%02X", str[i]);
|
|
|
|
else {
|
|
|
|
++backlog;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (backlog > 0) {
|
|
|
|
ret += write(fd, &str[i-backlog], backlog);
|
|
|
|
backlog = 0;
|
|
|
|
}
|
|
|
|
ret += writez(fd, "\e[31m");
|
|
|
|
ret += write(fd, buf, escapelen);
|
|
|
|
ret += writez(fd, reset_color);
|
|
|
|
}
|
|
|
|
if (backlog > 0)
|
|
|
|
ret += write(fd, &str[n-backlog], backlog);
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void render(bb_state_t *state, int lazy)
|
|
|
|
{
|
|
|
|
static int lastcursor = -1, lastscroll = -1;
|
2019-05-23 01:37:41 -07:00
|
|
|
if (lastcursor == -1 || lastscroll == -1)
|
|
|
|
lazy = 0;
|
|
|
|
|
|
|
|
if (lazy) {
|
|
|
|
char buf[32];
|
|
|
|
if (lastscroll > state->scroll) {
|
|
|
|
int n = sprintf(buf, "\e[3;%dr\e[%dT\e[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);
|
|
|
|
write(termfd, buf, n);
|
|
|
|
}
|
|
|
|
}
|
2019-05-22 19:05:56 -07:00
|
|
|
|
|
|
|
if (!lazy) {
|
|
|
|
term_move(0,0);
|
|
|
|
writez(termfd, "\e[0;1m"); // reset color + bold
|
|
|
|
write_escaped(termfd, state->path, strlen(state->path), "\e[0;1m");
|
|
|
|
writez(termfd, "\e[K");
|
|
|
|
|
|
|
|
term_move(0,1);
|
|
|
|
{ // Column labels
|
2019-05-23 01:37:41 -07:00
|
|
|
char buf[] = " \e[42;30m Size | Date | Perm| Name \e[0m";
|
|
|
|
buf[11] = state->sortmethod == SORT_SIZE ? (state->sort_reverse ? '-' : '+') : ' ';
|
|
|
|
buf[24] = state->sortmethod == SORT_TIME ? (state->sort_reverse ? '-' : '+') : ' ';
|
2019-05-23 02:13:13 -07:00
|
|
|
buf[39] = state->sortmethod == SORT_PERM ? (state->sort_reverse ? '-' : '+') : ' ';
|
2019-05-23 01:37:41 -07:00
|
|
|
buf[45] = state->sortmethod == SORT_ALPHA ? (state->sort_reverse ? '-' : '+') : ' ';
|
2019-05-22 19:05:56 -07:00
|
|
|
writez(termfd, buf);
|
|
|
|
writez(termfd, "\e[K");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
entry_t **files = state->files;
|
2019-05-22 19:05:56 -07:00
|
|
|
static const char *NORMAL_COLOR = "\e[0m";
|
|
|
|
static const char *CURSOR_COLOR = "\e[0;30;47m";
|
|
|
|
static const char *LINKDIR_COLOR = "\e[0;36m";
|
|
|
|
static const char *DIR_COLOR = "\e[0;34m";
|
|
|
|
static const char *LINK_COLOR = "\e[0;33m";
|
|
|
|
for (int i = state->scroll; i < state->scroll + height - 3; i++) {
|
|
|
|
if (lazy) {
|
|
|
|
if (i == state->cursor || i == lastcursor)
|
|
|
|
goto do_render;
|
|
|
|
if (i < lastscroll || i >= lastscroll + height - 3)
|
|
|
|
goto do_render;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
do_render:;
|
|
|
|
int y = i - state->scroll + 2;
|
|
|
|
term_move(0, y);
|
|
|
|
if (i >= state->nfiles) {
|
|
|
|
writez(termfd, "\e[K");
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
entry_t *entry = files[i];
|
2019-05-21 16:32:26 -07:00
|
|
|
struct stat info = {0};
|
|
|
|
lstat(entry->d_fullname, &info);
|
2019-05-21 02:17:11 -07:00
|
|
|
|
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-22 19:05:56 -07:00
|
|
|
const char *color;
|
|
|
|
if (i == state->cursor)
|
|
|
|
color = CURSOR_COLOR;
|
|
|
|
else if (entry->d_isdir && entry->d_type == DT_LNK)
|
|
|
|
color = LINKDIR_COLOR;
|
|
|
|
else if (entry->d_isdir)
|
|
|
|
color = DIR_COLOR;
|
|
|
|
else if (entry->d_type == DT_LNK)
|
|
|
|
color = LINK_COLOR;
|
|
|
|
else
|
|
|
|
color = NORMAL_COLOR;
|
|
|
|
writez(termfd, color);
|
|
|
|
|
2019-05-21 16:32:26 -07:00
|
|
|
{ // Filesize:
|
2019-05-21 02:17:11 -07:00
|
|
|
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]);
|
2019-05-21 02:17:11 -07:00
|
|
|
writez(termfd, buf);
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-21 16:32:26 -07:00
|
|
|
{ // Date:
|
2019-05-21 02:17:11 -07:00
|
|
|
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-21 02:17:11 -07:00
|
|
|
}
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-21 16:32:26 -07:00
|
|
|
{ // Name:
|
2019-05-22 19:05:56 -07:00
|
|
|
write_escaped(termfd, entry->d_name, entry->d_namlen, color);
|
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, " -> ");
|
2019-05-22 19:05:56 -07:00
|
|
|
write_escaped(termfd, linkpath, pathlen, color);
|
2019-05-21 16:32:26 -07:00
|
|
|
if (entry->d_isdir)
|
|
|
|
writez(termfd, "/");
|
|
|
|
}
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-22 19:05:56 -07:00
|
|
|
writez(termfd, " \e[0m\e[K"); // Reset color and attributes
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-22 00:25:25 -07:00
|
|
|
static const char *help = "Press '?' to see key bindings ";
|
2019-05-22 19:05:56 -07:00
|
|
|
term_move(0, height - 1);
|
|
|
|
if (state->nselected > 0) {
|
|
|
|
char buf[32] = {0};
|
|
|
|
int len = snprintf(buf, sizeof(buf), " %lu selected", state->nselected);
|
|
|
|
if (strlen(help) + len < width + 1) {
|
|
|
|
write(termfd, buf, len);
|
|
|
|
}
|
2019-05-22 00:25:25 -07:00
|
|
|
}
|
2019-05-22 19:05:56 -07:00
|
|
|
writez(termfd, "\e[K");
|
2019-05-22 00:25:25 -07:00
|
|
|
term_move(MAX(0, width - strlen(help)), height - 1);
|
|
|
|
writez(termfd, help);
|
2019-05-22 19:05:56 -07:00
|
|
|
lastcursor = state->cursor;
|
|
|
|
lastscroll = state->scroll;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-21 05:10:39 -07:00
|
|
|
static int compare_alpha(void *r, const void *v1, const void *v2)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-21 05:10:39 -07:00
|
|
|
int sign = *((int *)r) ? -1 : 1;
|
2019-05-21 02:17:11 -07:00
|
|
|
const entry_t *f1 = *((const entry_t**)v1), *f2 = *((const entry_t**)v2);
|
2019-05-21 05:10:39 -07:00
|
|
|
int diff = -(f1->d_isdir - f2->d_isdir);
|
|
|
|
if (diff) return -diff; // Always sort dirs before files
|
2019-05-21 02:17:11 -07:00
|
|
|
const char *p1 = f1->d_name, *p2 = f2->d_name;
|
2019-05-20 19:28:47 -07:00
|
|
|
while (*p1 && *p2) {
|
2019-05-21 05:10:39 -07:00
|
|
|
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);
|
2019-05-21 02:17:11 -07:00
|
|
|
diff = ((p1 - f1->d_name) - (p2 - f2->d_name)) || (n1 - n2);
|
2019-05-21 05:10:39 -07:00
|
|
|
if (diff) return diff*sign;
|
2019-05-20 19:28:47 -07:00
|
|
|
} else if (diff) {
|
2019-05-21 05:10:39 -07:00
|
|
|
return diff*sign;
|
2019-05-20 19:28:47 -07:00
|
|
|
} else {
|
|
|
|
++p1, ++p2;
|
|
|
|
}
|
|
|
|
}
|
2019-05-21 05:10:39 -07:00
|
|
|
return (*p1 - *p2)*sign;
|
|
|
|
}
|
|
|
|
|
2019-05-22 00:25:25 -07:00
|
|
|
static int compare_perm(void *r, const void *v1, const void *v2)
|
2019-05-21 05:10:39 -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
|
|
|
|
struct stat info1, info2;
|
2019-05-21 16:30:49 -07:00
|
|
|
lstat(f1->d_fullname, &info1);
|
|
|
|
lstat(f2->d_fullname, &info2);
|
2019-05-21 05:10:39 -07:00
|
|
|
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;
|
2019-05-21 16:30:49 -07:00
|
|
|
lstat(f1->d_fullname, &info1);
|
|
|
|
lstat(f2->d_fullname, &info2);
|
2019-05-21 05:10:39 -07:00
|
|
|
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;
|
2019-05-21 16:30:49 -07:00
|
|
|
lstat(f1->d_fullname, &info1);
|
|
|
|
lstat(f2->d_fullname, &info2);
|
2019-05-21 05:10:39 -07:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-05-22 01:39:15 -07:00
|
|
|
static void write_selection(int fd, entry_t *firstselected, char sep)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-21 02:17:11 -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');
|
2019-05-21 02:17:11 -07:00
|
|
|
if (!p2) p2 = p + strlen(p);
|
2019-05-20 19:28:47 -07:00
|
|
|
write(fd, p, p2 - p);
|
2019-05-22 01:39:15 -07:00
|
|
|
if (*p2 == '\n' && sep == '\n')
|
2019-05-20 19:28:47 -07:00
|
|
|
write(fd, "\\", 1);
|
|
|
|
p = p2;
|
|
|
|
}
|
2019-05-22 01:39:15 -07:00
|
|
|
write(fd, &sep, 1);
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-05-22 01:39:15 -07:00
|
|
|
static void explore(char *path, int print_dir, int print_selection, char sep)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-23 00:57:25 -07:00
|
|
|
char realpath_buf[MAX_PATH+1];
|
|
|
|
path = strdup(path);
|
2019-05-21 21:45:16 -07:00
|
|
|
if (!path) err("allocation failure");
|
2019-05-23 00:57:25 -07:00
|
|
|
char *original_path = strdup(path);
|
|
|
|
if (!original_path) err("allocation failure");
|
|
|
|
|
2019-05-21 17:13:28 -07:00
|
|
|
char to_select[MAX_PATH+1] = {0};
|
2019-05-21 02:17:11 -07:00
|
|
|
bb_state_t state = {0};
|
|
|
|
memset(&state, 0, sizeof(bb_state_t));
|
|
|
|
|
|
|
|
tail_call:
|
2019-05-23 00:57:25 -07:00
|
|
|
if (!realpath(path, realpath_buf))
|
|
|
|
err("realpath failed on %s", path);
|
|
|
|
path = realloc(path, strlen(realpath_buf));
|
|
|
|
strcpy(path, realpath_buf);
|
2019-05-21 02:17:11 -07:00
|
|
|
|
|
|
|
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;
|
2019-05-21 16:02:25 -07:00
|
|
|
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)
|
2019-05-23 00:57:25 -07:00
|
|
|
err("Failed to allocate %ld spaces for selecthash", hashsize);
|
2019-05-21 16:02:25 -07:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
2019-05-21 16:02:25 -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;
|
2019-05-21 16:29:01 -07:00
|
|
|
entry->visible = 1;
|
2019-05-21 16:02:25 -07:00
|
|
|
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;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-21 16:02:25 -07:00
|
|
|
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:;
|
2019-05-21 02:17:11 -07:00
|
|
|
}
|
2019-05-21 16:02:25 -07:00
|
|
|
closedir(dir);
|
|
|
|
free(selecthash);
|
|
|
|
if (state.nfiles == 0) err("No files found (not even '..')");
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
state.cursor = 0;
|
|
|
|
state.scroll = 0;
|
2019-05-21 05:10:39 -07:00
|
|
|
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;
|
2019-05-23 00:57:25 -07:00
|
|
|
if (state.sortmethod == SORT_TIME) cmp = compare_date;
|
2019-05-23 02:13:13 -07:00
|
|
|
if (state.sortmethod == SORT_PERM) cmp = compare_perm;
|
2019-05-21 05:10:39 -07:00
|
|
|
qsort_r(&state.files[1], state.nfiles-1, sizeof(entry_t*), &state.sort_reverse, cmp);
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-22 00:25:25 -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]) {
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|
2019-05-21 05:10:39 -07:00
|
|
|
to_select[0] = '\0';
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-22 19:05:56 -07:00
|
|
|
int picked, scrolloff, lazy = 0;
|
2019-05-20 19:28:47 -07:00
|
|
|
while (1) {
|
|
|
|
redraw:
|
2019-05-22 19:05:56 -07:00
|
|
|
render(&state, lazy);
|
|
|
|
lazy = 0;
|
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-22 15:47:47 -07:00
|
|
|
int key = term_getkey(termfd, &mouse_x, &mouse_y, KEY_DELAY);
|
2019-05-21 02:17:11 -07:00
|
|
|
switch (key) {
|
2019-05-20 19:28:47 -07:00
|
|
|
case KEY_MOUSE_LEFT: {
|
|
|
|
struct timespec clicktime;
|
|
|
|
clock_gettime(CLOCK_MONOTONIC, &clicktime);
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
2019-05-21 05:10:39 -07:00
|
|
|
if (mouse_y == 1) {
|
2019-05-22 00:25:25 -07:00
|
|
|
// Size Date Perm Name
|
2019-05-23 02:13:13 -07:00
|
|
|
sortmethod_t oldsort = state.sortmethod;
|
2019-05-21 05:10:39 -07:00
|
|
|
if (mouse_x <= 8)
|
2019-05-23 02:13:13 -07:00
|
|
|
state.sortmethod = SORT_SIZE;
|
2019-05-21 05:10:39 -07:00
|
|
|
else if (mouse_x <= 30)
|
2019-05-23 02:13:13 -07:00
|
|
|
state.sortmethod = SORT_TIME;
|
2019-05-21 05:10:39 -07:00
|
|
|
else if (mouse_x <= 35)
|
2019-05-23 02:13:13 -07:00
|
|
|
state.sortmethod = SORT_PERM;
|
2019-05-21 05:10:39 -07:00
|
|
|
else
|
2019-05-23 02:13:13 -07:00
|
|
|
state.sortmethod = SORT_ALPHA;
|
|
|
|
state.sort_reverse ^= state.sortmethod == oldsort;
|
|
|
|
goto sort_files;
|
2019-05-21 05:10:39 -07:00
|
|
|
} 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) {
|
2019-05-23 00:57:25 -07:00
|
|
|
lazy = 1;
|
2019-05-22 19:05:56 -07:00
|
|
|
// Single click
|
2019-05-20 19:28:47 -07:00
|
|
|
if (mouse_x == 0) {
|
2019-05-23 00:57:25 -07:00
|
|
|
if (IS_SELECTED(state.files[clicked]))
|
|
|
|
deselect_file(&state, state.files[clicked]);
|
|
|
|
else
|
|
|
|
select_file(&state, state.files[clicked]);
|
|
|
|
goto redraw;
|
2019-05-20 19:28:47 -07:00
|
|
|
} else {
|
2019-05-21 02:17:11 -07:00
|
|
|
state.cursor = clicked;
|
|
|
|
goto redraw;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Double click
|
2019-05-23 00:57:25 -07:00
|
|
|
// TODO: hacky
|
|
|
|
state.cursor = clicked;
|
|
|
|
key = '\n';
|
|
|
|
goto user_bindings;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
break;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-21 21:45:16 -07:00
|
|
|
case KEY_CTRL_C:
|
|
|
|
close_term();
|
|
|
|
if (print_dir)
|
|
|
|
printf("%s\n", original_path);
|
2019-05-22 15:33:18 -07:00
|
|
|
exit(1);
|
2019-05-21 21:45:16 -07:00
|
|
|
return;
|
|
|
|
|
2019-05-21 22:41:22 -07:00
|
|
|
case KEY_CTRL_Z:
|
|
|
|
close_term();
|
|
|
|
raise(SIGTSTP);
|
|
|
|
init_term();
|
|
|
|
goto redraw;
|
|
|
|
|
2019-05-21 03:59:30 -07:00
|
|
|
case '.':
|
|
|
|
state.showhidden ^= 1;
|
2019-05-23 00:57:25 -07:00
|
|
|
strcpy(to_select, state.files[state.cursor]->d_name);
|
2019-05-21 03:59:30 -07:00
|
|
|
goto tail_call;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-22 00:25:25 -07:00
|
|
|
case '?': {
|
2019-05-23 00:57:25 -07:00
|
|
|
char tmpname[] = "/tmp/bb_help.XXXXXX";
|
|
|
|
int fd = mkstemp(tmpname);
|
2019-05-22 00:25:25 -07:00
|
|
|
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++) {
|
2019-05-22 00:25:25 -07:00
|
|
|
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;
|
2019-05-23 00:57:25 -07:00
|
|
|
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");
|
2019-05-22 00:25:25 -07:00
|
|
|
writez(fd, "\e[0m\n");
|
|
|
|
}
|
|
|
|
writez(fd, "\n");
|
|
|
|
close(fd);
|
2019-05-23 00:57:25 -07:00
|
|
|
|
|
|
|
close_term();
|
|
|
|
char cmd[256];
|
|
|
|
sprintf(cmd, "less -r %s", tmpname);
|
|
|
|
system(cmd);
|
|
|
|
unlink(tmpname);
|
2019-05-22 00:25:25 -07:00
|
|
|
init_term();
|
|
|
|
goto redraw;
|
|
|
|
}
|
|
|
|
|
|
|
|
case -1:
|
|
|
|
goto skip_redraw;
|
|
|
|
|
|
|
|
default:
|
|
|
|
// Search user-defined key bindings from config.h:
|
2019-05-23 00:57:25 -07:00
|
|
|
user_bindings:
|
2019-05-22 00:25:25 -07:00
|
|
|
for (int i = 0; bindings[i].key > 0; i++) {
|
2019-05-21 20:06:53 -07:00
|
|
|
if (key == bindings[i].key) {
|
2019-05-21 22:41:22 -07:00
|
|
|
term_move(0, height-1);
|
2019-05-23 00:57:25 -07:00
|
|
|
//writez(termfd, "\e[K");
|
|
|
|
|
2019-05-22 00:25:25 -07:00
|
|
|
struct termios cur_tios;
|
2019-05-23 00:57:25 -07:00
|
|
|
if (bindings[i].flags & NORMAL_TERM) {
|
2019-05-21 20:06:53 -07:00
|
|
|
close_term();
|
2019-05-22 00:25:25 -07:00
|
|
|
} else {
|
|
|
|
tcgetattr(termfd, &cur_tios);
|
2019-05-22 14:33:14 -07:00
|
|
|
tcsetattr(termfd, TCSAFLUSH, &orig_termios);
|
2019-05-23 02:13:13 -07:00
|
|
|
//writez(termfd, "\e[?25h"); // Show cursor
|
2019-05-21 22:41:22 -07:00
|
|
|
}
|
2019-05-21 20:06:53 -07:00
|
|
|
|
2019-05-22 14:33:14 -07:00
|
|
|
run_cmd_on_selection(&state, bindings[i].command);
|
2019-05-22 00:25:25 -07:00
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
if (bindings[i].flags & NORMAL_TERM) {
|
|
|
|
lazy = 0;
|
2019-05-22 00:25:25 -07:00
|
|
|
init_term();
|
|
|
|
} else {
|
2019-05-23 00:57:25 -07:00
|
|
|
lazy = 1;
|
2019-05-22 00:25:25 -07:00
|
|
|
tcsetattr(termfd, TCSAFLUSH, &cur_tios);
|
2019-05-22 14:33:14 -07:00
|
|
|
writez(termfd, "\e[?25l"); // Hide cursor
|
2019-05-22 00:25:25 -07:00
|
|
|
}
|
2019-05-21 20:06:53 -07:00
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
// Scan for IPC requests
|
|
|
|
int needs_full_refresh = 0;
|
|
|
|
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';
|
2019-05-23 02:13:13 -07:00
|
|
|
char *value = strchr(line, ':');
|
|
|
|
if (value) ++value;
|
2019-05-23 00:57:25 -07:00
|
|
|
if (strcmp(line, "refresh") == 0) {
|
|
|
|
needs_full_refresh = 1;
|
|
|
|
} else if (strcmp(line, "quit") == 0) {
|
|
|
|
goto done;
|
2019-05-23 02:13:13 -07:00
|
|
|
} else if (startswith(line, "sort:")) {
|
|
|
|
sortmethod_t oldsort = state.sortmethod;
|
|
|
|
switch (value[0]) {
|
|
|
|
case '\0': continue;
|
|
|
|
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_PERM; break;
|
|
|
|
default: break;
|
|
|
|
}
|
|
|
|
if (value[1] == '+')
|
|
|
|
state.sort_reverse = 0;
|
|
|
|
else if (value[1] == '-')
|
|
|
|
state.sort_reverse = 1;
|
|
|
|
else if (state.sortmethod == oldsort)
|
|
|
|
state.sort_reverse ^= 1;
|
|
|
|
strcpy(to_select, state.files[state.cursor]->d_name);
|
|
|
|
goto sort_files;
|
2019-05-23 00:57:25 -07:00
|
|
|
} 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:")) {
|
|
|
|
lazy = 0;
|
2019-05-23 02:13:13 -07:00
|
|
|
if (strcmp(value, "*") == 0) {
|
2019-05-23 00:57:25 -07:00
|
|
|
for (int i = 0; i < state.nfiles; i++)
|
|
|
|
select_file(&state, state.files[i]);
|
|
|
|
} else {
|
2019-05-23 02:13:13 -07:00
|
|
|
entry_t *e = find_file(&state, value);
|
2019-05-23 00:57:25 -07:00
|
|
|
if (e) select_file(&state, e);
|
|
|
|
}
|
|
|
|
} else if (startswith(line, "deselect:")) {
|
|
|
|
lazy = 0;
|
2019-05-23 02:13:13 -07:00
|
|
|
if (strcmp(value, "*") == 0) {
|
2019-05-23 00:57:25 -07:00
|
|
|
clear_selection(&state);
|
|
|
|
} else {
|
2019-05-23 02:13:13 -07:00
|
|
|
entry_t *e = find_file(&state, value);
|
2019-05-23 00:57:25 -07:00
|
|
|
if (e) select_file(&state, e);
|
|
|
|
}
|
|
|
|
} else if (startswith(line, "cursor:")) {
|
|
|
|
for (int i = 0; i < state.nfiles; i++) {
|
2019-05-23 02:13:13 -07:00
|
|
|
if (strcmp(value[0] == '/' ?
|
2019-05-23 00:57:25 -07:00
|
|
|
state.files[i]->d_fullname : state.files[i]->d_name,
|
2019-05-23 02:13:13 -07:00
|
|
|
value) == 0) {
|
2019-05-23 00:57:25 -07:00
|
|
|
state.cursor = i;
|
|
|
|
goto found_it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(path);
|
2019-05-23 02:13:13 -07:00
|
|
|
char *lastslash = strrchr(value, '/');
|
2019-05-23 00:57:25 -07:00
|
|
|
if (!lastslash) goto found_it;
|
2019-05-23 02:13:13 -07:00
|
|
|
size_t len = lastslash - value;
|
2019-05-23 00:57:25 -07:00
|
|
|
path = calloc(len + 1, sizeof(char));
|
2019-05-23 02:13:13 -07:00
|
|
|
memcpy(path, value, len);
|
2019-05-23 00:57:25 -07:00
|
|
|
strcpy(to_select, lastslash+1);
|
|
|
|
goto tail_call;
|
|
|
|
found_it:;
|
|
|
|
} else if (startswith(line, "move:")) {
|
2019-05-23 01:37:41 -07:00
|
|
|
int expand_sel = 0;
|
|
|
|
if (*value == 'x') {
|
|
|
|
expand_sel = 1;
|
|
|
|
++value;
|
|
|
|
}
|
2019-05-23 00:57:25 -07:00
|
|
|
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;
|
|
|
|
|
|
|
|
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));
|
|
|
|
}
|
2019-05-23 01:37:41 -07:00
|
|
|
if (expand_sel) {
|
|
|
|
int sel = IS_SELECTED(state.files[oldcur]);
|
|
|
|
for (int i = state.cursor; i != oldcur; i += (oldcur > i ? 1 : -1)) {
|
|
|
|
if (sel && !IS_SELECTED(state.files[i]))
|
|
|
|
select_file(&state, state.files[i]);
|
|
|
|
else if (!sel && IS_SELECTED(state.files[i]))
|
|
|
|
deselect_file(&state, state.files[i]);
|
|
|
|
}
|
|
|
|
lazy &= abs(oldcur - state.cursor) <= 1;
|
|
|
|
}
|
2019-05-23 00:57:25 -07:00
|
|
|
} else if (startswith(line, "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);
|
|
|
|
}
|
2019-05-21 20:06:53 -07:00
|
|
|
|
2019-05-23 00:57:25 -07:00
|
|
|
if (needs_full_refresh) {
|
2019-05-21 21:45:16 -07:00
|
|
|
strcpy(to_select, state.files[state.cursor]->d_name);
|
2019-05-21 20:06:53 -07:00
|
|
|
goto tail_call;
|
2019-05-21 21:45:16 -07:00
|
|
|
}
|
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)
|
2019-05-22 01:39:15 -07:00
|
|
|
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 04:06:50 -07:00
|
|
|
char *path = ".";
|
2019-05-22 01:39:15 -07:00
|
|
|
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++) {
|
2019-05-22 01:55:34 -07:00
|
|
|
if (strcmp(argv[i], "--help") == 0) {
|
|
|
|
usage:
|
|
|
|
printf("bb - an itty bitty console TUI file browser\n");
|
|
|
|
printf("Usage: bb [-h/--help] [-s] [-b] [-0] [path]\n");
|
|
|
|
return 0;
|
|
|
|
}
|
2019-05-23 00:57:25 -07:00
|
|
|
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;
|
|
|
|
}
|
2019-05-22 01:50:46 -07:00
|
|
|
if (argv[i][0] == '-' && argv[i][1] == '-')
|
|
|
|
continue;
|
|
|
|
if (argv[i][0] == '-') {
|
|
|
|
for (char *c = &argv[i][1]; *c; c++) {
|
|
|
|
switch (*c) {
|
2019-05-22 01:55:34 -07:00
|
|
|
case 'h': goto usage;
|
2019-05-22 01:50:46 -07:00
|
|
|
case 'd':
|
|
|
|
print_dir = 1;
|
|
|
|
break;
|
|
|
|
case '0':
|
|
|
|
sep = '\0';
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
print_selection = 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (argv[i][0]) {
|
2019-05-21 04:06:50 -07:00
|
|
|
path = argv[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2019-05-21 21:45:16 -07:00
|
|
|
init_term();
|
2019-05-23 00:57:25 -07:00
|
|
|
explore(path, print_dir, print_selection, sep);
|
2019-05-20 19:28:47 -07:00
|
|
|
return 0;
|
|
|
|
}
|