libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
colors.cxx
1// C++
2#include <cassert>
3#include <iostream>
4#include <ostream>
5#include <vector>
6
7// cosmos
8#include <cosmos/error/UsageError.hxx>
9#include <cosmos/io/colors.hxx>
10#include <cosmos/io/Terminal.hxx>
11#include <cosmos/private/Initable.hxx>
12#include <cosmos/proc/process.hxx>
13
14namespace cosmos::term {
15
16namespace {
17
18 bool stdout_is_tty = true;
19 bool stderr_is_tty = true;
20
21 bool isTTYStream(const std::ostream &o) {
22 auto rdbuf = o.rdbuf();
23
24 if (rdbuf == std::cout.rdbuf())
25 return stdout_is_tty;
26 else if (rdbuf == std::cerr.rdbuf())
27 return stderr_is_tty;
28
29 return false;
30 }
31
32 std::string build_ansi_command(const std::vector<ANSICode> &cmd_list) {
33 // the ASCII escape character for ANSI control sequences
34 constexpr char ANSI_ESC_START = '\033';
35 // ANSI escape finish character
36 constexpr char ANSI_ESC_FINISH = 'm';
37
38 std::string ret(1, ANSI_ESC_START);
39 ret += '[';
40 for (const auto &cmd: cmd_list) {
41 ret += std::to_string(to_integral(cmd));
42 ret += ";";
43 }
44 // remove superfluous separator again
45 ret.pop_back();
46 ret += ANSI_ESC_FINISH;
47 return ret;
48 }
49
50 std::string build_ansi_command(const ANSICode code) {
51 return build_ansi_command(std::vector<ANSICode>({code}));
52 }
53
54 std::vector<const FeatureBase*> get_features(const FeatureBase &first) {
55 std::vector<const FeatureBase*> ret;
56 const FeatureBase *feature = &first;
57
58 while (true) {
59 ret.push_back(feature);
60
61 if (!feature->hasNextFeature())
62 break;
63
64 feature = &feature->getNextFeature();
65 }
66
67 return ret;
68 }
69
70} // end anon ns
71
72void refresh_tty_detection() {
73 if (proc::get_env_var("COSMOS_FORCE_COLOR_ON")) {
74 stdout_is_tty = stderr_is_tty = true;
75 return;
76 } else if (proc::get_env_var("COSMOS_FORCE_COLOR_OFF")) {
77 stdout_is_tty = stderr_is_tty = false;
78 return;
79 }
80
81 Terminal output{FileDescriptor{FileNum::STDOUT}};
82 Terminal error{FileDescriptor{FileNum::STDERR}};
83
84 stdout_is_tty = output.isTTY();
85 stderr_is_tty = error.isTTY();
86}
87
88ANSICode get_ansi_color_code(const ColorSpec &color) {
89 size_t code = color.isFrontColor() ? 30 : 40;
90 if (color.isBright())
91 code += 60;
92
93 return ANSICode{code + to_integral(color.getColor())};
94}
95
96TermControl get_off_control(const TermControl ctrl) {
97 switch(ctrl) {
98 case TermControl::UNDERLINE_ON: return TermControl::UNDERLINE_OFF;
99 case TermControl::BLINK_ON: return TermControl::BLINK_OFF;
100 case TermControl::INVERSE_ON: return TermControl::INVERSE_OFF;
101 default: cosmos_throw (UsageError("bad value"));
102 return TermControl::RESET; /* to silence compiler warning */
103 }
104}
105
106namespace {
107
109class CheckStdioTTYsInit :
110 public Initable {
111public: // functions
112
113 CheckStdioTTYsInit() : Initable(InitPrio::CHECK_STDIO_TTYS) {
114 }
115
116protected: // functions
117
118 void libInit() override {
119 refresh_tty_detection();
120 }
121
122 void libExit() override {
123 // no-op
124 }
125};
126
127CheckStdioTTYsInit g_check_stdio_ttys;
128
129} // end anon ns
130
131} // end ns
132
133
134std::ostream& operator<<(std::ostream &o, const cosmos::term::ColorSpec &fc) {
135 if (!cosmos::term::isTTYStream(o))
136 return o;
137 const auto code = get_ansi_color_code(fc);
138
139 return o << cosmos::term::build_ansi_command(code);
140}
141
142std::ostream& operator<<(std::ostream &o, const cosmos::term::TermControl p) {
143 if (!cosmos::term::isTTYStream(o))
144 return o;
145 return o << cosmos::term::build_ansi_command(cosmos::term::ANSICode{cosmos::to_integral(p)});
146}
147
148std::ostream& operator<<(std::ostream &o, const cosmos::term::FeatureBase &fb) {
149 using namespace cosmos::term;
150
151 if (!cosmos::term::isTTYStream(o)) {
152 if (!fb.hasNextFeature())
153 return o << fb.getText();
154 else {
155 const auto features = get_features(fb);
156 return o << features.back()->getText();
157 }
158 }
159
160 // simple case without dynamic allocation
161 if (!fb.hasNextFeature()) {
162 return o << build_ansi_command(fb.getOnCode()) << fb.getText() << build_ansi_command(fb.getOffCode());
163 }
164
165 // complex case with nested features
166 std::vector<cosmos::term::ANSICode> on_codes, off_codes;
167 const auto features = get_features(fb);
168 for (const auto feature: features) {
169 on_codes.push_back(feature->getOnCode());
170 off_codes.push_back(feature->getOffCode());
171 }
172
173 assert (features.back()->hasText());
174 return o << build_ansi_command(on_codes) << features.back()->getText() << build_ansi_command(off_codes);
175}
Complete color specification for ANSI terminals.
Definition colors.hxx:81
Base class to build nested ANSI feature objects.
Definition colors.hxx:165
ANSICode
A generic ANSI code e.g. for color indices.
Definition colors.hxx:126
TermControl
Various feature controls for ANSI terminals.
Definition colors.hxx:113