Attachment 'tables-1.1.py'
Download 1 #-*- coding: utf-8 -*-
2 '''
3 MoinMoin - tables parser, Version 1.1, 23.01.2011
4
5 A simple parser that reads key:value pairs from a listing of tables (moin table markup from page or attachment) and captions then formats these for display within an enclosing borderless table. Defaults to single column layout with captions above centred tables. If more key:value pairs are provided additional rows are added to the layout. The user can set options to (1) force double or triple column layout, (2) place captions to the left or right of the tables, (3) set tables left or right justified, (4) give tables a width other than the default of 66% (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.
6
7 The key:value list must be in the form " table_page/file:: caption" with a leading space (as moin dictionary entries). All non-conforming lines are ignored without warning.
8
9 The key is a page or attachment name. The parser checks for a page first. Processing instructions are permitted at the head of a page but not in an attachment. Comment lines beginning with `##` are permitted within the table. Otherwise, both the page and file must contain a markup for a single table only.
10
11 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.
12
13 Keywords
14
15 c-2 : forces double column layout, if the requested with is 50% or less. Overides c-3, if both options given, as double column takes precedence.
16
17 c-3 : forces triple column layout, if the requested width is 33% or less. Default is one columns, so no c-1 is not needed.
18
19 c-left : caption to left of image. Overides c-none or c-right, if more than one option given, as left takes precedence.
20
21 c-none : ignore captions.
22
23 c-right : caption to right of image. Overides c-none, if both options given, as right takes precedence over none.
24
25 t-left : tables set left. Overides f-right, if both options given, as left takes precedence.
26
27 t-right : tables set right.
28
29 nnn% : requested width of table (including caption if set to left or right), where n is digits from 25 to 100. If outside this range, the default 50% 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).
30
31 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.
32
33 # : 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 tables and remaining captions will increase in size proportionally. Values set by the nnn% argument therefore become nominal only.
34
35 Defaults are caption and table centred, width 66%, one column (so no options for these are needed).
36
37 Keywords can appear in any order, other than as indicated above for conflicting choices.
38
39 Dependency
40
41 The parser requires a macro RenderText.py to be installed in plugin/macros. This macro un-glues the inserted table's lines, renders the markup text and inserts it in the enclosing table cell.
42
43 {{{
44 from MoinMoin import wikiutil
45 from MoinMoin.parser.text_moin_wiki import Parser as wiki
46 def execute(macro, text):
47 text = (text.replace('&&', '\n')).replace(')+>>', ')>>')
48 return wikiutil.renderText(macro.request, wiki, text)
49 }}}
50
51 Usage
52
53 {{{#!tables [c-2|c-3 ][c-left|c-none|c-right ][t-left|t-right ][nnn% ][markup ]
54 table_page/file:: [#]caption
55 table_page/file:: [#]caption
56 ...
57 }}}
58
59 History
60
61 Version 1.1 - 23.01.2011: escaped macro syntax; revised RenderText macro accordingly
62 Version 1.0 - 22.01.2011: initial version
63
64 Developed with inspiration and code from:
65
66 figures.py parser - @copyright: 2011 Ian Riley
67 keyval.py parser - @copyright: 2006 by Matt Cooper <macooper@vt.edu>
68 sort.py parser - @copyright: 2005 ReimarBauer
69
70 Copyright
71
72 @copyright: 2011 Ian Riley <ian.riley@adelaide.edu.au>
73
74 License
75
76 GNU GPL, see COPYING for details.
77 '''
78
79 Dependencies = []
80
81 from MoinMoin.action import AttachFile
82 from MoinMoin.Page import Page
83 from MoinMoin.parser import text_moin_wiki as wiki
84 from MoinMoin import wikiutil
85
86 class Parser:
87 parsername = 'tables'
88
89 def __init__(self, raw, request, **kw):
90 self.raw = raw
91 self.request = request
92 self.form = request.form
93 self._ = request.getText
94 self.args = kw.get('format_args', '')
95
96 def format(self, formatter):
97 empty = u''
98 eol = u'\n'
99 rtn = u'\r'
100 glue = u'&&'
101 sep = u':: '
102 sep_dir = u'/'
103 space = u' '
104 monospace = u'` `'
105 macro_end = u')>>'
106 macro_esc = u')+>>'
107 ignore = u'#'
108 comment = u'##'
109 percent = u'%'
110 brk = u'<<BR>>'
111 parsing = (u"{{{", u"}}}", )
112 tab = u'||<tablestyle="width:100%; " '
113 cell = u'||<style="border:none; width:%(width)s%%; ">'
114 cella = u'||<%(align)s style="border:none; width:%(width)s%%; ">'
115 padcell = u'||<style="border:none; width:%(pad)s%%; ">'
116 celli = u'||'
117 cellf = u'||\n'
118 tbl = u'<<RenderText(%(tbl)s)>>'
119 cap = u'%(cap)s'
120 capc = u'<<RenderText(||<tablestyle="width:100%%; " style="border:none; ">%(cap)s ||)>>'
121 tfile = u'.txt'
122 warning_msg = u'||<style="color: #ff0000; font-weight: bold;">%s: %s||'
123 options, tables, rows, results = ([], [], [], [])
124 tbls, ignored, layout, padding = (0, 0, 0, 0, )
125 left, centre, right, void = (0, 1, 2, 3, )
126 top, mid, bot, ljust, cjust, rjust = (u'^', u'', u'v', u'(', u':', u')', )
127 max_cols = 1
128 std_width = 66
129 min_widths = (0, 25, 25, 25, )
130 max_widths = (0, 100, 50, 33, )
131 aligns = (mid, top, mid, mid, ) # TODO consider making this a selectable option
132 justs = (rjust, ljust, ljust, ljust, ) # leave centred (ie above) caption left justified
133 # for captions (left, centre, right, none,)
134 layouts = (cella+cap+cell+tbl, cell+capc+tbl, cell+tbl+cella+cap, cell+tbl, )
135
136 #set defaults which might get changed
137 cap_pos = centre
138 tbl_pos = centre
139 req_width = std_width
140
141 # set options
142 options = self.args.split(space)
143 if "c-none" in options:
144 cap_pos = void
145 if "c-right" in options: # do second as right takes precedence
146 cap_pos = right
147 if "c-left" in options: # do last as left takes precedence
148 cap_pos = left
149 if "c-3" in options:
150 max_cols = 3
151 if "c-2" in options: # do second as single column takes precedence
152 max_cols = 2
153 if "t-right" in options:
154 tbl_pos = right
155 if "t-left" in options: # do second as left takes precedence
156 tbl_pos = left
157 markup = "markup" in options
158 for option in options:
159 if option.endswith(percent):
160 try:
161 req_width = int(option.rstrip(percent)) # will check this value later
162 except:
163 pass
164 break # do not look for any more width options
165 del options
166 # check requested width in range or set default
167 if (req_width > max_widths[1]) or (req_width < min_widths[3]):
168 req_width = std_width
169 # dynamically set maximum columns depending on requested width
170 while (max_cols > 1) and (req_width > max_widths[max_cols]):
171 max_cols = max_cols - 1
172 # calculate widths for captions (left, centre, right)
173 widths = (int(req_width / 2), req_width, int(req_width / 2), req_width, )
174
175 def get_attachment(request, page_name, source):
176 if AttachFile.exists(request, page_name, source):
177 file_name = Page(request, page_name).getPagePath("attachments")
178 file_name = file_name+sep_dir+source
179 f = open(file_name,'r')
180 text = f.read()
181 f.close()
182 text = text.replace(rtn,empty)
183 if text == empty:
184 text = warning_msg % (source, 'attachment empty', )
185 else:
186 text = warning_msg % (source, 'not found', )
187 return text
188
189 def get_content(source):
190 content = []
191 page_name = self.request.page.page_name
192 abs_source = wikiutil.AbsPageName(page_name, source)
193 # is there a page?
194 if not Page(self.request, abs_source).isStandardPage(abs_source):
195 #try for an attachment instead
196 text = get_attachment(self.request, page_name, (source + tfile))
197 else:
198 if self.request.user.may.read(abs_source):
199 content = Page(self.request, abs_source)
200 text = content.getPageText()
201 if (content is None) or (text == empty):
202 text = warning_msg % (source, 'page empty', )
203 else:
204 text = warning_msg % (source, 'access denied', )
205 # crudely check that text conforms to table markup
206 text = text.strip()
207 lines = text.split(eol)
208 text = empty
209 for line in lines:
210 if line.startswith(comment):
211 continue
212 if (not line.startswith(celli)) or (not line.endswith(celli)):
213 text = warning_msg % (source, 'invalid table markup', )
214 break
215 text = text + line.strip() + glue
216 text = text.rstrip(glue)
217 text = text.replace(macro_end, macro_esc)
218 return text
219
220 # process the lines of text provided
221 lines = self.raw.split(eol)
222 for line in lines:
223 if line:
224 # handle table lines
225 if (line[0] == space) and (sep in line):
226 key, val = (line.lstrip(space)).split(sep)
227 val = val.replace(macro_end, macro_esc)
228 tables.append((key, val, get_content(key)))
229 if val.startswith(ignore):
230 ignored = ignored + 1
231 del lines
232
233 # construct a list of row column numbers
234 tbls = len(tables)
235 while tbls >= max_cols:
236 rows.append(max_cols)
237 tbls = tbls - max_cols
238 if tbls > 0:
239 rows.append(tbls)
240
241 # construct output for columns in rows
242 def calc_padding():
243 table_width = int(100 / max_cols) * max_cols
244 item_width = widths[cap_pos]
245 if (cap_pos in (left, right, )): # offset caption so double for image + caption
246 item_width = item_width * 2
247 if tbl_pos == centre: # centred so needs smaller interspersed padding
248 padding = int((table_width - (item_width * cols)) / (cols + 1))
249 else: # not centred so just one wider padding
250 padding = int(table_width - (item_width * cols))
251 return padding
252
253 def add_padding(seq):
254 yes_pad = False
255 if padding == 0:
256 return ''
257 if seq == 0:
258 yes_pad = (tbl_pos == right)
259 elif seq == 1:
260 yes_pad = (tbl_pos == centre) and \
261 ((cap_pos in (centre, void, )) or \
262 (first_col) or \
263 ((cap_pos == left) and (layout != void)))
264 elif seq == 2:
265 yes_pad = (tbl_pos == centre) and \
266 ((cols == 1) or \
267 ((cap_pos == right) and (layout != void)))
268 elif seq == 3:
269 yes_pad = (tbl_pos == left)
270
271 if yes_pad:
272 return (padcell % {'pad': padding})
273 else:
274 return ''
275
276 for cols in rows:
277 first_col = True
278 pending = ''
279 padding = calc_padding()
280 pending = pending + add_padding(0)
281 while cols > 0:
282 table = tables.pop(0)
283 if table[1].startswith(ignore):
284 layout = void
285 else:
286 layout = cap_pos
287 pending = pending + add_padding(1)
288 pending = pending + \
289 (layouts[layout] % {'align': aligns[cap_pos]+justs[cap_pos], \
290 'width': widths[cap_pos], \
291 'tbl': table[2], \
292 'cap': table[1]})
293 pending = pending + add_padding(2)
294 cols = cols - 1
295 first_col = False
296 pending = pending + add_padding(3)
297 pending = tab + pending[3:] + cellf # insert tablestyle and close table
298 results.append(pending)
299 del rows, tables
300
301 # escape markup if requested
302 if markup:
303 results.insert(0, parsing[0])
304 results.append(parsing[1])
305
306 wikiizer = wiki.Parser(eol.join(results), self.request)
307 wikiizer.format(formatter, inhibit_p=True)
You are not allowed to attach a file to this page.