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-26 13:41:09 -07:00
|
|
|
#include <limits.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>
|
2019-05-26 02:37:41 -07:00
|
|
|
#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"
|
2019-05-28 21:36:42 -07:00
|
|
|
#include "bterm.h"
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-29 00:58:33 -07:00
|
|
|
#define BB_VERSION "0.11.0"
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-05-26 13:41:09 -07:00
|
|
|
#ifndef PATH_MAX
|
|
|
|
#define PATH_MAX 4096
|
|
|
|
#endif
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
#define MAX(a,b) ((a) < (b) ? (b) : (a))
|
|
|
|
#define MIN(a,b) ((a) > (b) ? (b) : (a))
|
2019-05-23 01:37:41 -07:00
|
|
|
#define IS_SELECTED(p) (((p)->atme) != NULL)
|
2019-05-28 23:51:20 -07:00
|
|
|
#define OPTNUM(o) ((int)((o) & ~'0'))
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-28 00:07:43 -07:00
|
|
|
static const char *T_ENTER_BBMODE = T_OFF(T_SHOW_CURSOR) T_ON(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);
|
|
|
|
static const char *T_LEAVE_BBMODE_PARTIAL = T_OFF(T_MOUSE_XY ";" T_MOUSE_CELL ";" T_MOUSE_SGR);
|
2019-05-27 15:53:07 -07:00
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
#define err(...) do { \
|
2019-05-27 15:53:07 -07:00
|
|
|
close_term(); \
|
|
|
|
fputs(T_OFF(T_ALT_SCREEN), stdout); \
|
|
|
|
fflush(stdout); \
|
2019-05-23 19:04:17 -07:00
|
|
|
fprintf(stderr, __VA_ARGS__); \
|
2019-05-26 16:01:56 -07:00
|
|
|
if (errno) fprintf(stderr, "\n%s", strerror(errno)); \
|
2019-05-23 19:04:17 -07:00
|
|
|
fprintf(stderr, "\n"); \
|
2019-05-24 16:55:36 -07:00
|
|
|
cleanup_and_exit(1); \
|
2019-05-23 19:04:17 -07:00
|
|
|
} while (0)
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-25 04:30:51 -07:00
|
|
|
// Types
|
2019-05-21 05:10:39 -07:00
|
|
|
typedef enum {
|
2019-05-23 19:04:17 -07:00
|
|
|
SORT_NONE = 0,
|
|
|
|
SORT_NAME = 'n',
|
|
|
|
SORT_SIZE = 's',
|
|
|
|
SORT_PERM = 'p',
|
|
|
|
SORT_MTIME = 'm',
|
|
|
|
SORT_CTIME = 'c',
|
|
|
|
SORT_ATIME = 'a',
|
2019-05-24 20:49:24 -07:00
|
|
|
SORT_RANDOM = 'r',
|
2019-05-29 00:56:49 -07:00
|
|
|
SORT_SELECTED = '+',
|
2019-05-21 05:10:39 -07:00
|
|
|
} sortmethod_t;
|
|
|
|
|
2019-05-25 20:55:59 -07:00
|
|
|
/* 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.
|
2019-05-25 04:30:51 -07:00
|
|
|
*/
|
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-28 00:07:43 -07:00
|
|
|
char *name, *linkname;
|
2019-05-27 21:14:34 -07:00
|
|
|
struct stat info;
|
2019-05-28 21:36:42 -07:00
|
|
|
mode_t linkedmode;
|
2019-05-27 21:14:34 -07:00
|
|
|
int refcount : 2; // Should only be between 0-2
|
2019-05-28 21:36:42 -07:00
|
|
|
int no_esc : 1;
|
|
|
|
int link_no_esc : 1;
|
2019-05-28 22:04:02 -07:00
|
|
|
int shufflepos;
|
2019-05-28 00:07:43 -07:00
|
|
|
char fullname[1]; // Must be last
|
2019-05-21 02:17:11 -07:00
|
|
|
} entry_t;
|
|
|
|
|
2019-05-27 16:08:29 -07:00
|
|
|
typedef struct bb_s {
|
2019-05-25 04:30:51 -07:00
|
|
|
entry_t **files;
|
2019-05-27 16:08:29 -07:00
|
|
|
entry_t *firstselected;
|
2019-05-26 13:41:09 -07:00
|
|
|
char path[PATH_MAX];
|
2019-05-25 04:30:51 -07:00
|
|
|
int nfiles;
|
2019-05-21 02:17:11 -07:00
|
|
|
int scroll, cursor;
|
2019-05-27 19:51:04 -07:00
|
|
|
char options[128]; // General purpose options
|
2019-05-28 23:51:20 -07:00
|
|
|
char initialopts[128]; // Initial values of the options (after startupcmds)
|
2019-05-25 20:55:59 -07:00
|
|
|
char *marks[128]; // Mapping from key to directory
|
2019-05-27 19:51:04 -07:00
|
|
|
int colwidths[10];
|
2019-05-27 16:08:29 -07:00
|
|
|
} bb_t;
|
2019-05-21 02:17:11 -07:00
|
|
|
|
2019-05-28 21:36:42 -07:00
|
|
|
typedef enum { BB_NOP = 0, BB_INVALID, BB_REFRESH, BB_QUIT } bb_result_t;
|
|
|
|
|
2019-05-25 04:30:51 -07:00
|
|
|
// Functions
|
|
|
|
static void update_term_size(int sig);
|
|
|
|
static void init_term(void);
|
|
|
|
static void close_term(void);
|
2019-05-28 22:04:02 -07:00
|
|
|
static void cleanup_and_exit(int sig);
|
2019-05-25 04:30:51 -07:00
|
|
|
static void* memcheck(void *p);
|
2019-05-27 16:08:29 -07:00
|
|
|
static int run_cmd_on_selection(bb_t *bb, const char *cmd);
|
2019-05-28 21:36:42 -07:00
|
|
|
static int fputs_escaped(FILE *f, const char *str, const char *color);
|
2019-05-28 22:04:02 -07:00
|
|
|
static const char* color_of(mode_t mode);
|
2019-05-27 16:08:29 -07:00
|
|
|
static void render(bb_t *bb, int lazy);
|
2019-05-25 04:30:51 -07:00
|
|
|
static int compare_files(void *r, const void *v1, const void *v2);
|
2019-05-27 16:08:29 -07:00
|
|
|
static int find_file(bb_t *bb, const char *name);
|
|
|
|
static void clear_selection(bb_t *bb);
|
2019-05-28 22:29:22 -07:00
|
|
|
static void select_file(bb_t *bb, entry_t *e);
|
|
|
|
static void deselect_file(bb_t *bb, entry_t *e);
|
2019-05-28 23:51:20 -07:00
|
|
|
static void toggle_file(bb_t *bb, entry_t *e);
|
2019-05-27 16:08:29 -07:00
|
|
|
static void set_cursor(bb_t *bb, int i);
|
|
|
|
static void set_scroll(bb_t *bb, int i);
|
2019-05-28 22:04:02 -07:00
|
|
|
static entry_t* load_entry(const char *path);
|
2019-05-27 16:08:29 -07:00
|
|
|
static void populate_files(bb_t *bb, const char *path);
|
2019-05-28 21:36:42 -07:00
|
|
|
static bb_result_t execute_cmd(bb_t *bb, const char *cmd);
|
2019-05-28 22:29:22 -07:00
|
|
|
static void bb_browse(bb_t *bb, const char *path);
|
2019-05-25 04:30:51 -07:00
|
|
|
static void print_bindings(int verbose);
|
|
|
|
|
2019-05-26 16:34:48 -07:00
|
|
|
// Config options
|
2019-05-25 04:30:51 -07:00
|
|
|
extern binding_t bindings[];
|
2019-05-25 20:55:59 -07:00
|
|
|
extern const char *startupcmds[];
|
|
|
|
|
2019-05-26 16:34:48 -07:00
|
|
|
// Global variables
|
2019-05-27 14:12:47 -07:00
|
|
|
static struct termios orig_termios, bb_termios;
|
|
|
|
static FILE *tty_out = NULL, *tty_in = NULL;
|
2019-05-25 04:30:51 -07:00
|
|
|
static int termwidth, termheight;
|
|
|
|
static int mouse_x, mouse_y;
|
|
|
|
static char *cmdfilename = NULL;
|
2019-05-29 00:56:49 -07:00
|
|
|
static const int colsizew = 7, coldatew = 19, colpermw = 4, colnamew = 40, colselw = 1;
|
2019-05-25 04:30:51 -07:00
|
|
|
static struct timespec lastclick = {0, 0};
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Hanlder for SIGWINCH events
|
|
|
|
*/
|
2019-05-25 04:30:51 -07:00
|
|
|
void update_term_size(int sig)
|
2019-05-23 00:57:25 -07:00
|
|
|
{
|
2019-05-25 04:30:51 -07:00
|
|
|
(void)sig;
|
|
|
|
struct winsize sz = {0};
|
2019-05-27 14:12:47 -07:00
|
|
|
ioctl(fileno(tty_in), TIOCGWINSZ, &sz);
|
2019-05-25 04:30:51 -07:00
|
|
|
termwidth = sz.ws_col;
|
|
|
|
termheight = sz.ws_row;
|
2019-05-23 00:57:25 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Initialize the terminal files for /dev/tty and set up some desired
|
|
|
|
* attributes like passing Ctrl-c as a key instead of interrupting
|
|
|
|
*/
|
2019-05-25 04:30:51 -07:00
|
|
|
void init_term(void)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-27 14:12: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)(
|
2019-05-26 13:41:09 -07:00
|
|
|
IGNBRK | BRKINT | PARMRK | ISTRIP | INLCR | IGNCR | ICRNL | IXON);
|
2019-05-27 14:12:47 -07:00
|
|
|
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;
|
2019-05-28 21:36:42 -07:00
|
|
|
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);
|
2019-05-25 23:35:04 -07:00
|
|
|
// Initiate mouse tracking and disable text wrapping:
|
2019-05-27 15:53:07 -07:00
|
|
|
fputs(T_ENTER_BBMODE, tty_out);
|
2019-05-28 00:07:43 -07:00
|
|
|
fputs(T_OFF(T_WRAP), tty_out);
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Close the /dev/tty terminals and restore some of the attributes.
|
|
|
|
*/
|
2019-05-27 15:53:07 -07:00
|
|
|
void close_term(void)
|
|
|
|
{
|
2019-05-27 14:12:47 -07:00
|
|
|
if (tty_out) {
|
|
|
|
tcsetattr(fileno(tty_out), TCSAFLUSH, &orig_termios);
|
2019-05-28 21:36:42 -07:00
|
|
|
fputs(T_LEAVE_BBMODE_PARTIAL, tty_out);
|
|
|
|
fputs(T_ON(T_WRAP), tty_out);
|
|
|
|
fflush(tty_out);
|
2019-05-27 14:12:47 -07:00
|
|
|
fclose(tty_out);
|
|
|
|
tty_out = NULL;
|
|
|
|
fclose(tty_in);
|
|
|
|
tty_in = NULL;
|
2019-05-28 21:36:42 -07:00
|
|
|
} else {
|
|
|
|
fputs(T_LEAVE_BBMODE_PARTIAL, stdout);
|
|
|
|
fputs(T_ON(T_WRAP), stdout);
|
|
|
|
fflush(stdout);
|
2019-05-24 16:55:36 -07:00
|
|
|
}
|
2019-05-28 21:36:42 -07:00
|
|
|
signal(SIGWINCH, SIG_DFL);
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Close safely in a way that doesn't gunk up the terminal.
|
|
|
|
*/
|
2019-05-28 22:04:02 -07:00
|
|
|
void cleanup_and_exit(int sig)
|
|
|
|
{
|
|
|
|
(void)sig;
|
|
|
|
close_term();
|
|
|
|
unlink(cmdfilename);
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Memory allocation failures are unrecoverable in bb, so this wrapper just
|
|
|
|
* prints an error message and exits if that happens.
|
|
|
|
*/
|
2019-05-25 04:30:51 -07:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -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.
|
|
|
|
*/
|
2019-05-27 16:08:29 -07:00
|
|
|
int run_cmd_on_selection(bb_t *bb, const char *cmd)
|
2019-05-22 14:33:14 -07:00
|
|
|
{
|
|
|
|
pid_t child;
|
2019-05-23 04:02:11 -07:00
|
|
|
sig_t old_handler = signal(SIGINT, SIG_IGN);
|
2019-05-22 14:33:14 -07:00
|
|
|
if ((child = fork()) == 0) {
|
2019-05-23 19:04:17 -07:00
|
|
|
signal(SIGINT, SIG_DFL);
|
2019-05-22 14:33:14 -07:00
|
|
|
// TODO: is there a max number of args? Should this be batched?
|
2019-05-25 04:30:51 -07:00
|
|
|
size_t space = 32;
|
|
|
|
char **args = memcheck(calloc(space, sizeof(char*)));
|
|
|
|
size_t i = 0;
|
2019-05-22 14:33:14 -07:00
|
|
|
args[i++] = "sh";
|
|
|
|
args[i++] = "-c";
|
|
|
|
args[i++] = (char*)cmd;
|
2019-05-26 02:37:41 -07:00
|
|
|
#ifdef __APPLE__
|
2019-05-28 23:51:20 -07:00
|
|
|
args[i++] = "--"; // ensure files like "-i" are not interpreted as flags for sh
|
2019-05-26 02:37:41 -07:00
|
|
|
#endif
|
2019-05-27 16:08:29 -07:00
|
|
|
entry_t *first = bb->firstselected ? bb->firstselected : bb->files[bb->cursor];
|
2019-05-22 14:33:14 -07:00
|
|
|
for (entry_t *e = first; e; e = e->next) {
|
2019-05-28 23:51:20 -07:00
|
|
|
if (i >= space)
|
|
|
|
args = memcheck(realloc(args, (space += 100)*sizeof(char*)));
|
2019-05-28 00:07:43 -07:00
|
|
|
args[i++] = e->fullname;
|
2019-05-22 14:33:14 -07:00
|
|
|
}
|
|
|
|
args[i] = NULL;
|
|
|
|
|
2019-05-28 21:36:42 -07:00
|
|
|
setenv("BBSELECTED", bb->firstselected ? "1" : "", 1);
|
|
|
|
setenv("BBCURSOR", bb->files[bb->cursor]->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-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
2019-05-28 21:36:42 -07:00
|
|
|
int fputs_escaped(FILE *f, const char *str, const char *color)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-22 19:05:56 -07:00
|
|
|
static const char *escapes = " abtnvfr e";
|
2019-05-27 21:14:34 -07:00
|
|
|
int escaped = 0;
|
2019-05-25 20:55:59 -07:00
|
|
|
for (const char *c = str; *c; ++c) {
|
2019-05-27 21:14:34 -07:00
|
|
|
if (*c > 0 && *c <= '\x1b' && escapes[(int)*c] != ' ') { // "\n", etc.
|
2019-05-27 14:12:47 -07:00
|
|
|
fprintf(f, "\033[31m\\%c%s", escapes[(int)*c], color);
|
2019-05-27 21:14:34 -07:00
|
|
|
++escaped;
|
|
|
|
} else if (*c >= 0 && !(' ' <= *c && *c <= '~')) { // "\x02", etc.
|
2019-05-27 14:12:47 -07:00
|
|
|
fprintf(f, "\033[31m\\x%02X%s", *c, color);
|
2019-05-27 21:14:34 -07:00
|
|
|
++escaped;
|
|
|
|
} else {
|
2019-05-27 14:12:47 -07:00
|
|
|
fputc(*c, f);
|
2019-05-27 21:14:34 -07:00
|
|
|
}
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
2019-05-27 21:14:34 -07:00
|
|
|
return escaped;
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Returns the color of a file listing, given its mode.
|
|
|
|
*/
|
2019-05-28 22:04:02 -07:00
|
|
|
const char* color_of(mode_t mode)
|
2019-05-28 21:36:42 -07:00
|
|
|
{
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Draw everything to the screen.
|
|
|
|
* If lazy is true, then use terminal scrolling to move the file listing
|
|
|
|
* around and only update the files that have changed.
|
|
|
|
*/
|
2019-05-27 16:08:29 -07:00
|
|
|
void render(bb_t *bb, int lazy)
|
2019-05-22 19:05:56 -07:00
|
|
|
{
|
|
|
|
static int lastcursor = -1, lastscroll = -1;
|
2019-05-23 19:04:17 -07:00
|
|
|
char buf[64];
|
2019-05-23 01:37:41 -07:00
|
|
|
if (lastcursor == -1 || lastscroll == -1)
|
|
|
|
lazy = 0;
|
|
|
|
|
|
|
|
if (lazy) {
|
2019-05-23 19:04:17 -07:00
|
|
|
// Use terminal scrolling:
|
2019-05-27 16:08:29 -07:00
|
|
|
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);
|
2019-05-23 01:37:41 -07:00
|
|
|
}
|
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
|
|
|
|
// Canonicalize columns and column sizes
|
|
|
|
// TODO: make this lazy?
|
2019-05-27 20:55:36 -07:00
|
|
|
int cols = 0;
|
2019-05-27 19:51:04 -07:00
|
|
|
for (char *valid = &bb->options['0'], *p = &bb->options['0']; p <= &bb->options['9']; ++p) {
|
|
|
|
switch (*p) {
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'n': case 's': case 'm': case 'c': case 'a': case 'p': case '+':
|
2019-05-27 20:55:36 -07:00
|
|
|
++cols;
|
|
|
|
*(valid++) = *p;
|
2019-05-27 19:51:04 -07:00
|
|
|
}
|
|
|
|
if (p >= valid) *p = 0;
|
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
int tot = 0, space = termwidth - 1 - 3*cols;
|
2019-05-27 19:51:04 -07:00
|
|
|
for (int i = 0; i < cols; i++) {
|
|
|
|
switch (bb->options['0' + i]) {
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'a': case 'c': case 'm': bb->colwidths[i] = coldatew; break;
|
2019-05-27 19:51:04 -07:00
|
|
|
case 'n': bb->colwidths[i] = colnamew; break;
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'p': bb->colwidths[i] = colpermw; break;
|
|
|
|
case 's': bb->colwidths[i] = colsizew; break;
|
|
|
|
case '+': bb->colwidths[i] = colselw; break;
|
|
|
|
default: bb->colwidths[i] = 0; break;
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
int ratio = OPTNUM(bb->options['A' + i]);
|
|
|
|
if (ratio == 0) space -= bb->colwidths[i];
|
|
|
|
else tot += ratio;
|
2019-05-27 19:51:04 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
if (tot > 0) {
|
|
|
|
for (int i = 0; i < cols; i++) {
|
|
|
|
int ratio = OPTNUM(bb->options['A' + i]);
|
|
|
|
bb->colwidths[i] = MAX(bb->colwidths[i], (space * ratio) / tot);
|
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-23 04:44:48 -07:00
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
if (!lazy) {
|
|
|
|
// Path
|
2019-05-27 15:53:07 -07:00
|
|
|
move_cursor(tty_out, 0, 0);
|
2019-05-28 21:36:42 -07:00
|
|
|
const char *color = TITLE_COLOR;
|
|
|
|
fputs(color, tty_out);
|
2019-05-27 16:08:29 -07:00
|
|
|
fputs_escaped(tty_out, bb->path, color);
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(" \033[K\033[0m", tty_out);
|
2019-05-22 19:05:56 -07:00
|
|
|
|
2019-05-24 20:49:24 -07:00
|
|
|
// Columns
|
2019-05-27 19:51:04 -07:00
|
|
|
move_cursor(tty_out, 0, 1);
|
2019-05-29 00:56:49 -07:00
|
|
|
fputs("\033[0;44;30m\033[K", tty_out);
|
|
|
|
int x = 0;
|
2019-05-27 19:51:04 -07:00
|
|
|
for (int col = 0; col < cols; col++) {
|
|
|
|
move_cursor(tty_out, x, 1);
|
|
|
|
if (col > 0) {
|
2019-05-29 00:56:49 -07:00
|
|
|
fputs("│\033[K", tty_out);
|
|
|
|
x += 1;
|
2019-05-23 05:42:33 -07:00
|
|
|
}
|
2019-05-27 20:13:11 -07:00
|
|
|
const char *indicator = " ";
|
|
|
|
if (bb->options['s'] == bb->options['0' + col])
|
|
|
|
indicator = bb->options['r'] ? RSORT_INDICATOR : SORT_INDICATOR;
|
2019-05-27 19:51:04 -07:00
|
|
|
switch (bb->options['0' + col]) {
|
2019-05-29 00:56:49 -07:00
|
|
|
case '+': fprintf(tty_out, "%s+", indicator); break;
|
2019-05-27 20:13:11 -07:00
|
|
|
case 'n': fprintf(tty_out, "%sName", indicator); break;
|
2019-05-29 00:56:49 -07:00
|
|
|
case 's': move_cursor(tty_out, x + MAX(0, bb->colwidths[col] - (int)strlen(" Size")), 1);
|
2019-05-27 20:13:11 -07:00
|
|
|
fprintf(tty_out, "%sSize", indicator);
|
|
|
|
break;
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'p': move_cursor(tty_out, x + MAX(0, (bb->colwidths[col] - (int)strlen(" Permissions"))/2), 1);
|
|
|
|
fprintf(tty_out, "%sPermissions", indicator);
|
2019-05-27 20:13:11 -07:00
|
|
|
break;
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'm': move_cursor(tty_out, x + MAX(0, (bb->colwidths[col] - (int)strlen(" Modified"))/2), 1);
|
2019-05-27 20:13:11 -07:00
|
|
|
fprintf(tty_out, "%sModified", indicator);
|
|
|
|
break;
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'a': move_cursor(tty_out, x + MAX(0, (bb->colwidths[col] - (int)strlen(" Accessed"))/2), 1);
|
2019-05-27 20:13:11 -07:00
|
|
|
fprintf(tty_out, "%sAccessed", indicator);
|
|
|
|
break;
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'c': move_cursor(tty_out, x + MAX(0, (bb->colwidths[col] - (int)strlen(" Created"))/2), 1);
|
2019-05-27 20:13:11 -07:00
|
|
|
fprintf(tty_out, "%sCreated", indicator);
|
|
|
|
break;
|
2019-05-27 19:51:04 -07:00
|
|
|
default: continue;
|
|
|
|
}
|
|
|
|
x += bb->colwidths[col];
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(" \033[K\033[0m", tty_out);
|
2019-05-22 19:05:56 -07:00
|
|
|
}
|
|
|
|
|
2019-05-27 16:08:29 -07:00
|
|
|
entry_t **files = bb->files;
|
|
|
|
for (int i = bb->scroll; i < bb->scroll + termheight - 3; i++) {
|
2019-05-22 19:05:56 -07:00
|
|
|
if (lazy) {
|
2019-05-27 16:08:29 -07:00
|
|
|
if (i == bb->cursor || i == lastcursor)
|
2019-05-22 19:05:56 -07:00
|
|
|
goto do_render;
|
2019-05-25 04:30:51 -07:00
|
|
|
if (i < lastscroll || i >= lastscroll + termheight - 3)
|
2019-05-22 19:05:56 -07:00
|
|
|
goto do_render;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-05-26 13:41:09 -07:00
|
|
|
int y;
|
|
|
|
do_render:
|
2019-05-27 16:08:29 -07:00
|
|
|
y = i - bb->scroll + 2;
|
2019-05-27 15:53:07 -07:00
|
|
|
move_cursor(tty_out, 0, y);
|
2019-05-27 16:08:29 -07:00
|
|
|
if (i >= bb->nfiles) {
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("\033[K", tty_out);
|
2019-05-22 19:05:56 -07:00
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2019-05-21 02:17:11 -07:00
|
|
|
entry_t *entry = files[i];
|
2019-05-27 19:51:04 -07:00
|
|
|
|
2019-05-28 21:36:42 -07:00
|
|
|
fputs(IS_SELECTED(entry) ? SELECTED_INDICATOR : NOT_SELECTED_INDICATOR, tty_out);
|
|
|
|
|
2019-05-28 21:46:16 -07:00
|
|
|
char color[128];
|
|
|
|
strcpy(color, color_of(entry->info.st_mode));
|
|
|
|
if (i == bb->cursor) strcat(color, CURSOR_COLOR);
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(color, tty_out);
|
2019-05-22 19:05:56 -07:00
|
|
|
|
2019-05-29 00:56:49 -07:00
|
|
|
int x = 0;
|
2019-05-27 19:51:04 -07:00
|
|
|
for (int col = 0; col < cols; col++) {
|
2019-05-27 14:12:47 -07:00
|
|
|
fprintf(tty_out, "\033[%d;%dH\033[K", y+1, x+1);
|
2019-05-27 19:51:04 -07:00
|
|
|
if (col > 0) {
|
2019-05-29 00:56:49 -07:00
|
|
|
if (i == bb->cursor) fputs("│", tty_out);
|
|
|
|
else fputs("\033[37;2m│\033[22m", tty_out);
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(color, tty_out);
|
2019-05-29 00:56:49 -07:00
|
|
|
x += 1;
|
2019-05-26 16:01:56 -07:00
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
switch (bb->options['0' + col]) {
|
2019-05-29 00:56:49 -07:00
|
|
|
case '+':
|
|
|
|
fputs(IS_SELECTED(entry) ? SELECTED_INDICATOR : NOT_SELECTED_INDICATOR, tty_out);
|
|
|
|
fputs(color, tty_out);
|
|
|
|
break;
|
2019-05-23 19:04:17 -07:00
|
|
|
case 's': {
|
|
|
|
int j = 0;
|
|
|
|
const char* units = "BKMGTPEZY";
|
2019-05-25 20:55:59 -07:00
|
|
|
double bytes = (double)entry->info.st_size;
|
2019-05-23 19:04:17 -07:00
|
|
|
while (bytes > 1024) {
|
|
|
|
bytes /= 1024;
|
|
|
|
j++;
|
|
|
|
}
|
2019-05-27 20:13:11 -07:00
|
|
|
move_cursor(tty_out, x + bb->colwidths[col] - colsizew, y);
|
2019-05-27 14:12:47 -07:00
|
|
|
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-23 19:04:17 -07:00
|
|
|
case 'm':
|
2019-05-27 20:13:11 -07:00
|
|
|
move_cursor(tty_out, x + (bb->colwidths[col] - coldatew)/2, y);
|
2019-05-25 20:55:59 -07:00
|
|
|
strftime(buf, sizeof(buf), "%l:%M%p %b %e %Y", localtime(&(entry->info.st_mtime)));
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(buf, tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
break;
|
2019-05-21 03:59:30 -07:00
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
case 'c':
|
2019-05-27 20:13:11 -07:00
|
|
|
move_cursor(tty_out, x + (bb->colwidths[col] - coldatew)/2, y);
|
2019-05-25 20:55:59 -07:00
|
|
|
strftime(buf, sizeof(buf), "%l:%M%p %b %e %Y", localtime(&(entry->info.st_ctime)));
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(buf, tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
break;
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
case 'a':
|
2019-05-27 20:13:11 -07:00
|
|
|
move_cursor(tty_out, x + (bb->colwidths[col] - coldatew)/2, y);
|
2019-05-25 20:55:59 -07:00
|
|
|
strftime(buf, sizeof(buf), "%l:%M%p %b %e %Y", localtime(&(entry->info.st_atime)));
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(buf, tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'p':
|
2019-05-29 00:56:49 -07:00
|
|
|
move_cursor(tty_out, x + (bb->colwidths[col] - colpermw)/2, y);
|
|
|
|
fprintf(tty_out, " %c%c%c",
|
|
|
|
'0' + ((entry->info.st_mode >> 6) & 7),
|
|
|
|
'0' + ((entry->info.st_mode >> 3) & 7),
|
|
|
|
'0' + ((entry->info.st_mode >> 0) & 7));
|
2019-05-23 19:04:17 -07:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'n': {
|
2019-05-28 21:36:42 -07:00
|
|
|
if (entry->no_esc) fputs(entry->name, tty_out);
|
|
|
|
else entry->no_esc |= !fputs_escaped(tty_out, entry->name, color);
|
2019-05-27 19:51:04 -07:00
|
|
|
|
2019-05-28 21:36:42 -07:00
|
|
|
if (S_ISDIR(S_ISLNK(entry->info.st_mode) ? entry->linkedmode : entry->info.st_mode))
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("/", tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-05-28 00:07:43 -07:00
|
|
|
if (entry->linkname) {
|
2019-05-28 21:36:42 -07:00
|
|
|
if (i != bb->cursor)
|
|
|
|
fputs("\033[37m", tty_out);
|
|
|
|
fputs("\033[2m -> \033[22;3m", tty_out);
|
2019-05-28 21:46:16 -07:00
|
|
|
strcpy(color, color_of(entry->linkedmode));
|
|
|
|
if (i == bb->cursor) strcat(color, CURSOR_COLOR);
|
2019-05-28 21:36:42 -07:00
|
|
|
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))
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("/", tty_out);
|
2019-05-27 19:51:04 -07:00
|
|
|
|
2019-05-29 00:56:49 -07:00
|
|
|
fputs("\033[22;23m", tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
default: continue;
|
2019-05-21 16:32:26 -07:00
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
x += bb->colwidths[col];
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(" \033[K\033[0m", tty_out); // 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-27 15:53:07 -07:00
|
|
|
move_cursor(tty_out, 0, termheight - 1);
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("\033[K", tty_out);
|
2019-05-27 15:53:07 -07:00
|
|
|
move_cursor(tty_out, MAX(0, termwidth - (int)strlen(help)), termheight - 1);
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(help, tty_out);
|
2019-05-27 16:08:29 -07:00
|
|
|
lastcursor = bb->cursor;
|
|
|
|
lastscroll = bb->scroll;
|
2019-05-27 19:51:04 -07:00
|
|
|
int x = termwidth;
|
|
|
|
move_cursor(tty_out, --x, 0);
|
|
|
|
fputs("\033[0;2m]\033[1m", tty_out);
|
|
|
|
for (int o = 127, nopts = 0; o > 0; --o) {
|
2019-05-28 23:51:20 -07:00
|
|
|
if (bb->options[o] == bb->initialopts[o]) continue;
|
2019-05-27 19:51:04 -07:00
|
|
|
if (bb->options[o] <= 0) continue;
|
|
|
|
if (nopts > 0) {
|
|
|
|
x -= 1;
|
|
|
|
move_cursor(tty_out, x, 0);
|
|
|
|
fputs("\033[2m,\033[0m", tty_out);
|
|
|
|
}
|
|
|
|
if (bb->options[o] >= ' ') {
|
|
|
|
x -= 3;
|
|
|
|
move_cursor(tty_out, x, 0);
|
|
|
|
fprintf(tty_out, "\033[0;1m%c\033[0m=\033[33m%c\033[0m", (char)o, bb->options[o]);
|
|
|
|
} else if (bb->options[o] > 0) {
|
|
|
|
x -= 1;
|
|
|
|
move_cursor(tty_out, x, 0);
|
|
|
|
fprintf(tty_out, "\033[0;1m%c\033[0m", (char)o);
|
|
|
|
}
|
|
|
|
nopts++;
|
|
|
|
}
|
|
|
|
move_cursor(tty_out, --x, 0);
|
|
|
|
fputs("\033[0;2m[\033[0m", tty_out);
|
2019-05-27 14:12:47 -07:00
|
|
|
fflush(tty_out);
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Used for sorting, this function compares files according to the sorting-related options,
|
|
|
|
* like bb->options['s'], bb->options['i'], and bb->options['r']
|
|
|
|
*/
|
2019-05-27 19:51:04 -07:00
|
|
|
int compare_files(void *v, const void *v1, const void *v2)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-27 19:51:04 -07:00
|
|
|
bb_t *bb = (bb_t*)v;
|
|
|
|
char sort = bb->options['s'];
|
|
|
|
int sign = bb->options['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-28 23:51:20 -07:00
|
|
|
// *always* put ".." before everything else, then "."
|
2019-05-28 22:29:22 -07:00
|
|
|
int diff = (strcmp(f1->name, "..") == 0) - (strcmp(f2->name, "..") == 0);
|
|
|
|
if (diff) return -diff;
|
2019-05-28 23:51:20 -07:00
|
|
|
diff = (strcmp(f1->name, ".") == 0) - (strcmp(f2->name, ".") == 0);
|
|
|
|
if (diff) return -diff;
|
2019-05-29 00:56:49 -07:00
|
|
|
if (sort == '+') {
|
|
|
|
diff = -(IS_SELECTED(f1) - IS_SELECTED(f2)) * sign;
|
|
|
|
if (diff) return diff;
|
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
if (!bb->options['i']) {
|
2019-05-28 21:36:42 -07:00
|
|
|
// Unless interleave mode is on, sort dirs before files
|
|
|
|
int d1 = S_ISDIR(f1->info.st_mode) || (S_ISLNK(f1->info.st_mode) && S_ISDIR(f1->linkedmode));
|
|
|
|
int d2 = S_ISDIR(f2->info.st_mode) || (S_ISLNK(f2->info.st_mode) && S_ISDIR(f2->linkedmode));
|
|
|
|
if (d1 != d2) return d2 - d1;
|
2019-05-27 19:51:04 -07:00
|
|
|
}
|
2019-05-28 22:04:02 -07:00
|
|
|
switch (sort) {
|
2019-05-29 00:56:49 -07:00
|
|
|
case SORT_SELECTED:
|
|
|
|
goto sort_by_name;
|
2019-05-28 22:04:02 -07:00
|
|
|
case SORT_NAME: {
|
2019-05-29 00:56:49 -07:00
|
|
|
const char *n1, *n2;
|
|
|
|
sort_by_name:
|
|
|
|
n1 = f1->name;
|
|
|
|
n2 = f2->name;
|
|
|
|
while (*n1 && *n2) {
|
|
|
|
char c1 = *n1, c2 = *n2;
|
2019-05-28 22:04:02 -07:00
|
|
|
if ('A' <= c1 && 'Z' <= c1) c1 = c1 - 'A' + 'a';
|
|
|
|
if ('A' <= c2 && 'Z' <= c2) c2 = c2 - 'A' + 'a';
|
|
|
|
diff = (c1 - c2);
|
|
|
|
if ('0' <= c1 && c1 <= '9' && '0' <= c2 && c2 <= '9') {
|
2019-05-29 00:56:49 -07:00
|
|
|
long i1 = strtol(n1, (char**)&n1, 10);
|
|
|
|
long i2 = strtol(n2, (char**)&n2, 10);
|
|
|
|
diff = (int)(n1 - f1->name) - (int)(n2 - f2->name);
|
2019-05-28 22:04:02 -07:00
|
|
|
if (diff != 0)
|
|
|
|
return diff*sign;
|
2019-05-29 00:56:49 -07:00
|
|
|
if (i1 != i2)
|
|
|
|
return (i1 > i2 ? 1 : -1)*sign;
|
2019-05-28 22:04:02 -07:00
|
|
|
} else if (diff) {
|
2019-05-25 21:47:30 -07:00
|
|
|
return diff*sign;
|
2019-05-28 22:04:02 -07:00
|
|
|
} else {
|
2019-05-29 00:56:49 -07:00
|
|
|
++n1; ++n2;
|
2019-05-28 22:04:02 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
return (*n1 - *n2)*sign;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
case SORT_PERM:
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = -((f1->info.st_mode & 0x3FF) - (f2->info.st_mode & 0x3FF))*sign;
|
|
|
|
break;
|
2019-05-23 19:04:17 -07:00
|
|
|
case SORT_SIZE:
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = f1->info.st_size > f2->info.st_size ? -sign : sign;
|
|
|
|
break;
|
2019-05-23 19:04:17 -07:00
|
|
|
case SORT_MTIME:
|
2019-05-25 20:55:59 -07:00
|
|
|
if (f1->info.st_mtimespec.tv_sec == f2->info.st_mtimespec.tv_sec)
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = f1->info.st_mtimespec.tv_nsec > f2->info.st_mtimespec.tv_nsec ? -sign : sign;
|
2019-05-23 19:04:17 -07:00
|
|
|
else
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = f1->info.st_mtimespec.tv_sec > f2->info.st_mtimespec.tv_sec ? -sign : sign;
|
|
|
|
break;
|
2019-05-23 19:04:17 -07:00
|
|
|
case SORT_CTIME:
|
2019-05-25 20:55:59 -07:00
|
|
|
if (f1->info.st_ctimespec.tv_sec == f2->info.st_ctimespec.tv_sec)
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = f1->info.st_ctimespec.tv_nsec > f2->info.st_ctimespec.tv_nsec ? -sign : sign;
|
2019-05-23 19:04:17 -07:00
|
|
|
else
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = f1->info.st_ctimespec.tv_sec > f2->info.st_ctimespec.tv_sec ? -sign : sign;
|
|
|
|
break;
|
2019-05-23 19:04:17 -07:00
|
|
|
case SORT_ATIME:
|
2019-05-25 20:55:59 -07:00
|
|
|
if (f1->info.st_atimespec.tv_sec == f2->info.st_atimespec.tv_sec)
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = f1->info.st_atimespec.tv_nsec > f2->info.st_atimespec.tv_nsec ? -sign : sign;
|
2019-05-23 19:04:17 -07:00
|
|
|
else
|
2019-05-29 00:56:49 -07:00
|
|
|
diff = f1->info.st_atimespec.tv_sec > f2->info.st_atimespec.tv_sec ? -sign : sign;
|
|
|
|
break;
|
2019-05-28 22:04:02 -07:00
|
|
|
case SORT_RANDOM:
|
|
|
|
return f1->shufflepos - f2->shufflepos;
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
if (diff == 0) goto sort_by_name;
|
|
|
|
return diff;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-27 16:08:29 -07:00
|
|
|
int find_file(bb_t *bb, const char *name)
|
2019-05-23 00:57:25 -07:00
|
|
|
{
|
2019-05-27 16:08:29 -07:00
|
|
|
for (int i = 0; i < bb->nfiles; i++) {
|
|
|
|
entry_t *e = bb->files[i];
|
2019-05-28 00:07:43 -07:00
|
|
|
if (strcmp(name[0] == '/' ? e->fullname : e->name, name) == 0)
|
2019-05-25 04:30:51 -07:00
|
|
|
return i;
|
2019-05-23 00:57:25 -07:00
|
|
|
}
|
2019-05-25 04:30:51 -07:00
|
|
|
return -1;
|
2019-05-23 00:57:25 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Deselect all files
|
|
|
|
*/
|
2019-05-27 16:08:29 -07:00
|
|
|
void clear_selection(bb_t *bb)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-27 16:08:29 -07:00
|
|
|
for (entry_t *next, *e = bb->firstselected; e; e = next) {
|
2019-05-25 04:30:51 -07:00
|
|
|
next = e->next;
|
2019-05-28 21:36:42 -07:00
|
|
|
*(e->atme) = NULL;
|
|
|
|
e->atme = NULL;
|
2019-05-27 20:55:36 -07:00
|
|
|
if (--e->refcount <= 0) free(e);
|
2019-05-21 20:06:53 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Select a file
|
|
|
|
*/
|
|
|
|
void select_file(bb_t *bb, entry_t *e)
|
2019-05-23 00:57:25 -07:00
|
|
|
{
|
2019-05-28 22:29:22 -07:00
|
|
|
if (IS_SELECTED(e)) return;
|
|
|
|
if (strcmp(e->name, "..") == 0) return;
|
2019-05-28 23:51:20 -07:00
|
|
|
if (strcmp(e->name, ".") == 0) return;
|
2019-05-27 16:08:29 -07:00
|
|
|
if (bb->firstselected)
|
|
|
|
bb->firstselected->atme = &e->next;
|
|
|
|
e->next = bb->firstselected;
|
|
|
|
e->atme = &bb->firstselected;
|
2019-05-27 20:55:36 -07:00
|
|
|
++e->refcount;
|
2019-05-27 16:08:29 -07:00
|
|
|
bb->firstselected = e;
|
2019-05-23 00:57:25 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
2019-05-28 23:51:20 -07:00
|
|
|
* Deselect a file
|
2019-05-28 22:29:22 -07:00
|
|
|
*/
|
|
|
|
void deselect_file(bb_t *bb, entry_t *e)
|
2019-05-23 00:57:25 -07:00
|
|
|
{
|
2019-05-27 16:08:29 -07:00
|
|
|
(void)bb;
|
2019-05-28 22:29:22 -07:00
|
|
|
if (!IS_SELECTED(e)) return;
|
2019-05-27 16:08:29 -07:00
|
|
|
if (e->next)
|
|
|
|
e->next->atme = e->atme;
|
|
|
|
*(e->atme) = e->next;
|
2019-05-27 20:55:36 -07:00
|
|
|
--e->refcount;
|
2019-05-27 16:08:29 -07:00
|
|
|
e->next = NULL;
|
|
|
|
e->atme = NULL;
|
2019-05-23 00:57:25 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 23:51:20 -07:00
|
|
|
/*
|
|
|
|
* Toggle a file's selection state
|
|
|
|
*/
|
|
|
|
void toggle_file(bb_t *bb, entry_t *e)
|
|
|
|
{
|
|
|
|
if (IS_SELECTED(e)) deselect_file(bb, e);
|
|
|
|
else select_file(bb, e);
|
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Set bb's file cursor to the given index (and adjust the scroll as necessary)
|
|
|
|
*/
|
2019-05-27 16:08:29 -07:00
|
|
|
void set_cursor(bb_t *bb, int newcur)
|
2019-05-20 19:28:47 -07:00
|
|
|
{
|
2019-05-27 16:08:29 -07:00
|
|
|
if (newcur > bb->nfiles - 1) newcur = bb->nfiles - 1;
|
2019-05-25 04:30:51 -07:00
|
|
|
if (newcur < 0) newcur = 0;
|
2019-05-27 16:08:29 -07:00
|
|
|
bb->cursor = newcur;
|
|
|
|
if (bb->nfiles <= termheight - 4)
|
2019-05-25 04:30:51 -07:00
|
|
|
return;
|
|
|
|
|
2019-05-27 16:08:29 -07:00
|
|
|
if (newcur < bb->scroll + SCROLLOFF)
|
|
|
|
bb->scroll = newcur - SCROLLOFF;
|
|
|
|
else if (newcur > bb->scroll + (termheight-4) - SCROLLOFF)
|
|
|
|
bb->scroll = newcur - (termheight-4) + SCROLLOFF;
|
|
|
|
int max_scroll = bb->nfiles - (termheight-4) - 1;
|
2019-05-25 04:30:51 -07:00
|
|
|
if (max_scroll < 0) max_scroll = 0;
|
2019-05-27 16:08:29 -07:00
|
|
|
if (bb->scroll > max_scroll) bb->scroll = max_scroll;
|
|
|
|
if (bb->scroll < 0) bb->scroll = 0;
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Set bb's scroll to the given index (and adjust the cursor as necessary)
|
|
|
|
*/
|
2019-05-27 16:08:29 -07:00
|
|
|
void set_scroll(bb_t *bb, int newscroll)
|
2019-05-25 04:30:51 -07:00
|
|
|
{
|
2019-05-27 16:08:29 -07:00
|
|
|
newscroll = MIN(newscroll, bb->nfiles - (termheight-4) - 1);
|
2019-05-25 04:30:51 -07:00
|
|
|
newscroll = MAX(newscroll, 0);
|
2019-05-27 16:08:29 -07:00
|
|
|
bb->scroll = newscroll;
|
2019-05-25 04:30:51 -07:00
|
|
|
|
2019-05-27 16:08:29 -07:00
|
|
|
int delta = newscroll - bb->scroll;
|
|
|
|
int oldcur = bb->cursor;
|
|
|
|
if (bb->nfiles < termheight - 4) {
|
2019-05-25 04:30:51 -07:00
|
|
|
newscroll = 0;
|
|
|
|
} else {
|
2019-05-27 16:08:29 -07:00
|
|
|
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;
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
2019-05-27 16:08:29 -07:00
|
|
|
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;
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* 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.
|
|
|
|
*/
|
2019-05-28 22:04:02 -07:00
|
|
|
entry_t* load_entry(const char *path)
|
2019-05-28 21:36:42 -07:00
|
|
|
{
|
|
|
|
ssize_t linkpathlen = -1;
|
|
|
|
char linkbuf[PATH_MAX];
|
|
|
|
struct stat linkedstat, filestat;
|
|
|
|
if (lstat(path, &filestat) == -1) return NULL;
|
|
|
|
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 entry_size = sizeof(entry_t) + (strlen(path) + 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;
|
|
|
|
return entry;
|
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* 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`
|
|
|
|
*/
|
2019-05-27 16:08:29 -07:00
|
|
|
void populate_files(bb_t *bb, const char *path)
|
2019-05-25 04:30:51 -07:00
|
|
|
{
|
|
|
|
ino_t old_inode = 0;
|
2019-05-23 19:04:17 -07:00
|
|
|
// Clear old files (if any)
|
2019-05-27 16:08:29 -07:00
|
|
|
if (bb->files) {
|
2019-05-27 21:14:34 -07:00
|
|
|
old_inode = bb->files[bb->cursor]->info.st_ino;
|
2019-05-27 16:08:29 -07:00
|
|
|
for (int i = 0; i < bb->nfiles; i++) {
|
2019-05-27 20:55:36 -07:00
|
|
|
if (--bb->files[i]->refcount <= 0)
|
|
|
|
free(bb->files[i]);
|
2019-05-21 02:17:11 -07:00
|
|
|
}
|
2019-05-27 16:08:29 -07:00
|
|
|
free(bb->files);
|
|
|
|
bb->files = NULL;
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
2019-05-27 20:55:36 -07:00
|
|
|
int old_cursor = bb->cursor, old_scroll = bb->scroll;
|
2019-05-27 16:08:29 -07:00
|
|
|
bb->nfiles = 0;
|
|
|
|
bb->cursor = 0;
|
2019-05-25 20:55:59 -07:00
|
|
|
|
|
|
|
if (path == NULL)
|
|
|
|
return;
|
|
|
|
|
2019-05-27 16:08:29 -07:00
|
|
|
int samedir = strcmp(path, bb->path) == 0;
|
2019-05-25 21:47:30 -07:00
|
|
|
if (!samedir)
|
2019-05-27 16:08:29 -07:00
|
|
|
strcpy(bb->path, path);
|
2019-05-23 19:04:17 -07:00
|
|
|
|
|
|
|
// Hash inode -> entry_t with linear probing
|
2019-05-25 04:30:51 -07:00
|
|
|
int nselected = 0;
|
2019-05-27 16:08:29 -07:00
|
|
|
for (entry_t *p = bb->firstselected; p; p = p->next) ++nselected;
|
2019-05-25 04:30:51 -07:00
|
|
|
int hashsize = 2 * nselected;
|
|
|
|
entry_t **selecthash = NULL;
|
|
|
|
if (nselected > 0) {
|
|
|
|
selecthash = memcheck(calloc((size_t)hashsize, sizeof(entry_t*)));
|
2019-05-27 16:08:29 -07:00
|
|
|
for (entry_t *p = bb->firstselected; p; p = p->next) {
|
2019-05-27 21:14:34 -07:00
|
|
|
int probe = ((int)p->info.st_ino) % hashsize;
|
2019-05-25 04:30:51 -07:00
|
|
|
while (selecthash[probe])
|
|
|
|
probe = (probe + 1) % hashsize;
|
|
|
|
selecthash[probe] = p;
|
|
|
|
}
|
2019-05-21 02:17:11 -07:00
|
|
|
}
|
2019-05-21 16:02:25 -07:00
|
|
|
|
2019-05-27 16:08:29 -07:00
|
|
|
DIR *dir = opendir(bb->path);
|
2019-05-23 19:04:17 -07:00
|
|
|
if (!dir)
|
2019-05-27 16:08:29 -07:00
|
|
|
err("Couldn't open dir: %s", bb->path);
|
|
|
|
size_t pathlen = strlen(bb->path);
|
2019-05-23 19:04:17 -07:00
|
|
|
size_t filecap = 0;
|
2019-05-28 21:36:42 -07:00
|
|
|
char pathbuf[PATH_MAX];
|
|
|
|
strcpy(pathbuf, path);
|
|
|
|
pathbuf[pathlen] = '/';
|
2019-05-25 04:30:51 -07:00
|
|
|
for (struct dirent *dp; (dp = readdir(dir)) != NULL; ) {
|
2019-05-28 23:51:20 -07:00
|
|
|
/*
|
|
|
|
* Bit 1: 0 = hide ".." 1 = show ".."
|
|
|
|
* Bit 1: 0 = hide "." 1 = show "."
|
|
|
|
* Bit 2: 0 = hide .anything, 1 = show .anything
|
|
|
|
*/
|
|
|
|
if (dp->d_name[0] == '.') {
|
|
|
|
int o = OPTNUM(bb->options['.']);
|
|
|
|
if (dp->d_name[1] == '.' && dp->d_name[2] == '\0') {
|
|
|
|
if (!(o & 1)) continue;
|
|
|
|
} else if (dp->d_name[1] == '\0') {
|
|
|
|
if (!(o & 2)) continue;
|
|
|
|
} else if (!(o & 4)) continue;
|
|
|
|
}
|
2019-05-27 16:08:29 -07:00
|
|
|
if ((size_t)bb->nfiles >= filecap) {
|
2019-05-25 04:30:51 -07:00
|
|
|
filecap += 100;
|
2019-05-27 16:08:29 -07:00
|
|
|
bb->files = memcheck(realloc(bb->files, filecap*sizeof(entry_t*)));
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
// Hashed lookup from selected:
|
2019-05-25 04:30:51 -07:00
|
|
|
if (nselected > 0) {
|
2019-05-23 19:04:17 -07:00
|
|
|
for (int probe = ((int)dp->d_ino) % hashsize; selecthash[probe]; probe = (probe + 1) % hashsize) {
|
2019-05-27 21:14:34 -07:00
|
|
|
if (selecthash[probe]->info.st_ino == dp->d_ino) {
|
2019-05-27 20:55:36 -07:00
|
|
|
++selecthash[probe]->refcount;
|
2019-05-27 16:08:29 -07:00
|
|
|
bb->files[bb->nfiles++] = selecthash[probe];
|
2019-05-23 19:04:17 -07:00
|
|
|
goto next_file;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-25 04:30:51 -07:00
|
|
|
|
2019-05-28 21:36:42 -07:00
|
|
|
strcpy(&pathbuf[pathlen+1], dp->d_name);
|
|
|
|
entry_t *entry = load_entry(pathbuf);
|
|
|
|
if (!entry) err("Failed to load entry: '%s'", pathbuf);
|
2019-05-27 20:55:36 -07:00
|
|
|
++entry->refcount;
|
2019-05-27 16:08:29 -07:00
|
|
|
bb->files[bb->nfiles++] = entry;
|
2019-05-26 13:41:09 -07:00
|
|
|
next_file:
|
|
|
|
continue;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
closedir(dir);
|
|
|
|
free(selecthash);
|
2019-05-27 16:08:29 -07:00
|
|
|
if (bb->nfiles == 0) err("No files found (not even '..')");
|
2019-05-28 22:04:02 -07:00
|
|
|
// 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();
|
2019-05-25 04:30:51 -07:00
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
qsort_r(bb->files, (size_t)bb->nfiles, sizeof(entry_t*), bb, compare_files);
|
2019-05-25 21:47:30 -07:00
|
|
|
if (samedir) {
|
|
|
|
if (old_inode) {
|
2019-05-27 16:08:29 -07:00
|
|
|
for (int i = 0; i < bb->nfiles; i++) {
|
2019-05-27 21:14:34 -07:00
|
|
|
if (bb->files[i]->info.st_ino == old_inode) {
|
2019-05-27 16:08:29 -07:00
|
|
|
set_scroll(bb, old_scroll);
|
|
|
|
set_cursor(bb, i);
|
2019-05-25 21:47:30 -07:00
|
|
|
return;
|
|
|
|
}
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
|
|
|
}
|
2019-05-27 16:08:29 -07:00
|
|
|
set_cursor(bb, old_cursor);
|
|
|
|
set_scroll(bb, old_scroll);
|
2019-05-25 21:47:30 -07:00
|
|
|
} else {
|
2019-05-27 16:08:29 -07:00
|
|
|
set_cursor(bb, 0);
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Run a bb internal command (e.g. "+refresh") and return an indicator of what
|
|
|
|
* needs to happen next.
|
|
|
|
*/
|
2019-05-28 21:36:42 -07:00
|
|
|
bb_result_t execute_cmd(bb_t *bb, const char *cmd)
|
2019-05-25 20:55:59 -07:00
|
|
|
{
|
|
|
|
char *value = strchr(cmd, ':');
|
|
|
|
if (value) ++value;
|
|
|
|
switch (cmd[0]) {
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'r': { // +refresh
|
2019-05-27 16:08:29 -07:00
|
|
|
populate_files(bb, bb->path);
|
2019-05-25 20:55:59 -07:00
|
|
|
return BB_REFRESH;
|
2019-05-25 21:47:30 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'q': // +quit
|
2019-05-25 20:55:59 -07:00
|
|
|
return BB_QUIT;
|
2019-05-29 00:56:49 -07:00
|
|
|
case 's': // +select:, +scroll:, +spread:
|
2019-05-25 20:55:59 -07:00
|
|
|
switch (cmd[1]) {
|
|
|
|
case 'c': { // scroll:
|
2019-05-26 16:01:56 -07:00
|
|
|
if (!value) return BB_INVALID;
|
2019-05-25 20:55:59 -07:00
|
|
|
// TODO: figure out the best version of this
|
|
|
|
int isdelta = value[0] == '+' || value[0] == '-';
|
|
|
|
int n = (int)strtol(value, &value, 10);
|
|
|
|
if (*value == '%')
|
2019-05-27 16:08:29 -07:00
|
|
|
n = (n * (value[1] == 'n' ? bb->nfiles : termheight)) / 100;
|
2019-05-25 20:55:59 -07:00
|
|
|
if (isdelta)
|
2019-05-27 16:08:29 -07:00
|
|
|
set_scroll(bb, bb->scroll + n);
|
2019-05-25 20:55:59 -07:00
|
|
|
else
|
2019-05-27 16:08:29 -07:00
|
|
|
set_scroll(bb, n);
|
2019-05-25 20:55:59 -07:00
|
|
|
return BB_NOP;
|
|
|
|
}
|
|
|
|
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'p': // +spread:
|
2019-05-25 20:55:59 -07:00
|
|
|
goto move;
|
|
|
|
|
2019-05-29 00:56:49 -07:00
|
|
|
case '\0': case 'e': // +select:
|
2019-05-28 00:07:43 -07:00
|
|
|
if (!value) value = bb->files[bb->cursor]->name;
|
2019-05-25 20:55:59 -07:00
|
|
|
if (strcmp(value, "*") == 0) {
|
2019-05-27 16:08:29 -07:00
|
|
|
for (int i = 0; i < bb->nfiles; i++)
|
|
|
|
select_file(bb, bb->files[i]);
|
2019-05-25 20:55:59 -07:00
|
|
|
} else {
|
2019-05-27 16:08:29 -07:00
|
|
|
int f = find_file(bb, value);
|
|
|
|
if (f >= 0) select_file(bb, bb->files[f]);
|
2019-05-27 20:55:36 -07:00
|
|
|
// TODO: support selecting files in other directories
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-28 21:36:42 -07:00
|
|
|
return BB_REFRESH;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'c': { // +cd:
|
2019-05-27 19:51:04 -07:00
|
|
|
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);
|
|
|
|
return BB_NOP;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
if (chdir(rpbuf)) {
|
|
|
|
free(rpbuf);
|
|
|
|
return BB_INVALID;
|
|
|
|
}
|
|
|
|
char *oldpath = memcheck(strdup(bb->path));
|
|
|
|
populate_files(bb, rpbuf);
|
|
|
|
free(rpbuf);
|
|
|
|
if (strcmp(value, "..") == 0) {
|
|
|
|
int f = find_file(bb, oldpath);
|
|
|
|
if (f >= 0) set_cursor(bb, f);
|
|
|
|
}
|
|
|
|
free(oldpath);
|
|
|
|
return BB_REFRESH;
|
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
case 't': { // +toggle:
|
2019-05-28 00:07:43 -07:00
|
|
|
if (!value) value = bb->files[bb->cursor]->name;
|
2019-05-27 16:08:29 -07:00
|
|
|
int f = find_file(bb, value);
|
2019-05-25 20:55:59 -07:00
|
|
|
if (f < 0) return BB_INVALID;
|
2019-05-28 23:51:20 -07:00
|
|
|
toggle_file(bb, bb->files[f]);
|
2019-05-28 21:36:42 -07:00
|
|
|
return f == bb->cursor ? BB_NOP : BB_REFRESH;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'o': { // +options:
|
2019-05-27 19:51:04 -07:00
|
|
|
if (!value) return BB_INVALID;
|
2019-05-28 23:51:20 -07:00
|
|
|
|
|
|
|
for (char *key = value; *key; ) {
|
|
|
|
char *op = key;
|
|
|
|
while (*op != '\0' && *op != '^' && *op != '=' && *op != '%')
|
|
|
|
++op;
|
|
|
|
char *nextkey = op;
|
|
|
|
for ( ; key < op; key++)
|
|
|
|
switch (*op) {
|
|
|
|
case '\0': case ' ': case ',':
|
|
|
|
bb->options[(int)*key] = bb->initialopts[(int)*key];
|
|
|
|
break;
|
|
|
|
case '%': {
|
|
|
|
int n = (int)strtol(op+1, &nextkey, 10);
|
|
|
|
if (n < 0)
|
|
|
|
bb->options[(int)*key] = (char)((OPTNUM(bb->options[(int)*key]) - n - 1) % (-n));
|
|
|
|
else
|
|
|
|
bb->options[(int)*key] = (char)((OPTNUM(bb->options[(int)*key]) + 1) % n);
|
|
|
|
bb->options[(int)*key] += '0';
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
case '=':
|
|
|
|
bb->options[(int)*key] = op[1] == '0' ? 0 : op[1];
|
|
|
|
nextkey = op + 2;
|
|
|
|
break;
|
|
|
|
default: bb->options[(int)*key] = '1'; break;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-28 23:51:20 -07:00
|
|
|
key = nextkey;
|
|
|
|
while (*key == ' ' || *key == ',') ++key;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-27 19:51:04 -07:00
|
|
|
populate_files(bb, bb->path);
|
|
|
|
return BB_REFRESH;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'd': // +deselect:
|
2019-05-28 00:07:43 -07:00
|
|
|
if (!value) value = bb->files[bb->cursor]->name;
|
2019-05-27 19:51:04 -07:00
|
|
|
if (strcmp(value, "*") == 0) {
|
|
|
|
clear_selection(bb);
|
2019-05-28 21:36:42 -07:00
|
|
|
return BB_REFRESH;
|
2019-05-27 19:51:04 -07:00
|
|
|
} else {
|
|
|
|
int f = find_file(bb, value);
|
2019-05-28 23:51:20 -07:00
|
|
|
if (f < 0) return BB_INVALID;
|
|
|
|
select_file(bb, bb->files[f]);
|
2019-05-28 21:36:42 -07:00
|
|
|
return f == bb->cursor ? BB_NOP : BB_REFRESH;
|
2019-05-27 19:51:04 -07:00
|
|
|
}
|
|
|
|
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'g': { // +goto:
|
2019-05-25 20:55:59 -07:00
|
|
|
if (!value) return BB_INVALID;
|
2019-05-27 16:08:29 -07:00
|
|
|
int f = find_file(bb, value);
|
2019-05-25 20:55:59 -07:00
|
|
|
if (f >= 0) {
|
2019-05-27 16:08:29 -07:00
|
|
|
set_cursor(bb, f);
|
2019-05-25 20:55:59 -07:00
|
|
|
return BB_NOP;
|
|
|
|
}
|
|
|
|
char *path = memcheck(strdup(value));
|
|
|
|
char *lastslash = strrchr(path, '/');
|
|
|
|
if (!lastslash) return BB_INVALID;
|
|
|
|
*lastslash = '\0'; // Split in two
|
|
|
|
char *real = realpath(path, NULL);
|
2019-05-28 23:51:20 -07:00
|
|
|
if (!real || chdir(real)) return BB_INVALID;
|
2019-05-27 16:08:29 -07:00
|
|
|
populate_files(bb, real);
|
2019-05-25 20:55:59 -07:00
|
|
|
free(real); // estate
|
|
|
|
if (lastslash[1]) {
|
2019-05-27 16:08:29 -07:00
|
|
|
f = find_file(bb, lastslash + 1);
|
|
|
|
if (f >= 0) set_cursor(bb, f);
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
|
|
|
return BB_REFRESH;
|
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'm': { // +move:, +mark:
|
2019-05-25 20:55:59 -07:00
|
|
|
switch (cmd[1]) {
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'a': { // +mark:
|
2019-05-25 20:55:59 -07:00
|
|
|
if (!value) return BB_INVALID;
|
|
|
|
char key = value[0];
|
|
|
|
if (key < 0) return BB_INVALID;
|
2019-05-26 02:37:41 -07:00
|
|
|
value = strchr(value, '=');
|
2019-05-27 16:08:29 -07:00
|
|
|
if (!value) value = bb->path;
|
2019-05-25 20:55:59 -07:00
|
|
|
else ++value;
|
2019-05-27 16:08:29 -07:00
|
|
|
if (bb->marks[(int)key])
|
|
|
|
free(bb->marks[(int)key]);
|
|
|
|
bb->marks[(int)key] = memcheck(strdup(value));
|
2019-05-25 20:55:59 -07:00
|
|
|
return BB_NOP;
|
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
default: { // +move:
|
2019-05-26 13:41:09 -07:00
|
|
|
int oldcur, isdelta, n;
|
|
|
|
move:
|
2019-05-26 16:01:56 -07:00
|
|
|
if (!value) return BB_INVALID;
|
2019-05-27 16:08:29 -07:00
|
|
|
oldcur = bb->cursor;
|
2019-05-26 13:41:09 -07:00
|
|
|
isdelta = value[0] == '-' || value[0] == '+';
|
|
|
|
n = (int)strtol(value, &value, 10);
|
2019-05-25 20:55:59 -07:00
|
|
|
if (*value == '%')
|
2019-05-27 16:08:29 -07:00
|
|
|
n = (n * (value[1] == 'n' ? bb->nfiles : termheight)) / 100;
|
|
|
|
if (isdelta) set_cursor(bb, bb->cursor + n);
|
|
|
|
else set_cursor(bb, n);
|
2019-05-29 00:56:49 -07:00
|
|
|
if (cmd[0] == 's') { // +spread:
|
2019-05-27 16:08:29 -07:00
|
|
|
int sel = IS_SELECTED(bb->files[oldcur]);
|
|
|
|
for (int i = bb->cursor; i != oldcur; i += (oldcur > i ? 1 : -1)) {
|
2019-05-28 23:51:20 -07:00
|
|
|
if (sel != IS_SELECTED(bb->files[i]))
|
|
|
|
toggle_file(bb, bb->files[i]);
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
2019-05-27 16:08:29 -07:00
|
|
|
if (abs(oldcur - bb->cursor) > 1)
|
2019-05-28 21:36:42 -07:00
|
|
|
return BB_REFRESH;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
|
|
|
return BB_NOP;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
case 'j': { // +jump:
|
2019-05-25 20:55:59 -07:00
|
|
|
if (!value) return BB_INVALID;
|
|
|
|
char key = value[0];
|
2019-05-27 16:08:29 -07:00
|
|
|
if (bb->marks[(int)key]) {
|
|
|
|
value = bb->marks[(int)key];
|
2019-05-25 20:55:59 -07:00
|
|
|
goto cd;
|
|
|
|
}
|
|
|
|
return BB_INVALID;
|
|
|
|
}
|
2019-05-28 21:36:42 -07:00
|
|
|
default: err("UNKNOWN COMMAND: '%s'", cmd); break;
|
2019-05-25 20:55:59 -07:00
|
|
|
}
|
|
|
|
return BB_INVALID;
|
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Use bb to browse a path.
|
|
|
|
*/
|
|
|
|
void bb_browse(bb_t *bb, const char *path)
|
2019-05-25 04:30:51 -07:00
|
|
|
{
|
|
|
|
static long cmdpos = 0;
|
2019-05-25 20:55:59 -07:00
|
|
|
int lastwidth = termwidth, lastheight = termheight;
|
|
|
|
int lazy = 0, check_cmds = 1;
|
2019-05-25 04:30:51 -07:00
|
|
|
|
|
|
|
init_term();
|
2019-05-27 15:53:07 -07:00
|
|
|
fputs(T_ON(T_ALT_SCREEN), tty_out);
|
2019-05-25 04:30:51 -07:00
|
|
|
|
|
|
|
{
|
|
|
|
char *real = realpath(path, NULL);
|
|
|
|
if (!real || chdir(real))
|
|
|
|
err("Not a valid path: %s\n", path);
|
2019-05-27 16:08:29 -07:00
|
|
|
populate_files(bb, real);
|
2019-05-25 04:30:51 -07:00
|
|
|
free(real); // estate
|
|
|
|
}
|
|
|
|
|
2019-05-25 20:55:59 -07:00
|
|
|
for (int i = 0; startupcmds[i]; i++) {
|
|
|
|
if (startupcmds[i][0] == '+') {
|
2019-05-27 16:08:29 -07:00
|
|
|
if (execute_cmd(bb, startupcmds[i] + 1) == BB_QUIT)
|
2019-05-25 20:55:59 -07:00
|
|
|
goto quit;
|
|
|
|
} else {
|
2019-05-27 16:08:29 -07:00
|
|
|
run_cmd_on_selection(bb, startupcmds[i]);
|
2019-05-25 20:55:59 -07:00
|
|
|
check_cmds = 1;
|
|
|
|
}
|
|
|
|
}
|
2019-05-28 23:51:20 -07:00
|
|
|
memcpy(bb->initialopts, bb->options, sizeof(bb->initialopts));
|
2019-05-20 19:28:47 -07:00
|
|
|
|
2019-05-26 13:41:09 -07:00
|
|
|
refresh:
|
2019-05-25 20:55:59 -07:00
|
|
|
lazy = 0;
|
2019-05-23 19:04:17 -07:00
|
|
|
|
|
|
|
redraw:
|
2019-05-27 16:08:29 -07:00
|
|
|
render(bb, lazy);
|
2019-05-23 19:04:17 -07:00
|
|
|
lazy = 1;
|
|
|
|
|
|
|
|
next_input:
|
2019-05-25 04:30:51 -07:00
|
|
|
if (termwidth != lastwidth || termheight != lastheight) {
|
|
|
|
lastwidth = termwidth; lastheight = termheight;
|
2019-05-23 19:04:17 -07:00
|
|
|
lazy = 0;
|
|
|
|
goto redraw;
|
|
|
|
}
|
|
|
|
|
2019-05-25 20:55:59 -07:00
|
|
|
if (check_cmds) {
|
2019-05-24 16:55:36 -07:00
|
|
|
FILE *cmdfile = fopen(cmdfilename, "r");
|
2019-05-24 20:49:24 -07:00
|
|
|
if (!cmdfile) {
|
|
|
|
if (!lazy) goto redraw;
|
2019-05-25 20:55:59 -07:00
|
|
|
goto get_keyboard_input;
|
2019-05-24 20:49:24 -07:00
|
|
|
}
|
2019-05-24 16:55:36 -07:00
|
|
|
|
|
|
|
if (cmdpos)
|
|
|
|
fseek(cmdfile, cmdpos, SEEK_SET);
|
2019-05-26 13:41:09 -07:00
|
|
|
|
2019-05-24 16:55:36 -07:00
|
|
|
char *cmd = NULL;
|
|
|
|
size_t cap = 0;
|
2019-05-25 20:55:59 -07:00
|
|
|
while (cmdfile && getdelim(&cmd, &cap, '\0', cmdfile) >= 0) {
|
2019-05-24 16:55:36 -07:00
|
|
|
cmdpos = ftell(cmdfile);
|
|
|
|
if (!cmd[0]) continue;
|
2019-05-27 16:08:29 -07:00
|
|
|
switch (execute_cmd(bb, cmd)) {
|
2019-05-26 13:41:09 -07:00
|
|
|
case BB_INVALID:
|
|
|
|
break;
|
|
|
|
case BB_NOP:
|
2019-05-28 21:36:42 -07:00
|
|
|
free(cmd);
|
|
|
|
fclose(cmdfile);
|
2019-05-26 13:41:09 -07:00
|
|
|
goto redraw;
|
2019-05-25 20:55:59 -07:00
|
|
|
case BB_REFRESH:
|
|
|
|
free(cmd);
|
|
|
|
fclose(cmdfile);
|
2019-05-23 19:04:17 -07:00
|
|
|
goto refresh;
|
2019-05-25 20:55:59 -07:00
|
|
|
case BB_QUIT:
|
|
|
|
free(cmd);
|
|
|
|
fclose(cmdfile);
|
2019-05-24 20:49:24 -07:00
|
|
|
goto quit;
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
|
|
|
}
|
2019-05-25 20:55:59 -07:00
|
|
|
free(cmd);
|
|
|
|
fclose(cmdfile);
|
2019-05-24 16:55:36 -07:00
|
|
|
unlink(cmdfilename);
|
|
|
|
cmdpos = 0;
|
2019-05-25 20:55:59 -07:00
|
|
|
check_cmds = 0;
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-22 00:25:25 -07:00
|
|
|
|
2019-05-24 16:55:36 -07:00
|
|
|
int key;
|
2019-05-25 20:55:59 -07:00
|
|
|
get_keyboard_input:
|
2019-05-28 21:36:42 -07:00
|
|
|
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);
|
2019-05-25 04:30:51 -07:00
|
|
|
double dt_ms = 1e3*(double)(clicktime.tv_sec - lastclick.tv_sec);
|
|
|
|
dt_ms += 1e-6*(double)(clicktime.tv_nsec - lastclick.tv_nsec);
|
|
|
|
lastclick = clicktime;
|
2019-05-29 00:56:49 -07:00
|
|
|
// Get column:
|
|
|
|
char column = '?';
|
|
|
|
for (int col = 0, x = 0; col < 10; col++) {
|
|
|
|
if (col > 0) x += 1;
|
|
|
|
x += bb->colwidths[col];
|
|
|
|
if (x >= mouse_x) {
|
|
|
|
column = bb->options['0' + col];
|
|
|
|
break;
|
2019-05-23 04:44:48 -07:00
|
|
|
}
|
2019-05-29 00:56:49 -07:00
|
|
|
}
|
|
|
|
if (mouse_y == 1) {
|
|
|
|
if (bb->options['s'] == column)
|
|
|
|
bb->options['r'] = !bb->options['r'];
|
|
|
|
else bb->options['r'] = 0;
|
|
|
|
bb->options['s'] = column;
|
|
|
|
qsort_r(bb->files, (size_t)bb->nfiles, sizeof(entry_t*), bb, compare_files);
|
|
|
|
goto refresh;
|
|
|
|
} else if (2 <= mouse_y && mouse_y <= termheight - 2) {
|
2019-05-27 16:08:29 -07:00
|
|
|
int clicked = bb->scroll + (mouse_y - 2);
|
2019-05-29 00:56:49 -07:00
|
|
|
if (clicked > bb->nfiles - 1) goto next_input;
|
|
|
|
if (column == '+') {
|
2019-05-28 23:51:20 -07:00
|
|
|
toggle_file(bb, bb->files[clicked]);
|
2019-05-29 00:56:49 -07:00
|
|
|
lazy = 0;
|
2019-05-23 05:42:33 -07:00
|
|
|
goto redraw;
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-27 16:08:29 -07:00
|
|
|
set_cursor(bb, clicked);
|
2019-05-26 02:37:41 -07:00
|
|
|
if (dt_ms <= 200) {
|
2019-05-23 19:04:17 -07:00
|
|
|
key = KEY_MOUSE_DOUBLE_LEFT;
|
2019-05-26 02:37:41 -07:00
|
|
|
goto user_bindings;
|
|
|
|
}
|
|
|
|
goto redraw;
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2019-05-23 05:42:33 -07:00
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
case KEY_CTRL_C:
|
2019-05-24 16:55:36 -07:00
|
|
|
cleanup_and_exit(SIGINT);
|
2019-05-25 04:30:51 -07:00
|
|
|
goto quit; // Unreachable
|
2019-05-23 19:04:17 -07:00
|
|
|
|
|
|
|
case KEY_CTRL_Z:
|
2019-05-28 21:36:42 -07:00
|
|
|
fputs(T_OFF(T_ALT_SCREEN), tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
close_term();
|
|
|
|
raise(SIGTSTP);
|
|
|
|
init_term();
|
2019-05-27 15:53:07 -07:00
|
|
|
fputs(T_ON(T_ALT_SCREEN), tty_out);
|
2019-05-24 16:55:36 -07:00
|
|
|
lazy = 0;
|
2019-05-23 19:04:17 -07:00
|
|
|
goto redraw;
|
|
|
|
|
|
|
|
case KEY_CTRL_H: {
|
2019-05-27 15:53:07 -07:00
|
|
|
move_cursor(tty_out, 0,termheight-1);
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("\033[K\033[33;1mPress any key...\033[0m", tty_out);
|
2019-05-28 21:36:42 -07:00
|
|
|
while ((key = bgetkey(tty_in, &mouse_x, &mouse_y, 1000)) == -1)
|
2019-05-23 19:04:17 -07:00
|
|
|
;
|
2019-05-27 15:53:07 -07:00
|
|
|
move_cursor(tty_out, 0,termheight-1);
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("\033[K\033[1m<\033[33m", tty_out);
|
2019-05-28 21:36:42 -07:00
|
|
|
const char *name = bkeyname(key);
|
2019-05-27 14:12:47 -07:00
|
|
|
if (name) fputs(name, tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
else if (' ' <= key && key <= '~')
|
2019-05-27 14:12:47 -07:00
|
|
|
fputc((char)key, tty_out);
|
|
|
|
else
|
|
|
|
fprintf(tty_out, "\033[31m\\x%02X", key);
|
2019-05-23 05:42:33 -07:00
|
|
|
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("\033[0;1m> is bound to: \033[34;1m", tty_out);
|
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]) {
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs(bindings[i].description, tty_out);
|
|
|
|
fputs("\033[0m", tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
goto next_input;
|
2019-05-23 03:40:04 -07:00
|
|
|
}
|
2019-05-21 20:06:53 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs("--- nothing ---\033[0m", tty_out);
|
2019-05-23 19:04:17 -07:00
|
|
|
goto next_input;
|
|
|
|
}
|
2019-05-23 05:51:11 -07:00
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
case -1:
|
|
|
|
goto next_input;
|
|
|
|
|
2019-05-26 13:41:09 -07:00
|
|
|
default: {
|
2019-05-23 19:04:17 -07:00
|
|
|
// Search user-defined key bindings from config.h:
|
|
|
|
binding_t *binding;
|
2019-05-26 13:41:09 -07:00
|
|
|
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]) {
|
2019-05-25 20:55:59 -07:00
|
|
|
// 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 04:44:48 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-25 20:55:59 -07:00
|
|
|
// Nothing matched
|
|
|
|
goto next_input;
|
2019-05-23 19:04:17 -07:00
|
|
|
|
|
|
|
run_binding:
|
2019-05-24 16:55:36 -07:00
|
|
|
if (cmdpos != 0)
|
|
|
|
err("Command file still open");
|
2019-05-25 20:55:59 -07:00
|
|
|
if (binding->command[0] == '+') {
|
2019-05-27 16:08:29 -07:00
|
|
|
switch (execute_cmd(bb, binding->command + 1)) {
|
2019-05-25 20:55:59 -07:00
|
|
|
case BB_INVALID: break;
|
|
|
|
case BB_NOP: goto redraw;
|
|
|
|
case BB_REFRESH: goto refresh;
|
|
|
|
case BB_QUIT: goto quit;
|
|
|
|
}
|
|
|
|
goto get_keyboard_input;
|
|
|
|
}
|
2019-05-27 15:53:07 -07:00
|
|
|
move_cursor(tty_out, 0, termheight-1);
|
|
|
|
if (binding->flags & NORMAL_TERM) {
|
2019-05-28 21:36:42 -07:00
|
|
|
fputs(T_OFF(T_ALT_SCREEN), tty_out);
|
|
|
|
fputs(T_ON(T_WRAP), tty_out);
|
2019-05-27 15:53:07 -07:00
|
|
|
}
|
2019-05-28 21:36:42 -07:00
|
|
|
if (binding->flags & AT_CURSOR && !bb->firstselected) {
|
|
|
|
move_cursor(tty_out, 0, 3 + bb->cursor - bb->scroll);
|
|
|
|
fputs("\033[K", tty_out);
|
|
|
|
}
|
|
|
|
close_term();
|
2019-05-27 16:08:29 -07:00
|
|
|
run_cmd_on_selection(bb, binding->command);
|
2019-05-23 19:04:17 -07:00
|
|
|
init_term();
|
2019-05-27 15:53:07 -07:00
|
|
|
if (binding->flags & NORMAL_TERM)
|
|
|
|
fputs(T_ON(T_ALT_SCREEN), tty_out);
|
|
|
|
if (binding->flags & NORMAL_TERM)
|
2019-05-23 19:04:17 -07:00
|
|
|
lazy = 0;
|
2019-05-25 20:55:59 -07:00
|
|
|
check_cmds = 1;
|
|
|
|
goto redraw;
|
2019-05-26 13:41:09 -07:00
|
|
|
}
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-23 19:04:17 -07:00
|
|
|
goto next_input;
|
|
|
|
|
2019-05-24 20:49:24 -07:00
|
|
|
quit:
|
2019-05-27 16:08:29 -07:00
|
|
|
populate_files(bb, NULL);
|
2019-05-27 15:53:07 -07:00
|
|
|
fputs(T_LEAVE_BBMODE, tty_out);
|
2019-05-25 04:30:51 -07:00
|
|
|
close_term();
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
|
|
|
|
2019-05-28 22:29:22 -07:00
|
|
|
/*
|
|
|
|
* Print the current key bindings
|
|
|
|
*/
|
2019-05-25 04:30:51 -07:00
|
|
|
void print_bindings(int verbose)
|
2019-05-23 03:40:04 -07:00
|
|
|
{
|
|
|
|
struct winsize sz = {0};
|
|
|
|
ioctl(STDOUT_FILENO, TIOCGWINSZ, &sz);
|
2019-05-24 20:49:24 -07:00
|
|
|
int _width = sz.ws_col;
|
|
|
|
if (_width == 0) _width = 80;
|
2019-05-23 03:40:04 -07:00
|
|
|
|
|
|
|
char buf[1024];
|
|
|
|
char *kb = "Key Bindings";
|
2019-05-24 20:49:24 -07:00
|
|
|
printf("\n\033[33;1;4m\033[%dG%s\033[0m\n\n", (_width-(int)strlen(kb))/2, kb);
|
2019-05-23 03:40:04 -07:00
|
|
|
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];
|
2019-05-28 21:36:42 -07:00
|
|
|
const char *name = bkeyname(key);
|
2019-05-23 04:44:48 -07:00
|
|
|
if (name)
|
2019-05-25 20:55:59 -07:00
|
|
|
p = stpcpy(p, name);
|
2019-05-23 04:44:48 -07:00
|
|
|
else if (' ' <= key && key <= '~')
|
|
|
|
p += sprintf(p, "%c", (char)key);
|
|
|
|
else
|
2019-05-24 20:49:24 -07:00
|
|
|
p += sprintf(p, "\033[31m\\x%02X", key);
|
2019-05-23 03:40:04 -07:00
|
|
|
}
|
|
|
|
*p = '\0';
|
2019-05-24 20:49:24 -07:00
|
|
|
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);
|
2019-05-23 03:40:04 -07:00
|
|
|
if (verbose) {
|
2019-05-24 20:49:24 -07:00
|
|
|
printf("\n\033[%dG\033[0;32m", MAX(1, (_width - (int)strlen(bindings[i].command))/2));
|
2019-05-27 14:12:47 -07:00
|
|
|
fputs_escaped(stdout, bindings[i].command, "\033[0;32m");
|
2019-05-23 03:40:04 -07:00
|
|
|
fflush(stdout);
|
|
|
|
}
|
2019-05-24 20:49:24 -07:00
|
|
|
printf("\033[0m\n");
|
2019-05-23 03:40:04 -07:00
|
|
|
}
|
|
|
|
printf("\n");
|
|
|
|
}
|
|
|
|
|
2019-05-20 19:28:47 -07:00
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
2019-05-25 04:30:51 -07:00
|
|
|
char *initial_path = NULL, *depthstr;
|
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;
|
2019-05-25 04:30:51 -07:00
|
|
|
|
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];
|
|
|
|
}
|
2019-05-24 16:55:36 -07:00
|
|
|
if (!initial_path && cmd_args == 0) initial_path = ".";
|
|
|
|
|
2019-05-23 19:04:17 -07:00
|
|
|
FILE *cmdfile = NULL;
|
2019-05-24 16:55:36 -07:00
|
|
|
if (initial_path) {
|
2019-05-25 21:47:30 -07:00
|
|
|
has_initial_path:
|
2019-05-24 16:55:36 -07:00
|
|
|
cmdfilename = memcheck(strdup(CMDFILE_FORMAT));
|
|
|
|
if (!mktemp(cmdfilename)) err("Couldn't create tmpfile\n");
|
|
|
|
cmdfile = fopen(cmdfilename, "a");
|
|
|
|
if (!cmdfile) err("Couldn't create cmdfile: '%s'\n", cmdfilename);
|
2019-05-25 04:30:51 -07:00
|
|
|
// Set up environment variables
|
|
|
|
depthstr = getenv("BB_DEPTH");
|
|
|
|
int depth = depthstr ? atoi(depthstr) : 0;
|
|
|
|
if (asprintf(&depthstr, "%d", depth + 1) < 0)
|
|
|
|
err("Allocation failure");
|
|
|
|
setenv("BB_DEPTH", depthstr, 1);
|
|
|
|
setenv("BBCMD", cmdfilename, 1);
|
2019-05-24 16:55:36 -07:00
|
|
|
} else if (cmd_args > 0) {
|
|
|
|
char *parent_bbcmd = getenv("BBCMD");
|
|
|
|
if (!parent_bbcmd || parent_bbcmd[0] == '\0') {
|
2019-05-25 21:47:30 -07:00
|
|
|
initial_path = ".";
|
|
|
|
goto has_initial_path;
|
2019-05-23 19:04:17 -07:00
|
|
|
}
|
2019-05-24 16:55:36 -07:00
|
|
|
cmdfile = fopen(parent_bbcmd, "a");
|
|
|
|
if (!cmdfile) 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++) {
|
2019-05-28 21:36:42 -07:00
|
|
|
if (argv[i][0] == '?') {
|
|
|
|
fclose(cmdfile);
|
|
|
|
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-24 16:55:36 -07:00
|
|
|
fwrite(argv[i]+1, sizeof(char), strlen(argv[i]+1)+1, cmdfile);
|
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] [-0] [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
|
|
|
}
|
2019-05-23 05:42:33 -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-23 05:42:33 -07:00
|
|
|
}
|
2019-05-22 01:50:46 -07:00
|
|
|
if (argv[i][0] == '-') {
|
|
|
|
for (char *c = &argv[i][1]; *c; c++) {
|
|
|
|
switch (*c) {
|
2019-05-26 13:41:09 -07:00
|
|
|
case 'h':goto usage;
|
2019-05-25 23:31:03 -07:00
|
|
|
case 'v': goto version;
|
2019-05-26 13:41:09 -07:00
|
|
|
case 'd': print_dir = 1;
|
|
|
|
break;
|
|
|
|
case '0': sep = '\0';
|
|
|
|
break;
|
|
|
|
case 's': print_selection = 1;
|
|
|
|
break;
|
|
|
|
case 'b': print_bindings(0);
|
|
|
|
return 0;
|
|
|
|
case 'B': print_bindings(1);
|
|
|
|
return 0;
|
2019-05-22 01:50:46 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
continue;
|
|
|
|
}
|
2019-05-21 04:06:50 -07:00
|
|
|
}
|
2019-05-24 16:55:36 -07:00
|
|
|
|
|
|
|
if (cmdfile)
|
2019-05-23 19:04:17 -07:00
|
|
|
fclose(cmdfile);
|
2019-05-24 16:55:36 -07:00
|
|
|
|
2019-05-28 21:36:42 -07:00
|
|
|
if (!initial_path)
|
2019-05-24 16:55:36 -07:00
|
|
|
return 0;
|
2019-05-28 21:36:42 -07:00
|
|
|
|
|
|
|
// Default values
|
|
|
|
setenv("SHELL", "bash", 0);
|
|
|
|
setenv("EDITOR", "nano", 0);
|
|
|
|
setenv("PAGER", "less", 0);
|
2019-05-23 19:04:17 -07:00
|
|
|
|
2019-05-24 16:55:36 -07:00
|
|
|
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);
|
2019-05-27 19:51:04 -07:00
|
|
|
signal(SIGSEGV, cleanup_and_exit);
|
2019-05-24 16:55:36 -07:00
|
|
|
|
2019-05-25 04:30:51 -07:00
|
|
|
char *real = realpath(initial_path, NULL);
|
|
|
|
if (!real || chdir(real)) err("Not a valid path: %s\n", initial_path);
|
2019-05-27 16:14:14 -07:00
|
|
|
|
|
|
|
bb_t *bb = memcheck(calloc(1, sizeof(bb_t)));
|
2019-05-27 19:51:04 -07:00
|
|
|
bb->options['0'] = 'n';
|
|
|
|
bb->options['s'] = 'n';
|
2019-05-28 22:29:22 -07:00
|
|
|
bb_browse(bb, real);
|
2019-05-25 04:30:51 -07:00
|
|
|
free(real);
|
2019-05-24 16:55:36 -07:00
|
|
|
|
2019-05-27 16:14:14 -07:00
|
|
|
if (bb->firstselected && print_selection) {
|
|
|
|
for (entry_t *e = bb->firstselected; e; e = e->next) {
|
2019-05-28 00:07:43 -07:00
|
|
|
const char *p = e->fullname;
|
2019-05-27 16:08:29 -07:00
|
|
|
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);
|
2019-05-25 04:30:51 -07:00
|
|
|
}
|
|
|
|
if (print_dir) {
|
2019-05-23 19:04:17 -07:00
|
|
|
printf("%s\n", initial_path);
|
|
|
|
}
|
2019-05-27 16:14:14 -07:00
|
|
|
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
|
|
|
|
2019-05-25 04:30:51 -07:00
|
|
|
return 0;
|
2019-05-20 19:28:47 -07:00
|
|
|
}
|
2019-05-24 17:35:16 -07:00
|
|
|
|
|
|
|
// vim: ts=4 sw=0 et cino=L2,l1,(0,W4,m1
|