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.