置顶公告:【置顶】关于临时开启评论区所有功能的公告(2022.10.22) | 【置顶】关于本站Widget恢复使用的公告
  • 你好~!欢迎来到萌娘百科镜像站!如需查看或编辑,请联系本站管理员注册账号。
  • 本镜像站和其他萌娘百科的镜像站无关,请注意分别。

Module:Citation

贴贴♀百科,万娘皆可贴的百科全书!转载请标注来源页面的网页链接,并声明引自贴贴百科。内容不可商用。
跳到导航 跳到搜索
Template-info.svg 模块文档  [创建] [刷新]
  1. ---------------------------------------------------------------------
  2. -- Module:Citation - Lua module for Citation auxiliary templates
  3. ---------------------------------------------------------------------
  4. -- For the {{citation}} formatting functions, see: Module:Citation/CS1
  5. -- (see NOTES at bottom)
  6. --require "mw.text"
  7. local z = {
  8. wikitext = require("Module:Wikitext"),
  9. extensiontags = {
  10. nowiki = true,
  11. ref = true,
  12. gallery = true,
  13. pre = true,
  14. source = true,
  15. categorytree = true,
  16. charinsert = true,
  17. hiero = true,
  18. imagemap = true,
  19. inputbox = true,
  20. math = true,
  21. poem = true,
  22. ref = true,
  23. references = true,
  24. syntaxhighlight = true,
  25. timeline = true,
  26. }
  27. }
  28. function trim( str )
  29. if str == nil then
  30. return nil;
  31. end
  32. return str:match( "^%s*(.-)%s*$" );
  33. end
  34. function hideinprint(content)
  35. return content
  36. end
  37. function onlyinprint(content)
  38. return ""
  39. end
  40. -- This returns a string with HTML character entities for wikitext markup characters.
  41. function wikiescape(text)
  42. text = text:gsub( '[&\'%[%]{|}]', {
  43. ['&'] = '&',
  44. ["'"] = ''',
  45. ['['] = '[',
  46. [']'] = ']',
  47. ['{'] = '{',
  48. ['|'] = '|',
  49. ['}'] = '}' } );
  50. return text;
  51. end
  52. function createTag(t, frame)
  53. local name = t.name or "!-- --"
  54. local content = t.contents or ""
  55. local attrs = {}
  56. if ( z.extensiontags[name] ) then
  57. -- We have to preprocess these, so that they are properly turned into so-called "strip markers" in the generated wikitext.
  58. if ( not frame ) then error ("Please supply an extra frame argument to the createTag() function.") end
  59. local params = {}
  60. for n,v in pairs(t.params) do
  61. table.insert(params, "|" .. n .. "=" .. v)
  62. end
  63. return frame:preprocess("{{#tag:" .. name .. "|" .. content .. table.concat(params) .. "}}")
  64. else
  65. for n,v in pairs(t.params) do
  66. if (v) then
  67. table.insert(attrs, n .. "=\"" .. wikiescape(v) .. "\"")
  68. else
  69. table.insert(attrs, n)
  70. end
  71. end
  72. if ("" == content) then
  73. return "<" .. name .. " " .. table.concat(attrs, " ") .. "/>"
  74. else
  75. return "<" .. name .. " " .. table.concat(attrs, " ") .. ">" .. content .. "</" .. name .. ">"
  76. end
  77. end
  78. end
  79. --[[
  80. This is a clone of mw.text.nowiki. When the mw.text library is installed,
  81. this can be replaced by a call to that library. ]]
  82. function nowiki( s )
  83. -- string.gsub is safe here, because we're only caring about ASCII chars
  84. s = string.gsub( s, '["&\'<=>%[%]{|}]', {
  85. ['"'] = '&#34;',
  86. ['&'] = '&#38;',
  87. ["'"] = '&#39;',
  88. ['<'] = '&#60;',
  89. ['='] = '&#61;',
  90. ['>'] = '&#62;',
  91. ['['] = '&#91;',
  92. [']'] = '&#93;',
  93. ['{'] = '&#123;',
  94. ['|'] = '&#124;',
  95. ['}'] = '&#125;',
  96. } )
  97. s = string.sub( string.gsub( '\n' .. s, '\n[#*:;]', {
  98. ["\n#"] = "\n&#35;",
  99. ["\n*"] = "\n&#42;",
  100. ["\n:"] = "\n&#58;",
  101. ["\n;"] = "\n&#59;",
  102. } ), 2 )
  103. s = string.gsub( s, '://', '&#58;//' )
  104. s = string.gsub( s, 'ISBN ', 'ISBN&#32;' )
  105. s = string.gsub( s, 'RFC ', 'RFC&#32;' )
  106. return s
  107. end
  108. function externallinkid(args)
  109. local sep = args.separator or "&nbsp;"
  110. args.suffix = args.suffix or ""
  111. local url_string = args.id
  112. if args.encode == true or args.encode == nil then
  113. url_string = mw.uri.encode( url_string );
  114. end
  115. local t0 = onlyinprint(args.label .. sep .. args.id)
  116. local t1 = hideinprint("[[" .. args.link .. "|" .. args.label .. "]]" .. sep .. "[" .. args.prefix .. url_string .. args.suffix .. " " .. nowiki(args.id) .. "]")
  117. return t0 .. t1
  118. end
  119. function doi(id, inactive, nocat)
  120. local cat = ""
  121. local text;
  122. if ( inactive ~= nil ) then
  123. text = "[[Digital object identifier|doi]]:" .. id;
  124. cat = cat .. "[[Category:Pages with DOIs inactive since " .. selectyear(inactive) .. "]]"
  125. inactive = " (inactive " .. inactive .. ")"
  126. else
  127. text = externallinkid({link="Digital object identifier",label="doi",prefix="http://dx.doi.org/",id=id,separator=":"})
  128. inactive = ""
  129. end
  130. if ( string.sub(id,1,3) ~= "10." ) then
  131. cat = cat .. "[[Category:Pages with DOI errors]]" .. '<span class="error"> Bad DOI (expected "10." prefix) in code number</span>'
  132. end
  133. if ( nocat and nocat ~= "" ) then cat = "" end
  134. return text .. inactive .. cat
  135. end
  136. function selectyear( str )
  137. local lang = mw.getContentLanguage();
  138. local good, result;
  139. good, result = pcall( lang.formatDate, lang, 'Y', str )
  140. if good then
  141. return result;
  142. else
  143. return '';
  144. end
  145. end
  146. function anchorid(label, args)
  147. local P1 = trim(args[1]) or ""
  148. local P2 = trim(args[2]) or ""
  149. local P3 = trim(args[3]) or ""
  150. local P4 = trim(args[4]) or ""
  151. local P5 = trim(args[5]) or ""
  152. local anchor = P1 .. P2 .. P3 .. P4 .. P5;
  153. if anchor ~= '' then -- See bug description in Citation/CS1
  154. anchor = mw.uri.anchorEncode( anchor );
  155. end
  156. return label .. anchor
  157. end
  158. function refid(label, args)
  159. local p = args.p or ""
  160. local pp = args.pp or ""
  161. local loc = args.loc or ""
  162. return anchorid(label, args) .. p .. pp .. loc
  163. end
  164. function name(args)
  165. local P1 = trim(args[1]) or ""
  166. if ( args[5] ~= nil) then
  167. return P1 .. " et al."
  168. else
  169. local P2 = trim(args[2]) or ""
  170. local P3 = trim(args[3]) or ""
  171. local P4 = trim(args[4]) or ""
  172. if ( args[4] ~= nil ) then
  173. P4 = " " .. P4
  174. P3 = " &amp; " .. P3
  175. P2 = ", " .. P2
  176. elseif ( args[3] ~= nil ) then
  177. P3 = " " .. P3
  178. P2 = " &amp; " .. P2
  179. elseif ( args[2] ~= nil ) then
  180. P2 = " " .. P2
  181. end
  182. return P1 .. P2 .. P3 .. P4
  183. end
  184. end
  185. function crossref(frame, label, args)
  186. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  187. local LB = config.BracketLeft or ""
  188. local RB = config.BracketRight or ""
  189. local anchor = args.ref or args.Ref or anchorid( label, args)
  190. local text = name(args)
  191. local loc = args.loc
  192. local page
  193. local pages = args.pp or args.pages
  194. if pages == nil or pages == '' then
  195. page = args.p or args.page;
  196. end
  197. if nil == loc then loc = "" else loc = " " .. loc end
  198. if ( page ~= nil ) then
  199. local pagesep = config.PageSep or ", p.&nbsp;"
  200. loc = loc .. pagesep .. page
  201. end
  202. if ( pages ~= nil ) then
  203. local pagessep = config.PagesSep or ", pp.&nbsp;"
  204. loc = loc .. pagessep .. pages
  205. end
  206. local pagename = args.pagename or ""
  207. local ps = args.Postscript or ""
  208. return LB .. "[[" .. pagename .. "#" .. anchor .. "|" .. text .. "]]" .. loc .. RB .. ps
  209. end
  210. function r0(frame, name, group, page)
  211. if ( name == nil ) then return "" end
  212. if ( group == nil ) then group = "" end
  213. local p = ""
  214. if ( page ~= nil ) then
  215. local contents = ":" .. page
  216. p = createTag({name="sup",contents=contents,params={class="reference",style="white-space:nowrap;"}})
  217. end
  218. return createTag({name="ref",contents="",params={name=name,group=group}}, frame) .. p
  219. end
  220. function reflist0(frame, config, args)
  221. local contents = args.refs or ""
  222. local liststyle = args.liststyle
  223. local count = args[1]
  224. local width = args.colwidth
  225. local group = args.group or config.default_group
  226. if ( nil == tonumber(count) and nil == width ) then
  227. width = count
  228. count = nil
  229. end
  230. if ( nil == liststyle ) then
  231. if ( "upper-alpha" == group or "lower-alpha" == group or "upper-roman" == group or "lower-roman" == group or "upper-greek" == group or "lower-greek" == group ) then
  232. liststyle = group
  233. else
  234. liststyle = config.default_liststyle
  235. end
  236. end
  237. local params = {}
  238. params.class = "reflist"
  239. params.style = z.wikitext.liststyle(liststyle)
  240. if ( nil ~= count ) then
  241. params.class = params.class .. " references-column-count references-column-count-" .. count
  242. params.style = params.style .. " " .. z.wikitext.columncountstyle(count)
  243. end
  244. if ( nil ~= width ) then
  245. params.class = params.class .. " references-column-width"
  246. params.style = params.style .. " " .. z.wikitext.columnwidthstyle(width)
  247. end
  248. local references = createTag({name="references",contents=contents,params={group=group}}, frame)
  249. return createTag({name="div",contents=references,params=params})
  250. end
  251. function refbegin0(frame, config, args)
  252. local liststyle = args.liststyle
  253. local indent = args.indent
  254. local indentsize = args.indentsize
  255. local count = args[1]
  256. local width = args.colwidth
  257. if ( nil == tonumber(count) and nil == width ) then
  258. width = count
  259. count = nil
  260. end
  261. if ( nil == liststyle ) then
  262. if ( "upper-alpha" == group or "lower-alpha" == group or "upper-roman" == group or "lower-roman" == group or "upper-greek" == group or "lower-greek" == group ) then
  263. liststyle = group
  264. else
  265. liststyle = config.default_liststyle
  266. end
  267. end
  268. local params = {}
  269. params.class = "refbegin"
  270. params.style = z.wikitext.liststyle(liststyle)
  271. if ( nil ~= count ) then
  272. params.class = params.class .. " references-column-count references-column-count-" .. count
  273. params.style = params.style .. " " .. z.wikitext.columncountstyle(count)
  274. end
  275. if ( nil ~= width ) then
  276. params.class = params.class .. " references-column-width"
  277. params.style = params.style .. " " .. z.wikitext.columnwidthstyle(width)
  278. end
  279. local dlopen
  280. if ( nil ~= indent ) then
  281. dlopen = z.wikitext.OpenHTMLTag({name="dl",params={style="text-indent: -" .. (indentsize or "3.2") .. "em;"}})
  282. else
  283. dlopen = ""
  284. end
  285. return z.wikitext.OpenHTMLTag({name="div",params=params}) .. dlopen
  286. end
  287. function refend0(frame, config, args)
  288. local indent = args.indent
  289. local dlclose
  290. if ( nil ~= indent ) then
  291. dlclose = "</dl>"
  292. else
  293. dlclose = ""
  294. end
  295. return dlclose .. "</div>"
  296. end
  297. -- This is used by {{doi}} to create DOI links in the style used in citations.
  298. function z.doi(frame)
  299. local pframe = frame:getParent()
  300. local id = pframe.args.id or pframe.args[1] or ""
  301. return doi(id)
  302. end
  303. -- This is used by {{ISSN}} to create ISSN links in the style used in citations.
  304. function z.ISSN(frame)
  305. local pframe = frame:getParent()
  306. local Name = pframe.args[1] or ""
  307. return hideinprint("[[International Standard Serial Number|ISSN]]&nbsp;[http://www.worldcat.org/search?fq=x0:jrnl&q=n2:" .. Name .. " " .. Name .. "]")
  308. end
  309. -- This is used by templates such as {{SfnRef}} to create the (encoded) anchor name for a Harvard cross-reference hyperlink.
  310. function z.SFNID(frame)
  311. local pframe = frame:getParent()
  312. return anchorid('FOOTNOTE', pframe.args)
  313. end
  314. -- This is used by templates such as {{Harvard citation}} to create the Harvard cross-reference text.
  315. function z.Harvard(frame)
  316. local pframe = frame:getParent()
  317. return crossref(frame, pframe.args)
  318. end
  319. -- This is used by templates such as {{sfn}} to create the entire cross-reference.
  320. function z.sfn(frame)
  321. local pframe = frame:getParent()
  322. pframe.args.Postscript = pframe.args.postscript or pframe.args.ps or ".";
  323. local content = crossref(frame, 'CITEREF', pframe.args)
  324. local args = { name = refid( 'FOOTNOTE', pframe.args) }
  325. return createTag({name = "ref", contents = content, params = args}, frame)
  326. end
  327. -- This is used by template {{r}}.
  328. function z.r(frame)
  329. local pframe = frame:getParent()
  330. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  331. local args = pframe.args -- the arguments passed TO the template, in the wikitext that instantiates the template
  332. args.page1 = args.page1 or args.page
  333. local text = ""
  334. -- This would be shorter using ipairs(), but that doesn't work on an arguments table supplied to a template.
  335. local index = 1
  336. while args[index] ~= nil do
  337. local arg = args[index]
  338. local t = r0(frame, arg, args.group, args["page" .. index])
  339. text = text .. t
  340. index = index + 1
  341. end
  342. return text
  343. end
  344. -- This is used by template {{ref label}}.
  345. function z.reflabel(frame)
  346. local pframe = frame:getParent()
  347. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  348. local args = pframe.args -- the arguments passed TO the template, in the wikitext that instantiates the template
  349. local P1 = args[1] or ""
  350. local P2 = args[2] or ""
  351. local P3 = args[3] or ""
  352. local id = nil
  353. local contents = "[[#endnote_" .. P1 .. P3 .. "|&#91;" .. P2 .. "&#93;]]"
  354. local params = {}
  355. params.class="reference"
  356. if ( args.noid == nil or args.noid == "" ) then params.id = "ref_" .. P1 .. P3 end
  357. return createTag({name="sup",contents=contents,params=params})
  358. end
  359. -- This is used by template {{note label}}.
  360. function z.notelabel(frame)
  361. local pframe = frame:getParent()
  362. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  363. local args = pframe.args -- the arguments passed TO the template, in the wikitext that instantiates the template
  364. local id = args[1] or ""
  365. local arrow = args[3] or ""
  366. local postscript = args[4] or ""
  367. local contents
  368. if arrow ~= "" then
  369. local sup_arrow = createTag({name="sup",contents=arrow,params={}})
  370. contents = "[[#ref_" .. id .. arrow .. "|<b>" .. sup_arrow .. "</b>]]" .. postscript
  371. if "none" == arrow then arrow = "^" end -- Change this AFTER using it in the ID parameter and the contents.
  372. else
  373. contents = (args[2] or "") .. postscript
  374. end
  375. local params = { class="citation wikicite" }
  376. if id ~= "" and ( args.noid == nil or args.noid == "" ) then
  377. params.id = mw.uri.anchorEncode("endnote_" .. id .. arrow)
  378. end
  379. return createTag({name="span",contents=contents,params=params})
  380. end
  381. -- This is used by templates {{reflist}} and {{notelist}}.
  382. function z.reflist(frame)
  383. local pframe = frame:getParent()
  384. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  385. local args = pframe.args -- the arguments passed TO the template, in the wikitext that instantiates the template
  386. return reflist0(frame, config, args)
  387. end
  388. -- This is used by template {{refbegin}}.
  389. function z.refbegin(frame)
  390. local pframe = frame:getParent()
  391. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  392. local args = pframe.args -- the arguments passed TO the template, in the wikitext that instantiates the template
  393. return refbegin0(frame, config, args)
  394. end
  395. -- This is used by template {{refend}}.
  396. function z.refend(frame)
  397. local pframe = frame:getParent()
  398. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  399. local args = pframe.args -- the arguments passed TO the template, in the wikitext that instantiates the template
  400. return refend0(frame, config, args)
  401. end
  402. -- This is used by template {{efn}}.
  403. function z.efn(frame)
  404. local pframe = frame:getParent()
  405. local config = frame.args -- the arguments passed BY the template, in the wikitext of the template itself
  406. local args = pframe.args -- the arguments passed TO the template, in the wikitext that instantiates the template
  407. return createTag({name="ref",contents=(args[1] or ""),params={name=args.name,group=config.default_group}}, frame)
  408. end
  409. return z
  410. ---------------------------------------------------------------------
  411. --NOTES
  412. --
  413. -- NOTE A1: This Lua module was originally designed to handle a mix
  414. -- of citation styles, crossing Vancouver style with Wikipedia's
  415. -- local Citation Style 1 (CS1) from {Template:Citation/core}.
  416. -- However, the conflicting positions of parameters, scattered
  417. -- in twisted locations across this module, led to a separate
  418. -- variation just to untangle the CS1 format of citations.
  419. --
  420. -- NOTE D2: The placement of dots and other separators between the
  421. -- displayed parameters has been a continual headache, to keep
  422. -- coordinated with the data in parentheses "(data)". There
  423. -- has been a need to pre-check for the existence of related
  424. -- options, to keep from putting double-dots ".." in some cases.
  425. -- In particular, the omission of the "title=" parameter has led
  426. -- to several cases of a spurious dot ". ." because the original
  427. -- design had treated the title as a mandatory parameter.
  428. --
  429. ------------------------------------------------------------------------
  430. --HISTORY:
  431. --18Oct2012 Fixed lead-space in Chapter by omitting " ".
  432. --18Oct2012 Fixed lead-space in Chapter/Title as end " " of Authors/Date/...
  433. --19Oct2012 Put HISTORY comments to log major changes (not typos).
  434. --19Oct2012 Fixed extra dot ".." in Title by omitting at end of "tcommon=...".
  435. --19Oct2012 For pages, put &nbsp in "p.&nbsp;" etc.
  436. --19Oct2012 Enhanced "pages=" to detect lone page as "p." else "pp." prefix.
  437. --19Oct2012 Fixed to show "." after Periodical name (work, newspaper...).
  438. --19Oct2012 Fixed web-link to have spaces "[... Archived] from the original".
  439. --19Oct2012 Fixed to show ";" between authors & coauthors.
  440. --19Oct2012 Fixed to omit extra "." after coauthors.
  441. --20Oct2012 Fixed COinS data to not urlencode all, as "ctx_ver=Z39.88-2004"
  442. --20Oct2012 Fixed COinS to not end as "&" but use lead "&rft...=" form.
  443. --20Oct2012 Fixed COinS to not url.encode page's "rfr_id=..." pagename.
  444. --20Oct2012 Fixed COinS data when "web" to default to rft.genre "book".
  445. --05Nov2012 Add a span wrapper even when there is no Ref parameter
  446. --15Feb2013 Added Agency for "agency=xx".
  447. --19Feb2013 Put NOTES comments to explain module operation.
  448. --19Feb2013 Copied as Module:Citation/CS1 to alter to match wp:CS1 form.
  449. --19Feb2013 Changed OrigYear to use [__] for CS1 style.
  450. --19Feb2013 Fixed to not show duplicate Publisher/Agency.
  451. --19Feb2013 Moved page-number parameters to after final date.
  452. --19Feb2013 Fixed to not put double-dots after title again.
  453. --20Feb2013 Changed to omit dot "." if already ends with dot.
  454. --20Feb2013 If class "journal" shows Publisher after Periodical/Series.
  455. --20Feb2013 Shifted Format to after Language, and Others after Volume.
  456. --20Feb2013 Set AccessDate + <span class="reference-accessdate">
  457. --20Feb2013 Fixed url when deadurl=no.
  458. --20Feb2013 Added sepc for separator character between parameters.
  459. --20Feb2013 Put "OCLC" for "Online Computer Library Center".
  460. --20Feb2013 Fix empty "authorlink=" as person.link ~= "".
  461. --20Feb2013 Added space after AuthorSep & AuthorNameSep.
  462. --21Feb2013 Added args.contributor (was missing parameter).
  463. --21Feb2013 Fixed EditorSep (was misspelled "EdithorSep").
  464. --21Feb2013 Set OCinSdata.rft_val_fmt = "info:ofi/fmt:kev:mtx:book"
  465. --21Feb2013 Checked to omit blank codes (asin= | doi= etc.).
  466. --21Feb2013 Set enddot to end line if not config.CitationClass "citation".
  467. --21Feb2013 Fixed to show "issn=x" as the ISSN code.
  468. --21Feb2013 Fixed to show "id=x" after Zbl code.
  469. --21Feb2013 Changed to omit double-dot before date when already dot.
  470. --21Feb2013 Order config.CitationClass "citation": Volume, Issue, Publisher.
  471. --21Feb2013 Put warning "Bad DOI (expected "10."..)" in DOI result.
  472. --21Feb2013 Automatically unbolded volume+comma when > 4 long.
  473. --21Feb2013 Changed to allow lowercase "asin-tld".
  474. --22Feb2013 Fixed ref=harv to extract Year from Date.
  475. --22Feb2013 Set Harvard refer. span id if config.CitationClass "citation".
  476. --22Feb2013 Fixed config.CitationClass "citation" as span class="citation".
  477. --22Feb2013 Capitalized "Archived/Retrieved" only when sepc is dot ".".
  478. --23Feb2013 Fixed author editor for "in" or "In" and put space after sepc.
  479. --23Feb2013 Changed to omit dot in "et al." when sepc is "." separator.
  480. --23Feb2013 Fixed "author1-first" to also get args.given or args.given1.
  481. --23Feb2013 Fixed args.article to set Title, after Periodical is Title.
  482. --23Feb2013 Fixed to allow blank Title (such as "contribution=mytitle").
  483. --23Feb2013 Fixed double-dot ".." at end of Editors list
  484. --26Feb2013 Moved "issue=" data to show before "page=".
  485. --26Feb2013 Moved "type=" data to show after "format=".
  486. --26Feb2013 For "pmc=" link, omitted suffix "/?tool=pmcentrez".
  487. --27Feb2013 For coauthors, omitted extra separator after authors.
  488. --27Feb2013 For date, allowed empty date to use month/day/year.
  489. --27Feb2013 Fixed double-dot ".." at end of authors/coauthors list.
  490. --27Feb2013 Reset editor suffix as ", ed." when date exists.
  491. --27Feb2013 Removed duplicate display of "others=" data.
  492. --27Feb2013 Removed parentheses "( )" around "department" TitleNote.
  493. --05Mar2013 Moved Language to follow Periodical or Series.
  494. --05Mar2013 Fixed Edition to follow Series or Volume.
  495. --05Mar2013 Fixed class encyclopaedia to show article as quoted Chapter.
  496. --05Mar2013 Fixed class encyclopaedia to show page as "pp." or "p.".
  497. --07Mar2013 Changed class encyclopaedia to omit "( )" around publisher.
  498. --07Mar2013 Fixed end double-dot by string.sub(idcommon,-1,-1) was "-1,1".
  499. --13Mar2013 Removed enddot "." after "quote=" parameter.
  500. --13Mar2013 Changed config.CitationClass "news" to use "p." page format.
  501. --13Mar2013 Fixed missing "location=" when "web" or "encyclopaedia".
  502. --14Mar2013 Fixed end double-dot after book/work title.
  503. --14Mar2013 Fixed double-dot before "p." or "pp." page number.
  504. --14Mar2013 Fixed config.CitationClass "book" to use p./pp. page.
  505. --
  506. --End