-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathtable.py
854 lines (805 loc) · 27.4 KB
/
table.py
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
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
//
// Copyright (c) 2011 Ronaldo Carpio
//
// Permission to use, copy, modify, distribute and sell this software
// and its documentation for any purpose is hereby granted without fee,
// provided that the above copyright notice appear in all copies and
// that both that copyright notice and this permission notice appear
// in supporting documentation. The authors make no representations
// about the suitability of this software for any purpose.
// It is provided "as is" without express or implied warranty.
//
"""
Provides a simple table class. A SimpleTable is essentially
a list of lists plus some formatting functionality.
Dependencies: the Python 2.5+ standard library.
Installation: just copy this module into your working directory (or
anywhere in your pythonpath).
Basic use::
mydata = [[11,12],[21,22]] # data MUST be 2-dimensional
myheaders = [ "Column 1", "Column 2" ]
mystubs = [ "Row 1", "Row 2" ]
tbl = SimpleTable(mydata, myheaders, mystubs, title="Title")
print( tbl )
print( tbl.as_csv() )
A SimpleTable is inherently (but not rigidly) rectangular.
You should create it from a *rectangular* (2d!) iterable of data.
Each item in your rectangular iterable will become the data
of a single Cell. In principle, items can be any object,
not just numbers and strings. However, default conversion
during table production is by simple string interpolation.
(So you cannot have a tuple as a data item *and* rely on
the default conversion.)
A SimpleTable allows only one column (the first) of stubs at
initilization, concatenation of tables allows you to produce tables
with interior stubs. (You can also assign the datatype 'stub' to the
cells in any column, or use ``insert_stubs``.) A SimpleTable can be
concatenated with another SimpleTable or extended by another
SimpleTable. ::
table1.extend_right(table2)
table1.extend(table2)
A SimpleTable can be initialized with `datatypes`: a list of ints that
provide indexes into `data_fmts` and `data_aligns`. Each data cell is
assigned a datatype, which will control formatting. If you do not
specify the `datatypes` list, it will be set to ``range(ncols)`` where
`ncols` is the number of columns in the data. (I.e., cells in a
column have their own datatype.) This means that you can just specify
`data_fmts` without bothering to provide a `datatypes` list. If
``len(datatypes)<ncols`` then datatype assignment will cycle across a
row. E.g., if you provide 10 columns of data with ``datatypes=[0,1]``
then you will have 5 columns of datatype 0 and 5 columns of datatype
1, alternating. Correspoding to this specification, you should provide
a list of two ``data_fmts`` and a list of two ``data_aligns``.
Cells can be assigned labels as their `datatype` attribute.
You can then provide a format for that lable.
Us the SimpleTable's `label_cells` method to do this. ::
def mylabeller(cell):
if cell.data is np.nan:
return 'missing'
mytable.label_cells(mylabeller)
print(mytable.as_text(missing='-'))
Potential problems for Python 3
-------------------------------
- Calls ``next`` instead of ``__next__``.
The 2to3 tool should handle that no problem.
(We will switch to the `next` function if 2.5 support is ever dropped.)
- from __future__ import division, with_statement
- from itertools import izip as zip
- Let me know if you find other problems.
:contact: alan dot isaac at gmail dot com
:requires: Python 2.5.1+
:note: current version
:note: HTML data format currently specifies tags
:todo: support a bit more of http://www.oasis-open.org/specs/tr9503.html
:todo: add labels2formatters method, that associates a cell formatter with a datatype
:todo: add colspan support to Cell
:since: 2008-12-21
:change: 2010-05-02 eliminate newlines that came before and after table
:change: 2010-05-06 add `label_cells` to `SimpleTable`
"""
from __future__ import division, with_statement
import logging
try: #plan for Python 3
from itertools import izip_longest, izip as zip
pass # accommodate 2to3 tool
except ImportError:
pass
from itertools import cycle
from collections import defaultdict
import csv
def csv2st(csvfile, headers=False, stubs=False, title=None):
"""Return SimpleTable instance,
created from the data in `csvfile`,
which is in comma separated values format.
The first row may contain headers: set headers=True.
The first column may contain stubs: set stubs=True.
Can also supply headers and stubs as tuples of strings.
"""
rows = list()
with open(csvfile,'r') as fh:
reader = csv.reader(fh)
if headers is True:
try:
headers = next(reader)
except AttributeError: #must be Python 2.5 or earlier
headers = reader.next()
elif headers is False:
headers=()
if stubs is True:
stubs = list()
for row in reader:
if row:
stubs.append(row[0])
rows.append(row[1:])
else: #no stubs, or stubs provided
for row in reader:
if row:
rows.append(row)
if stubs is False:
stubs = ()
nrows = len(rows)
ncols = len(rows[0])
if any(len(row)!=ncols for row in rows):
raise IOError('All rows of CSV file must have same length.')
return SimpleTable(data=rows, headers=headers, stubs=stubs)
class SimpleTable(list):
"""Produce a simple ASCII, CSV, HTML, or LaTeX table from a
*rectangular* (2d!) array of data, not necessarily numerical.
Directly supports at most one header row,
which should be the length of data[0].
Directly supports at most one stubs column,
which must be the length of data.
(But see `insert_stubs` method.)
See globals `default_txt_fmt`, `default_csv_fmt`, `default_html_fmt`,
and `default_latex_fmt` for formatting options.
Sample uses::
mydata = [[11,12],[21,22]] # data MUST be 2-dimensional
myheaders = [ "Column 1", "Column 2" ]
mystubs = [ "Row 1", "Row 2" ]
tbl = text.SimpleTable(mydata, myheaders, mystubs, title="Title")
print( tbl )
print( tbl.as_html() )
# set column specific data formatting
tbl = text.SimpleTable(mydata, myheaders, mystubs,
data_fmts=["%3.2f","%d"])
print( tbl.as_csv() )
with open('c:/temp/temp.tex','w') as fh:
fh.write( tbl.as_latex_tabular() )
"""
def __init__(self, data, headers=None, stubs=None, title='',
datatypes=None,
csv_fmt=None, txt_fmt=None, ltx_fmt=None, html_fmt=None,
celltype= None, rowtype=None,
**fmt_dict):
"""
Parameters
----------
data : list of lists or 2d array (not matrix!)
R rows by K columns of table elements
headers : list (or tuple) of str
sequence of K strings, one per header
stubs : list (or tuple) of str
sequence of R strings, one per stub
title : string
title of the table
datatypes : list of int
indexes to `data_fmts`
txt_fmt : dict
text formatting options
ltx_fmt : dict
latex formatting options
csv_fmt : dict
csv formatting options
hmtl_fmt : dict
hmtl formatting options
celltype : class
the cell class for the table (default: Cell)
rowtype : class
the row class for the table (default: Row)
fmt_dict : dict
general formatting options
"""
#self._raw_data = data
self.title = title
self._datatypes = datatypes or range(len(data[0]))
#start with default formatting
self._txt_fmt = default_txt_fmt.copy()
self._latex_fmt = default_latex_fmt.copy()
self._csv_fmt = default_csv_fmt.copy()
self._html_fmt = default_html_fmt.copy()
#substitute any general user specified formatting
#:note: these will be overridden by output specific arguments
self._csv_fmt.update(fmt_dict)
self._txt_fmt.update(fmt_dict)
self._latex_fmt.update(fmt_dict)
self._html_fmt.update(fmt_dict)
#substitute any output-type specific formatting
self._csv_fmt.update(csv_fmt or dict())
self._txt_fmt.update(txt_fmt or dict())
self._latex_fmt.update(ltx_fmt or dict())
self._html_fmt.update(html_fmt or dict())
self.output_formats = dict(
txt=self._txt_fmt,
csv=self._csv_fmt,
html=self._html_fmt,
latex=self._latex_fmt
)
self._Cell = celltype or Cell
self._Row = rowtype or Row
rows = self._data2rows(data) # a list of Row instances
list.__init__(self, rows)
self._add_headers_stubs(headers, stubs)
def __str__(self):
return self.as_text()
def _add_headers_stubs(self, headers, stubs):
"""Return None. Adds headers and stubs to table,
if these were provided at initialization.
Parameters
----------
headers : list of strings
K strings, where K is number of columns
stubs : list of strings
R strings, where R is number of non-header rows
:note: a header row does not receive a stub!
"""
if headers:
self.insert_header_row(0, headers, dec_below='header_dec_below')
if stubs:
self.insert_stubs(0, stubs)
def insert(self, idx, row, datatype=None):
"""Return None. Insert a row into a table.
"""
if datatype is None:
try:
datatype = row.datatype
except AttributeError:
pass
row = self._Row(row, datatype=datatype, table=self)
list.insert(self, idx, row)
def insert_header_row(self, rownum, headers, dec_below='header_dec_below'):
"""Return None. Insert a row of headers,
where ``headers`` is a sequence of strings.
(The strings may contain newlines, to indicated multiline headers.)
"""
header_rows = [header.split('\n') for header in headers]
#rows in reverse order
rows = list(izip_longest(*header_rows, fillvalue=''))
rows.reverse()
for i, row in enumerate(rows):
self.insert(rownum, row, datatype='header')
if i == 0:
self[rownum].dec_below = dec_below
else:
self[rownum].dec_below = None
def insert_stubs(self, loc, stubs):
"""Return None. Insert column of stubs at column `loc`.
If there is a header row, it gets an empty cell.
So ``len(stubs)`` should equal the number of non-header rows.
"""
_Cell = self._Cell
stubs = iter(stubs)
for row in self:
if row.datatype == 'header':
empty_cell = _Cell('', datatype='empty')
row.insert(loc, empty_cell)
else:
try:
row.insert_stub(loc, next(stubs))
except AttributeError: #Python 2.5 or earlier
row.insert_stub(loc, stubs.next())
except StopIteration:
raise ValueError('length of stubs must match table length')
def _data2rows(self, raw_data):
"""Return list of Row,
the raw data as rows of cells.
"""
logging.debug('Enter SimpleTable.data2rows.')
_Cell = self._Cell
_Row = self._Row
rows = []
for datarow in raw_data:
dtypes = cycle(self._datatypes)
newrow = _Row(datarow, datatype='data', table=self, celltype=_Cell)
for cell in newrow:
try:
cell.datatype = next(dtypes)
except AttributeError: #Python 2.5 or earlier
cell.datatype = dtypes.next()
cell.row = newrow #a cell knows its row
rows.append(newrow)
logging.debug('Exit SimpleTable.data2rows.')
return rows
def pad(self, s, width, align):
"""DEPRECATED: just use the pad function"""
return pad(s, width, align)
def get_colwidths(self, output_format, **fmt_dict):
output_format = get_output_format(output_format)
fmt = self.output_formats[output_format].copy()
fmt.update(fmt_dict)
ncols = max(len(row) for row in self)
request = fmt.get('colwidths')
if request is 0: #assume no extra space desired (e.g, CSV)
return [0] * ncols
elif request is None: #assume no extra space desired (e.g, CSV)
request = [0] * ncols
elif isinstance(request, int):
request = [request] * ncols
elif len(request) < ncols:
request = [request[i%len(request)] for i in range(ncols)]
min_widths = []
for col in zip(*self):
maxwidth = max(len(c.format(0,output_format,**fmt)) for c in col)
min_widths.append(maxwidth)
result = map(max, min_widths, request)
return result
def _get_fmt(self, output_format, **fmt_dict):
"""Return dict, the formatting options.
"""
output_format = get_output_format(output_format)
#first get the default formatting
try:
fmt = self.output_formats[output_format].copy()
except KeyError:
raise ValueError('Unknown format: %s' % output_format)
#then, add formatting specific to this call
fmt.update(fmt_dict)
return fmt
def as_csv(self, **fmt_dict):
"""Return string, the table in CSV format.
Currently only supports comma separator."""
#fetch the format, which may just be default_csv_format
fmt = self._get_fmt('csv', **fmt_dict)
return self.as_text(**fmt)
def as_text(self, **fmt_dict):
"""Return string, the table as text."""
#fetch the text format, override with fmt_dict
fmt = self._get_fmt('txt', **fmt_dict)
#get rows formatted as strings
formatted_rows = [ row.as_string('text', **fmt) for row in self ]
rowlen = len(formatted_rows[-1]) #don't use header row
#place decoration above the table body, if desired
table_dec_above = fmt.get('table_dec_above','=')
if table_dec_above:
formatted_rows.insert(0, table_dec_above * rowlen)
#next place a title at the very top, if desired
#:note: user can include a newlines at end of title if desired
title = self.title
if title:
title = pad(self.title, rowlen, fmt.get('title_align','c'))
formatted_rows.insert(0, title)
#add decoration below the table, if desired
table_dec_below = fmt.get('table_dec_below','-')
if table_dec_below:
formatted_rows.append(table_dec_below * rowlen)
return '\n'.join(formatted_rows)
def as_html(self, **fmt_dict):
"""Return string.
This is the default formatter for HTML tables.
An HTML table formatter must accept as arguments
a table and a format dictionary.
"""
#fetch the text format, override with fmt_dict
fmt = self._get_fmt('html', **fmt_dict)
formatted_rows = ['<table class="simpletable">']
if self.title:
title = '<caption>%s</caption>' % self.title
formatted_rows.append(title)
formatted_rows.extend( row.as_string('html', **fmt) for row in self )
formatted_rows.append('</table>')
return '\n'.join(formatted_rows)
def as_latex_tabular(self, **fmt_dict):
'''Return string, the table as a LaTeX tabular environment.
Note: will equire the booktabs package.'''
#fetch the text format, override with fmt_dict
fmt = self._get_fmt('latex', **fmt_dict)
aligns = self[-1].get_aligns('latex', **fmt)
formatted_rows = [ r'\begin{tabular}{%s}' % aligns ]
table_dec_above = fmt['table_dec_above']
if table_dec_above:
formatted_rows.append(table_dec_above)
formatted_rows.extend(
row.as_string(output_format='latex', **fmt) for row in self )
table_dec_below = fmt['table_dec_below']
if table_dec_below:
formatted_rows.append(table_dec_below)
formatted_rows.append(r'\end{tabular}')
#tabular does not support caption, but make it available for figure environment
if self.title:
title = r'%%\caption{%s}' % self.title
formatted_rows.append(title)
return '\n'.join(formatted_rows)
"""
if fmt_dict['strip_backslash']:
ltx_stubs = [stub.replace('\\',r'$\backslash$') for stub in self.stubs]
ltx_headers = [header.replace('\\',r'$\backslash$') for header in self.headers]
ltx_headers = self.format_headers(fmt_dict, ltx_headers)
else:
ltx_headers = self.format_headers(fmt_dict)
ltx_stubs = self.format_stubs(fmt_dict, ltx_stubs)
"""
def extend_right(self, table):
"""Return None.
Extend each row of `self` with corresponding row of `table`.
Does **not** import formatting from ``table``.
This generally makes sense only if the two tables have
the same number of rows, but that is not enforced.
:note: To extend append a table below, just use `extend`,
which is the ordinary list method. This generally makes sense
only if the two tables have the same number of columns,
but that is not enforced.
"""
for row1, row2 in zip(self, table):
row1.extend(row2)
def label_cells(self, func):
"""Return None. Labels cells based on `func`.
If ``func(cell) is None`` then its datatype is
not changed; otherwise it is set to ``func(cell)``.
"""
for row in self:
for cell in row:
label = func(cell)
if label is not None:
cell.datatype = label
@property
def data(self):
return [row.data for row in self]
#END: class SimpleTable
def pad(s, width, align):
"""Return string padded with spaces,
based on alignment parameter."""
if align == 'l':
s = s.ljust(width)
elif align == 'r':
s = s.rjust(width)
else:
s = s.center(width)
return s
class Row(list):
"""Provides a table row as a list of cells.
A row can belong to a SimpleTable, but does not have to.
"""
def __init__(self, seq, datatype='data', table=None, celltype=None,
dec_below='row_dec_below', **fmt_dict):
"""
Parameters
----------
seq : sequence of data or cells
table : SimpleTable
datatype : str ('data' or 'header')
dec_below : str
(e.g., 'header_dec_below' or 'row_dec_below')
decoration tag, identifies the decoration to go below the row.
(Decoration is repeated as needed for text formats.)
"""
self.datatype = datatype
self.table = table
if celltype is None:
if table is None:
celltype = Cell
else:
celltype = table._Cell
self._Cell = celltype
self._fmt = fmt_dict
self.special_fmts = dict() #special formatting for any output format
self.dec_below = dec_below
list.__init__(self, (celltype(cell,row=self) for cell in seq))
def add_format(self, output_format, **fmt_dict):
"""
Return None. Adds row-instance specific formatting
for the specified output format.
Example: myrow.add_format('txt', row_dec_below='+-')
"""
output_format = get_output_format(output_format)
if output_format not in self.special_fmts:
self.special_fmts[output_format] = dict()
self.special_fmts[output_format].update(fmt_dict)
def insert_stub(self, loc, stub):
"""Return None. Inserts a stub cell
in the row at `loc`.
"""
_Cell = self._Cell
if not isinstance(stub, _Cell):
stub = stub
stub = _Cell(stub, datatype='stub', row=self)
self.insert(loc, stub)
def _get_fmt(self, output_format, **fmt_dict):
"""Return dict, the formatting options.
"""
output_format = get_output_format(output_format)
#first get the default formatting
try:
fmt = default_fmts[output_format].copy()
except KeyError:
raise ValueError('Unknown format: %s' % output_format)
#second get table specific formatting (if possible)
try:
fmt.update(self.table.output_formats[output_format])
except AttributeError:
pass
#finally, add formatting for this row and this call
fmt.update(self._fmt)
fmt.update(fmt_dict)
special_fmt = self.special_fmts.get(output_format, None)
if special_fmt is not None:
fmt.update(special_fmt)
return fmt
def get_aligns(self, output_format, **fmt_dict):
"""Return string, sequence of column alignments.
Ensure comformable data_aligns in `fmt_dict`."""
fmt = self._get_fmt(output_format, **fmt_dict)
return ''.join( cell.alignment(output_format, **fmt) for cell in self )
def as_string(self, output_format='txt', **fmt_dict):
"""Return string: the formatted row.
This is the default formatter for rows.
Override this to get different formatting.
A row formatter must accept as arguments
a row (self) and an output format,
one of ('html', 'txt', 'csv', 'latex').
"""
fmt = self._get_fmt(output_format, **fmt_dict)
#get column widths
try:
colwidths = self.table.get_colwidths(output_format, **fmt)
except AttributeError:
colwidths = fmt.get('colwidths')
if colwidths is None:
colwidths = (0,) * len(self)
colsep = fmt['colsep']
row_pre = fmt.get('row_pre','')
row_post = fmt.get('row_post','')
formatted_cells = []
for cell, width in zip(self, colwidths):
content = cell.format(width, output_format=output_format, **fmt)
formatted_cells.append(content)
formatted_row = row_pre + colsep.join(formatted_cells) + row_post
formatted_row = self._decorate_below(formatted_row, output_format, **fmt)
return formatted_row
def _decorate_below(self, row_as_string, output_format, **fmt_dict):
"""This really only makes sense for the text and latex output formats."""
dec_below = fmt_dict.get(self.dec_below, None)
if dec_below is None:
result = row_as_string
else:
output_format = get_output_format(output_format)
if output_format == 'txt':
row0len = len(row_as_string)
dec_len = len (dec_below)
repeat, addon = divmod(row0len, dec_len)
result = row_as_string + "\n" + (dec_below * repeat + dec_below[:addon])
elif output_format == 'latex':
result = row_as_string + "\n" + dec_below
else:
raise ValueError("I can't decorate a %s header."%output_format)
return result
@property
def data(self):
return [cell.data for cell in self]
#END class Row
class Cell(object):
"""Provides a table cell.
A cell can belong to a Row, but does not have to.
"""
def __init__(self, data='', datatype=None, row=None, **fmt_dict):
try: #might have passed a Cell instance
self.data = data.data
self._datatype = data.datatype
self._fmt = data._fmt
except AttributeError: #passed ordinary data
self.data = data
self._datatype = datatype
self._fmt = dict()
self._fmt.update(fmt_dict)
self.row = row
def __str__(self):
return '%s' % self.data
def _get_fmt(self, output_format, **fmt_dict):
"""Return dict, the formatting options.
"""
output_format = get_output_format(output_format)
#first get the default formatting
try:
fmt = default_fmts[output_format].copy()
except KeyError:
raise ValueError('Unknown format: %s' % output_format)
#then get any table specific formtting
try:
fmt.update(self.row.table.output_formats[output_format])
except AttributeError:
pass
#then get any row specific formtting
try:
fmt.update(self.row._fmt)
except AttributeError:
pass
#finally add formatting for this instance and call
fmt.update(self._fmt)
fmt.update(fmt_dict)
return fmt
def alignment(self, output_format, **fmt_dict):
fmt = self._get_fmt(output_format, **fmt_dict)
datatype = self.datatype
data_aligns = fmt.get('data_aligns','c')
if isinstance(datatype, int):
align = data_aligns[datatype % len(data_aligns)]
elif datatype == 'stub':
#still support deprecated `stubs_align`
align = fmt.get('stubs_align') or fmt.get('stub_align','l')
elif datatype in fmt:
label_align = '%s_align' % datatype
align = fmt.get(label_align,'c')
else:
raise ValueError('Unknown cell datatype: %s'%datatype)
return align
def format(self, width, output_format='txt', **fmt_dict):
"""Return string.
This is the default formatter for cells.
Override this to get different formating.
A cell formatter must accept as arguments
a cell (self) and an output format,
one of ('html', 'txt', 'csv', 'latex').
It will generally respond to the datatype,
one of (int, 'header', 'stub').
"""
fmt = self._get_fmt(output_format, **fmt_dict)
data = self.data
datatype = self.datatype
data_fmts = fmt.get('data_fmts')
if data_fmts is None:
#chk allow for deprecated use of data_fmt
data_fmt = fmt.get('data_fmt')
if data_fmt is None:
data_fmt = '%s'
data_fmts = [data_fmt]
data_aligns = fmt.get('data_aligns','c')
if isinstance(datatype, int):
datatype = datatype % len(data_fmts) #constrain to indexes
content = data_fmts[datatype] % data
elif datatype in fmt:
dfmt = fmt.get(datatype)
try:
content = dfmt % data
except TypeError: #dfmt is not a substitution string
content = dfmt
else:
raise ValueError('Unknown cell datatype: %s'%datatype)
align = self.alignment(output_format, **fmt)
return pad(content, width, align)
def get_datatype(self):
if self._datatype == None:
dtype = self.row.datatype
else:
dtype = self._datatype
return dtype
def set_datatype(self, val):
#TODO: add checking
self._datatype = val
datatype = property(get_datatype, set_datatype)
#END class Cell
######### begin: default formats for SimpleTable ##############
""" Some formatting suggestions:
- if you want rows to have no extra spacing,
set colwidths=0 and colsep=''.
(Naturally the columns will not align.)
- if you want rows to have minimal extra spacing,
set colwidths=1. The columns will align.
- to get consistent formatting, you should leave
all field width handling to SimpleTable:
use 0 as the field width in data_fmts. E.g., ::
data_fmts = ["%#0.6g","%#0.6g","%#0.4g","%#0.4g"],
colwidths = 14,
data_aligns = "r",
"""
default_txt_fmt = dict(
fmt = 'txt',
#basic table formatting
table_dec_above='=',
table_dec_below='-',
title_align='c',
#basic row formatting
row_pre = '',
row_post = '',
header_dec_below = '-',
row_dec_below = None,
colwidths = None,
colsep=' ',
data_aligns = "c",
#data formats
#data_fmt = "%s", #deprecated; use data_fmts
data_fmts = ["%s"],
#labeled alignments
#stubs_align = 'l', #deprecated; use data_fmts
stub_align = 'l',
header_align = 'c',
#labeled formats
header_fmt = '%s', #deprecated; just use 'header'
stub_fmt = '%s', #deprecated; just use 'stub'
header='%s',
stub='%s',
empty_cell = '', #deprecated; just use 'empty'
empty = '',
missing='--',
)
default_csv_fmt = dict(
fmt = 'csv',
table_dec_above = '',
table_dec_below = '',
#basic row formatting
row_pre = '',
row_post = '',
header_dec_below = '',
row_dec_below = None,
title_align = '',
data_aligns = "l",
colwidths = None,
colsep = ',',
#data formats
data_fmt = '%s', #deprecated; use data_fmts
data_fmts = ['%s'],
#labeled alignments
#stubs_align = 'l', #deprecated; use data_fmts
stub_align = "l",
header_align = 'c',
#labeled formats
header_fmt = '"%s"', #deprecated; just use 'header'
stub_fmt = '"%s"', #deprecated; just use 'stub'
empty_cell = '', #deprecated; just use 'empty'
header='%s',
stub='%s',
empty = '',
missing='--',
)
default_html_fmt = dict(
#basic table formatting
table_dec_above=None,
table_dec_below=None,
header_dec_below=None,
row_dec_below = None,
title_align='c',
#basic row formatting
colwidths = None,
colsep=' ',
row_pre = '<tr>\n ',
row_post = '\n</tr>',
data_aligns = "c",
#data formats
data_fmts = ['<td>%s</td>'],
data_fmt = "<td>%s</td>", #deprecated; use data_fmts
#labeled alignments
#stubs_align = 'l', #deprecated; use data_fmts
stub_align = 'l',
header_align = 'c',
#labeled formats
header_fmt = '<th>%s</th>', #deprecated; just use `header`
stub_fmt = '<th>%s</th>', #deprecated; just use `stub`
empty_cell = '<td></td>', #deprecated; just use `empty`
header='<th>%s</th>',
stub='<th>%s</th>',
empty = '<td></td>',
missing='<td>--</td>',
)
default_latex_fmt = dict(
fmt = 'ltx',
#basic table formatting
table_dec_above = r'\toprule',
table_dec_below = r'\bottomrule',
header_dec_below = r'\midrule',
row_dec_below = None,
strip_backslash = True, # NotImplemented
#row formatting
row_post = r' \\',
data_aligns = 'c',
colwidths = None,
colsep = ' & ',
#data formats
data_fmts = ['%s'],
data_fmt = '%s', #deprecated; use data_fmts
#labeled alignments
#stubs_align = 'l', #deprecated; use data_fmts
stub_align = 'l',
header_align = 'c',
#labeled formats
header_fmt = r'\textbf{%s}', #deprecated; just use 'header'
stub_fmt = r'\textbf{%s}', #deprecated; just use 'stub'
empty_cell = '', #deprecated; just use 'empty'
header = r'\textbf{%s}',
stub = r'\textbf{%s}',
empty = '',
missing = '--'
)
default_fmts = dict(
html= default_html_fmt,
txt=default_txt_fmt,
latex=default_latex_fmt,
csv=default_csv_fmt
)
output_format_translations = dict(
htm='html',
text='txt',
ltx='latex'
)
def get_output_format(output_format):
if output_format not in ('html', 'txt', 'latex', 'csv'):
try: output_format = output_format_translations[output_format]
except KeyError: raise ValueError('unknown output format %s'%output_format)
return output_format
######### end: default formats ##############