Attachment 'figures-1.5.py'
Download 1 #-*- coding: utf-8 -*-
2 '''
3 MoinMoin - figures parser, Version 1.5, 14.01.2012
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.5 - 14.10.2012: included time dependency to properly handle Ref macro in captions
48 Version 1.4 - 22.01.2011: fixed dynamic column allocation; minor edits
49 Version 1.3 - 15.01.2011: fixed the wiki formatting by setting inhibit_p to True.
50 Version 1.2 - 13.01.2011: allowed for caption to overridden; right justified left positioned captions; other minor fixes
51 Version 1.1 - 11.01.2011: reverted to Python 2.5 string formatting
52 Version 1.0 - 11.01.2011: initial version
53
54 Developed with inspiration and code from:
55
56 keyval.py parser - @copyright: 2006 by Matt Cooper <macooper@vt.edu>
57 sort.py parser - @copyright: 2005 ReimarBauer
58
59 Copyright
60
61 @copyright: 2012 Ian Riley <ian.riley@internode.on.net>
62
63 License
64
65 GNU GPL, see COPYING for details.
66 '''
67
68 from MoinMoin.parser import text_moin_wiki as wiki
69 from MoinMoin import wikiutil
70
71 Dependencies = ["time"] # for captions with <<Ref()>>, which uses <<FootNote()>>, so no caching
72
73 class Parser:
74 parsername = 'figures'
75
76 def __init__(self, raw, request, **kw):
77 self.raw = raw
78 self.request = request
79 self.form = request.form
80 self._ = request.getText
81 self.args = kw.get('format_args', '')
82
83 def format(self, formatter):
84 eol = u'\n'
85 sep = u':: '
86 space = u' '
87 ignore = u'#'
88 percent = u'%'
89 brk = u'<<BR>>'
90 parsing = (u"{{{", u"}}}", )
91 tab = u'||<tablestyle="width:100%; " '
92 cell = u'||<%(align)s style="border:none; width:%(width)s%%; ">'
93 padcell = u'||<style="border:none; width:%(pad)s%%; ">'
94 cellf = u'||\n'
95 att = u'{{attachment:%(att)s||width=100%%}}'
96 cap = u'%(cap)s'
97 options, figures, rows, results = ([], [], [], [])
98 figs, ignored, layout, padding = (0, 0, 0, 0, )
99 left, centre, right, void = (0, 1, 2, 3, )
100 top, mid, bot, ljust, cjust, rjust = (u'^', u'', u'v', u'(', u':', u')', )
101 max_cols = 3
102 std_width = 33
103 min_widths = (0, 25, 25, 25, )
104 max_widths = (0, 100, 50, 33, )
105 aligns = (mid, top, mid, mid, ) # TODO consider making this a selectable option
106 justs = (rjust, ljust, ljust, ljust, ) # leave centred (ie underneath) caption left justified
107 # for captions (left, centre, right, none,)
108 layouts = (cell+cap+cell+att, cell+att+brk+cap, cell+att+cell+cap, cell+att, )
109
110 # TODO add a 4 column option min and max of 25%
111
112 #set defaults which might get changed
113 cap_pos = centre
114 fig_pos = centre
115 req_width = std_width
116
117 # set options
118 options = self.args.split(space)
119 if "c-none" in options:
120 cap_pos = void
121 if "c-right" in options: # do second as right takes precedence
122 cap_pos = right
123 if "c-left" in options: # do last as left takes precedence
124 cap_pos = left
125 if "c-2" in options:
126 max_cols = 2
127 if "c-1" in options: # do second as single column takes precedence
128 max_cols = 1
129 if "f-right" in options:
130 fig_pos = right
131 if "f-left" in options: # do second as left takes precedence
132 fig_pos = left
133 markup = "markup" in options
134 for option in options:
135 if option.endswith(percent):
136 try:
137 req_width = int(option.rstrip(percent)) # will check this value later
138 except:
139 pass
140 break # do not look for any more width options
141 del options
142 # check requested width in range or set default
143 if (req_width > max_widths[1]) or (req_width < min_widths[3]):
144 req_width = std_width
145 # dynamically set maximum columns depending on requested width
146 while (max_cols > 1) and (req_width > max_widths[max_cols]):
147 max_cols = max_cols - 1
148 # calculate widths for captions (left, centre, right)
149 widths = (int(req_width / 2), req_width, int(req_width / 2), req_width, )
150
151 # process the lines of text provided
152 lines = self.raw.split(eol)
153 for line in lines:
154 if line:
155 # handle figure lines
156 if (line[0] == space) and (sep in line):
157 key, val = (line.lstrip(space)).split(sep)
158 figures.append((key, val, ))
159 if val.startswith(ignore):
160 ignored = ignored + 1
161 del lines
162
163 # construct a list of row column numbers
164 figs = len(figures)
165 while figs >= max_cols:
166 rows.append(max_cols)
167 figs = figs - max_cols
168 if figs > 0:
169 rows.append(figs)
170
171 # construct output for columns in rows
172 def calc_padding():
173 table_width = int(100 / max_cols) * max_cols
174 item_width = widths[cap_pos]
175 if (cap_pos in (left, right, )): # offset caption so double for image + caption
176 item_width = item_width * 2
177 if fig_pos == centre: # centred so needs smaller interspersed padding
178 padding = int((table_width - (item_width * cols)) / (cols + 1))
179 else: # not centred so wider padding
180 padding = int(table_width - (item_width * cols))
181 return padding
182
183 def add_padding(seq):
184 yes_pad = False
185 if padding == 0:
186 return ''
187 if seq == 0:
188 yes_pad = (fig_pos == right)
189 elif seq == 1:
190 yes_pad = (fig_pos == centre) and \
191 ((cap_pos in (centre, void, )) or \
192 (first_col) or \
193 ((cap_pos == left) and (layout != void)))
194 elif seq == 2:
195 yes_pad = (fig_pos == centre) and \
196 ((cols == 1) or \
197 ((cap_pos == right) and (layout != void)))
198 elif seq == 3:
199 yes_pad = (fig_pos == left)
200
201 if yes_pad:
202 return (padcell % {'pad': padding})
203 else:
204 return ''
205
206 for cols in rows:
207 first_col = True
208 pending = ''
209 padding = calc_padding()
210 pending = pending + add_padding(0)
211 while cols > 0:
212 figure = figures.pop(0)
213 if figure[1].startswith(ignore):
214 layout = void
215 else:
216 layout = cap_pos
217 pending = pending + add_padding(1)
218 pending = pending + \
219 (layouts[layout] % {'align': aligns[cap_pos]+justs[cap_pos], \
220 'width': widths[cap_pos], \
221 'att': figure[0], \
222 'cap': figure[1]})
223 pending = pending + add_padding(2)
224 cols = cols - 1
225 first_col = False
226 pending = pending + add_padding(3)
227 pending = tab + pending[3:] + cellf # insert tablestyle and close table
228 results.append(pending)
229 del rows, figures
230
231 # escape markup if requested
232 if markup:
233 results.insert(0, parsing[0])
234 results.append(parsing[1])
235
236 wikiizer = wiki.Parser(eol.join(results), self.request)
237 wikiizer.format(formatter, inhibit_p=True)
You are not allowed to attach a file to this page.