Attachment 'figures-1.4.py'
Download 1 #-*- coding: utf-8 -*-
2 '''
3 MoinMoin - figures parser, Version 1.4, 22.01.2011
4
5 A simple parser that reads key:value pairs from a listing of figures (attachments) and captions then formats these for display in as tables. Defaults to three column table with captions below centred figures (centred if less than three in the row). If more than three key:value pairs are provided additional rows are added to the table. The user can set options to (1) force a single or double column table, (2) place captions to the left or right of the figures, (3) set figures left or right justified, (4) give figures a width other than the default of 33% (a value from 25 to 100% is permitted), and (5) show only the markup for re-use with modifications, if desired, or for debugging purposes. The number of columns is set dynamically, if specified widths are two wide for 3 columns (ie if 34-50% then two columns forced and if >50% a single column is forced).
6
7 The key:value list must be in the form " figure_file:: caption" with a leading space (as moin dictionary entries). All non-conforming lines are ignored without warning.
8
9 The parser is intended for page section use only. The parser name is followed by a list of keywords separated by spaces to specify any options required.
10
11 Keywords
12
13 c-1 : forces single column only. Overides c-2, if both options given, as single column takes precedence.
14
15 c-2 : forces double column only, if the requested width 50% or less. Default is three columns, so no col-3 is not needed.
16
17 c-left : caption to left of image. Overides c-none or c-right, if more than one option given, as left takes precedence.
18
19 c-none : ignore captions.
20
21 c-right : caption to right of image. Overides c-none, if both options given, as right takes precedence over none.
22
23 f-left : figures set left. Overides f-right, if both options given, as left takes precedence.
24
25 f-right : figures set right.
26
27 nnn% : requested width of figure (including caption if set to left or right), where n is digits from 25 to 100. If outside this range, the default 33% is applied and no warning is given. If two or more width options are given, the first takes precedence and others are ignored (even if the first is invalid and the default widith applied). The value is checked and if 34-50% two columns are forced, and if >50% then a single column is forced.
28
29 markup : shows the markup if you wish to re-use this with modification (assuming the default tabulation does not suit a specific purpose) or for debugging purposes.
30
31 # : inserted at the beginning of the caption overrides caption inclusion and any associated padding. Inclusion of caption text is option and effectively acts as a comment. Overidding captions can result in uneven multiple row layout where rows have different number of images and/or overidden captions. Therefore, this feature in recommended only for single row layouts. Also, caption and padding cells are not generated, so the sum of the cell widths is less than the table width and the figures and remaining captions will increase in size proportionally. Values set by the nnn% argument therefore become nominal only.
32
33 Defaults are caption and figure centred, width 33%, three columns (so no options for these are needed).
34
35 Keywords can appear in any order, other than as indicated above for conflicting choices.
36
37 Usage
38
39 {{{#!figures [c-1|c-2 ][c-left|c-none|c-right ][f-left|f-right ][nnn% ][markup ]
40 figure_file:: [#]caption
41 figure_file:: [#]caption
42 ...
43 }}}
44
45 History
46
47 Version 1.4 - 22.01.2011: fixed dynamic column allocation; minor edits
48 Version 1.3 - 15.01.2011: fixed the wiki formatting by setting inhibit_p to True.
49 Version 1.2 - 13.01.2011: allowed for caption to overridden; right justified left positioned captions; other minor fixes
50 Version 1.1 - 11.01.2011: reverted to Python 2.5 string formatting
51 Version 1.0 - 11.01.2011: initial version
52
53 Developed with inspiration and code from:
54
55 keyval.py parser - @copyright: 2006 by Matt Cooper <macooper@vt.edu>
56 sort.py parser - @copyright: 2005 ReimarBauer
57
58 Copyright
59
60 @copyright: 2011 Ian Riley <ian.riley@adelaide.edu.au>
61
62 License
63
64 GNU GPL, see COPYING for details.
65 '''
66
67 Dependencies = []
68
69 from MoinMoin.parser import text_moin_wiki as wiki
70 from MoinMoin import wikiutil
71
72 class Parser:
73 parsername = 'figures'
74
75 def __init__(self, raw, request, **kw):
76 self.raw = raw
77 self.request = request
78 self.form = request.form
79 self._ = request.getText
80 self.args = kw.get('format_args', '')
81
82 def format(self, formatter):
83 eol = u'\n'
84 sep = u':: '
85 space = u' '
86 ignore = u'#'
87 percent = u'%'
88 brk = u'<<BR>>'
89 parsing = (u"{{{", u"}}}", )
90 tab = u'||<tablestyle="width:100%; " '
91 cell = u'||<%(align)s style="border:none; width:%(width)s%%; ">'
92 padcell = u'||<style="border:none; width:%(pad)s%%; ">'
93 cellf = u'||\n'
94 att = u'{{attachment:%(att)s||width=100%%}}'
95 cap = u'%(cap)s'
96 options, figures, rows, results = ([], [], [], [])
97 figs, ignored, layout, padding = (0, 0, 0, 0, )
98 left, centre, right, void = (0, 1, 2, 3, )
99 top, mid, bot, ljust, cjust, rjust = (u'^', u'', u'v', u'(', u':', u')', )
100 max_cols = 3
101 std_width = 33
102 min_widths = (0, 25, 25, 25, )
103 max_widths = (0, 100, 50, 33, )
104 aligns = (mid, top, mid, mid, ) # TODO consider making this a selectable option
105 justs = (rjust, ljust, ljust, ljust, ) # leave centred (ie underneath) caption left justified
106 # for captions (left, centre, right, none,)
107 layouts = (cell+cap+cell+att, cell+att+brk+cap, cell+att+cell+cap, cell+att, )
108
109 # TODO add a 4 column option min and max of 25%
110
111 #set defaults which might get changed
112 cap_pos = centre
113 fig_pos = centre
114 req_width = std_width
115
116 # set options
117 options = self.args.split(space)
118 if "c-none" in options:
119 cap_pos = void
120 if "c-right" in options: # do second as right takes precedence
121 cap_pos = right
122 if "c-left" in options: # do last as left takes precedence
123 cap_pos = left
124 if "c-2" in options:
125 max_cols = 2
126 if "c-1" in options: # do second as single column takes precedence
127 max_cols = 1
128 if "f-right" in options:
129 fig_pos = right
130 if "f-left" in options: # do second as left takes precedence
131 fig_pos = left
132 markup = "markup" in options
133 for option in options:
134 if option.endswith(percent):
135 try:
136 req_width = int(option.rstrip(percent)) # will check this value later
137 except:
138 pass
139 break # do not look for any more width options
140 del options
141 # check requested width in range or set default
142 if (req_width > max_widths[1]) or (req_width < min_widths[3]):
143 req_width = std_width
144 # dynamically set maximum columns depending on requested width
145 while (max_cols > 1) and (req_width > max_widths[max_cols]):
146 max_cols = max_cols - 1
147 # calculate widths for captions (left, centre, right)
148 widths = (int(req_width / 2), req_width, int(req_width / 2), req_width, )
149
150 # process the lines of text provided
151 lines = self.raw.split(eol)
152 for line in lines:
153 if line:
154 # handle figure lines
155 if (line[0] == space) and (sep in line):
156 key, val = (line.lstrip(space)).split(sep)
157 figures.append((key, val, ))
158 if val.startswith(ignore):
159 ignored = ignored + 1
160 del lines
161
162 # construct a list of row column numbers
163 figs = len(figures)
164 while figs >= max_cols:
165 rows.append(max_cols)
166 figs = figs - max_cols
167 if figs > 0:
168 rows.append(figs)
169
170 # construct output for columns in rows
171 def calc_padding():
172 table_width = int(100 / max_cols) * max_cols
173 item_width = widths[cap_pos]
174 if (cap_pos in (left, right, )): # offset caption so double for image + caption
175 item_width = item_width * 2
176 if fig_pos == centre: # centred so needs smaller interspersed padding
177 padding = int((table_width - (item_width * cols)) / (cols + 1))
178 else: # not centred so just one wider padding
179 padding = int(table_width - (item_width * cols))
180 return padding
181
182 def add_padding(seq):
183 yes_pad = False
184 if padding == 0:
185 return ''
186 if seq == 0:
187 yes_pad = (fig_pos == right)
188 elif seq == 1:
189 yes_pad = (fig_pos == centre) and \
190 ((cap_pos in (centre, void, )) or \
191 (first_col) or \
192 ((cap_pos == left) and (layout != void)))
193 elif seq == 2:
194 yes_pad = (fig_pos == centre) and \
195 ((cols == 1) or \
196 ((cap_pos == right) and (layout != void)))
197 elif seq == 3:
198 yes_pad = (fig_pos == left)
199
200 if yes_pad:
201 return (padcell % {'pad': padding})
202 else:
203 return ''
204
205 for cols in rows:
206 first_col = True
207 pending = ''
208 padding = calc_padding()
209 pending = pending + add_padding(0)
210 while cols > 0:
211 figure = figures.pop(0)
212 if figure[1].startswith(ignore):
213 layout = void
214 else:
215 layout = cap_pos
216 pending = pending + add_padding(1)
217 pending = pending + \
218 (layouts[layout] % {'align': aligns[cap_pos]+justs[cap_pos], \
219 'width': widths[cap_pos], \
220 'att': figure[0], \
221 'cap': figure[1]})
222 pending = pending + add_padding(2)
223 cols = cols - 1
224 first_col = False
225 pending = pending + add_padding(3)
226 pending = tab + pending[3:] + cellf # insert tablestyle and close table
227 results.append(pending)
228 del rows, figures
229
230 # escape markup if requested
231 if markup:
232 results.insert(0, parsing[0])
233 results.append(parsing[1])
234
235 wikiizer = wiki.Parser(eol.join(results), self.request)
236 wikiizer.format(formatter, inhibit_p=True)
You are not allowed to attach a file to this page.