bb/bb.c

1482 lines
49 KiB
C
Raw Normal View History

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>
#include <limits.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 <sys/wait.h>
2019-05-20 19:28:47 -07:00
#include <termios.h>
#include <time.h>
#include <unistd.h>
2019-05-21 20:06:53 -07:00
#include "config.h"
#include "bterm.h"
2019-05-20 19:28:47 -07:00
2019-05-30 18:08:49 -07:00
#define BB_VERSION "0.11.1"
2019-05-23 19:04:17 -07:00
#ifndef PATH_MAX
#define PATH_MAX 4096
#endif
2019-05-30 00:33:51 -07:00
#define MAX_COLS 12
#define MAX_SORT (2*MAX_COLS)
// Power of 2:
#define HASH_SIZE 1024
#define HASH_MASK (HASH_SIZE - 1)
2019-05-20 19:28:47 -07:00
#define MAX(a,b) ((a) < (b) ? (b) : (a))
#define MIN(a,b) ((a) > (b) ? (b) : (a))
#define IS_SELECTED(p) (((p)->selected.atme) != NULL)
#define LOWERCASE(c) ('A' <= (c) && (c) <= 'Z' ? ((c) + 'a' - 'A') : (c))
#define E_ISDIR(e) (S_ISDIR(S_ISLNK((e)->info.st_mode) ? (e)->linkedmode : (e)->info.st_mode))
2019-05-20 19:28:47 -07:00
2019-05-23 19:04:17 -07:00
#define err(...) do { \
2019-05-30 00:33:51 -07:00
cleanup(); \
2019-05-23 19:04:17 -07:00
fprintf(stderr, __VA_ARGS__); \
if (errno) fprintf(stderr, "\n%s", strerror(errno)); \
2019-05-23 19:04:17 -07:00
fprintf(stderr, "\n"); \
2019-05-30 00:33:51 -07:00
exit(EXIT_FAILURE); \
2019-05-23 19:04:17 -07:00
} while (0)
2019-05-20 19:28:47 -07:00
// Types
typedef enum {
2019-05-30 00:33:51 -07:00
COL_NONE = 0,
COL_DIR = '/',
COL_NAME = 'n',
COL_SIZE = 's',
COL_PERM = 'p',
COL_MTIME = 'm',
COL_CTIME = 'c',
COL_ATIME = 'a',
COL_RANDOM = 'r',
COL_SELECTED = '*',
} column_t;
/* entry_t uses intrusive linked lists. This means entries can only belong to
* one list at a time, in this case the list of selected entries. 'atme' is an
* indirect pointer to either the 'next' field of the previous list member, or
* the variable that points to the first list member. In other words,
* item->next->atme == &item->next and firstitem->atme == &firstitem.
*/
typedef struct {
2019-05-21 03:59:30 -07:00
struct entry_s *next, **atme;
} llnode_t;
typedef struct entry_s {
llnode_t selected, hash;
char *name, *linkname;
struct stat info;
mode_t linkedmode;
unsigned int refcount : 2; // Should only be between 0-2
int no_esc : 1;
int link_no_esc : 1;
int shufflepos;
int index;
char fullname[1];
// ------- fullname must be last! --------------
} entry_t;
typedef struct bb_s {
entry_t *hash[HASH_SIZE];
entry_t **files;
entry_t *firstselected;
char path[PATH_MAX];
int nfiles;
int scroll, cursor;
char *marks[128]; // Mapping from key to directory
char sort[MAX_SORT+1];
char columns[MAX_COLS+1];
char aligns[MAX_COLS+1]; // l,r,c
int colwidths[MAX_COLS+1];
2019-05-31 00:03:48 -07:00
unsigned int dirty : 1;
unsigned int show_dotdot : 1;
unsigned int show_dot : 1;
unsigned int show_dotfiles : 1;
} bb_t;
2019-05-30 22:25:22 -07:00
typedef enum { BB_OK = 0, BB_INVALID, BB_QUIT } bb_result_t;
// Functions
static void update_term_size(int sig);
static void init_term(void);
static void close_term(void);
static void cleanup_and_exit(int sig);
2019-05-30 00:33:51 -07:00
static void cleanup(void);
static void* memcheck(void *p);
static int run_cmd_on_selection(bb_t *bb, const char *cmd);
static int fputs_escaped(FILE *f, const char *str, const char *color);
static const char* color_of(mode_t mode);
static void set_sort(bb_t *bb, const char *sort);
static void render(bb_t *bb);
2019-05-31 00:03:48 -07:00
static int compare_files(const void *v1, const void *v2);
static void clear_selection(bb_t *bb);
static void select_entry(bb_t *bb, entry_t *e);
static void deselect_entry(bb_t *bb, entry_t *e);
static void toggle_entry(bb_t *bb, entry_t *e);
static void set_cursor(bb_t *bb, int i);
static void set_scroll(bb_t *bb, int i);
static entry_t* load_entry(bb_t *bb, const char *path);
static void remove_entry(entry_t *e);
static void sort_files(bb_t *bb);
static void populate_files(bb_t *bb, const char *path);
static bb_result_t execute_cmd(bb_t *bb, const char *cmd);
static void bb_browse(bb_t *bb, const char *path);
static void print_bindings(void);
// Config options
extern binding_t bindings[];
extern const char *startupcmds[];
2019-05-30 00:33:51 -07:00
// Constants
static const char *T_ENTER_BBMODE = T_OFF(T_SHOW_CURSOR) T_ON(T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR);
2019-05-31 00:03:48 -07:00
static const char *T_LEAVE_BBMODE = T_OFF(T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR ";" T_ALT_SCREEN) T_ON(T_SHOW_CURSOR);
2019-05-30 00:33:51 -07:00
static const char *T_LEAVE_BBMODE_PARTIAL = T_OFF(T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR);
// Global variables
static struct termios orig_termios, bb_termios;
static FILE *tty_out = NULL, *tty_in = NULL;
static int termwidth, termheight;
static int mouse_x, mouse_y;
static char *cmdfilename = NULL;
static const int colsizew = 7, coldatew = 19, colpermw = 5, colnamew = 40,
colselw = 2, coldirw = 1, colrandw = 2;
static struct timespec lastclick = {0, 0};
2019-05-31 00:03:48 -07:00
static bb_t *bb = NULL;
2019-05-20 19:28:47 -07:00
/*
* Hanlder for SIGWINCH events
*/
void update_term_size(int sig)
{
(void)sig;
struct winsize sz = {0};
ioctl(fileno(tty_in), TIOCGWINSZ, &sz);
termwidth = sz.ws_col;
termheight = sz.ws_row;
}
/*
* Initialize the terminal files for /dev/tty and set up some desired
* attributes like passing Ctrl-c as a key instead of interrupting
*/
void init_term(void)
2019-05-20 19:28:47 -07:00
{
tty_in = fopen("/dev/tty", "r");
tty_out = fopen("/dev/tty", "w");
tcgetattr(fileno(tty_out), &orig_termios);
memcpy(&bb_termios, &orig_termios, sizeof(bb_termios));
bb_termios.c_iflag &= ~(unsigned long)(
IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR | ICRNL | IXON);
bb_termios.c_oflag &= (unsigned long)~OPOST;
bb_termios.c_lflag &= (unsigned long)~(ECHO | ECHONL | ICANON | ISIG | IEXTEN);
bb_termios.c_cflag &= (unsigned long)~(CSIZE | PARENB);
bb_termios.c_cflag |= (unsigned long)CS8;
bb_termios.c_cc[VMIN] = 0;
bb_termios.c_cc[VTIME] = 0;
if (tcsetattr(fileno(tty_out), TCSAFLUSH, &bb_termios) == -1)
err("Couldn't tcsetattr");
2019-05-23 04:02:11 -07:00
update_term_size(0);
signal(SIGWINCH, update_term_size);
// Initiate mouse tracking and disable text wrapping:
fputs(T_ENTER_BBMODE, tty_out);
fputs(T_OFF(T_WRAP), tty_out);
2019-05-20 19:28:47 -07:00
}
/*
* Close the /dev/tty terminals and restore some of the attributes.
*/
void close_term(void)
{
if (tty_out) {
tcsetattr(fileno(tty_out), TCSAFLUSH, &orig_termios);
fputs(T_LEAVE_BBMODE_PARTIAL, tty_out);
fputs(T_ON(T_WRAP), tty_out);
fflush(tty_out);
fclose(tty_out);
tty_out = NULL;
fclose(tty_in);
tty_in = NULL;
} else {
fputs(T_LEAVE_BBMODE_PARTIAL, stdout);
fputs(T_ON(T_WRAP), stdout);
fflush(stdout);
}
signal(SIGWINCH, SIG_DFL);
2019-05-20 19:28:47 -07:00
}
/*
* Close safely in a way that doesn't gunk up the terminal.
*/
void cleanup_and_exit(int sig)
{
2019-05-30 00:33:51 -07:00
static volatile sig_atomic_t error_in_progress = 0;
if (error_in_progress)
raise(sig);
error_in_progress = 1;
cleanup();
signal(sig, SIG_DFL);
raise(sig);
}
/*
* Close the terminal, reset the screen, and delete the cmdfile
*/
void cleanup(void)
{
if (cmdfilename) {
unlink(cmdfilename);
cmdfilename = NULL;
}
close_term();
2019-05-30 00:33:51 -07:00
fputs(T_OFF(T_ALT_SCREEN), stdout);
fflush(stdout);
}
/*
* Memory allocation failures are unrecoverable in bb, so this wrapper just
* prints an error message and exits if that happens.
*/
void* memcheck(void *p)
2019-05-20 19:28:47 -07:00
{
2019-05-23 19:04:17 -07:00
if (!p) err("Allocation failure");
return p;
2019-05-20 19:28:47 -07:00
}
/*
* Run a command with the selected files passed as sequential arguments to the
* command (or pass the cursor file if none are selected).
* Return the exit status of the command.
*/
int run_cmd_on_selection(bb_t *bb, const char *cmd)
{
pid_t child;
2019-05-31 00:03:48 -07:00
void (*old_handler)(int) = signal(SIGINT, SIG_IGN);
if ((child = fork()) == 0) {
2019-05-23 19:04:17 -07:00
signal(SIGINT, SIG_DFL);
// TODO: is there a max number of args? Should this be batched?
size_t space = 32;
char **args = memcheck(calloc(space, sizeof(char*)));
size_t i = 0;
args[i++] = "sh";
args[i++] = "-c";
args[i++] = (char*)cmd;
args[i++] = "--"; // ensure files like "-i" are not interpreted as flags for sh
entry_t *first = bb->firstselected ? bb->firstselected : bb->files[bb->cursor];
for (entry_t *e = first; e; e = e->selected.next) {
if (i >= space)
2019-05-31 00:03:48 -07:00
args = memcheck(realloc(args, (space += 100)*sizeof(char*)));
args[i++] = e->fullname;
}
args[i] = NULL;
setenv("BBSELECTED", bb->firstselected ? "1" : "", 1);
setenv("BBCURSOR", bb->files[bb->cursor]->fullname, 1);
execvp("sh", args);
err("Failed to execute command: '%s'", cmd);
return -1;
}
if (child == -1)
err("Failed to fork");
int status;
waitpid(child, &status, 0);
signal(SIGINT, old_handler);
return status;
}
/*
* Print a string, but replacing bytes like '\n' with a red-colored "\n".
* The color argument is what color to put back after the red.
* Returns the number of bytes that were escaped.
*/
int fputs_escaped(FILE *f, const char *str, const char *color)
2019-05-20 19:28:47 -07:00
{
static const char *escapes = " abtnvfr e";
int escaped = 0;
for (const char *c = str; *c; ++c) {
if (*c > 0 && *c <= '\x1b' && escapes[(int)*c] != ' ') { // "\n", etc.
fprintf(f, "\033[31m\\%c%s", escapes[(int)*c], color);
++escaped;
} else if (*c >= 0 && !(' ' <= *c && *c <= '~')) { // "\x02", etc.
fprintf(f, "\033[31m\\x%02X%s", *c, color);
++escaped;
} else {
fputc(*c, f);
}
}
return escaped;
}
/*
* Returns the color of a file listing, given its mode.
*/
const char* color_of(mode_t mode)
{
if (S_ISDIR(mode))
return DIR_COLOR;
else if (S_ISLNK(mode))
return LINK_COLOR;
else if (mode & (S_IXUSR | S_IXGRP | S_IXOTH))
return EXECUTABLE_COLOR;
else
return NORMAL_COLOR;
}
void set_sort(bb_t *bb, const char *sort)
{
for (const char *s = sort; s[0] && s[1]; s += 2) {
char *found;
if ((found = strchr(bb->sort, s[1]))) {
memmove(found-1, found+1, strlen(found+1)+1);
}
}
size_t len = MIN(MAX_SORT, strlen(sort));
memmove(bb->sort + len, bb->sort, MAX_SORT+1 - len);
memmove(bb->sort, sort, len);
}
/*
* Draw everything to the screen.
* If bb->dirty is false, then use terminal scrolling to move the file listing
* around and only update the files that have changed.
*/
void render(bb_t *bb)
{
static int lastcursor = -1, lastscroll = -1;
2019-05-23 19:04:17 -07:00
char buf[64];
if (lastcursor == -1 || lastscroll == -1)
bb->dirty = 1;
if (!bb->dirty) {
2019-05-23 19:04:17 -07:00
// Use terminal scrolling:
if (lastscroll > bb->scroll) {
fprintf(tty_out, "\033[3;%dr\033[%dT\033[1;%dr", termheight-1, lastscroll - bb->scroll, termheight);
} else if (lastscroll < bb->scroll) {
fprintf(tty_out, "\033[3;%dr\033[%dS\033[1;%dr", termheight-1, bb->scroll - lastscroll, termheight);
}
}
if (bb->dirty) {
2019-05-23 19:04:17 -07:00
// Path
move_cursor(tty_out, 0, 0);
const char *color = TITLE_COLOR;
fputs(color, tty_out);
fputs_escaped(tty_out, bb->path, color);
fputs(" \033[K\033[0m", tty_out);
// Columns
move_cursor(tty_out, 0, 1);
fputs("\033[0;44;30m\033[K", tty_out);
int x = 0;
for (int col = 0; bb->columns[col]; col++) {
const char *title = NULL;
switch (bb->columns[col]) {
2019-05-30 00:33:51 -07:00
case COL_SELECTED: title = "*"; break;
case COL_DIR: title = "/"; break;
case COL_RANDOM: title = "Random"; break;
case COL_NAME: title = "Name"; break;
case COL_SIZE: title = "Size"; break;
case COL_PERM: title = "Permissions"; break;
case COL_MTIME: title = "Modified"; break;
case COL_ATIME: title = "Accessed"; break;
case COL_CTIME: title = "Created"; break;
default: title = ""; break;
}
move_cursor(tty_out, x, 1);
if (col > 0) {
fputs("\033[K", tty_out);
x += 1;
}
int k = bb->aligns[col] == 'c' ? 1 : (bb->aligns[col] == 'r' ? 2 : 0);
const char *indicator = " ";
char *found;
if ((found = strchr(bb->sort, bb->columns[col])))
indicator = found[-1] == '-' ? RSORT_INDICATOR : SORT_INDICATOR;
move_cursor(tty_out, x + MAX(0, ((bb->colwidths[col] - (int)strlen(title) - 1)*k)/2), 1);
if (bb->columns[col] == bb->sort[1])
fputs("\033[1m", tty_out);
fputs(indicator, tty_out);
fputs(title, tty_out);
if (bb->columns[col] == bb->sort[1])
fputs("\033[22m", tty_out);
x += bb->colwidths[col];
}
fputs(" \033[K\033[0m", tty_out);
}
entry_t **files = bb->files;
for (int i = bb->scroll; i < bb->scroll + termheight - 3; i++) {
if (!bb->dirty) {
if (i == bb->cursor || i == lastcursor)
goto do_render;
if (i < lastscroll || i >= lastscroll + termheight - 3)
goto do_render;
continue;
}
int y;
do_render:
y = i - bb->scroll + 2;
move_cursor(tty_out, 0, y);
if (i == bb->scroll && bb->nfiles == 0) {
const char *s = "...no files here...";
2019-05-29 19:45:23 -07:00
fprintf(tty_out, "\033[37;2m%s\033[0m\033[K\033[J", s);
break;
}
if (i >= bb->nfiles) {
fputs("\033[J", tty_out);
break;
}
entry_t *entry = files[i];
if (i == bb->cursor) fputs(CURSOR_COLOR, tty_out);
int x = 0;
for (int col = 0; bb->columns[col]; col++) {
fprintf(tty_out, "\033[%d;%dH\033[K", y+1, x+1);
if (col > 0) {
if (i == bb->cursor) fputs("", tty_out);
else fputs("\033[37;2m│\033[22m", tty_out);
2019-05-29 19:45:23 -07:00
fputs(i == bb->cursor ? CURSOR_COLOR : "\033[0m", tty_out);
x += 1;
}
int k = bb->aligns[col] == 'c' ? 1 : (bb->aligns[col] == 'r' ? 2 : 0);
switch (bb->columns[col]) {
2019-05-30 00:33:51 -07:00
case COL_SELECTED:
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - colselw))/2), y);
fputs(IS_SELECTED(entry) ? SELECTED_INDICATOR : NOT_SELECTED_INDICATOR, tty_out);
fputs(i == bb->cursor ? CURSOR_COLOR : "\033[0m", tty_out);
break;
2019-05-30 00:33:51 -07:00
case COL_DIR:
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - coldirw))/2), y);
if (E_ISDIR(entry)) fputs("/", tty_out);
else fputs(" ", tty_out);
break;
2019-05-30 00:33:51 -07:00
case COL_RANDOM:
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - colrandw))/2), y);
fprintf(tty_out, "\033[48;5;%dm \033[0m%s", 232 + (entry->shufflepos / (RAND_MAX / (255-232))),
i == bb->cursor ? CURSOR_COLOR : "\033[0m");
break;
2019-05-30 00:33:51 -07:00
case COL_SIZE: {
2019-05-23 19:04:17 -07:00
int j = 0;
const char* units = "BKMGTPEZY";
double bytes = (double)entry->info.st_size;
2019-05-23 19:04:17 -07:00
while (bytes > 1024) {
bytes /= 1024;
j++;
}
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - colsizew))/2), y);
fprintf(tty_out, "%6.*f%c", j > 0 ? 1 : 0, bytes, units[j]);
2019-05-23 19:04:17 -07:00
break;
}
2019-05-20 19:28:47 -07:00
2019-05-30 00:33:51 -07:00
case COL_MTIME:
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - coldatew))/2), y);
2019-05-31 00:03:48 -07:00
strftime(buf, sizeof(buf), "%I:%M%p %b %e %Y", localtime(&(entry->info.st_mtime)));
fputs(buf, tty_out);
2019-05-23 19:04:17 -07:00
break;
2019-05-21 03:59:30 -07:00
2019-05-30 00:33:51 -07:00
case COL_CTIME:
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - coldatew))/2), y);
2019-05-31 00:03:48 -07:00
strftime(buf, sizeof(buf), "%I:%M%p %b %e %Y", localtime(&(entry->info.st_ctime)));
fputs(buf, tty_out);
2019-05-23 19:04:17 -07:00
break;
2019-05-20 19:28:47 -07:00
2019-05-30 00:33:51 -07:00
case COL_ATIME:
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - coldatew))/2), y);
2019-05-31 00:03:48 -07:00
strftime(buf, sizeof(buf), "%I:%M%p %b %e %Y", localtime(&(entry->info.st_atime)));
fputs(buf, tty_out);
2019-05-23 19:04:17 -07:00
break;
2019-05-30 00:33:51 -07:00
case COL_PERM:
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - colpermw))/2), y);
fprintf(tty_out, " %03o", entry->info.st_mode & 0777);
2019-05-23 19:04:17 -07:00
break;
2019-05-30 00:33:51 -07:00
case COL_NAME: {
2019-05-29 19:45:23 -07:00
char color[128];
strcpy(color, color_of(entry->info.st_mode));
if (i == bb->cursor) strcat(color, CURSOR_COLOR);
move_cursor(tty_out, x + MAX(0, (k*(bb->colwidths[col] - (int)strlen(entry->name)))/2), y);
fputs(color, tty_out);
if (entry->no_esc) fputs(entry->name, tty_out);
else entry->no_esc |= !fputs_escaped(tty_out, entry->name, color);
if (E_ISDIR(entry)) fputs("/", tty_out);
2019-05-23 19:04:17 -07:00
if (entry->linkname) {
if (i != bb->cursor)
fputs("\033[37m", tty_out);
fputs("\033[2m -> \033[22;3m", tty_out);
strcpy(color, color_of(entry->linkedmode));
if (i == bb->cursor) strcat(color, CURSOR_COLOR);
fputs(color, tty_out);
if (entry->link_no_esc) fputs(entry->linkname, tty_out);
else entry->link_no_esc |= !fputs_escaped(tty_out, entry->linkname, color);
if (S_ISDIR(entry->linkedmode))
fputs("/", tty_out);
fputs("\033[22;23m", tty_out);
2019-05-23 19:04:17 -07:00
}
fputs(i == bb->cursor ? CURSOR_COLOR : "\033[0m", tty_out);
2019-05-29 19:45:23 -07:00
fputs("\033[K", tty_out);
2019-05-23 19:04:17 -07:00
break;
}
default: break;
2019-05-21 16:32:26 -07:00
}
x += bb->colwidths[col];
2019-05-20 19:28:47 -07:00
}
fputs(" \033[K\033[0m", tty_out); // Reset color and attributes
2019-05-20 19:28:47 -07:00
}
static const char *help = "Press '?' to see key bindings ";
move_cursor(tty_out, 0, termheight - 1);
fputs("\033[K", tty_out);
move_cursor(tty_out, MAX(0, termwidth - (int)strlen(help)), termheight - 1);
fputs(help, tty_out);
lastcursor = bb->cursor;
lastscroll = bb->scroll;
fflush(tty_out);
2019-05-20 19:28:47 -07:00
}
/*
* Used for sorting, this function compares files according to the sorting-related options,
* like bb->sort
*/
2019-05-31 00:03:48 -07:00
int compare_files(const void *v1, const void *v2)
2019-05-20 19:28:47 -07:00
{
#define COMPARE(a, b) if ((a) != (b)) { return sign*((a) < (b) ? 1 : -1); }
#define COMPARE_TIME(t1, t2) COMPARE((t1).tv_sec, (t2).tv_sec) COMPARE((t1).tv_nsec, (t2).tv_nsec)
const entry_t *e1 = *((const entry_t**)v1), *e2 = *((const entry_t**)v2);
for (char *sort = bb->sort + 1; *sort; sort += 2) {
int sign = sort[-1] == '-' ? -1 : 1;
switch (*sort) {
case COL_DIR: COMPARE(E_ISDIR(e1), E_ISDIR(e2)); break;
case COL_SELECTED: COMPARE(IS_SELECTED(e1), IS_SELECTED(e2)); break;
2019-05-30 00:33:51 -07:00
case COL_NAME: {
/* This sorting method is not identical to strverscmp(). Notably, bb's sort
* will order: [0, 1, 9, 00, 01, 09, 10, 000, 010] instead of strverscmp()'s
* order: [000, 00, 01, 010, 09, 0, 1, 9, 10]. I believe bb's sort is consistent
* with how people want their files grouped: all files padded to n digits
* will be grouped together, and files with the same padding will be sorted
* ordinally. This version also does case-insensitivity by lowercasing words,
* so the following characters come before all letters: [\]^_`
*/
const char *n1 = e1->name, *n2 = e2->name;
while (*n1 && *n2) {
char c1 = LOWERCASE(*n1), c2 = LOWERCASE(*n2);
if ('0' <= c1 && c1 <= '9' && '0' <= c2 && c2 <= '9') {
long i1 = strtol(n1, (char**)&n1, 10);
long i2 = strtol(n2, (char**)&n2, 10);
// Shorter numbers always go before longer. In practice, I assume
// filenames padded to the same number of digits should be grouped
// together, instead of
// [1.png, 0001.png, 2.png, 0002.png, 3.png], it makes more sense to have:
// [1.png, 2.png, 3.png, 0001.png, 0002.png]
COMPARE((n2 - e2->name), (n1 - e1->name));
COMPARE(i2, i1);
} else {
COMPARE(c2, c1);
++n1; ++n2;
}
}
COMPARE(LOWERCASE(*n2), LOWERCASE(*n1));
break;
2019-05-23 19:04:17 -07:00
}
case COL_PERM: COMPARE((e1->info.st_mode & 0x3FF), (e2->info.st_mode & 0x3FF)); break;
case COL_SIZE: COMPARE(e1->info.st_size, e2->info.st_size); break;
2019-05-31 00:03:48 -07:00
case COL_MTIME: COMPARE_TIME(e1->info.st_mtim, e2->info.st_mtim); break;
case COL_CTIME: COMPARE_TIME(e1->info.st_ctim, e2->info.st_ctim); break;
case COL_ATIME: COMPARE_TIME(e1->info.st_atim, e2->info.st_atim); break;
case COL_RANDOM: COMPARE(e1->shufflepos, e2->shufflepos); break;
2019-05-20 19:28:47 -07:00
}
2019-05-23 19:04:17 -07:00
}
return 0;
#undef COMPARE
#undef COMPARE_TIME
2019-05-20 19:28:47 -07:00
}
/*
* Deselect all files
*/
void clear_selection(bb_t *bb)
2019-05-20 19:28:47 -07:00
{
for (entry_t *next, *e = bb->firstselected; e; e = next) {
next = e->selected.next;
*(e->selected.atme) = NULL;
e->selected.atme = NULL;
if (--e->refcount <= 0) remove_entry(e);
2019-05-21 20:06:53 -07:00
}
}
/*
* Select a file
*/
void select_entry(bb_t *bb, entry_t *e)
{
if (IS_SELECTED(e)) return;
if (bb->nfiles > 0 && e != bb->files[bb->cursor])
bb->dirty = 1;
if (bb->firstselected)
bb->firstselected->selected.atme = &e->selected.next;
e->selected.next = bb->firstselected;
e->selected.atme = &bb->firstselected;
++e->refcount;
bb->firstselected = e;
}
/*
* Deselect a file
*/
void deselect_entry(bb_t *bb, entry_t *e)
{
(void)bb;
if (!IS_SELECTED(e)) return;
if (bb->nfiles > 0 && e != bb->files[bb->cursor])
bb->dirty = 1;
if (e->selected.next)
e->selected.next->selected.atme = e->selected.atme;
*(e->selected.atme) = e->selected.next;
--e->refcount;
e->selected.next = NULL;
e->selected.atme = NULL;
}
/*
* Toggle a file's selection state
*/
void toggle_entry(bb_t *bb, entry_t *e)
{
if (IS_SELECTED(e)) deselect_entry(bb, e);
else select_entry(bb, e);
}
/*
* Set bb's file cursor to the given index (and adjust the scroll as necessary)
*/
void set_cursor(bb_t *bb, int newcur)
2019-05-20 19:28:47 -07:00
{
if (newcur > bb->nfiles - 1) newcur = bb->nfiles - 1;
if (newcur < 0) newcur = 0;
bb->cursor = newcur;
if (bb->nfiles <= termheight - 4)
return;
if (newcur < bb->scroll + SCROLLOFF)
bb->scroll = newcur - SCROLLOFF;
else if (newcur > bb->scroll + (termheight-4) - SCROLLOFF)
bb->scroll = newcur - (termheight-4) + SCROLLOFF;
if (bb->nfiles <= termheight - 4) {
bb->scroll = 0;
} else {
int max_scroll = bb->nfiles - (termheight-4) - 1;
if (max_scroll < 0) max_scroll = 0;
if (bb->scroll > max_scroll) bb->scroll = max_scroll;
if (bb->scroll < 0) bb->scroll = 0;
}
}
/*
* Set bb's scroll to the given index (and adjust the cursor as necessary)
*/
void set_scroll(bb_t *bb, int newscroll)
{
newscroll = MIN(newscroll, bb->nfiles - (termheight-4) - 1);
newscroll = MAX(newscroll, 0);
if (bb->nfiles <= termheight - 4)
newscroll = 0;
int delta = newscroll - bb->scroll;
int oldcur = bb->cursor;
if (bb->nfiles > termheight - 4) {
if (bb->cursor > newscroll + (termheight-4) - SCROLLOFF && bb->scroll < bb->nfiles - (termheight-4) - 1)
bb->cursor = newscroll + (termheight-4) - SCROLLOFF;
else if (bb->cursor < newscroll + SCROLLOFF && bb->scroll > 0)
bb->cursor = newscroll + SCROLLOFF;
} else {
newscroll = 0;
}
bb->scroll = newscroll;
if (abs(bb->cursor - oldcur) < abs(delta))
bb->cursor += delta - (bb->cursor - oldcur);
if (bb->cursor > bb->nfiles - 1) bb->cursor = bb->nfiles - 1;
if (bb->cursor < 0) bb->cursor = 0;
}
/*
* Load a file's info into an entry_t and return it (if found).
* The returned entry must be free()ed by the caller.
* Warning: this does not deduplicate entries, and it's best if there aren't
* duplicate entries hanging around.
*/
entry_t* load_entry(bb_t *bb, const char *path)
{
struct stat linkedstat, filestat;
if (lstat(path, &filestat) == -1) return NULL;
char path2[PATH_MAX];
if (path[0] != '/') {
strcpy(path2, bb->path);
strcat(path2, "/");
strcat(path2, path);
path = path2;
}
// Check for pre-existing:
for (entry_t *e = bb->hash[(int)filestat.st_ino & HASH_MASK]; e; e = e->hash.next) {
if (e->info.st_ino == filestat.st_ino && e->info.st_dev == filestat.st_dev
&& strcmp(e->fullname, path) == 0)
return e;
}
ssize_t linkpathlen = -1;
char linkbuf[PATH_MAX];
if (S_ISLNK(filestat.st_mode)) {
linkpathlen = readlink(path, linkbuf, sizeof(linkbuf));
if (linkpathlen < 0) err("Couldn't read link: '%s'", path);
linkbuf[linkpathlen] = 0;
if (stat(path, &linkedstat) == -1) memset(&linkedstat, 0, sizeof(linkedstat));
}
size_t pathlen = strlen(path);
size_t entry_size = sizeof(entry_t) + (pathlen + 1) + (size_t)(linkpathlen + 1);
entry_t *entry = memcheck(calloc(entry_size, 1));
char *end = stpcpy(entry->fullname, path);
if (linkpathlen >= 0)
entry->linkname = strcpy(end + 1, linkbuf);
entry->name = strrchr(entry->fullname, '/');
if (!entry->name) err("No slash found in '%s'", entry->fullname);
++entry->name;
if (S_ISLNK(filestat.st_mode))
entry->linkedmode = linkedstat.st_mode;
entry->info = filestat;
if (bb->hash[(int)filestat.st_ino & HASH_MASK])
bb->hash[(int)filestat.st_ino & HASH_MASK]->hash.atme = &entry->hash.next;
entry->hash.next = bb->hash[(int)filestat.st_ino & HASH_MASK];
entry->hash.atme = &bb->hash[(int)filestat.st_ino & HASH_MASK];
bb->hash[(int)filestat.st_ino & HASH_MASK] = entry;
return entry;
}
void remove_entry(entry_t *e)
{
if (e->refcount > 0) err("Attempt to remove in-use entry");
if (e->hash.next)
e->hash.next->hash.atme = e->hash.atme;
*(e->hash.atme) = e->hash.next;
e->hash.atme = NULL;
e->hash.next = NULL;
free(e);
}
void sort_files(bb_t *bb)
{
2019-05-31 00:03:48 -07:00
qsort(bb->files, (size_t)bb->nfiles, sizeof(entry_t*), compare_files);
for (int i = 0; i < bb->nfiles; i++) {
bb->files[i]->index = i;
}
bb->dirty = 1;
}
/*
* Remove all the files currently stored in bb->files and if `path` is non-NULL,
* update `bb` with a listing of the files in `path`
*/
void populate_files(bb_t *bb, const char *path)
{
bb->dirty = 1;
2019-05-23 19:04:17 -07:00
// Clear old files (if any)
if (bb->files) {
for (int i = 0; i < bb->nfiles; i++) {
if (--bb->files[i]->refcount <= 0)
remove_entry(bb->files[i]);
else
bb->files[i]->index = -1;
bb->files[i] = NULL;
}
free(bb->files);
bb->files = NULL;
}
int old_scroll = bb->scroll;
bb->nfiles = 0;
bb->cursor = 0;
bb->scroll = 0;
if (path == NULL || !path[0])
return;
DIR *dir = opendir(path);
2019-05-23 19:04:17 -07:00
if (!dir)
err("Couldn't open dir: %s", path);
size_t cap = 0;
char pathbuf[PATH_MAX];
strcpy(pathbuf, path);
strcat(pathbuf, "/");
size_t pathbuflen = strlen(pathbuf);
for (struct dirent *dp; (dp = readdir(dir)) != NULL; ) {
if (dp->d_name[0] == '.') {
if (dp->d_name[1] == '.' && dp->d_name[2] == '\0') {
if (!bb->show_dotdot) continue;
} else if (dp->d_name[1] == '\0') {
if (!bb->show_dot) continue;
} else if (!bb->show_dotfiles) continue;
}
if ((size_t)bb->nfiles + 1 > cap) {
cap += 100;
2019-05-31 00:03:48 -07:00
bb->files = memcheck(realloc(bb->files, cap*sizeof(void*)));
2019-05-23 19:04:17 -07:00
}
strcpy(&pathbuf[pathbuflen], dp->d_name);
entry_t *entry = load_entry(bb, pathbuf);
if (!entry) err("Failed to load entry: '%s'", dp->d_name);
++entry->refcount;
entry->index = bb->nfiles;
bb->files[bb->nfiles++] = entry;
2019-05-20 19:28:47 -07:00
}
2019-05-23 19:04:17 -07:00
closedir(dir);
if (path != bb->path)
strcpy(bb->path, path);
// TODO: this may have some weird aliasing issues, but eh, it's simple and effective
for (int i = 0; i < bb->nfiles; i++)
bb->files[i]->shufflepos = rand();
sort_files(bb);
set_scroll(bb, old_scroll);
2019-05-23 19:04:17 -07:00
}
2019-05-20 19:28:47 -07:00
/*
* Run a bb internal command (e.g. "+refresh") and return an indicator of what
* needs to happen next.
*/
bb_result_t execute_cmd(bb_t *bb, const char *cmd)
{
char *value = strchr(cmd, ':');
if (value) ++value;
2019-05-30 00:33:51 -07:00
#define set_bool(target) do { if (!value) { target = !target; } else { target = value[0] == '1'; } } while (0)
switch (cmd[0]) {
case '.': { // +..:, +.:
if (cmd[1] == '.') // +..:
set_bool(bb->show_dotdot);
else // +.:
set_bool(bb->show_dot);
populate_files(bb, bb->path);
2019-05-30 22:25:22 -07:00
return BB_OK;
}
case 'a': { // +align:
if (!value) return BB_INVALID;
strncpy(bb->aligns, value, MAX_COLS);
bb->dirty = 1;
2019-05-30 22:25:22 -07:00
return BB_OK;
}
case 'c': { // +cd:, +columns:
switch (cmd[1]) {
case 'd': { // +cd:
char pbuf[PATH_MAX];
cd:
if (!value) return BB_INVALID;
if (value[0] == '~') {
char *home;
if (!(home = getenv("HOME")))
return BB_INVALID;
strcpy(pbuf, home);
strcat(pbuf, value+1);
value = pbuf;
}
char *rpbuf = realpath(value, NULL);
if (!rpbuf) return BB_INVALID;
if (strcmp(rpbuf, bb->path) == 0) {
free(rpbuf);
2019-05-30 22:25:22 -07:00
return BB_OK;
}
if (chdir(rpbuf)) {
free(rpbuf);
return BB_INVALID;
}
char *oldpath = memcheck(strdup(bb->path));
populate_files(bb, rpbuf);
free(rpbuf);
if (strcmp(value, "..") == 0) {
entry_t *old = load_entry(bb, oldpath);
if (old) {
if (old->index >= 0)
set_cursor(bb, old->index);
if (old->refcount <= 0)
remove_entry(old);
}
}
free(oldpath);
2019-05-30 22:25:22 -07:00
return BB_OK;
}
case 'o': { // +columns:
if (!value) return BB_INVALID;
strncpy(bb->columns, value, MAX_COLS);
for (int i = 0; i < value[i] && i < MAX_COLS; i++) {
int *colw = &bb->colwidths[i];
switch (value[i]) {
2019-05-30 00:33:51 -07:00
case COL_CTIME: case COL_MTIME: case COL_ATIME:
*colw = coldatew; break;
case COL_SIZE: *colw = colsizew; break;
case COL_PERM: *colw = colpermw; break;
case COL_NAME: *colw = colnamew; break;
case COL_SELECTED: *colw = colselw; break;
case COL_RANDOM: *colw = colrandw; break;
case COL_DIR: *colw = coldirw; break;
}
}
bb->dirty = 1;
2019-05-30 22:25:22 -07:00
return BB_OK;
}
}
return BB_INVALID;
}
case 'd': { // +deselect:, +dotfiles:
switch (cmd[1]) {
case 'e': { // +deselect:
if (!value) value = bb->files[bb->cursor]->name;
if (strcmp(value, "*") == 0) {
clear_selection(bb);
2019-05-30 22:25:22 -07:00
return BB_OK;
} else {
entry_t *e = load_entry(bb, value);
if (e) {
deselect_entry(bb, e);
if (e->refcount <= 0)
remove_entry(e);
}
2019-05-30 22:25:22 -07:00
return BB_OK;
}
}
case 'o': { // +dotfiles:
set_bool(bb->show_dotfiles);
populate_files(bb, bb->path);
2019-05-30 22:25:22 -07:00
return BB_OK;
}
}
}
case 'g': { // +goto:
if (!value) return BB_INVALID;
entry_t *e = load_entry(bb, value);
if (!e) return BB_INVALID;
if (e->index >= 0) {
set_cursor(bb, e->index);
2019-05-30 22:25:22 -07:00
return BB_OK;
}
char *real = realpath(e->fullname, NULL);
if (!real) return BB_INVALID;
char *lastslash = strrchr(real, '/');
if (!lastslash) {
free(real); // estate
return BB_INVALID;
}
*lastslash = '\0'; // Split in two
if (chdir(real)) return BB_INVALID;
populate_files(bb, real);
free(real); // estate
if (e->index >= 0)
set_cursor(bb, e->index);
2019-05-30 22:25:22 -07:00
return BB_OK;
}
case 'j': { // +jump:
if (!value) return BB_INVALID;
char key = value[0];
if (bb->marks[(int)key]) {
value = bb->marks[(int)key];
goto cd;
}
return BB_INVALID;
}
case 'm': { // +move:, +mark:
switch (cmd[1]) {
case 'a': { // +mark:
if (!value) return BB_INVALID;
char key = value[0];
if (key < 0) return BB_INVALID;
value = strchr(value, '=');
if (!value) value = bb->path;
else ++value;
if (bb->marks[(int)key])
free(bb->marks[(int)key]);
bb->marks[(int)key] = memcheck(strdup(value));
2019-05-30 22:25:22 -07:00
return BB_OK;
}
default: { // +move:
int oldcur, isdelta, n;
move:
if (!value) return BB_INVALID;
oldcur = bb->cursor;
isdelta = value[0] == '-' || value[0] == '+';
n = (int)strtol(value, &value, 10);
if (*value == '%')
n = (n * (value[1] == 'n' ? bb->nfiles : termheight)) / 100;
if (isdelta) set_cursor(bb, bb->cursor + n);
else set_cursor(bb, n);
if (cmd[0] == 's') { // +spread:
int sel = IS_SELECTED(bb->files[oldcur]);
for (int i = bb->cursor; i != oldcur; i += (oldcur > i ? 1 : -1)) {
if (sel != IS_SELECTED(bb->files[i]))
toggle_entry(bb, bb->files[i]);
}
}
2019-05-30 22:25:22 -07:00
return BB_OK;
}
}
}
case 'q': // +quit
return BB_QUIT;
2019-05-30 22:25:22 -07:00
case 'r': // +refresh
populate_files(bb, bb->path);
2019-05-30 22:25:22 -07:00
return BB_OK;
case 's': // +scroll:, +select:, +sort:, +spread:
switch (cmd[1]) {
case 'c': { // scroll:
if (!value) return BB_INVALID;
// TODO: figure out the best version of this
int isdelta = value[0] == '+' || value[0] == '-';
int n = (int)strtol(value, &value, 10);
if (*value == '%')
n = (n * (value[1] == 'n' ? bb->nfiles : termheight)) / 100;
if (isdelta)
set_scroll(bb, bb->scroll + n);
else
set_scroll(bb, n);
2019-05-30 22:25:22 -07:00
return BB_OK;
}
case '\0': case 'e': // +select:
if (!value) value = bb->files[bb->cursor]->name;
if (strcmp(value, "*") == 0) {
for (int i = 0; i < bb->nfiles; i++) {
if (strcmp(bb->files[i]->name, ".")
&& strcmp(bb->files[i]->name, ".."))
select_entry(bb, bb->files[i]);
}
} else {
entry_t *e = load_entry(bb, value);
if (e) select_entry(bb, e);
}
2019-05-30 22:25:22 -07:00
return BB_OK;
case 'o': // +sort:
if (!value) return BB_INVALID;
set_sort(bb, value);
sort_files(bb);
2019-05-30 22:25:22 -07:00
return BB_OK;
case 'p': // +spread:
goto move;
}
case 't': { // +toggle:
if (!value) value = bb->files[bb->cursor]->name;
entry_t *e = load_entry(bb, value);
if (e) {
toggle_entry(bb, e);
if (e->refcount <= 0) remove_entry(e);
}
2019-05-30 22:25:22 -07:00
return BB_OK;
}
default: err("UNKNOWN COMMAND: '%s'", cmd); break;
}
return BB_INVALID;
}
/*
* Use bb to browse a path.
*/
void bb_browse(bb_t *bb, const char *path)
{
static long cmdpos = 0;
int lastwidth = termwidth, lastheight = termheight;
int check_cmds = 1;
for (int i = 0; startupcmds[i]; i++) {
if (startupcmds[i][0] == '+') {
if (execute_cmd(bb, startupcmds[i] + 1) == BB_QUIT)
goto quit;
} else {
run_cmd_on_selection(bb, startupcmds[i]);
check_cmds = 1;
}
}
populate_files(bb, path);
init_term();
fputs(T_ON(T_ALT_SCREEN), tty_out);
bb->scroll = 0;
bb->cursor = 0;
2019-05-20 19:28:47 -07:00
refresh:
bb->dirty = 1;
2019-05-23 19:04:17 -07:00
redraw:
render(bb);
bb->dirty = 0;
2019-05-23 19:04:17 -07:00
next_input:
if (termwidth != lastwidth || termheight != lastheight) {
lastwidth = termwidth; lastheight = termheight;
bb->dirty = 1;
2019-05-23 19:04:17 -07:00
goto redraw;
}
if (check_cmds) {
FILE *cmdfile = fopen(cmdfilename, "r");
if (!cmdfile) {
if (bb->dirty) goto redraw;
goto get_keyboard_input;
}
if (cmdpos)
fseek(cmdfile, cmdpos, SEEK_SET);
char *cmd = NULL;
size_t cap = 0;
while (cmdfile && getdelim(&cmd, &cap, '\0', cmdfile) >= 0) {
cmdpos = ftell(cmdfile);
if (!cmd[0]) continue;
2019-05-30 22:25:22 -07:00
if (execute_cmd(bb, cmd) == BB_QUIT) {
free(cmd);
fclose(cmdfile);
goto quit;
2019-05-23 19:04:17 -07:00
}
}
free(cmd);
fclose(cmdfile);
unlink(cmdfilename);
cmdpos = 0;
check_cmds = 0;
goto refresh;
2019-05-23 19:04:17 -07:00
}
int key;
get_keyboard_input:
key = bgetkey(tty_in, &mouse_x, &mouse_y, KEY_DELAY);
2019-05-23 19:04:17 -07:00
switch (key) {
case KEY_MOUSE_LEFT: {
struct timespec clicktime;
clock_gettime(CLOCK_MONOTONIC, &clicktime);
double dt_ms = 1e3*(double)(clicktime.tv_sec - lastclick.tv_sec);
dt_ms += 1e-6*(double)(clicktime.tv_nsec - lastclick.tv_nsec);
lastclick = clicktime;
// Get column:
char column[3] = "+?";
for (int col = 0, x = 0; bb->columns[col]; col++) {
if (col > 0) x += 1;
x += bb->colwidths[col];
if (x >= mouse_x) {
column[1] = bb->columns[col];
break;
}
}
if (mouse_y == 1) {
char *pos;
if ((pos = strstr(bb->sort, column)) && pos == bb->sort)
column[0] = '-';
set_sort(bb, column);
sort_files(bb);
goto refresh;
} else if (2 <= mouse_y && mouse_y <= termheight - 2) {
int clicked = bb->scroll + (mouse_y - 2);
if (clicked > bb->nfiles - 1) goto next_input;
2019-05-30 00:33:51 -07:00
if (column[1] == COL_SELECTED) {
toggle_entry(bb, bb->files[clicked]);
bb->dirty = 1;
goto redraw;
2019-05-23 19:04:17 -07:00
}
set_cursor(bb, clicked);
if (dt_ms <= 200) {
2019-05-23 19:04:17 -07:00
key = KEY_MOUSE_DOUBLE_LEFT;
goto user_bindings;
}
goto redraw;
2019-05-23 19:04:17 -07:00
}
break;
}
2019-05-23 19:04:17 -07:00
case KEY_CTRL_C:
cleanup_and_exit(SIGINT);
2019-05-23 19:04:17 -07:00
case KEY_CTRL_Z:
fputs(T_OFF(T_ALT_SCREEN), tty_out);
2019-05-23 19:04:17 -07:00
close_term();
raise(SIGTSTP);
init_term();
fputs(T_ON(T_ALT_SCREEN), tty_out);
bb->dirty = 1;
2019-05-23 19:04:17 -07:00
goto redraw;
case -1:
goto next_input;
default: {
2019-05-23 19:04:17 -07:00
// Search user-defined key bindings from config.h:
binding_t *binding;
user_bindings:
2019-05-23 19:04:17 -07:00
for (int i = 0; bindings[i].keys[0] > 0; i++) {
for (int j = 0; bindings[i].keys[j]; j++) {
if (key == bindings[i].keys[j]) {
// Move to front optimization:
if (i > 2) {
binding_t tmp;
tmp = bindings[0];
bindings[0] = bindings[i];
bindings[i] = tmp;
i = 0;
}
2019-05-23 19:04:17 -07:00
binding = &bindings[i];
goto run_binding;
}
}
2019-05-23 19:04:17 -07:00
}
// Nothing matched
goto next_input;
2019-05-23 19:04:17 -07:00
run_binding:
if (cmdpos != 0)
err("Command file still open");
if (binding->command[0] == '+') {
2019-05-30 22:25:22 -07:00
if (execute_cmd(bb, binding->command + 1) == BB_QUIT)
goto quit;
goto refresh;
}
move_cursor(tty_out, 0, termheight-1);
if (binding->flags & NORMAL_TERM) {
fputs(T_OFF(T_ALT_SCREEN), tty_out);
fputs(T_ON(T_WRAP), tty_out);
}
if (binding->flags & AT_CURSOR && !bb->firstselected) {
move_cursor(tty_out, 0, 3 + bb->cursor - bb->scroll);
fputs("\033[K", tty_out);
}
close_term();
run_cmd_on_selection(bb, binding->command);
2019-05-23 19:04:17 -07:00
init_term();
if (binding->flags & NORMAL_TERM)
fputs(T_ON(T_ALT_SCREEN), tty_out);
if (binding->flags & NORMAL_TERM)
bb->dirty = 1;
check_cmds = 1;
goto redraw;
}
2019-05-20 19:28:47 -07:00
}
2019-05-23 19:04:17 -07:00
goto next_input;
quit:
populate_files(bb, NULL);
fputs(T_LEAVE_BBMODE, tty_out);
2019-05-30 00:33:51 -07:00
cleanup();
2019-05-20 19:28:47 -07:00
}
/*
* Print the current key bindings
*/
void print_bindings(void)
{
struct winsize sz = {0};
ioctl(STDOUT_FILENO, TIOCGWINSZ, &sz);
int width = sz.ws_col;
if (width == 0) width = 80;
char buf[1024];
char *kb = "Key Bindings";
printf("\n\033[33;1;4m\033[%dG%s\033[0m\n\n", (width-(int)strlen(kb))/2, kb);
for (int i = 0; bindings[i].keys[0]; i++) {
char *p = buf;
for (int j = 0; bindings[i].keys[j]; j++) {
if (j > 0) *(p++) = ',';
int key = bindings[i].keys[j];
const char *name = bkeyname(key);
if (name)
p = stpcpy(p, name);
else if (' ' <= key && key <= '~')
p += sprintf(p, "%c", (char)key);
else
p += sprintf(p, "\033[31m\\x%02X", key);
}
*p = '\0';
printf("\033[1m\033[%dG%s\033[0m", width/2 - 1 - (int)strlen(buf), buf);
printf("\033[0m\033[%dG\033[34;1m%s\033[0m", width/2 + 1, bindings[i].description);
printf("\033[0m\n");
}
printf("\n");
}
2019-05-20 19:28:47 -07:00
int main(int argc, char *argv[])
{
2019-05-31 00:03:48 -07:00
char *initial_path = NULL, depthstr[64] = {0};
char sep = '\n';
2019-05-21 04:06:50 -07:00
int print_dir = 0, print_selection = 0;
2019-05-23 19:04:17 -07:00
int cmd_args = 0;
2019-05-21 04:06:50 -07:00
for (int i = 1; i < argc; i++) {
2019-05-23 19:04:17 -07:00
if (argv[i][0] == '+') {
++cmd_args;
continue;
}
if (strcmp(argv[i], "--") == 0) {
if (i + 1 < argc) initial_path = argv[i+1];
break;
}
if (argv[i][0] != '-' && !initial_path)
initial_path = argv[i];
}
if (!initial_path && cmd_args == 0) initial_path = ".";
2019-05-31 00:03:48 -07:00
int cmdfd = -1;
if (initial_path) {
has_initial_path:
cmdfilename = memcheck(strdup(CMDFILE_FORMAT));
2019-05-31 00:03:48 -07:00
if ((cmdfd = mkostemp(cmdfilename, O_APPEND)) == -1)
err("Couldn't create tmpfile: '%s'", CMDFILE_FORMAT);
// Set up environment variables
2019-05-31 00:03:48 -07:00
char *curdepth = getenv("BB_DEPTH");
int depth = curdepth ? atoi(curdepth) : 0;
sprintf(depthstr, "%d", depth + 1);
setenv("BB_DEPTH", depthstr, 1);
setenv("BBCMD", cmdfilename, 1);
} else if (cmd_args > 0) {
char *parent_bbcmd = getenv("BBCMD");
if (!parent_bbcmd || parent_bbcmd[0] == '\0') {
initial_path = ".";
goto has_initial_path;
2019-05-23 19:04:17 -07:00
}
2019-05-31 00:03:48 -07:00
cmdfd = open(parent_bbcmd, O_RDWR | O_APPEND);
if (cmdfd == -1) err("Couldn't open cmdfile: '%s'\n", parent_bbcmd);
2019-05-23 19:04:17 -07:00
}
int i;
for (i = 1; i < argc; i++) {
if (argv[i][0] == '?') {
2019-05-31 00:03:48 -07:00
if (cmdfd != -1)
close(cmdfd);
init_term();
char *line = breadline(tty_in, tty_out, argv[i] + 1, argv[i+1]);
close_term();
if (!line) return 1;
fputs(line, stdout);
free(line);
fflush(stdout);
return 0;
}
2019-05-23 19:04:17 -07:00
if (argv[i][0] == '+') {
2019-05-31 00:03:48 -07:00
write(cmdfd, argv[i]+1, strlen(argv[i]+1)+1);
2019-05-23 19:04:17 -07:00
continue;
}
if (strcmp(argv[i], "--") == 0) break;
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] [-d] [-0] (+command)* [path]\n");
2019-05-25 23:31:03 -07:00
return 0;
}
if (strcmp(argv[i], "--version") == 0) {
version:
printf("bb " BB_VERSION "\n");
return 0;
2019-05-22 01:55:34 -07:00
}
if (argv[i][0] == '-' && argv[i][1] == '-') {
if (argv[i][2] == '\0') break;
2019-05-22 01:50:46 -07:00
continue;
}
2019-05-22 01:50:46 -07:00
if (argv[i][0] == '-') {
for (char *c = &argv[i][1]; *c; c++) {
switch (*c) {
case 'h':goto usage;
2019-05-25 23:31:03 -07:00
case 'v': goto version;
case 'd': print_dir = 1;
break;
case '0': sep = '\0';
break;
case 's': print_selection = 1;
break;
case 'b': print_bindings();
return 0;
2019-05-22 01:50:46 -07:00
}
}
continue;
}
2019-05-21 04:06:50 -07:00
}
2019-05-31 00:03:48 -07:00
if (cmdfd != -1) {
close(cmdfd);
cmdfd = -1;
}
if (!initial_path)
return 0;
// Default values
setenv("SHELL", "bash", 0);
setenv("EDITOR", "nano", 0);
setenv("PAGER", "less", 0);
2019-05-23 19:04:17 -07:00
2019-05-30 00:33:51 -07:00
atexit(cleanup);
signal(SIGTERM, cleanup_and_exit);
signal(SIGINT, cleanup_and_exit);
signal(SIGXCPU, cleanup_and_exit);
signal(SIGXFSZ, cleanup_and_exit);
signal(SIGVTALRM, cleanup_and_exit);
signal(SIGPROF, cleanup_and_exit);
signal(SIGSEGV, cleanup_and_exit);
char *real = realpath(initial_path, NULL);
if (!real || chdir(real)) err("Not a valid path: %s\n", initial_path);
2019-05-31 00:03:48 -07:00
bb = memcheck(calloc(1, sizeof(bb_t)));
bb->columns[0] = COL_NAME;
strcpy(bb->sort, "+/+n");
bb_browse(bb, real);
free(real);
if (bb->firstselected && print_selection) {
for (entry_t *e = bb->firstselected; e; e = e->selected.next) {
const char *p = e->fullname;
while (*p) {
const char *p2 = strchr(p, '\n');
if (!p2) p2 = p + strlen(p);
write(STDOUT_FILENO, p, (size_t)(p2 - p));
if (*p2 == '\n' && sep == '\n')
write(STDOUT_FILENO, "\\", 1);
p = p2;
}
write(STDOUT_FILENO, &sep, 1);
}
fflush(stdout);
}
if (print_dir) {
2019-05-23 19:04:17 -07:00
printf("%s\n", initial_path);
}
for (int m = 0; m < 128; m++)
if (bb->marks[m]) free(bb->marks[m]);
free(bb);
2019-05-23 19:04:17 -07:00
return 0;
2019-05-20 19:28:47 -07:00
}
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1