-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPrettyTable.h
471 lines (416 loc) · 13.6 KB
/
PrettyTable.h
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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
// PrettyTable.h
#ifndef PRETTY_TABLE_H_
#define PRETTY_TABLE_H_
#include <list>
#include <string>
#include <vector>
#include <limits>
#include <iomanip>
#include <sstream>
#include <numeric>
#include <iterator>
#include <algorithm>
#include <functional>
#include "AlphaNum.h"
//#define PRETTY_TABLE_DEBUG 1
#ifdef PRETTY_TABLE_DEBUG
#include <iostream>
#endif
// 1. The function receives a reference to an array as its parameter.
// 2. It returns a reference to a char array as its output.
// 3. The function lacks a definition and is never invoked.
// 4. The reference to the array retains its size information.
// 5. The `sizeof` operator correctly returns the size of the char array.
template <typename T, size_t size>
char (&arraysize_helper(T (&)[size]))[size];
#define ARRAYSIZE(arr) (sizeof(arraysize_helper(arr)))
#define ARRAYEND(arr) ((arr) + ARRAYSIZE(arr))
// A class designed to disable the copy constructor and copy assignment
// operator. By making the copy constructor and assignment operator private,
// any class derived from CNonCopyable becomes non-copyable.
// Usage example:
// class CFooBar : public CNonCopyable {};
class CNonCopyable
{
protected:
CNonCopyable()
{}
~CNonCopyable()
{}
private:
CNonCopyable(const CNonCopyable&);
void operator=(const CNonCopyable&);
};
// Allows iterations on an enum, as ++i
template <typename Enum>
Enum& enum_increment(Enum& value, const Enum& first, const Enum& last)
{ return value = (value == last) ? first : static_cast<Enum>(value + 1); }
// Dummy type is used as default parameter for template CPrettyTable.
struct PrettyTableStubType
{
char buf[3]; // There is no native type with that size.
};
enum ePrettyTableSortOrder
{
ePTSO_Ascending,
ePTSO_Descending
};
// This function will not be called, but compiler needs it.
std::ostream& operator<<(std::ostream& os, const PrettyTableStubType&);
// This template prints data in formatted table. It calculates indentations
// after all fields are known. There is example code:
//
// CPrettyTable<int, const char*, double> tbl("num", "name", "score");
// for (int i = 0; i < 5; ++i)
// tbl.Add(i + 1, "FooBar", 2.5 * (i + 1));
// tbl.SetCaption("FooBar");
// tbl.Dump(std::cerr);
//
// The code prints following line to standard error stream:
// +-----+[ FooBar ]------+
// | num | name | score |
// +-----+--------+-------+
// | 1 | FooBar | 2.5 |
// | 2 | FooBar | 5 |
// | 3 | FooBar | 7.5 |
// | 4 | FooBar | 10 |
// | 5 | FooBar | 12.5 |
// +-----+--------+-------+
template<typename T01=PrettyTableStubType,
typename T02=PrettyTableStubType,
typename T03=PrettyTableStubType,
typename T04=PrettyTableStubType,
typename T05=PrettyTableStubType,
typename T06=PrettyTableStubType,
typename T07=PrettyTableStubType,
typename T08=PrettyTableStubType,
typename T09=PrettyTableStubType,
typename T10=PrettyTableStubType,
typename T11=PrettyTableStubType,
typename T12=PrettyTableStubType,
typename T13=PrettyTableStubType,
typename T14=PrettyTableStubType,
typename T15=PrettyTableStubType,
typename T16=PrettyTableStubType,
typename T17=PrettyTableStubType,
typename T18=PrettyTableStubType,
typename T19=PrettyTableStubType,
typename T20=PrettyTableStubType>
class CPrettyTable : CNonCopyable
{
public:
// Produces table without title.
CPrettyTable() : m_has_header(false) {}
// Gets column titles as parameters.
CPrettyTable(const char* p01,
const char* p02 = NULL,
const char* p03 = NULL,
const char* p04 = NULL,
const char* p05 = NULL,
const char* p06 = NULL,
const char* p07 = NULL,
const char* p08 = NULL,
const char* p09 = NULL,
const char* p10 = NULL,
const char* p11 = NULL,
const char* p12 = NULL,
const char* p13 = NULL,
const char* p14 = NULL,
const char* p15 = NULL,
const char* p16 = NULL,
const char* p17 = NULL,
const char* p18 = NULL,
const char* p19 = NULL,
const char* p20 = NULL) : m_has_header(true)
{
const char* names[] = {
p01, p02, p03, p04, p05, p06, p07, p08, p09, p10,
p11, p12, p13, p14, p15, p16, p17, p18, p19, p20
};
std::vector<Val> row;
row.reserve(ARRAYSIZE(names));
for (const char** name = names;
name != ARRAYEND(names) && NULL != *name;
++name)
row.push_back(std::make_pair(*name, false));
m_db.push_back(row);
}
void SetCaption(const char* s) { m_caption = (NULL != s) ? s : ""; }
void Add(const T01& p01 = PrettyTableStubType(),
const T02& p02 = PrettyTableStubType(),
const T03& p03 = PrettyTableStubType(),
const T04& p04 = PrettyTableStubType(),
const T05& p05 = PrettyTableStubType(),
const T06& p06 = PrettyTableStubType(),
const T07& p07 = PrettyTableStubType(),
const T08& p08 = PrettyTableStubType(),
const T09& p09 = PrettyTableStubType(),
const T10& p10 = PrettyTableStubType(),
const T11& p11 = PrettyTableStubType(),
const T12& p12 = PrettyTableStubType(),
const T13& p13 = PrettyTableStubType(),
const T14& p14 = PrettyTableStubType(),
const T15& p15 = PrettyTableStubType(),
const T16& p16 = PrettyTableStubType(),
const T17& p17 = PrettyTableStubType(),
const T18& p18 = PrettyTableStubType(),
const T19& p19 = PrettyTableStubType(),
const T20& p20 = PrettyTableStubType())
{
std::vector<Val> row;
row.reserve(20);
AddCell(row, p01);
AddCell(row, p02);
AddCell(row, p03);
AddCell(row, p04);
AddCell(row, p05);
AddCell(row, p06);
AddCell(row, p07);
AddCell(row, p08);
AddCell(row, p09);
AddCell(row, p10);
AddCell(row, p11);
AddCell(row, p12);
AddCell(row, p13);
AddCell(row, p14);
AddCell(row, p15);
AddCell(row, p16);
AddCell(row, p17);
AddCell(row, p18);
AddCell(row, p19);
AddCell(row, p20);
m_db.push_back(row);
}
unsigned int Size() const
{
if (IsEmpty())
return 0;
return m_db.size() - 1;
}
bool IsEmpty() const
{
if (m_db.empty())
return true;
if (m_has_header && 1 == m_db.size())
return true;
return false;
}
std::string Get() const
{
std::ostringstream buf;
Dump(buf);
return buf.str();
}
void Clear()
{
if (m_has_header)
m_db.erase(++m_db.begin(), m_db.end());
else
m_db.clear();
}
void Dump(std::ostream& out) const
{
if (m_db.empty())
return;
std::vector<unsigned int> lens;
DB::const_iterator row;
for (row = m_db.begin(); row != m_db.end(); ++row)
{
if (row->size() > lens.size())
lens.insert(lens.end(), row->size() - lens.size(), 0);
std::vector<Val>::const_iterator col;
for (col = row->begin(); col != row->end(); ++col)
{
unsigned int pos = row->size() - std::distance(col, row->end());
if (lens[pos] < col->first.length())
lens[pos] = col->first.length();
}
}
if (lens.empty())
return;
// The first line, PrintDelimiter starts with '\n'.
if (!m_caption.empty())
PrintCaption(lens, out);
else
PrintDelimiter(lens, out);
// Titles, PrintRow starts with a new line, there is '\n'.
PrintRow(*m_db.begin(), lens, out);
// The line after the title.
if (m_has_header)
PrintDelimiter(lens, out);
// Other lines.
row = m_db.begin();
for (++row; row != m_db.end(); ++row)
PrintRow(*row, lens, out);
// The last line.
PrintDelimiter(lens, out);
}
// Sorts table by certain column. First column has 0 col value. There is CPU
// consuming operation, do not use it when CPU productivity is critical.
void Sort(unsigned int col, ePrettyTableSortOrder order = ePTSO_Ascending)
{
// Prevents the title from sorting.
DB title;
if (m_has_header)
title.splice(title.begin(), m_db, m_db.begin());
m_db.sort(RowCmp(col, order));
// Moves the title back.
if (m_has_header)
m_db.splice(m_db.begin(), title, title.begin());
}
private:
class RowCmp
{
public:
RowCmp(unsigned int c, ePrettyTableSortOrder o) : m_col(c), m_order(o) {}
bool operator()(const std::vector<std::pair<std::string, bool> >& lhs,
const std::vector<std::pair<std::string, bool> >& rhs) const
{
if (m_col >= lhs.size() || m_col >= rhs.size())
return true;
// Uses alpha numeric comparator to get more human results.
int res = doj::alphanum_comp(lhs[m_col].first, rhs[m_col].first);
return (ePTSO_Ascending == m_order) ? res < 0 : res >= 0;
}
private:
unsigned int m_col;
ePrettyTableSortOrder m_order;
};
template<typename T>
void AddCell(std::vector<std::pair<std::string, bool> >& row, const T& p)
{
// Distinguishes default parameter and caller's parameter.
if (sizeof(PrettyTableStubType) == sizeof(T))
return;
std::ostringstream buf;
buf << p;
// Each numeric type has is_specialized=true. It will be printed with
// right positioning of the fill characters. Other types, e.g. string,
// will be printed with left positioning.
row.push_back(std::make_pair(buf.str(),
std::numeric_limits<T>::is_specialized));
}
void PrintRow(std::list<std::vector<std::pair<std::string, bool> > >::const_reference row,
const std::vector<unsigned int>& lens,
std::ostream& out) const
{
out << std::endl << '|';
std::vector<Val>::const_iterator col = row.begin();
std::vector<unsigned int>::const_iterator len = lens.begin();
for ( ; col != row.end() && len != lens.end(); ++col, ++len)
out << " "
<< (col->second ? std::right : std::left)
<< std::setw(*len)
<< col->first << " |";
}
void PrintDelimiter(const std::vector<unsigned int>& lens,
std::ostream& out) const
{
out << std::endl << '+';
std::vector<unsigned int>::const_iterator len;
for (len = lens.begin(); len != lens.end(); ++len)
out << std::string(*len + 2, '-') << '+';
}
// Caption is embedded to the first line delimiter as following
// +----+-[ FooBar ]-------+
void PrintCaption(std::vector<unsigned int>& lens,
std::ostream& out) const
{
#ifdef PRETTY_TABLE_DEBUG
std::cerr << "lens[]: ";
std::copy(lens.begin(), lens.end(),
std::ostream_iterator<unsigned int>(std::cerr, " "));
std::cerr << '\n';
#endif
// Counts delimieter size.
const char del[] = { '-', '+', '-' };
unsigned int dsz = ARRAYSIZE(del);
// Modifies caption.
std::ostringstream buf;
buf << "-[ " << m_caption << " ]-";
std::string caption = buf.str();
// Width of the table includes delimiter size for each column and '+' on
// each border.
unsigned int all =
std::accumulate(lens.begin(), lens.end(), 2 + dsz * lens.size());
// 2 is for two '+' by sides
if (caption.length() >= all - 2)
{
out << '+' << caption << '+';
// Expands cell lengths to fit to the caption.
unsigned int gap = caption.length() - all + 2 + 1;
if (gap >= lens.size())
{
unsigned int pad = gap / lens.size();
unsigned int pa0 =
(0 != (gap % lens.size())) ? pad + gap % lens.size() : pad;
lens[0] += pa0;
// Adds the pad to each column.
std::transform(++lens.begin(), lens.end(), ++lens.begin(),
std::bind2nd(std::plus<unsigned int>(), pad));
#ifdef PRETTY_TABLE_DEBUG
std::cerr << '\n'
<< "gap:" << gap
<< ", pad:" << pad
<< ", pa0:" << pa0 << "\n";
#endif
}
else
{
lens[0] += gap;
}
return;
}
// Prepares regular delimiter without the caption.
std::stringstream bu2;
PrintDelimiter(lens, bu2);
// Injects the caption to the prepared string.
std::string line = bu2.str();
unsigned int gap = (all - caption.length()) / 2;
out << std::string(line, 0, gap)
<< caption
<< std::string(line, gap+caption.length());
}
typedef std::pair<std::string, bool> Val;
typedef std::vector<Val> Raw;
typedef std::list<Raw> DB;
DB m_db;
bool m_has_header;
std::string m_caption;
};
#ifdef PRETTY_TABLE_DEBUG
int main()
{
CPrettyTable<int, const char*, double, const char*>
tbl("num", "name", "score", "value");
for (int j = 0; j < 20; ++j)
{
std::stringstream buf;
buf << "foo-" << j << '-' << (char)'a' + j << "-bar-" << 20 - j;
tbl.Add(j, buf.str().c_str(), j * 2.5, "looooooooooooong string");
}
tbl.Sort(1);
std::cerr << "Sort ascending\n" << tbl.Get() << '\n';
tbl.Sort(1, ePTSO_Descending);
std::cerr << "Sort descending\n" << tbl.Get() << '\n';
tbl.Clear();
std::cerr << "Caption\n";
tbl.Add(100, "short", 0, "long");
for (int i = 0; i < 30; ++i)
{
std::stringstream buf;
buf << i << ' ' << std::string(i, 'F');
tbl.SetCaption(buf.str().c_str());
std::cerr << tbl.Get() << '\n';
}
std::cerr << "Resource consumption\n";
for (int k = 0; k < 20000; ++k)
{
tbl.Add(k, std::string(1000, '+').c_str(), 0, "looooooooooooong string");
}
tbl.Clear();
std::cerr << tbl.Get() << "\nDone.\n";
return 0;
}
#endif // PRETTY_TABLE_DEBUG
#endif // PRETTY_TABLE_H_