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 <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>
|
|
|
|
|
|
|
|
#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 const int SCROLLOFF = 5;
|
|
|
|
|
|
|
|
static struct termios orig_termios;
|
|
|
|
static int termfd;
|
|
|
|
static int width, height;
|
|
|
|
static int mouse_x, mouse_y;
|
|
|
|
|
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 02:17:11 -07:00
|
|
|
int visible : 1;
|
2019-05-21 03:59:30 -07:00
|
|
|
int 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 03:59:30 -07:00
|
|
|
int showhidden : 1;
|
2019-05-21 02:17:11 -07:00
|
|
|
} 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 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-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);
|
2019-05-21 02:17:11 -07:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
static pid_t run_cmd(int *child_out, int *child_in, const char *cmd, ...)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-21 02:17:11 -07:00
|
|
|
int child_outfds[2], child_infds[2];
|
2019-05-20 19:28:47 -07:00
|
|
|
pid_t child;
|
2019-05-21 02:17:11 -07:00
|
|
|
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");
|
2019-05-21 02:17:11 -07:00
|
|
|
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 {
|
2019-05-21 02:17:11 -07:00
|
|
|
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);
|
2019-05-21 02:17:11 -07:00
|
|
|
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);
|
2019-05-21 02:17:11 -07:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
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);
|
2019-05-21 02:17:11 -07:00
|
|
|
writez(termfd, state->path);
|
2019-05-21 03:59:30 -07:00
|
|
|
|
|
|
|
term_move(0,1);
|
|
|
|
writez(termfd, "\e[32m Size Date Bits Name");
|
2019-05-20 19:28:47 -07:00
|
|
|
writez(termfd, "\e[0m"); // Reset color
|
|
|
|
|
|
|
|
char fullpath[MAX_PATH];
|
2019-05-21 02:17:11 -07:00
|
|
|
size_t pathlen = strlen(state->path);
|
|
|
|
strncpy(fullpath, state->path, pathlen + 1);
|
|
|
|
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-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);
|
|
|
|
|
|
|
|
// Selection box:
|
2019-05-21 02:17:11 -07:00
|
|
|
if (IS_SELECTED(entry))
|
2019-05-21 03:59:30 -07:00
|
|
|
writez(termfd, "\e[43m \e[0m");
|
2019-05-20 19:28:47 -07:00
|
|
|
else
|
|
|
|
writez(termfd, " ");
|
|
|
|
|
2019-05-21 03:59:30 -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 02:17:11 -07:00
|
|
|
struct stat info = {0};
|
2019-05-20 19:28:47 -07:00
|
|
|
fullpath[pathlen] = '/';
|
2019-05-21 02:17:11 -07:00
|
|
|
strncpy(fullpath + pathlen + 1, entry->d_name, entry->d_namlen);
|
|
|
|
fullpath[pathlen + 1 + entry->d_namlen] = '\0';
|
2019-05-20 19:28:47 -07:00
|
|
|
lstat(fullpath, &info);
|
|
|
|
|
2019-05-21 02:17:11 -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]);
|
2019-05-21 02:17:11 -07:00
|
|
|
writez(termfd, buf);
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -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, " │ ");
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
// Permissions:
|
|
|
|
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
|
|
|
|
|
|
|
// Name:
|
2019-05-21 02:17:11 -07:00
|
|
|
write(termfd, entry->d_name, entry->d_namlen);
|
2019-05-21 03:59:30 -07:00
|
|
|
if (entry->d_isdir)
|
2019-05-20 19:28:47 -07:00
|
|
|
writez(termfd, "/");
|
2019-05-21 03:59:30 -07:00
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
if (entry->d_type == DT_LNK) {
|
2019-05-20 19:28:47 -07:00
|
|
|
char linkpath[MAX_PATH] = {0};
|
|
|
|
ssize_t pathlen;
|
2019-05-21 02:17:11 -07:00
|
|
|
if ((pathlen = readlink(entry->d_name, linkpath, sizeof(linkpath))) < 0)
|
2019-05-20 19:28:47 -07:00
|
|
|
err("readlink() failed");
|
2019-05-21 03:59:30 -07:00
|
|
|
//writez(termfd, "\e[36m -> "); // Cyan FG
|
|
|
|
writez(termfd, " -> ");
|
2019-05-20 19:28:47 -07:00
|
|
|
write(termfd, linkpath, pathlen);
|
2019-05-21 03:59:30 -07:00
|
|
|
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
|
|
|
}
|
|
|
|
|
|
|
|
term_move(0, height - 1);
|
|
|
|
char buf[32] = {0};
|
2019-05-21 02:17:11 -07:00
|
|
|
int len = snprintf(buf, sizeof(buf), "%lu selected", state->nselected);
|
2019-05-20 19:28:47 -07:00
|
|
|
write(termfd, buf, len);
|
|
|
|
}
|
|
|
|
|
|
|
|
static int compare_alpha(const void *v1, const void *v2)
|
|
|
|
{
|
2019-05-21 02:17:11 -07:00
|
|
|
const entry_t *f1 = *((const entry_t**)v1), *f2 = *((const entry_t**)v2);
|
2019-05-20 19:28:47 -07:00
|
|
|
int diff;
|
2019-05-21 03:59:30 -07:00
|
|
|
diff = -(f1->d_isdir - f2->d_isdir);
|
2019-05-20 19:28:47 -07:00
|
|
|
if (diff) return -diff;
|
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) {
|
|
|
|
int diff = (*p1 - *p2);
|
|
|
|
if ('0' <= *p1 && *p1 <= '9' && '0' <= *p2 && *p2 <= '9') {
|
|
|
|
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-20 19:28:47 -07:00
|
|
|
if (diff) return diff;
|
|
|
|
} else if (diff) {
|
|
|
|
return diff;
|
|
|
|
} else {
|
|
|
|
++p1, ++p2;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return *p1 - *p2;
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
static void write_selection(int fd, entry_t *firstselected)
|
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);
|
|
|
|
if (*p2 == '\n')
|
|
|
|
write(fd, "\\", 1);
|
|
|
|
p = p2;
|
|
|
|
}
|
|
|
|
write(fd, "\n", 1);
|
2019-05-21 02:17:11 -07:00
|
|
|
firstselected = firstselected->next;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static int term_get_event()
|
|
|
|
{
|
|
|
|
char c;
|
|
|
|
if (read(termfd, &c, 1) != 1)
|
|
|
|
return -1;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
|
|
|
if (c != '\x1b')
|
|
|
|
return c;
|
|
|
|
|
|
|
|
// Actual escape key:
|
|
|
|
if (read(termfd, &c, 1) != 1)
|
|
|
|
return KEY_ESC;
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
switch (c) {
|
2019-05-21 02:17:11 -07:00
|
|
|
case '[':
|
2019-05-20 19:28:47 -07:00
|
|
|
if (read(termfd, &c, 1) != 1)
|
|
|
|
return -1;
|
|
|
|
switch (c) {
|
2019-05-21 02:17:11 -07:00
|
|
|
case 'H': return KEY_HOME;
|
|
|
|
case 'F': return KEY_END;
|
|
|
|
case '<': // Mouse clicks
|
|
|
|
{
|
|
|
|
int buttons = 0, x = 0, y = 0;
|
|
|
|
char buf;
|
|
|
|
while (read(termfd, &buf, 1) == 1 && '0' <= buf && buf <= '9')
|
|
|
|
buttons = buttons * 10 + (buf - '0');
|
|
|
|
if (buf != ';') return -1;
|
|
|
|
while (read(termfd, &buf, 1) == 1 && '0' <= buf && buf <= '9')
|
|
|
|
x = x * 10 + (buf - '0');
|
|
|
|
if (buf != ';') return -1;
|
|
|
|
while (read(termfd, &buf, 1) == 1 && '0' <= buf && buf <= '9')
|
|
|
|
y = y * 10 + (buf - '0');
|
|
|
|
if (buf != 'm' && buf != 'M') return -1;
|
|
|
|
|
|
|
|
mouse_x = x - 1, mouse_y = y - 1;
|
|
|
|
|
|
|
|
if (buf == 'm')
|
|
|
|
return KEY_MOUSE_RELEASE;
|
|
|
|
switch (buttons) {
|
|
|
|
case 0: return KEY_MOUSE_LEFT;
|
|
|
|
case 1: return KEY_MOUSE_RIGHT;
|
|
|
|
case 2: return KEY_MOUSE_MIDDLE;
|
|
|
|
default: return -1;
|
|
|
|
}
|
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
|
|
|
default:
|
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
|
|
|
return -1;
|
2019-05-20 19:28:47 -07:00
|
|
|
default:
|
2019-05-21 02:17:11 -07:00
|
|
|
return -1;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
static void explore(char *path)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-21 02:17:11 -07:00
|
|
|
char *tmp = path;
|
|
|
|
path = malloc(strlen(tmp) + 1);
|
|
|
|
strcpy(path, tmp);
|
|
|
|
tmp = NULL;
|
2019-05-20 19:28:47 -07:00
|
|
|
char to_select[MAX_PATH] = {0};
|
2019-05-21 02:17:11 -07:00
|
|
|
bb_state_t state = {0};
|
|
|
|
memset(&state, 0, sizeof(bb_state_t));
|
|
|
|
|
|
|
|
tail_call:
|
|
|
|
|
|
|
|
state.path = path;
|
|
|
|
|
|
|
|
DIR *dir = opendir(state.path);
|
2019-05-20 19:28:47 -07:00
|
|
|
if (!dir)
|
2019-05-21 02:17:11 -07:00
|
|
|
err("Couldn't open dir: %s", state.path);
|
|
|
|
if (chdir(state.path) != 0)
|
|
|
|
err("Couldn't chdir into %s", state.path);
|
2019-05-20 19:28:47 -07:00
|
|
|
struct dirent *dp;
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
// Hash inode -> entry_t with linear probing
|
|
|
|
size_t hashsize = 2 * state.nselected;
|
|
|
|
entry_t **selecthash = calloc(hashsize, sizeof(entry_t*));
|
2019-05-20 19:28:47 -07:00
|
|
|
if (!selecthash)
|
|
|
|
err("Failed to allocate %d spaces for selecthash", hashsize);
|
2019-05-21 02:17:11 -07:00
|
|
|
for (entry_t *p = state.firstselected; p; p = p->next) {
|
|
|
|
int probe = ((int)p->d_ino) % hashsize;
|
2019-05-20 19:28:47 -07:00
|
|
|
while (selecthash[probe])
|
|
|
|
probe = (probe + 1) % hashsize;
|
2019-05-21 02:17:11 -07:00
|
|
|
selecthash[probe] = p;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
size_t pathlen = strlen(state.path);
|
|
|
|
size_t filecap = 0;
|
|
|
|
state.nfiles = 0;
|
2019-05-20 19:28:47 -07:00
|
|
|
while ((dp = readdir(dir)) != NULL) {
|
|
|
|
if (dp->d_name[0] == '.' && dp->d_name[1] == '\0')
|
|
|
|
continue;
|
2019-05-21 03:59:30 -07:00
|
|
|
if (!state.showhidden && dp->d_name[0] == '.' && !(dp->d_name[1] == '.' && dp->d_name[2] == '\0'))
|
|
|
|
continue;
|
2019-05-21 02:17:11 -07:00
|
|
|
// Hashed lookup from selected:
|
2019-05-21 03:59:30 -07:00
|
|
|
if (state.nselected > 0) {
|
2019-05-20 19:28:47 -07:00
|
|
|
for (int probe = ((int)dp->d_ino) % hashsize; selecthash[probe]; probe = (probe + 1) % hashsize) {
|
2019-05-21 02:17:11 -07:00
|
|
|
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 02:17:11 -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;
|
2019-05-21 03:59:30 -07:00
|
|
|
entry->d_isdir = dp->d_type == DT_DIR;
|
|
|
|
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);
|
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
entry->d_namlen = dp->d_namlen;
|
2019-05-21 03:59:30 -07:00
|
|
|
entry->next = NULL, entry->atme = NULL;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
|
|
|
if (state.nfiles >= filecap) {
|
|
|
|
filecap += 100;
|
|
|
|
state.files = realloc(state.files, filecap*sizeof(entry_t*));
|
|
|
|
}
|
|
|
|
state.files[state.nfiles++] = entry;
|
|
|
|
next_file:;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
free(selecthash);
|
2019-05-21 02:17:11 -07:00
|
|
|
if (state.nfiles == 0) err("No files found (not even '..')");
|
|
|
|
qsort(state.files, state.nfiles, sizeof(entry_t*), compare_alpha);
|
2019-05-20 19:28:47 -07:00
|
|
|
closedir(dir);
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
state.cursor = 0;
|
|
|
|
state.scroll = 0;
|
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-20 19:28:47 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
clock_gettime(CLOCK_MONOTONIC, &state.lastclick);
|
2019-05-20 19:28:47 -07:00
|
|
|
int picked, scrolloff;
|
|
|
|
|
|
|
|
while (1) {
|
|
|
|
redraw:
|
2019-05-21 02:17:11 -07:00
|
|
|
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 02:17:11 -07:00
|
|
|
int key = term_get_event();
|
|
|
|
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 03:59:30 -07:00
|
|
|
if (mouse_y > 0 && state.scroll + (mouse_y - 2) < state.nfiles) {
|
|
|
|
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 {
|
2019-05-21 02:17:11 -07:00
|
|
|
state.cursor = clicked;
|
|
|
|
goto redraw;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Double click
|
|
|
|
picked = clicked;
|
|
|
|
goto open_file;
|
|
|
|
}
|
|
|
|
}
|
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
|
|
|
|
|
|
|
case 'q': case 'Q': case KEY_CTRL_C:
|
2019-05-20 19:28:47 -07:00
|
|
|
goto done;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case KEY_CTRL_D:
|
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))
|
|
|
|
state.scroll = state.nfiles - (height - 4);
|
2019-05-20 19:28:47 -07:00
|
|
|
goto redraw;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case KEY_CTRL_U:
|
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;
|
2019-05-21 02:17:11 -07:00
|
|
|
if (state.scroll < 0)
|
|
|
|
state.scroll = 0;
|
2019-05-20 19:28:47 -07:00
|
|
|
goto redraw;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-21 03:59:30 -07:00
|
|
|
case 'g':
|
|
|
|
state.cursor = 0;
|
|
|
|
state.scroll = 0;
|
|
|
|
goto redraw;
|
|
|
|
|
|
|
|
case 'G':
|
|
|
|
state.cursor = state.nfiles - 1;
|
|
|
|
if (state.nfiles > height - 4)
|
|
|
|
state.scroll = state.nfiles - (height - 4);
|
|
|
|
goto redraw;
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
case ' ':
|
|
|
|
picked = state.cursor;
|
2019-05-20 19:28:47 -07:00
|
|
|
toggle:
|
2019-05-21 03:59:30 -07:00
|
|
|
if (picked == 0) goto skip_redraw;
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
2019-05-21 02:17:11 -07:00
|
|
|
--state.nselected;
|
2019-05-20 19:28:47 -07:00
|
|
|
} else {
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
2019-05-21 02:17:11 -07:00
|
|
|
state.firstselected = e;
|
|
|
|
++state.nselected;
|
|
|
|
}
|
|
|
|
goto redraw;
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
case KEY_ESC:
|
|
|
|
for (entry_t *e = state.firstselected; e; e = e->next) {
|
|
|
|
e->next = NULL;
|
2019-05-21 03:59:30 -07:00
|
|
|
e->atme = NULL;
|
2019-05-21 02:17:11 -07:00
|
|
|
if (!e->visible) free(e);
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
state.firstselected = NULL;
|
|
|
|
state.nselected = 0;
|
2019-05-20 19:28:47 -07:00
|
|
|
goto redraw;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'j':
|
2019-05-21 02:17:11 -07:00
|
|
|
if (state.cursor >= state.nfiles - 1)
|
2019-05-20 19:28:47 -07:00
|
|
|
goto skip_redraw;
|
2019-05-21 02:17:11 -07:00
|
|
|
++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)) {
|
2019-05-21 02:17:11 -07:00
|
|
|
++state.scroll;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
goto redraw;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'k':
|
2019-05-21 02:17:11 -07:00
|
|
|
if (state.cursor <= 0)
|
2019-05-20 19:28:47 -07:00
|
|
|
goto skip_redraw;
|
2019-05-21 02:17:11 -07:00
|
|
|
--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-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'J':
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
2019-05-21 02:17:11 -07:00
|
|
|
} else {
|
|
|
|
picked = ++state.cursor;
|
2019-05-21 03:59:30 -07:00
|
|
|
if (IS_SELECTED(state.files[picked]))
|
|
|
|
goto toggle_off;
|
2019-05-21 02:17:11 -07:00
|
|
|
}
|
2019-05-21 03:59:30 -07:00
|
|
|
goto redraw;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
goto skip_redraw;
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'K':
|
2019-05-21 02:17:11 -07:00
|
|
|
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;
|
2019-05-21 02:17:11 -07:00
|
|
|
} else {
|
|
|
|
picked = --state.cursor;
|
2019-05-21 03:59:30 -07:00
|
|
|
if (IS_SELECTED(state.files[picked]))
|
|
|
|
goto toggle_off;
|
2019-05-21 02:17:11 -07:00
|
|
|
}
|
2019-05-21 03:59:30 -07:00
|
|
|
goto redraw;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
goto skip_redraw;
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'h':
|
2019-05-21 03:59:30 -07:00
|
|
|
picked = 0;
|
|
|
|
goto open_file;
|
|
|
|
|
|
|
|
case '.':
|
|
|
|
state.showhidden ^= 1;
|
|
|
|
goto tail_call;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
|
|
|
case 'l': case '\r':
|
|
|
|
picked = state.cursor;
|
2019-05-20 19:28:47 -07:00
|
|
|
open_file:
|
|
|
|
{
|
2019-05-21 03:59:30 -07:00
|
|
|
if (state.files[picked]->d_isdir) {
|
2019-05-21 02:17:11 -07:00
|
|
|
if (strcmp(state.files[picked]->d_name, "..") == 0) {
|
|
|
|
char *p = strrchr(state.path, '/');
|
2019-05-20 19:28:47 -07:00
|
|
|
if (p) strcpy(to_select, p+1);
|
|
|
|
else to_select[0] = '\0';
|
|
|
|
} else to_select[0] = '\0';
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-21 03:59:30 -07:00
|
|
|
char tmp[MAX_PATH];
|
|
|
|
if (!realpath(state.files[picked]->d_fullname, tmp))
|
2019-05-20 19:28:47 -07:00
|
|
|
err("realpath failed");
|
2019-05-21 02:17:11 -07:00
|
|
|
free(path);
|
2019-05-21 03:59:30 -07:00
|
|
|
path = calloc(strlen(tmp) + 1, sizeof(char));
|
|
|
|
strcpy(path, tmp);
|
2019-05-20 19:28:47 -07:00
|
|
|
goto tail_call;
|
|
|
|
} else {
|
2019-05-21 02:17:11 -07:00
|
|
|
char *name = state.files[picked]->d_name;
|
2019-05-20 19:28:47 -07:00
|
|
|
close_term();
|
|
|
|
pid_t child = run_cmd(NULL, NULL,
|
|
|
|
#ifdef __APPLE__
|
2019-05-21 02:17:11 -07:00
|
|
|
"if file -bI %s | grep '^text/' >/dev/null; then $EDITOR %s; else open %s; fi",
|
2019-05-20 19:28:47 -07:00
|
|
|
#else
|
2019-05-21 02:17:11 -07:00
|
|
|
"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;
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'm':
|
2019-05-21 02:17:11 -07:00
|
|
|
if (state.nselected) {
|
2019-05-20 19:28:47 -07:00
|
|
|
int fd;
|
2019-05-21 02:17:11 -07:00
|
|
|
run_cmd(NULL, &fd, "xargs -I {} mv {} .");
|
|
|
|
write_selection(fd, state.firstselected);
|
2019-05-20 19:28:47 -07:00
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
break;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'd':
|
2019-05-21 02:17:11 -07:00
|
|
|
if (state.nselected) {
|
2019-05-20 19:28:47 -07:00
|
|
|
int fd;
|
|
|
|
run_cmd(NULL, &fd, "xargs rm -rf");
|
2019-05-21 02:17:11 -07:00
|
|
|
write_selection(fd, state.firstselected);
|
2019-05-20 19:28:47 -07:00
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
break;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
case 'p':
|
2019-05-21 02:17:11 -07:00
|
|
|
if (state.nselected) {
|
|
|
|
int fd;
|
|
|
|
run_cmd(NULL, &fd, "xargs -I {} cp {} .");
|
|
|
|
write_selection(fd, state.firstselected);
|
|
|
|
close(fd);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
2019-05-21 03:59:30 -07:00
|
|
|
case '|': {
|
|
|
|
close_term();
|
|
|
|
char *buf = NULL;
|
|
|
|
size_t bufsize = 0;
|
|
|
|
printf("> ");
|
|
|
|
fflush(stdout);
|
|
|
|
getline(&buf, &bufsize, stdin);
|
|
|
|
|
|
|
|
int fd;
|
|
|
|
pid_t child = run_cmd(NULL, &fd, buf);
|
|
|
|
if (state.nselected > 0) {
|
2019-05-21 02:17:11 -07:00
|
|
|
write_selection(fd, state.firstselected);
|
2019-05-21 03:59:30 -07:00
|
|
|
} else {
|
|
|
|
for (int i = 0; i < state.nfiles; i++) {
|
|
|
|
write(fd, state.files[i]->d_name, state.files[i]->d_namlen);
|
|
|
|
write(fd, "\n", 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
close(fd);
|
|
|
|
waitpid(child, NULL, 0);
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-21 03:59:30 -07:00
|
|
|
printf("press enter to continue...");
|
|
|
|
fflush(stdout);
|
|
|
|
getline(&buf, &bufsize, stdin);
|
|
|
|
free(buf);
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-21 03:59:30 -07:00
|
|
|
init_term();
|
|
|
|
goto tail_call;
|
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
default:
|
|
|
|
goto skip_redraw;
|
|
|
|
}
|
|
|
|
goto skip_redraw;
|
|
|
|
}
|
|
|
|
done:
|
|
|
|
close_term();
|
2019-05-21 02:17:11 -07:00
|
|
|
write_selection(STDOUT_FILENO, state.firstselected);
|
2019-05-20 19:28:47 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
init_term();
|
|
|
|
char path[MAX_PATH];
|
|
|
|
if (!realpath(argc > 1 ? argv[1] : ".", path))
|
|
|
|
err("realpath failed");
|
|
|
|
explore(path);
|
|
|
|
done:
|
|
|
|
return 0;
|
|
|
|
}
|