2020-02-23 16:30:07 -08:00
|
|
|
/*
|
|
|
|
* This file contains logic for drawing columns.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static void lpad(char *buf, int width)
|
|
|
|
{
|
|
|
|
int len = strlen(buf);
|
|
|
|
if (len < width) {
|
|
|
|
int pad = width - len;
|
|
|
|
memmove(&buf[pad], buf, (size_t)len + 1);
|
|
|
|
while (pad > 0) buf[--pad] = ' ';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static char* stpcpy_escaped(char *buf, const char *str, const char *color)
|
|
|
|
{
|
|
|
|
static const char *escapes = " abtnvfr e";
|
|
|
|
for (const char *c = str; *c; ++c) {
|
|
|
|
if (*c > 0 && *c <= '\x1b' && escapes[(int)*c] != ' ') { // "\n", etc.
|
|
|
|
buf += sprintf(buf, "\033[31m\\%c%s", escapes[(int)*c], color);
|
|
|
|
} else if (*c >= 0 && !(' ' <= *c && *c <= '~')) { // "\x02", etc.
|
|
|
|
buf += sprintf(buf, "\033[31m\\x%02X%s", *c, color);
|
|
|
|
} else {
|
|
|
|
*(buf++) = *c;
|
|
|
|
}
|
|
|
|
}
|
2020-02-23 19:48:24 -08:00
|
|
|
*buf = '\0';
|
2020-02-23 16:30:07 -08:00
|
|
|
return buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void timeago(char *buf, time_t t)
|
|
|
|
{
|
|
|
|
const int SECOND = 1;
|
|
|
|
const int MINUTE = 60 * SECOND;
|
|
|
|
const int HOUR = 60 * MINUTE;
|
|
|
|
const int DAY = 24 * HOUR;
|
|
|
|
const int MONTH = 30 * DAY;
|
|
|
|
const int YEAR = 365 * DAY;
|
|
|
|
|
|
|
|
time_t now = time(0);
|
|
|
|
double delta = difftime(now, t);
|
|
|
|
|
|
|
|
if (delta < 1.5)
|
|
|
|
sprintf(buf, "a second");
|
|
|
|
else if (delta < 1 * MINUTE)
|
|
|
|
sprintf(buf, "%d seconds", (int)delta);
|
|
|
|
else if (delta < 2 * MINUTE)
|
|
|
|
sprintf(buf, "a minute");
|
|
|
|
else if (delta < 1 * HOUR)
|
|
|
|
sprintf(buf, "%d minutes", (int)delta/MINUTE);
|
|
|
|
else if (delta < 2 * HOUR)
|
|
|
|
sprintf(buf, "an hour");
|
|
|
|
else if (delta < 1 * DAY)
|
|
|
|
sprintf(buf, "%d hours", (int)delta/HOUR);
|
|
|
|
else if (delta < 2 * DAY)
|
|
|
|
sprintf(buf, "yesterday");
|
|
|
|
else if (delta < 1 * MONTH)
|
|
|
|
sprintf(buf, "%d days", (int)delta/DAY);
|
|
|
|
else if (delta < 2 * MONTH)
|
|
|
|
sprintf(buf, "a month");
|
|
|
|
else if (delta < 1 * YEAR)
|
|
|
|
sprintf(buf, "%d months", (int)delta/MONTH);
|
|
|
|
else if (delta < 2 * YEAR)
|
|
|
|
sprintf(buf, "a year");
|
|
|
|
else
|
|
|
|
sprintf(buf, "%d years", (int)delta/YEAR);
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_mreltime(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color;
|
2020-02-23 16:30:07 -08:00
|
|
|
timeago(buf, entry->info.st_mtime);
|
|
|
|
lpad(buf, width);
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_areltime(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color;
|
2020-02-23 16:30:07 -08:00
|
|
|
timeago(buf, entry->info.st_atime);
|
|
|
|
lpad(buf, width);
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_creltime(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color;
|
2020-02-23 16:30:07 -08:00
|
|
|
timeago(buf, entry->info.st_ctime);
|
|
|
|
lpad(buf, width);
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_mtime(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color;
|
2020-02-23 16:30:07 -08:00
|
|
|
strftime(buf, (size_t)width, BB_TIME_FMT, localtime(&(entry->info.st_mtime)));
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_atime(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color;
|
2020-02-23 16:30:07 -08:00
|
|
|
strftime(buf, (size_t)width, BB_TIME_FMT, localtime(&(entry->info.st_atime)));
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_ctime(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color;
|
2020-02-23 16:30:07 -08:00
|
|
|
strftime(buf, (size_t)width, BB_TIME_FMT, localtime(&(entry->info.st_ctime)));
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_selected(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)width;
|
2020-02-23 18:13:33 -08:00
|
|
|
buf = stpcpy(buf, IS_SELECTED(entry) ? SELECTED_INDICATOR : NOT_SELECTED_INDICATOR);
|
2020-02-23 19:48:24 -08:00
|
|
|
buf = stpcpy(buf, color);
|
2020-02-23 16:30:07 -08:00
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_perm(entry_t *entry, const char *color, char *buf, int width) {
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color; (void)width;
|
2020-02-23 16:30:07 -08:00
|
|
|
sprintf(buf, " %03o", entry->info.st_mode & 0777);
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_random(entry_t *entry, const char *color, char *buf, int width)
|
2020-02-23 16:30:07 -08:00
|
|
|
{
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color;
|
|
|
|
sprintf(buf, "%*d", width, entry->shufflepos);
|
2020-02-23 16:30:07 -08:00
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_size(entry_t *entry, const char *color, char *buf, int width)
|
2020-02-23 16:30:07 -08:00
|
|
|
{
|
2020-02-23 19:48:24 -08:00
|
|
|
(void)color; (void)width;
|
2020-02-23 16:30:07 -08:00
|
|
|
int j = 0;
|
|
|
|
const char* units = "BKMGTPEZY";
|
|
|
|
double bytes = (double)entry->info.st_size;
|
|
|
|
while (bytes > 1024) {
|
|
|
|
bytes /= 1024;
|
|
|
|
j++;
|
|
|
|
}
|
|
|
|
sprintf(buf, " %6.*f%c ", j > 0 ? 1 : 0, bytes, units[j]);
|
|
|
|
}
|
|
|
|
|
2020-04-10 00:12:57 -07:00
|
|
|
static void col_name(entry_t *entry, const char *color, char *buf, int width)
|
2020-02-23 16:30:07 -08:00
|
|
|
{
|
|
|
|
(void)width;
|
2020-02-23 19:48:24 -08:00
|
|
|
if (entry->no_esc) buf = stpcpy(buf, entry->name);
|
|
|
|
else buf = stpcpy_escaped(buf, entry->name, color);
|
2020-02-23 16:30:07 -08:00
|
|
|
|
|
|
|
if (E_ISDIR(entry)) buf = stpcpy(buf, "/");
|
|
|
|
|
|
|
|
if (!entry->linkname) return;
|
|
|
|
|
|
|
|
buf = stpcpy(buf, "\033[2m -> \033[3m");
|
|
|
|
buf = stpcpy(buf, color);
|
|
|
|
if (entry->link_no_esc) buf = stpcpy(buf, entry->linkname);
|
|
|
|
else buf = stpcpy_escaped(buf, entry->linkname, color);
|
|
|
|
|
|
|
|
if (S_ISDIR(entry->linkedmode))
|
|
|
|
buf = stpcpy(buf, "/");
|
|
|
|
|
|
|
|
buf = stpcpy(buf, "\033[22;23m");
|
|
|
|
}
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
const char *name;
|
2020-02-23 19:48:24 -08:00
|
|
|
void (*render)(entry_t*, const char*, char*, int);
|
|
|
|
unsigned int stretchy : 1;
|
2020-02-23 16:30:07 -08:00
|
|
|
} column_t;
|
|
|
|
|
|
|
|
static column_t columns[255] = {
|
|
|
|
['*'] = {.name = "*", .render = col_selected},
|
2020-02-23 19:48:24 -08:00
|
|
|
['n'] = {.name = "Name", .render = col_name, .stretchy = 1},
|
2020-03-01 12:09:25 -08:00
|
|
|
['s'] = {.name = " Size", .render = col_size},
|
2020-02-23 16:30:07 -08:00
|
|
|
['p'] = {.name = "Perm", .render = col_perm},
|
2020-02-23 16:53:27 -08:00
|
|
|
['m'] = {.name = " Modified", .render = col_mreltime},
|
2020-02-23 19:48:24 -08:00
|
|
|
['M'] = {.name = " Modified ", .render = col_mtime},
|
2020-02-23 16:53:27 -08:00
|
|
|
['a'] = {.name = " Accessed", .render = col_areltime},
|
2020-02-23 19:48:24 -08:00
|
|
|
['A'] = {.name = " Accessed ", .render = col_atime},
|
2020-02-23 16:53:27 -08:00
|
|
|
['c'] = {.name = " Created", .render = col_creltime},
|
2020-02-23 19:48:24 -08:00
|
|
|
['C'] = {.name = " Created ", .render = col_ctime},
|
|
|
|
['r'] = {.name = "Random", .render = col_random},
|
2020-02-23 16:30:07 -08:00
|
|
|
};
|