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.