Attachment 'Ref-2.6.py'

Download

   1 # -*- coding: iso-8859-1 -*-
   2 """
   3 MoinMoin - Ref macro, Version 2.6, 17.11.2012
   4 
   5 A macro to cite references from a dictionary page (a ' key:: value' listing).
   6 
   7 Searches one or more dictionary pages to match key (Argument 1, with or without 'ref=', quoted string). Matches are included in the page as (1) bookmarks, (2) search key text followed by a footnote link, (3) search key text as the link to the footnote, (4) search key text with found reference in a popup text box on mouseover, (5) found reference text only and (6) search key text only. 
   8 
   9 The dictionary page is set the RefDict by default, or can be set for (a) the whole page by #pragma ref-dict, or (b) for individual calls by Argument 3.
  10 
  11 For options (2) and (3), search key text will have 'et al' set to Italics (which can be switched off for the whole page by setting #pragma ref-ital off). Also, the year (last word of the key assumed to be a year) can be surrounded in brackets. By default these are just ' (year)' but can be set by #pragma ref-year-bkt to give, for example ', (^)' (where ^ indicates the insertion point for year), if the desired format in the in-text citation differs from the key text in the dictionary.
  12 
  13 Exclamation marks (as WikiName escapes) can be included in the search key, they are removed for matching in dictpage(s) and restored for display under options (2) and (3).
  14 
  15 Additionally superscripting of the numerical link in options (2) and (3) can be switched off and default square brackets are used to distinguish the link from the text. The bracketing can customised using #pragma ref-link-bkt as described above for bracketing the year.
  16 
  17 Option (4) can be used to show reference text in context as a popup tip. This option can also be used for other information, such as definitions of words that might be unfamilar to the reading, with the definitiions stored in a dictionary page. 
  18 
  19 Option (5) is used to create reference (or further reading) list at any point in the the page, without hyperlinks to or from the citation. Or, just to put the reference into the text perhaps in brackets.
  20 
  21 Option (6) simply inserts the search key text into the page. If reference text is inserted by Option (5) on the same or separate page, inserting references into the text using this macro allows them to be found by searching for the macro. Additionally, if this citation type is set at the page level (by #pragma), it can be temporarily reset to make a footnote list on the page for proof reading purposes.
  22 
  23 These options can be selected for each instance (Argument 2, with or without 'cite=', set to bkt, bktfn, bktlk, fn, key, keyfn, keylk, tip or val).
  24 
  25 Case of the inserted key text can be overridden for options (2), (3) and (5) (`Argument 4`, with or without ` case = `, set to ` lower, title ` or  ` upper`).
  26 
  27 If 0 or >1 match is found, or dictpage is not found, a warning is inserted into the page.
  28 
  29 <<Ref()>>
  30 
  31 Inserts '[citation needed]' to act as a reminder and/or placeholder.
  32 
  33 <<Ref([ref=]"key"[,[cite=]bkt|bktfn|bktlk|fn|key|keyfn|keylk|tip|val)] \
  34                  [,[dict=]["^]dictpage["]] \
  35                  [,[case=]lower|title|upper])>>
  36 
  37 Argument labels (ref=, cite=, dict= and case=) are optional. If unlabelled, aruguments must appear in oder. Labelled arguments can appear in any order, but only after any unlablled arguments. A value for ref is required, all others are optional.
  38 
  39 ref="key": search key to be matched in dictpage(s), set in single or double quotes. Setting ref="" is equivalent to <<Ref()>> (see above).
  40 
  41 cite=bkt: insert only the key (reference code with the year bracketed) into the page. No link or footnote is created.
  42 
  43 cite=bktfn: insert a footnote into the page as the search key (reference code with the year bracketed) followed by a numerical link to the found value (reference text) as a footnote.
  44 
  45 cite=bktlk: insert a footnote into the page as the search key (reference code with the year bracketed) as the link to the found value (reference text) as a footnote.
  46 
  47 cite=fn: insert a footnote into the page as a numerical link only with the found value (reference text) as the footnote.
  48 
  49 cite=key: insert only the key (reference code) into the page. No link or footnote is created.
  50 
  51 cite=keylk: insert a footnote into the page as the search key (reference code) as the link to the found value (reference text) as the footnote.
  52 
  53 cite=keyfn: insert a footnote into the page as the search key (reference code) followed by a numerical link to the found value (reference text) as the footnote (default).
  54 
  55 cite=tip: insert key (reference code) and provide found value (reference text) as a popup tip. Tips can also be turned on for all references (other than cite=val) on the page using #pragma ref-tips on. 
  56 
  57 cite=val: insert only the found value (reference text) into the page. No link or footnote is generated.
  58 
  59 dictpage(s): dictionary page name. If starting with ^, it is processed as a regex of dictpages to search (case sensitive). If containing any spaces or non-alphanumeric characters, it must be as a quoted string.
  60 
  61 case=lower|title|upper: convert displayed text to lower, upper or title case, respectively (not for cite=val).
  62 
  63 #pragma ref-cite citetype: set the default cite type for the page.
  64 
  65 #pragma ref-debug wiki: shows the wiki markup without processing it.
  66 
  67 #pragma ref-debug html: shows the markup as html.
  68 
  69 #pragma ref-dict dictpage(s): dictionary to be used by default for that page, replacing the wiki-wide default of RefDict. If starting with ^ (not as a quoted string), it is processed as a regex of dictpages to search (case sensitive).
  70 
  71 #pragma ref-ital off: stops 'et al' in key being converted to italics.
  72 
  73 #pragma ref-link-bkt string: to distinguish the link when ref-super=off, sets string to bracket footnote forward link number for cite=fn, keyfn or bktfn, replacing the default '[^]'; where ^ indicates the insertion point of the link (only one ^ can appear in the string). Set in single or double quotes, or backslash pair.
  74 
  75 #pragma ref-super off: turns off superscripting of numerical forward link to the footnote.
  76 
  77 #pragma ref-tips on: turns on popup tips for all citation tips other than cite=val.
  78 
  79 #pragma ref-year-bkt string: string to bracket year when cite=bktfn, bktlk or bkt, replacing the default ' (^)'; where ^ indicates the insertion point of the year (only one ^ can appear in the string). Set in single or double quotes, or backslash pair.
  80 
  81 @copyright: 2010 Ian Riley <ian.riley@adelaide.edu.au>
  82 @licence: GNU GPL, see COPYING for details.
  83 
  84 History
  85 
  86 Version 2.6 - 17.11.2012: added option to overide case of inserted text, changed to wikiutil to for parsing parameters, improved error messages including message when a dictpage is not found..
  87 
  88 Version 2.5 - 20.10.2012: added option to display found text as popup tips.
  89 
  90 Version 2.4 - 10.10.2012: forced all search strings to be escaped for regex specials, to allow for example the inclusion of a '?' in the date of the reference key.
  91 
  92 Version 2.3.- 09.05.2011: minor bug fix
  93 
  94 Version 2.2 - 12.01.2011: streamlined the unwrap function.
  95 
  96 Version 2.1 - 27.12.2010: some html security improvements.
  97 
  98 Version 2.0 - 25.12.2010: dict specification optional (for cleaner markup), added key linking and display, added additional pragma to set cite type for the whole page (again for cleaner markup), to turn off superscripting and to add alternative demarkation of numbered links and to change year demarkation text. Macro call without arguments to raise a citation needed reminder.
  99 
 100 Version 1.0 - 19.12.2010: initial version.
 101 
 102 Developed from:
 103 
 104 (1) MoinMoin - Include macro
 105 
 106 @copyright: 2000-2004 Juergen Hermann <jh@web.de>,
 107             2000-2001 Richard Jones <richard@bizarsoftware.com.au>
 108 @license: GNU GPL, see COPYING for details.
 109 
 110 (2) http://moinmo.in/FeatureRequests/IncludeMacroInline using code of Junyx 2008-03-24 20:36:25
 111 
 112 """
 113 
 114 Dependencies = ["time"] 
 115 
 116 import re
 117 from MoinMoin import wikiutil
 118 from MoinMoin.Page import Page
 119 from MoinMoin.parser.text_moin_wiki import Parser as WikiParser
 120 
 121 # remove character bracketing 'wrapped', if it is in 'wrappers'
 122 def unwrap(wrapped,wrappers):
 123     if (wrapped[0] in wrappers) and wrapped.endswith(wrapped[0]):
 124         wrapped = wrapped[1:-1]
 125     return wrapped
 126 
 127 _msg = u'<strong class="error">Ref(): %s</strong>'
 128 _msg_bad_arg = _msg % (u'invalid arguments {%s}')
 129 _msg_bad_pragma = _msg % (u'invalid #pragma {%s}')
 130 _msg_no_dict = _msg % (u'dict={%s} not found')
 131 _msg_recursion = _msg % (u'recursive use of {%s} forbidden')
 132 _msg_match_none = _msg % (u'no match for key={%s} in dict={%s}')
 133 _msg_match_many = _msg % (u'%s matches for key={%s} in dict={%s}')
 134 _msg_bad_insert = _msg % (u'Single insertion point ^ in {#pragma %s %s} obligatory')
 135 _msg_ref_needed = u'[citation needed]'
 136 
 137 _kw_case = [None,'lower','title','upper']
 138 _kw_cite = [None,'fn','keyfn','bktfn','keylk','bktlk','tip','key','val','bkt']
 139 
 140 _df_cite = u'keyfn'
 141 _df_dict = u'RefDict'
 142 
 143 _out_fn = u'<<FootNote(%s)>>'
 144 _out_keyfn = u'%s' + _out_fn
 145 _out_year_bkt = u' (%s)'
 146 _out_esc = u'{{{%s}}}'
 147 _out_link_bkt = u'[%s]'
 148 _out_tip = u'<span style="cursor: help; border-bottom: 1px dashed;" title="%s">%s</span>'
 149 
 150 def execute(macro, args):
 151     request = macro.request
 152     _ = request.getText
 153 
 154     def dec_page():
 155         # decrease or remove pagelist marker
 156         if this_page._macroRef_pagelist[inc_name] > 1:
 157             this_page._macroRef_pagelist[inc_name] = this_page._macroRef_pagelist[inc_name] - 1
 158         else:
 159             del this_page._macroRef_pagelist[inc_name]
 160 
 161     # return immediately if getting links for the current page
 162     if request.mode_getpagelinks:
 163         return ('')
 164 
 165     # parse and check arguments
 166     try:
 167         args_parser = wikiutil.ParameterParser('%(ref)s%(cite)s%(dict)s%(case)s')
 168         (count,args_dict) = args_parser.parse_parameters(args)
 169     except:
 170         return (_msg_bad_arg % ('cannot parse: ' + args))
 171 
 172     # flag reference needed
 173     text = args_dict['ref']
 174     if (text == '') or (text == None):
 175         return (_msg_ref_needed)
 176 
 177     # get cite or apply page default
 178     cite = args_dict['cite']
 179     if not cite in _kw_cite:
 180         return (_msg_bad_arg % ('invalid cite: ' + cite))
 181     if cite == None: 
 182         cite = macro.request.getPragma('ref-cite')
 183         if not cite in _kw_cite:
 184             return (_msg_bad_pragma % ('invalid #pragma ref-cite: ' + cite))
 185         if cite == None:
 186             cite = _df_cite
 187 
 188     # establish dictionary page to be searched
 189     dict_name = args_dict['dict']
 190     if dict_name == None:
 191         dict_name = macro.request.getPragma('ref-dict')
 192         if dict_name == None:
 193             dict_name = _df_dict
 194 
 195     # get case change or leave as is
 196     case = args_dict['case']
 197     if not case in _kw_case:
 198         return (_msg_bad_arg % ('invalid case: ' + case))
 199      
 200     # prepare page
 201     result = []
 202     result_pending = []
 203     this_page = macro.formatter.page
 204     if not hasattr(this_page, '_macroRef_pagelist'):
 205         this_page._macroRef_pagelist = {}
 206 
 207     # get list of dictionary pages to search
 208     inc_name = wikiutil.AbsPageName(this_page.page_name, dict_name)
 209     pagelist = [inc_name]
 210     if inc_name.startswith("^"):
 211         try:
 212             inc_match = re.compile(inc_name)
 213         except re.error:
 214             pass # treat as plain page name
 215         else:
 216             # get user filtered readable page list
 217             pagelist = request.rootpage.getPageList(filter = inc_match.match)
 218 
 219     ref_key = ''
 220     ref_match_count = 0
 221     val_end = '\n'
 222 
 223     # iterate over pages
 224     for inc_name in pagelist:
 225         if not request.user.may.read(inc_name):
 226             continue
 227         if inc_name in this_page._macroRef_pagelist:
 228             dec_page()
 229             return (_msg_recursion % (inc_name, ))
 230 
 231         fmt = macro.formatter.__class__(request, is_included=True)
 232         fmt._base_depth = macro.formatter._base_depth
 233         inc_page = Page(request, inc_name, formatter=fmt)
 234         if not inc_page.exists():
 235            return (_msg_no_dict % (inc_name, ))
 236         inc_page._macroRef_pagelist = this_page._macroRef_pagelist
 237 
 238         # prepare for ref_key search 
 239         body = inc_page.get_raw_body() + '\n'
 240         ref_pos = 0
 241         end_pos = -1
 242         ref_found = True
 243 
 244         # need to reset ref_key within loop, as it might have been modified
 245         ref_key = args_dict['ref']
 246 
 247         # remove any WikiName escape character
 248         ref_key = ref_key.replace("!","")
 249 
 250         if ref_key:
 251             #try:
 252             #    ref_match = re.compile(ref_key + ':: ', re.M).search(body)
 253             #except re.error:
 254             ref_match = re.compile(re.escape(ref_key + ':: '), re.M).search(body)
 255             if ref_match:
 256                 ref_pos = ref_match.end() 
 257             else:
 258                 ref_found = False
 259 
 260         if val_end:
 261             #try:
 262             #    end_match = re.compile(val_end, re.M).search(body, ref_pos)
 263             #except re.error:
 264             end_match = re.compile(re.escape(val_end), re.M).search(body, ref_pos)
 265             if end_match:
 266                 end_pos = end_match.start()
 267             else:
 268                 ref_found = False
 269 
 270         if ref_found:
 271             inc_page.set_raw_body(body[ref_pos:end_pos], modified=True)
 272             ref_match_count = ref_match_count + 1
 273 
 274         #TODO: loop back over remainder of body to find duplicate keys to allow user to be warned.
 275         # Perhaps is would be better to use WikiDict calls to get the match, 
 276         # but I think this can only find the first match, not duplicates.
 277         
 278         # set or increment pagelist marker
 279         this_page._macroRef_pagelist[inc_name] = this_page._macroRef_pagelist.get(inc_name, 0) + 1
 280 
 281         # need to reset ref_key within loop, as it might have been modified
 282         ref_key = args_dict['ref']
 283         
 284         if case:
 285             if case == 'upper': ref_key = ref_key.upper()
 286             elif case == 'title': ref_key = ref_key.title()
 287             elif case == 'lower': ref_key = ref_key.lower()
 288 
 289         # make et al italic unless #pragma ref-ital off
 290         if macro.request.getPragma('ref-ital') != 'off':
 291             ref_key = ref_key.replace("et al","''et al''")
 292 
 293         # bracket year (last 'word' of ref key) for cite=bktfn, bktlk or bkt
 294         if cite in ['bktfn', 'bktlk', 'bkt']:
 295             ref_key = ref_key.rstrip()
 296             ref_len = len(ref_key)
 297             year_pos = ref_key.rfind(' ')
 298             year_bkt = _out_year_bkt
 299             if macro.request.getPragma('ref-year-bkt') != None:
 300                 #TODO check if the escape call really needed or does getPragma handle the html risk
 301                 year_bkt = wikiutil.escape(macro.request.getPragma('ref-year-bkt'))
 302                 if year_bkt.count("^") != 1:
 303                     dec_page()
 304                     return (_msg_bad_insert % ('ref-year-bkt', year_bkt)) 
 305                 year_bkt = year_bkt.replace("_"," ").replace("^","%s",1)
 306                 year_bkt = unwrap(year_bkt,"\\\"'")
 307             year_str = year_bkt % (ref_key[year_pos+1:ref_len])
 308             ref_key = ref_key[0:year_pos] + year_str
 309 
 310         if ref_found:
 311             # insert the ref value into the page as a footnote only
 312             if cite == 'fn':
 313                 result_pending = _out_fn % (inc_page.get_body())
 314 
 315             # insert the ref key (modified) into the page followed by the ref value into the page as a footnote
 316             elif cite in ['keyfn','bktfn','keylk','bktlk']:
 317                 result_pending = _out_keyfn % (ref_key, inc_page.get_body())
 318             
 319             # do nothing now, but later insert ref key with ref value as tip
 320             elif cite == 'tip':
 321                 result_pending = u''
 322 
 323             # insert the ref key into the page only
 324             elif cite in ['key', 'bkt']:
 325                 result_pending = ref_key
 326 
 327             # insert the ref value into the page only
 328             elif cite == 'val':
 329                 result_pending = inc_page.get_body()
 330 
 331             if macro.request.getPragma('ref-debug') == 'wiki':
 332                 result_pending = _out_esc % (result_pending)
 333 
 334             result_pending = wikiutil.renderText(request, WikiParser, result_pending).rstrip(' ')
 335 
 336             # post formating of html citation and link
 337             if cite in ['keylk','bktlk']:
 338                 ref_key = ref_key.replace("!","")
 339                 ref_key = ref_key.replace("''et al''","<em>et al</em>")
 340                 result_pending = (result_pending.replace("<sup>","")
 341                                                 .replace("</sup>","")
 342                                                 .replace(ref_key,"")
 343                                                 .replace("</a>","")
 344                                                 .rstrip(" 0123456789") + ref_key + "</a>")
 345 
 346             if cite in ['fn','keyfn','bktfn']:
 347                 if macro.request.getPragma('ref-super') == 'off':
 348                     link_bkt = _out_link_bkt
 349                     if macro.request.getPragma('ref-link-bkt') != None:
 350                         #TODO check if the escape call really needed or does getPragma handle the html risk
 351                         link_bkt = wikiutil.escape(macro.request.getPragma('ref-link-bkt'))
 352                         if link_bkt.count("^") != 1:
 353                             dec_page()
 354                             return (_msg_bad_insert % ('ref-link-bkt', link_bkt, ))
 355                         link_bkt = (unwrap(link_bkt,"\\\"'").replace("_"," ")
 356                                                             .replace("^","%s",1))
 357                     t = tuple(result_pending.replace("<sup>","<>")
 358                                             .replace("</a>","</a><>")
 359                                             .replace("</sup>","")
 360                                             .split("<>"))
 361                     result_pending = ('%s' + link_bkt + '%s') % (t[0],t[1],t[2])
 362                     
 363             # escape HTML including quotes
 364             tip = wikiutil.escape(inc_page.get_body(), quote=True)
 365                 
 366             #convert to html then remove tags to remove wiki markup
 367             tip = wikiutil.renderText(request, WikiParser, tip)
 368             tip = re.sub('<[^>]*>', '', tip)
 369                 
 370             if cite == 'tip':
 371                 #markup ref_key
 372                 result_pending = wikiutil.renderText(request, WikiParser, ref_key).rstrip(' ')
 373 
 374             if (cite == 'tip') or (cite != 'val' and macro.request.getPragma('ref-tips') == 'on'):
 375 
 376                 result_pending = _out_tip % (tip, result_pending)
 377 
 378             if macro.request.getPragma('ref-debug') == 'html':
 379                 result_pending = _out_esc % (result_pending)
 380                 result_pending = wikiutil.renderText(request, WikiParser, result_pending).rstrip(' ')
 381 
 382             result.append(result_pending)
 383 
 384         # decrease or remove pagelist marker
 385         dec_page()
 386 
 387     # no match and last page in list, so insert warning message
 388     # need to reset ref_key within loop, as it might have been modified
 389     ref_key = args_dict['ref']
 390     if ref_match_count == 0:
 391         result.append(_msg_match_none % (ref_key, dict_name, ))
 392     elif ref_match_count > 1:
 393         result.append(_msg_match_many % (ref_match_count, ref_key, dict_name, ))
 394 
 395     return ''.join(result)

You are not allowed to attach a file to this page.