2016-05-02 05:29:21 +00:00
|
|
|
// Generic output functions.
|
|
|
|
//
|
|
|
|
// Constants for various character classifications. Each character of a command string can be
|
|
|
|
// classified as one of the following types.
|
2005-10-04 15:11:39 +00:00
|
|
|
#ifndef FISH_OUTPUT_H
|
|
|
|
#define FISH_OUTPUT_H
|
|
|
|
|
2016-05-02 05:29:21 +00:00
|
|
|
#include <stddef.h>
|
2017-02-14 04:37:27 +00:00
|
|
|
|
2016-05-02 05:29:21 +00:00
|
|
|
#include <vector>
|
2016-04-21 06:00:54 +00:00
|
|
|
|
2016-05-02 05:29:21 +00:00
|
|
|
#include "color.h"
|
2016-04-21 06:00:54 +00:00
|
|
|
#include "fallback.h" // IWYU pragma: keep
|
2005-10-04 15:11:39 +00:00
|
|
|
|
2017-08-05 22:08:39 +00:00
|
|
|
class env_var_t;
|
|
|
|
|
2018-10-06 20:32:08 +00:00
|
|
|
class outputter_t {
|
|
|
|
/// Storage for buffered contents.
|
|
|
|
std::string contents_;
|
|
|
|
|
2020-08-23 10:38:25 +00:00
|
|
|
/// Count of how many outstanding begin_buffering() calls there are.
|
2021-02-17 21:27:34 +00:00
|
|
|
uint32_t buffer_count_{0};
|
2018-10-06 20:32:08 +00:00
|
|
|
|
|
|
|
/// fd to output to.
|
|
|
|
int fd_{-1};
|
|
|
|
|
|
|
|
rgb_color_t last_color = rgb_color_t::normal();
|
|
|
|
rgb_color_t last_color2 = rgb_color_t::normal();
|
|
|
|
bool was_bold = false;
|
|
|
|
bool was_underline = false;
|
|
|
|
bool was_italics = false;
|
|
|
|
bool was_dim = false;
|
|
|
|
bool was_reverse = false;
|
|
|
|
|
2021-01-11 19:53:11 +00:00
|
|
|
void reset_modes() {
|
|
|
|
was_bold = false;
|
|
|
|
was_underline = false;
|
|
|
|
was_italics = false;
|
|
|
|
was_dim = false;
|
|
|
|
was_reverse = false;
|
|
|
|
}
|
|
|
|
|
2018-10-06 20:32:08 +00:00
|
|
|
/// Construct an outputter which outputs to a given fd.
|
|
|
|
explicit outputter_t(int fd) : fd_(fd) {}
|
|
|
|
|
|
|
|
/// Flush output, if we have a set fd and our buffering count is 0.
|
|
|
|
void maybe_flush() {
|
2021-02-17 21:27:34 +00:00
|
|
|
if (fd_ >= 0 && buffer_count_ == 0) flush_to(fd_);
|
2018-10-06 20:32:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
/// Construct an outputter which outputs to its string.
|
|
|
|
outputter_t() = default;
|
|
|
|
|
|
|
|
/// Unconditionally write the color string to the output.
|
|
|
|
bool write_color(rgb_color_t color, bool is_fg);
|
|
|
|
|
|
|
|
/// Set the foreground and background color.
|
2021-08-17 22:45:42 +00:00
|
|
|
void set_color(rgb_color_t fg, rgb_color_t bg);
|
2018-10-06 20:32:08 +00:00
|
|
|
|
|
|
|
/// Write a wide character to the receiver.
|
2021-02-17 23:38:46 +00:00
|
|
|
void writech(wchar_t ch) { writestr(&ch, 1); }
|
2018-10-06 20:32:08 +00:00
|
|
|
|
|
|
|
/// Write a NUL-terminated wide character string to the receiver.
|
2021-02-17 23:38:46 +00:00
|
|
|
void writestr(const wchar_t *str) { writestr(str, wcslen(str)); }
|
2018-10-06 20:32:08 +00:00
|
|
|
|
|
|
|
/// Write a wide character string to the receiver.
|
2021-02-17 23:38:46 +00:00
|
|
|
void writestr(const wcstring &str) { writestr(str.data(), str.size()); }
|
2018-10-06 20:32:08 +00:00
|
|
|
|
|
|
|
/// Write the given terminfo string to the receiver, like tputs().
|
|
|
|
int term_puts(const char *str, int affcnt);
|
|
|
|
|
2021-02-17 23:38:46 +00:00
|
|
|
/// Write a wide string of the given length.
|
|
|
|
void writestr(const wchar_t *str, size_t len);
|
|
|
|
|
2018-10-06 20:32:08 +00:00
|
|
|
/// Write a narrow string of the given length.
|
|
|
|
void writestr(const char *str, size_t len) {
|
|
|
|
contents_.append(str, len);
|
|
|
|
maybe_flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Write a narrow NUL-terminated string.
|
2019-03-12 22:07:07 +00:00
|
|
|
void writestr(const char *str) { writestr(str, std::strlen(str)); }
|
2018-10-06 20:32:08 +00:00
|
|
|
|
|
|
|
/// Write a narrow character.
|
|
|
|
void push_back(char c) {
|
|
|
|
contents_.push_back(c);
|
|
|
|
maybe_flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// \return the "output" contents.
|
|
|
|
const std::string &contents() const { return contents_; }
|
|
|
|
|
|
|
|
/// Output any buffered data to the given \p fd.
|
|
|
|
void flush_to(int fd);
|
|
|
|
|
|
|
|
/// Begins buffering. Output will not be automatically flushed until a corresponding
|
2020-08-23 10:38:25 +00:00
|
|
|
/// end_buffering() call.
|
|
|
|
void begin_buffering() {
|
2021-02-17 21:27:34 +00:00
|
|
|
buffer_count_++;
|
|
|
|
assert(buffer_count_ > 0 && "bufferCount_ overflow");
|
2018-10-06 20:32:08 +00:00
|
|
|
}
|
|
|
|
|
2020-08-23 10:38:25 +00:00
|
|
|
/// Balance a begin_buffering() call.
|
|
|
|
void end_buffering() {
|
2021-02-17 21:27:34 +00:00
|
|
|
assert(buffer_count_ > 0 && "bufferCount_ underflow");
|
|
|
|
buffer_count_--;
|
2018-10-06 20:32:08 +00:00
|
|
|
maybe_flush();
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Accesses the singleton stdout outputter.
|
|
|
|
/// This can only be used from the main thread.
|
|
|
|
/// This outputter flushes its buffer after every write operation.
|
|
|
|
static outputter_t &stdoutput();
|
|
|
|
};
|
|
|
|
|
|
|
|
void writembs_check(outputter_t &outp, const char *mbs, const char *mbs_name, bool critical,
|
|
|
|
const char *file, long line);
|
|
|
|
#define writembs(outp, mbs) writembs_check((outp), (mbs), #mbs, true, __FILE__, __LINE__)
|
|
|
|
#define writembs_nofail(outp, mbs) writembs_check((outp), (mbs), #mbs, false, __FILE__, __LINE__)
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2019-11-19 00:54:36 +00:00
|
|
|
rgb_color_t parse_color(const env_var_t &var, bool is_background);
|
2005-09-20 13:26:39 +00:00
|
|
|
|
2016-05-02 05:29:21 +00:00
|
|
|
/// Sets what colors are supported.
|
|
|
|
enum { color_support_term256 = 1 << 0, color_support_term24bit = 1 << 1 };
|
2021-09-22 00:57:25 +00:00
|
|
|
using color_support_t = unsigned int;
|
2014-09-19 22:37:31 +00:00
|
|
|
color_support_t output_get_color_support();
|
2019-11-19 00:54:36 +00:00
|
|
|
void output_set_color_support(color_support_t val);
|
2012-02-13 17:52:17 +00:00
|
|
|
|
2019-11-19 00:54:36 +00:00
|
|
|
rgb_color_t best_color(const std::vector<rgb_color_t> &candidates, color_support_t support);
|
2014-11-10 00:42:35 +00:00
|
|
|
|
2013-02-14 23:50:24 +00:00
|
|
|
unsigned char index_for_color(rgb_color_t c);
|
|
|
|
|
2005-10-04 15:11:39 +00:00
|
|
|
#endif
|