summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorPřemysl Janouch <p.janouch@gmail.com>2017-06-30 21:29:10 +0200
committerPřemysl Janouch <p.janouch@gmail.com>2017-06-30 21:29:10 +0200
commit4a9662b00a14ceceb35be4d053fa34d66df1db78 (patch)
treeb120b7f5afeb4cfb43b760323f666aaba6794509
parentc90fa0a060a6f51262afafd90278a438e7e7aa3f (diff)
downloadsdn-4a9662b00a14ceceb35be4d053fa34d66df1db78.tar.gz
sdn-4a9662b00a14ceceb35be4d053fa34d66df1db78.tar.xz
sdn-4a9662b00a14ceceb35be4d053fa34d66df1db78.zip
Configurable colors
-rw-r--r--README.adoc11
-rw-r--r--sdn.cpp130
2 files changed, 126 insertions, 15 deletions
diff --git a/README.adoc b/README.adoc
index e0c7dce..b1c78f0 100644
--- a/README.adoc
+++ b/README.adoc
@@ -56,6 +56,17 @@ bindkey '\eo' navigate
As far as I'm aware, bash cannot be used for this, as there is no command to
reset the prompt from within a `bind -x` handler.
+Colors
+------
+Here is an example of a '~/.config/sdn/look' file; the format is similar to
+that of git, only named colors aren't supported:
+....
+cursor 231 202
+bar 16 255 ul
+cwd bold
+input
+....
+
Contributing and Support
------------------------
Use this project's GitHub to report any bugs, request features, or submit pull
diff --git a/sdn.cpp b/sdn.cpp
index aeca397..8f2749b 100644
--- a/sdn.cpp
+++ b/sdn.cpp
@@ -23,6 +23,8 @@
#include <algorithm>
#include <cwchar>
#include <climits>
+#include <cstdlib>
+#include <fstream>
#include <ncurses.h>
#include <unistd.h>
@@ -41,6 +43,10 @@
// Trailing return types make C++ syntax suck considerably less
#define fun static auto
+#ifndef A_ITALIC
+#define A_ITALIC 0
+#endif
+
using namespace std;
// For some reason handling of encoding in C and C++ is extremely annoying
@@ -78,6 +84,19 @@ fun prefix_length (const wstring &in, const wstring &of) -> int {
return score;
}
+fun split (const string &s, const string &sep, vector<string> &out) {
+ size_t mark = 0, p = s.find (sep);
+ for (; p != string::npos; p = s.find (sep, (mark = p + sep.length ())))
+ if (mark < p)
+ out.push_back (s.substr (mark, p - mark));
+ if (mark < s.length ())
+ out.push_back (s.substr (mark));
+}
+
+fun split (const string &s, const string &sep) -> vector<string> {
+ vector<string> result; split (s, sep, result); return result;
+}
+
fun shell_escape (const string &v) -> string {
string result;
for (auto c : v)
@@ -110,6 +129,35 @@ fun decode_mode (mode_t m) -> wstring {
};
}
+template<class T> fun shift (vector<T> &v) -> T {
+ auto front = v.front (); v.erase (begin (v)); return front;
+}
+
+// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
+
+fun xdg_config_home () -> string {
+ const char *user_dir = getenv ("XDG_CONFIG_HOME");
+ if (user_dir && user_dir[0] == '/')
+ return user_dir;
+
+ const char *home_dir = getenv ("HOME");
+ return string (home_dir ? home_dir : "") + "/.config";
+}
+
+// In C++17 we will get <optional> but until then there's unique_ptr
+fun xdg_config_find (const string &suffix) -> unique_ptr<ifstream> {
+ vector<string> dirs {xdg_config_home ()};
+ const char *system_dirs = getenv ("XDG_CONFIG_DIRS");
+ split (system_dirs ? system_dirs : "/etc/xdg", ":", dirs);
+ for (const auto &dir : dirs) {
+ if (dir[0] != '/')
+ continue;
+ if (ifstream ifs {dir + suffix})
+ return make_unique<ifstream> (move (ifs));
+ }
+ return nullptr;
+}
+
// - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
using ncstring = basic_string<cchar_t>;
@@ -175,6 +223,30 @@ fun align (const ncstring &nc, int target) -> ncstring {
: apply_attrs (wstring (missing, L' '), 0) + nc;
}
+fun allocate_pair (short fg, short bg) -> short {
+ static short counter = 1; init_pair (counter, fg, bg); return counter++;
+}
+
+fun decode_attrs (const vector<string> &attrs) -> chtype {
+ chtype result = 0; int fg = -1, bg = -1, colors = 0;
+ for (const auto &s : attrs) {
+ char *end; auto color = strtol (s.c_str (), &end, 10);
+ if (!*end && color >= -1 && color < COLORS) {
+ if (++colors == 1) fg = color;
+ else if (colors == 2) bg = color;
+ }
+ else if (s == "bold") result |= A_BOLD;
+ else if (s == "dim") result |= A_DIM;
+ else if (s == "ul") result |= A_UNDERLINE;
+ else if (s == "blink") result |= A_BLINK;
+ else if (s == "reverse") result |= A_REVERSE;
+ else if (s == "italic") result |= A_ITALIC;
+ }
+ if (fg != -1 || bg != -1)
+ result |= COLOR_PAIR (allocate_pair (fg, bg));
+ return result;
+}
+
// --- Application -------------------------------------------------------------
#define CTRL 31 &
@@ -208,6 +280,10 @@ static struct {
wchar_t editor; // Prompt character for editing
wstring editor_line; // Current user input
+
+ enum { AT_CURSOR, AT_BAR, AT_CWD, AT_INPUT, AT_COUNT };
+ chtype attrs[AT_COUNT] = {A_REVERSE, 0, A_BOLD, 0};
+ const char *attr_names[AT_COUNT] = {"cursor", "bar", "cwd", "input"};
} g;
fun make_row (const string &filename, const struct stat &info) -> row {
@@ -255,33 +331,32 @@ fun update () {
int available = visible_lines ();
int used = min (available, int (g.entries.size ()) - g.offset);
for (int i = 0; i < used; i++) {
- attrset (0);
+ auto index = g.offset + i;
+ attrset (index == g.cursor ? g.attrs[g.AT_CURSOR] : 0);
move (available - used + i, 0);
- int index = g.offset + i;
- if (index == g.cursor)
- attron (A_REVERSE);
-
- auto limit = COLS, used = 0;
+ auto used = 0;
for (int col = start_column; col < row::COLUMNS; col++) {
const auto &field = g.entries[index].row.cols[col];
auto aligned = align (field, alignment[col] * g.max_widths[col]);
- used += print (aligned + apply_attrs (L" ", 0), limit - used);
+ used += print (aligned + apply_attrs (L" ", 0), COLS - used);
}
- hline (' ', limit - used);
+ hline (' ', COLS - used);
}
- attrset (A_BOLD);
- mvprintw (LINES - 2, 0, "%s", g.cwd.c_str ());
+ auto bar = apply_attrs (to_wide (g.cwd), g.attrs[g.AT_CWD]);
if (g.out_of_date)
- addstr (" [+]");
+ bar += apply_attrs (L" [+]", 0);
+
+ move (LINES - 2, 0);
+ attrset (g.attrs[g.AT_BAR]);
+ hline (' ', COLS - print (bar, COLS));
- attrset (0);
+ attrset (g.attrs[g.AT_INPUT]);
if (g.editor) {
move (LINES - 1, 0);
- wchar_t prefix[] = { g.editor, L' ', L'\0' };
- addwstr (prefix);
- move (LINES - 1, print (apply_attrs (g.editor_line, 0), COLS - 3) + 2);
+ auto p = apply_attrs ({g.editor, L' ', L'\0'}, 0);
+ move (LINES - 1, print (p + apply_attrs (g.editor_line, 0), COLS - 1));
curs_set (1);
} else
curs_set (0);
@@ -492,6 +567,29 @@ fun inotify_check () {
update ();
}
+fun load_configuration () {
+ auto config = xdg_config_find ("/" PROJECT_NAME "/look");
+ if (!config)
+ return;
+
+ // Bail out on dumb terminals, there's not much one can do about them
+ if (!has_colors () || start_color () == ERR || use_default_colors () == ERR)
+ return;
+
+ string line;
+ while (getline (*config, line)) {
+ vector<string> tokens = split (line, " ");
+ if (tokens.empty () || line.front () == '#')
+ continue;
+ auto name = shift (tokens);
+ for (int i = 0; i < g.AT_COUNT; i++)
+ if (name == g.attr_names[i])
+ g.attrs[i] = decode_attrs (tokens);
+ }
+
+ // TODO: load and use LS_COLORS
+}
+
int main (int argc, char *argv[]) {
(void) argc;
(void) argv;
@@ -518,6 +616,8 @@ int main (int argc, char *argv[]) {
cerr << "cannot initialize screen" << endl;
return 1;
}
+
+ load_configuration ();
reload ();
auto start_dir = g.cwd;