1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
|
#ifndef _FORMAT_H
#define _FORMAT_H
#include "journal.h"
#include "expr.h"
#include "walk.h"
namespace ledger {
DECLARE_EXCEPTION(format_error, std::runtime_error);
class format_t : public noncopyable
{
struct element_t : public noncopyable
{
#define ELEMENT_ALIGN_LEFT 0x01
#define ELEMENT_HIGHLIGHT 0x02
enum kind_t {
STRING,
EXPR,
#if 0
SPACER,
DEPTH_SPACER
#endif
};
kind_t type;
unsigned char flags;
unsigned char min_width;
unsigned char max_width;
string chars;
expr_t expr;
scoped_ptr<struct element_t> next;
element_t()
: type(STRING), flags(false), min_width(0), max_width(0) {
TRACE_CTOR(element_t, "");
}
~element_t() {
TRACE_DTOR(element_t);
}
friend inline void mark_red(std::ostream& out, const element_t * elem) {
out.setf(std::ios::left);
out.width(0);
out << "\e[31m";
if (elem->flags & ELEMENT_ALIGN_LEFT)
out << std::left;
else
out << std::right;
if (elem->min_width > 0)
out.width(elem->min_width);
}
};
string format_string;
scoped_ptr<element_t> elements;
enum elision_style_t {
TRUNCATE_TRAILING,
TRUNCATE_MIDDLE,
TRUNCATE_LEADING,
ABBREVIATE
};
static elision_style_t elision_style;
static int abbrev_length;
static bool ansi_codes;
static bool ansi_invert;
public:
format_t() {
TRACE_CTOR(format_t, "");
}
format_t(const string& _format) {
TRACE_CTOR(format_t, "const string&");
parse(_format);
}
~format_t() {
TRACE_DTOR(format_t);
}
void parse(const string& _format) {
elements.reset(parse_elements(_format));
format_string = _format;
}
void format(std::ostream& out, scope_t& scope) const;
private:
static element_t * parse_elements(const string& fmt);
static string truncate(const string& str, unsigned int width,
const bool is_account = false);
};
} // namespace ledger
#endif // _FORMAT_H
|