2019-09-30 17:06:27 -07:00
|
|
|
|
/*
|
|
|
|
|
* Bitty Browser (bb)
|
|
|
|
|
* Copyright 2019 Bruce Hill
|
|
|
|
|
* Released under the MIT license
|
|
|
|
|
*
|
|
|
|
|
* This file contains definitions and customization for `bb`.
|
|
|
|
|
*/
|
|
|
|
|
#include <dirent.h>
|
|
|
|
|
#include <fcntl.h>
|
|
|
|
|
#include <limits.h>
|
|
|
|
|
#include <signal.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>
|
|
|
|
|
#include <termios.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
|
|
#include "bterm.h"
|
|
|
|
|
|
|
|
|
|
// Macros:
|
2019-10-13 21:21:20 -07:00
|
|
|
|
#define BB_VERSION "0.17.2"
|
2019-09-30 17:06:27 -07:00
|
|
|
|
|
|
|
|
|
#ifndef PATH_MAX
|
|
|
|
|
#define PATH_MAX 4096
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#define MAX_COLS 12
|
|
|
|
|
#define MAX_SORT (2*MAX_COLS)
|
|
|
|
|
#define HASH_SIZE 1024
|
|
|
|
|
#define HASH_MASK (HASH_SIZE - 1)
|
|
|
|
|
#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 IS_VIEWED(p) ((p)->index >= 0)
|
|
|
|
|
#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))
|
|
|
|
|
#define ONSCREEN (termheight - 3)
|
|
|
|
|
|
|
|
|
|
#ifdef __APPLE__
|
|
|
|
|
#define mtime(s) (s).st_mtimespec
|
|
|
|
|
#define atime(s) (s).st_atimespec
|
|
|
|
|
#define ctime(s) (s).st_ctimespec
|
|
|
|
|
#else
|
|
|
|
|
#define mtime(s) (s).st_mtim
|
|
|
|
|
#define atime(s) (s).st_atim
|
|
|
|
|
#define ctime(s) (s).st_ctim
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#define err(...) do { \
|
|
|
|
|
cleanup(); \
|
|
|
|
|
fprintf(stderr, __VA_ARGS__); \
|
|
|
|
|
if (errno) fprintf(stderr, "\n%s", strerror(errno)); \
|
|
|
|
|
fprintf(stderr, "\n"); \
|
|
|
|
|
exit(EXIT_FAILURE); \
|
|
|
|
|
} while (0)
|
|
|
|
|
|
2019-10-13 19:34:16 -07:00
|
|
|
|
#define warn(...) do { \
|
2019-10-27 14:58:23 -07:00
|
|
|
|
move_cursor(tty_out, 0, termheight-1); \
|
|
|
|
|
fputs("\033[41;33;1m", tty_out); \
|
|
|
|
|
fprintf(tty_out, __VA_ARGS__); \
|
|
|
|
|
fputs(" Press any key to continue...\033[0m ", tty_out); \
|
|
|
|
|
fflush(tty_out); \
|
|
|
|
|
while (bgetkey(tty_in, NULL, NULL) == -1) usleep(100); \
|
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
#define LL_PREPEND(head, node, name) do { \
|
|
|
|
|
((node)->name).atme = &(head); \
|
|
|
|
|
((node)->name).next = head; \
|
|
|
|
|
if (head) ((head)->name).atme = &(((node)->name).next); \
|
|
|
|
|
head = node; \
|
|
|
|
|
} while (0)
|
|
|
|
|
|
|
|
|
|
#define LL_REMOVE(node, name) do { \
|
|
|
|
|
if (((node)->name).next) \
|
|
|
|
|
((__typeof__(node))(node)->name.next)->name.atme = ((node)->name).atme; \
|
|
|
|
|
*(((node)->name).atme) = ((node)->name).next; \
|
|
|
|
|
((node)->name).atme = NULL; \
|
|
|
|
|
((node)->name).next = NULL; \
|
2019-10-13 19:34:16 -07:00
|
|
|
|
} while (0)
|
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
|
// Types:
|
|
|
|
|
typedef struct {
|
|
|
|
|
int key;
|
|
|
|
|
char *script;
|
|
|
|
|
char *description;
|
|
|
|
|
} binding_t;
|
|
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
|
int width;
|
|
|
|
|
const char *name;
|
|
|
|
|
} column_t;
|
|
|
|
|
|
|
|
|
|
typedef enum {
|
|
|
|
|
COL_NONE = 0,
|
|
|
|
|
COL_NAME = 'n',
|
|
|
|
|
COL_SIZE = 's',
|
|
|
|
|
COL_PERM = 'p',
|
|
|
|
|
COL_MTIME = 'm',
|
|
|
|
|
COL_CTIME = 'c',
|
|
|
|
|
COL_ATIME = 'a',
|
|
|
|
|
COL_RANDOM = 'r',
|
|
|
|
|
COL_SELECTED = '*',
|
|
|
|
|
} column_e;
|
|
|
|
|
|
|
|
|
|
/* 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 entry_s {
|
2019-10-27 14:58:23 -07:00
|
|
|
|
struct {
|
|
|
|
|
struct entry_s *next, **atme;
|
|
|
|
|
} selected, hash;
|
2019-09-30 17:06:27 -07:00
|
|
|
|
char *name, *linkname;
|
|
|
|
|
struct stat info;
|
|
|
|
|
mode_t linkedmode;
|
|
|
|
|
int no_esc : 1;
|
|
|
|
|
int link_no_esc : 1;
|
|
|
|
|
int shufflepos;
|
|
|
|
|
int index;
|
|
|
|
|
char fullname[1];
|
|
|
|
|
// ------- fullname must be last! --------------
|
|
|
|
|
// When entries are allocated, extra space on the end is reserved to fill
|
|
|
|
|
// in fullname.
|
|
|
|
|
} entry_t;
|
|
|
|
|
|
|
|
|
|
typedef struct bb_s {
|
|
|
|
|
entry_t *hash[HASH_SIZE];
|
|
|
|
|
entry_t **files;
|
|
|
|
|
entry_t *firstselected;
|
|
|
|
|
char path[PATH_MAX];
|
|
|
|
|
char prev_path[PATH_MAX];
|
|
|
|
|
int nfiles;
|
|
|
|
|
int scroll, cursor;
|
|
|
|
|
|
|
|
|
|
char sort[MAX_SORT+1];
|
|
|
|
|
char columns[MAX_COLS+1];
|
|
|
|
|
unsigned int dirty : 1;
|
|
|
|
|
unsigned int show_dotdot : 1;
|
|
|
|
|
unsigned int show_dot : 1;
|
|
|
|
|
unsigned int show_dotfiles : 1;
|
|
|
|
|
unsigned int interleave_dirs : 1;
|
|
|
|
|
unsigned int should_quit : 1;
|
|
|
|
|
} bb_t;
|
|
|
|
|
|
2019-10-27 14:58:23 -07:00
|
|
|
|
// For keeping track of child processes
|
|
|
|
|
typedef struct proc_s {
|
|
|
|
|
pid_t pid;
|
|
|
|
|
struct {
|
|
|
|
|
struct proc_s *next, **atme;
|
|
|
|
|
} running;
|
|
|
|
|
} proc_t;
|
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
|
// Configurable options:
|
|
|
|
|
#define SCROLLOFF MIN(5, (termheight-4)/2)
|
|
|
|
|
#define CMDFILE_FORMAT "/tmp/bb.XXXXXX"
|
|
|
|
|
#define SORT_INDICATOR "↓"
|
|
|
|
|
#define RSORT_INDICATOR "↑"
|
|
|
|
|
#define SELECTED_INDICATOR " \033[31;7m \033[0m"
|
|
|
|
|
#define NOT_SELECTED_INDICATOR " "
|
|
|
|
|
// Colors (using ANSI escape sequences):
|
|
|
|
|
#define TITLE_COLOR "\033[37;1m"
|
|
|
|
|
#define NORMAL_COLOR "\033[37m"
|
|
|
|
|
#define CURSOR_COLOR "\033[43;30;1m"
|
|
|
|
|
#define LINK_COLOR "\033[35m"
|
|
|
|
|
#define DIR_COLOR "\033[34m"
|
|
|
|
|
#define EXECUTABLE_COLOR "\033[31m"
|
|
|
|
|
|
|
|
|
|
#define MAX_BINDINGS 1024
|
|
|
|
|
|
|
|
|
|
binding_t bindings[MAX_BINDINGS];
|
|
|
|
|
|
|
|
|
|
// Column widths and titles:
|
|
|
|
|
const column_t columns[] = {
|
|
|
|
|
['*'] = {2, "*"},
|
|
|
|
|
['a'] = {21, " Accessed"},
|
|
|
|
|
['c'] = {21, " Created"},
|
|
|
|
|
['m'] = {21, " Modified"},
|
|
|
|
|
['n'] = {40, "Name"},
|
|
|
|
|
['p'] = {5, "Permissions"},
|
|
|
|
|
['r'] = {2, "Random"},
|
|
|
|
|
['s'] = {9, " Size"},
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
// Functions
|
2019-10-13 21:20:00 -07:00
|
|
|
|
void bb_browse(bb_t *bb);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
static void cleanup(void);
|
2019-10-27 14:58:23 -07:00
|
|
|
|
static void cleanup_and_raise(int sig);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
static const char* color_of(mode_t mode);
|
|
|
|
|
#ifdef __APPLE__
|
|
|
|
|
static int compare_files(void *v, const void *v1, const void *v2);
|
|
|
|
|
#else
|
|
|
|
|
static int compare_files(const void *v1, const void *v2, void *v);
|
|
|
|
|
#endif
|
|
|
|
|
static int fputs_escaped(FILE *f, const char *str, const char *color);
|
|
|
|
|
static void init_term(void);
|
|
|
|
|
static int is_simple_bbcmd(const char *s);
|
|
|
|
|
static entry_t* load_entry(bb_t *bb, const char *path, int clear_dots);
|
2019-10-13 18:33:46 -07:00
|
|
|
|
static inline int matches_cmd(const char *str, const char *cmd);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
static void* memcheck(void *p);
|
|
|
|
|
static void normalize_path(const char *root, const char *path, char *pbuf, int clear_dots);
|
2019-10-13 21:20:00 -07:00
|
|
|
|
static int populate_files(bb_t *bb, const char *path);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
static void print_bindings(int fd);
|
2019-10-13 19:34:16 -07:00
|
|
|
|
static void run_bbcmd(bb_t *bb, const char *cmd);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
static void render(bb_t *bb);
|
|
|
|
|
static void restore_term(const struct termios *term);
|
|
|
|
|
static int run_script(bb_t *bb, const char *cmd);
|
|
|
|
|
static void set_cursor(bb_t *bb, int i);
|
|
|
|
|
static void set_selected(bb_t *bb, entry_t *e, int selected);
|
|
|
|
|
static void set_scroll(bb_t *bb, int i);
|
|
|
|
|
static void set_sort(bb_t *bb, const char *sort);
|
|
|
|
|
static void sort_files(bb_t *bb);
|
|
|
|
|
static char *trim(char *s);
|
|
|
|
|
static int try_free_entry(entry_t *e);
|
|
|
|
|
static void update_term_size(int sig);
|
2019-10-27 14:58:23 -07:00
|
|
|
|
static int wait_for_process(proc_t **proc);
|
2019-09-30 17:06:27 -07:00
|
|
|
|
|
|
|
|
|
// Constants
|
|
|
|
|
static const char *T_ENTER_BBMODE = T_OFF(T_SHOW_CURSOR ";" T_WRAP) T_ON(T_ALT_SCREEN ";" T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR);
|
|
|
|
|
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 ";" T_WRAP);
|
|
|
|
|
static const char *T_LEAVE_BBMODE_PARTIAL = T_OFF(T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR) T_ON(T_WRAP);
|
|
|
|
|
static const struct termios default_termios = {
|
|
|
|
|
.c_iflag = ICRNL,
|
|
|
|
|
.c_oflag = OPOST | ONLCR | NL0 | CR0 | TAB0 | BS0 | VT0 | FF0,
|
|
|
|
|
.c_lflag = ISIG | ICANON | IEXTEN | ECHO | ECHOE | ECHOK | ECHOCTL | ECHOKE,
|
|
|
|
|
.c_cflag = CS8 | CREAD,
|
|
|
|
|
.c_cc[VINTR] = '',
|
|
|
|
|
.c_cc[VQUIT] = '',
|
|
|
|
|
.c_cc[VERASE] = 127,
|
|
|
|
|
.c_cc[VKILL] = '',
|
|
|
|
|
.c_cc[VEOF] = '',
|
|
|
|
|
.c_cc[VSTART] = '',
|
|
|
|
|
.c_cc[VSTOP] = '',
|
|
|
|
|
.c_cc[VSUSP] = '',
|
|
|
|
|
.c_cc[VREPRINT] = '',
|
|
|
|
|
.c_cc[VWERASE] = '',
|
|
|
|
|
.c_cc[VLNEXT] = '',
|
|
|
|
|
.c_cc[VDISCARD] = '',
|
|
|
|
|
.c_cc[VMIN] = 1,
|
|
|
|
|
.c_cc[VTIME] = 0,
|
|
|
|
|
};
|
|
|
|
|
|
2019-10-03 13:26:31 -07:00
|
|
|
|
static const char *description_str = "bb - an itty bitty console TUI file browser\n";
|
|
|
|
|
static const char *usage_str = "Usage: bb (-h/--help | -v/--version | -s | -d | -0 | +command | path)*\n";
|
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
|
// Shell functions
|
|
|
|
|
static const char *bbcmdfn = "bb() {\n"
|
|
|
|
|
" if test $# -eq 0; then cat >> $BBCMD; return; fi\n"
|
|
|
|
|
" for arg; do\n"
|
|
|
|
|
" shift;\n"
|
2019-10-02 01:40:40 -07:00
|
|
|
|
" if expr \"$arg\" : \"^+[^:]*:$\" >/dev/null; then\n"
|
|
|
|
|
" if test $# -gt 0; then printf \"$arg%s\\0\" \"$@\" >> $BBCMD;\n"
|
|
|
|
|
" else sed \"s/\\([^\\x00]\\+\\)/$arg\\1/g\" >> $BBCMD; fi;\n"
|
|
|
|
|
" return;\n"
|
|
|
|
|
" fi;\n"
|
|
|
|
|
" printf \"%s\\0\" \"$arg\" >> $BBCMD;\n"
|
|
|
|
|
" done;\n"
|
2019-09-30 17:06:27 -07:00
|
|
|
|
"}\n"
|
|
|
|
|
"ask() {\n"
|
|
|
|
|
#ifdef ASK
|
|
|
|
|
ASK ";\n"
|
|
|
|
|
#else
|
2019-10-03 14:57:34 -07:00
|
|
|
|
" [ $# -lt 2 ] && printf '\033[31;1mNot enough args to ask!\033[0m\n' && return 1;\n"
|
2019-10-11 19:07:05 -07:00
|
|
|
|
" printf \"\033[1m%s\033[0m\" \"$2\" >/dev/tty;\n"
|
|
|
|
|
" tput cvvis >/dev/tty;\n"
|
|
|
|
|
" read $1 </dev/tty >/dev/tty;\n"
|
2019-09-30 17:06:27 -07:00
|
|
|
|
#endif
|
|
|
|
|
"}\n"
|
|
|
|
|
"ask1() {\n"
|
|
|
|
|
#ifdef ASK1
|
|
|
|
|
ASK1 ";\n"
|
|
|
|
|
#else
|
2019-10-11 19:07:05 -07:00
|
|
|
|
" tput civis >/dev/tty;\n"
|
2019-09-30 17:06:27 -07:00
|
|
|
|
" printf \"\033[1m%s\033[0m\" \"$2\" >/dev/tty;\n"
|
|
|
|
|
" stty -icanon -echo >/dev/tty;\n"
|
|
|
|
|
" eval \"$1=\\$(dd bs=1 count=1 2>/dev/null </dev/tty)\";\n"
|
|
|
|
|
" stty icanon echo >/dev/tty;\n"
|
2019-10-11 19:07:05 -07:00
|
|
|
|
" tput cvvis >/dev/tty;\n"
|
2019-09-30 17:06:27 -07:00
|
|
|
|
#endif
|
|
|
|
|
"}\n"
|
|
|
|
|
"confirm() {\n"
|
|
|
|
|
#ifdef CONFIRM
|
|
|
|
|
CONFIRM ";\n"
|
|
|
|
|
#else
|
|
|
|
|
" ask1 REPLY \"\033[1mIs that okay? [y/N] \" && [ \"$REPLY\" = 'y' ];\n"
|
|
|
|
|
#endif
|
|
|
|
|
"}\n"
|
|
|
|
|
"pause() {\n"
|
|
|
|
|
#ifdef PAUSE
|
|
|
|
|
PAUSE ";\n"
|
|
|
|
|
#else
|
2019-09-30 17:08:59 -07:00
|
|
|
|
" ask1 REPLY \"\033[0;2mPress any key to continue...\033[0m\";"
|
2019-09-30 17:06:27 -07:00
|
|
|
|
#endif
|
|
|
|
|
"}\n"
|
|
|
|
|
"pick() {\n"
|
|
|
|
|
#ifdef PICK
|
|
|
|
|
PICK ";\n"
|
|
|
|
|
#else
|
|
|
|
|
" ask query \"$1\" && awk '{print length, $1}' | sort -n | cut -d' ' -f2- |\n"
|
|
|
|
|
" grep -i -m1 \"$(echo \"$query\" | sed 's;.;[^/&]*[&];g')\";\n"
|
|
|
|
|
#endif
|
|
|
|
|
"}\n"
|
|
|
|
|
"spin() {\n"
|
|
|
|
|
#ifdef SPIN
|
|
|
|
|
SPIN ";\n"
|
|
|
|
|
#else
|
|
|
|
|
" eval \"$@\" &\n"
|
|
|
|
|
" pid=$!;\n"
|
|
|
|
|
" spinner='-\\|/';\n"
|
|
|
|
|
" sleep 0.01;\n"
|
|
|
|
|
" while kill -0 $pid 2>/dev/null; do\n"
|
|
|
|
|
" printf '%c\\033[D' \"$spinner\" >/dev/tty;\n"
|
|
|
|
|
" spinner=\"$(echo $spinner | sed 's/\\(.\\)\\(.*\\)/\\2\\1/')\";\n"
|
|
|
|
|
" sleep 0.1;\n"
|
|
|
|
|
" done;\n"
|
|
|
|
|
" wait $pid;\n"
|
|
|
|
|
#endif
|
|
|
|
|
"}\n"
|
|
|
|
|
#ifdef SH
|
|
|
|
|
"alias sh=" SH";\n"
|
|
|
|
|
#else
|
|
|
|
|
#define SH "sh"
|
|
|
|
|
#endif
|
|
|
|
|
;
|
|
|
|
|
|
2019-09-30 17:22:17 -07:00
|
|
|
|
const char *runstartup =
|
|
|
|
|
"[ ! \"$XDG_CONFIG_HOME\" ] && XDG_CONFIG_HOME=~/.config;\n"
|
|
|
|
|
"[ ! \"$sysconfdir\" ] && sysconfdir=/etc;\n"
|
2019-10-09 15:01:50 -07:00
|
|
|
|
"for path in \"$XDG_CONFIG_HOME/bb\" \"$sysconfdir/xdg/bb\" .; do\n"
|
|
|
|
|
" if [ -e \"$path/bbstartup.sh\" ]; then\n"
|
|
|
|
|
" . \"$path/bbstartup.sh\";\n"
|
|
|
|
|
" break;\n"
|
|
|
|
|
" fi;\n"
|
|
|
|
|
"done\n";
|
2019-09-30 17:22:17 -07:00
|
|
|
|
|
2019-09-30 17:06:27 -07:00
|
|
|
|
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1
|