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>
14namespace cosmos::term {
18 bool stdout_is_tty =
true;
19 bool stderr_is_tty =
true;
21 bool isTTYStream(
const std::ostream &o) {
22 auto rdbuf = o.rdbuf();
24 if (rdbuf == std::cout.rdbuf())
26 else if (rdbuf == std::cerr.rdbuf())
32 std::string build_ansi_command(
const std::vector<ANSICode> &cmd_list) {
34 constexpr char ANSI_ESC_START =
'\033';
36 constexpr char ANSI_ESC_FINISH =
'm';
38 std::string ret(1, ANSI_ESC_START);
40 for (
const auto &cmd: cmd_list) {
41 ret += std::to_string(to_integral(cmd));
46 ret += ANSI_ESC_FINISH;
50 std::string build_ansi_command(
const ANSICode code) {
51 return build_ansi_command(std::vector<ANSICode>({code}));
54 std::vector<const FeatureBase*> get_features(
const FeatureBase &first) {
55 std::vector<const FeatureBase*> ret;
56 const FeatureBase *feature = &first;
59 ret.push_back(feature);
61 if (!feature->hasNextFeature())
64 feature = &feature->getNextFeature();
72void refresh_tty_detection() {
73 if (proc::get_env_var(
"COSMOS_FORCE_COLOR_ON")) {
74 stdout_is_tty = stderr_is_tty =
true;
76 }
else if (proc::get_env_var(
"COSMOS_FORCE_COLOR_OFF")) {
77 stdout_is_tty = stderr_is_tty =
false;
81 Terminal output{FileDescriptor{FileNum::STDOUT}};
82 Terminal error{FileDescriptor{FileNum::STDERR}};
84 stdout_is_tty = output.isTTY();
85 stderr_is_tty = error.isTTY();
88ANSICode get_ansi_color_code(
const ColorSpec &color) {
89 size_t code = color.isFrontColor() ? 30 : 40;
93 return ANSICode{code + to_integral(color.getColor())};
96TermControl get_off_control(
const TermControl 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;
109class CheckStdioTTYsInit :
113 CheckStdioTTYsInit() : Initable(InitPrio::CHECK_STDIO_TTYS) {
118 void libInit()
override {
119 refresh_tty_detection();
122 void libExit()
override {
127CheckStdioTTYsInit g_check_stdio_ttys;
135 if (!cosmos::term::isTTYStream(o))
137 const auto code = get_ansi_color_code(fc);
139 return o << cosmos::term::build_ansi_command(code);
143 if (!cosmos::term::isTTYStream(o))
149 using namespace cosmos::term;
151 if (!cosmos::term::isTTYStream(o)) {
152 if (!fb.hasNextFeature())
153 return o << fb.getText();
155 const auto features = get_features(fb);
156 return o << features.back()->getText();
161 if (!fb.hasNextFeature()) {
162 return o << build_ansi_command(fb.getOnCode()) << fb.getText() << build_ansi_command(fb.getOffCode());
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());
173 assert (features.back()->hasText());
174 return o << build_ansi_command(on_codes) << features.back()->getText() << build_ansi_command(off_codes);
Complete color specification for ANSI terminals.
Base class to build nested ANSI feature objects.
ANSICode
A generic ANSI code e.g. for color indices.
TermControl
Various feature controls for ANSI terminals.