libcosmos
Linux C++ System Programming Library
Loading...
Searching...
No Matches
colors.hxx
Go to the documentation of this file.
1#pragma once
2
3// C++
4#include <iosfwd>
5#include <string_view>
6#include <variant>
7
8// cosmos
9#include <cosmos/dso_export.h>
10#include <cosmos/utils.hxx>
11
37namespace cosmos::term {
38
40
44enum class TermColor : size_t {
45 // the integer values denote the offset from the base ANSI escape code
46 // for front/back bright/normal colors.
47 BLACK = 0,
48 RED,
49 GREEN,
50 YELLOW,
51 BLUE,
52 MAGENTA,
53 CYAN,
54 WHITE
55};
56
58enum class ColorKind {
59 FRONT,
60 BACK
61};
62
64enum class ColorIntensity {
65 NORMAL,
66 BRIGHT
67};
68
70
81class ColorSpec {
82public: // functions
83
84 ColorSpec(const TermColor color, const ColorKind kind, const ColorIntensity intensity) :
85 m_color(color), m_kind(kind), m_intensity(intensity) {}
86
87 TermColor getColor() const { return m_color; }
88 bool isBright() const { return m_intensity == ColorIntensity::BRIGHT; }
89 bool isNormal() const { return !isBright(); }
90 bool isFrontColor() const { return m_kind == ColorKind::FRONT; }
91 bool isBackColor() const { return !isFrontColor(); }
92
93protected: // data
94
95 TermColor m_color;
96 ColorKind m_kind = ColorKind::FRONT;
97 ColorIntensity m_intensity = ColorIntensity::NORMAL;
98};
99
101struct FrontColor : public ColorSpec {
102 explicit FrontColor(const TermColor c) : ColorSpec(c, ColorKind::FRONT, ColorIntensity::NORMAL) {}
103 FrontColor& setBright() { m_intensity = ColorIntensity::BRIGHT; return *this; }
104};
105
107struct BackColor : public ColorSpec {
108 explicit BackColor(const TermColor c) : ColorSpec(c, ColorKind::BACK, ColorIntensity::NORMAL) {}
109 BackColor& setBright() { m_intensity = ColorIntensity::BRIGHT; return *this; }
110};
111
113enum class TermControl : size_t {
114 RESET = 0,
115 UNDERLINE_ON = 4,
116 UNDERLINE_OFF = 24,
117 BLINK_ON = 5,
118 BLINK_OFF = 25,
119 INVERSE_ON = 7,
120 INVERSE_OFF = 27,
121 DEFAULT_FG_COLOR = 39,
122 DEFAULT_BG_COLOR = 49
123};
124
126enum class ANSICode : size_t {
127};
128
130
133TermControl COSMOS_API get_off_control(const TermControl ctrl);
135ANSICode COSMOS_API get_ansi_color_code(const ColorSpec &color);
136
138
150COSMOS_API void refresh_tty_detection();
151
153
166public:
167 ANSICode getOnCode() const { return m_on_code; }
168 ANSICode getOffCode() const { return m_off_code; }
169 bool hasText() const { return std::holds_alternative<const std::string_view*>(m_info); }
170 const std::string_view& getText() const { return *std::get<const std::string_view*>(m_info); }
171 bool hasNextFeature() const { return std::holds_alternative<const FeatureBase*>(m_info); }
172 const FeatureBase& getNextFeature() const { return *std::get<const FeatureBase*>(m_info); }
173protected:
174 FeatureBase(const std::string_view &text, const ANSICode on_code, const ANSICode off_code) :
175 m_info(&text), m_on_code(on_code), m_off_code(off_code)
176 {}
177
178 FeatureBase(const FeatureBase &next, const ANSICode on_code, const ANSICode off_code) :
179 m_info(&next), m_on_code(on_code), m_off_code(off_code)
180 {}
181protected:
182 // TODO: this way of stacking features is prone to memory corruption
183 // if the used pointers lose validity ... it works for a stack of
184 // temporary objects but in some non-standard use cases errors can
185 // occur.
187 std::variant<const std::string_view*, const FeatureBase*> m_info;
188 const ANSICode m_on_code;
189 const ANSICode m_off_code;
190};
191
194 public FeatureBase {
195protected:
196 TextEffect(const TermControl feature, const std::string_view &text) :
198 text,
199 ANSICode{to_integral(feature)},
200 ANSICode{to_integral(get_off_control(feature))}
201 }
202 {}
203
204 TextEffect(const TermControl feature, const FeatureBase &next) :
206 next,
207 ANSICode{to_integral(feature)},
208 ANSICode{to_integral(get_off_control(feature))}
209 }
210 {}
211};
212
214template <TermControl effect>
215struct TextEffectT : public TextEffect {
216 explicit TextEffectT(const std::string_view &text) :
217 TextEffect{effect, text} {}
218 explicit TextEffectT(const FeatureBase &next) :
219 TextEffect{effect, next} {}
220};
221
228
230struct ColoredText : public FeatureBase {
231 explicit ColoredText(const std::string_view &text, const TermColor c, const ColorKind kind, const ColorIntensity intensity) :
233 text,
234 get_ansi_color_code(ColorSpec{c, kind, intensity}),
235 getOffCode(kind)
236 }
237 {}
238
239 explicit ColoredText(const FeatureBase &next, const TermColor c, const ColorKind kind, const ColorIntensity intensity) :
241 next,
242 get_ansi_color_code(ColorSpec{c, kind, intensity}),
243 getOffCode(kind)
244 }
245 {}
246
247protected: // functions
248
249 ANSICode getOffCode(const ColorKind kind) const {
250 auto ret = kind == ColorKind::FRONT ? TermControl::DEFAULT_FG_COLOR : TermControl::DEFAULT_BG_COLOR;
251 return ANSICode{to_integral(ret)};
252 }
253};
254
256template <TermColor color, ColorIntensity intensity = ColorIntensity::NORMAL>
257struct ColoredTextT : public ColoredText {
258 explicit ColoredTextT(const std::string_view &text) : ColoredText{text, color, ColorKind::FRONT, intensity} {}
259 explicit ColoredTextT(const FeatureBase &next) : ColoredText{next, color, ColorKind::FRONT, intensity} {}
260};
261
263template <TermColor color, ColorIntensity intensity = ColorIntensity::NORMAL>
264struct TextOnColorT : public ColoredText {
265 explicit TextOnColorT(const std::string_view &text) : ColoredText{text, color, ColorKind::BACK, intensity} {}
266 explicit TextOnColorT(const FeatureBase &next) : ColoredText{next, color, ColorKind::BACK, intensity} {}
267};
268
269/*
270 * Helpers to apply colored text and background easily on an ostream.
271 *
272 * These only enable the color for the given string argument and disable
273 * coloring after the output operation again.
274 *
275 * These helpers can be nested with text effects or background helpers to
276 * create readable chained effects for a string like:
277 *
278 * std::cout << Green(OnWhite(Underlined("text"))) << std::endl;
279 */
280
285
290
295
300
305
310
315
320
321} // end ns
322
323COSMOS_API std::ostream& operator<<(std::ostream &o, const cosmos::term::ColorSpec &fc);
324COSMOS_API std::ostream& operator<<(std::ostream &o, const cosmos::term::TermControl p);
325COSMOS_API std::ostream& operator<<(std::ostream &o, const cosmos::term::FeatureBase &fb);
Complete color specification for ANSI terminals.
Definition colors.hxx:81
Base class to build nested ANSI feature objects.
Definition colors.hxx:165
std::variant< const std::string_view *, const FeatureBase * > m_info
either a terminal string or a pointer to the next feature to apply.
Definition colors.hxx:187
Base class for easy feature TermControl application on ostreams.
Definition colors.hxx:194
TermColor
Primitive Colors for ANSI Terminals.
Definition colors.hxx:44
ANSICode
A generic ANSI code e.g. for color indices.
Definition colors.hxx:126
ColorKind
Differentiation between text (front) and background color.
Definition colors.hxx:58
TermControl
Various feature controls for ANSI terminals.
Definition colors.hxx:113
@ INVERSE_ON
Inverse fg/bg colors.
@ DEFAULT_BG_COLOR
Set default bg color.
@ RESET
Remove all attributes currently set (including colors)
@ DEFAULT_FG_COLOR
Set default fg color.
@ BLINK_ON
Blinking text.
@ UNDERLINE_ON
Underlined text.
ColorIntensity
Differentiation of normal and bright color intensity.
Definition colors.hxx:64
Simple type to represent an ANSI background color in bright or normal intensity.
Definition colors.hxx:107
Template for definition of concrete color text helpers.
Definition colors.hxx:257
Base class for easy colored text application on ostreams.
Definition colors.hxx:230
Simple type to represent an ANSI foreground color in bright or normal intensity.
Definition colors.hxx:101
Template for definition of concrete text effect helpers.
Definition colors.hxx:215
Template for definition of concrete background color helpers.
Definition colors.hxx:264