You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

281 lines
19 KiB

  1. <?php
  2. // Project: Web Reference Database (refbase) <http://www.refbase.net>
  3. // Copyright: Matthias Steffens <mailto:refbase@extracts.de> and the file's
  4. // original author(s).
  5. //
  6. // This code is distributed in the hope that it will be useful,
  7. // but WITHOUT ANY WARRANTY. Please see the GNU General Public
  8. // License for more details.
  9. //
  10. // File: ./cite/formats/cite_latex_bbl.php
  11. // Repository: $HeadURL$
  12. // Author(s): Matthias Steffens <mailto:refbase@extracts.de>
  13. //
  14. // Created: 20-Jul-07, 19:42
  15. // Modified: $Date: 2017-04-13 02:00:18 +0000 (Thu, 13 Apr 2017) $
  16. // $Author$
  17. // $Revision: 1416 $
  18. // This is a citation format file (which must reside within the 'cite/formats/' sub-directory of your refbase root directory). It contains a version of
  19. // the 'citeRecords()' function that outputs a reference list from selected records in LateX .bbl format, to be used with LaTeX/BibTeX. The .bbl file
  20. // generated by refbase can be used as a replacement of the BibTeX-generated .bbl file. Make sure to use "citeOrder=author" when generating the .bbl file.
  21. // NOTE: In its current incarnation, this script outputs a .bbl variant that is intended to be used with the natbib package.
  22. // You may want to adopt the style-specific settings below if necessary. The natbib package is available at:
  23. // <http://www.ctan.org/tex-archive/macros/latex/contrib/natbib/>
  24. // --------------------------------------------------------------------
  25. // --- BEGIN CITATION FORMAT ---
  26. function citeRecords($result, $rowsFound, $query, $queryURL, $showQuery, $showLinks, $rowOffset, $showRows, $previousOffset, $nextOffset, $wrapResults, $citeStyle, $citeOrder, $citeType, $orderBy, $headerMsg, $userID, $viewType)
  27. {
  28. global $contentTypeCharset; // defined in 'ini.inc.php'
  29. global $client;
  30. // The array '$transtab_refbase_latex' contains search & replace patterns for conversion from refbase markup to LaTeX markup & entities.
  31. // Converts refbase fontshape markup (italic, bold) into LaTeX commands of the 'textcomp' package, super- and subscript as well as greek
  32. // symbols get converted into the respective commands in math mode. You may need to adopt the LaTeX markup to suit your individual needs.
  33. global $transtab_refbase_latex; // defined in 'transtab_refbase_latex.inc.php'
  34. // The arrays '$transtab_latin1_latex' and '$transtab_unicode_latex' provide translation tables for best-effort conversion of higher ASCII
  35. // characters from ISO-8859-1 (or Unicode, respectively) to LaTeX entities.
  36. global $transtab_latin1_latex; // defined in 'transtab_latin1_latex.inc.php'
  37. global $transtab_unicode_latex; // defined in 'transtab_unicode_latex.inc.php'
  38. // --- Begin style-specific settings ----------------------------------
  39. // NOTE: the given settings are meant to be used with the natbib package;
  40. // you'll have to adopt these settings if you'd like to generate a
  41. // .bbl file for another style/package
  42. // 1) Define header with .bst style-specific declarations:
  43. // NOTE: since this header is normally inserted by BibTeX from the used .bst style, I'm not sure if there's a variant of this header that's
  44. // universally accepted by natbib. So, if necessary adopt it to your needs.
  45. // -- by default, we use this header (as inserted by a .bst style that was generated from 'merlin.mbs' via the docstrip utility):
  46. $bblHeader = "\\providecommand{\\natexlab}[1]{#1}" . "\n"
  47. . "\\providecommand{\\url}[1]{\\texttt{#1}}" . "\n"
  48. . "\\providecommand{\\urlprefix}{URL }" . "\n"
  49. . "\\providecommand{\\eprint}[2][]{\\url{#2}}" . "\n\n";
  50. // -- here's another header example (as inserted by 'elsart-harv.bst'):
  51. // $bblHeader = "\\expandafter\\ifx\\csname natexlab\\endcsname\\relax\\def\\natexlab#1{#1}\\fi" . "\n"
  52. // . "\\expandafter\\ifx\\csname url\\endcsname\\relax" . "\n"
  53. // . " \\def\\url#1{\\texttt{#1}}\\fi" . "\n"
  54. // . "\\expandafter\\ifx\\csname urlprefix\\endcsname\\relax\\def\\urlprefix{URL }\\fi" . "\n\n";
  55. // 2) Define name of \bibitem command:
  56. $bibItemCommand = "\\bibitem"; // use "\\harvarditem" for the Harvard family of styles
  57. // 3) Define variables and placeholder strings which together will form the \bibitem option string:
  58. // -- variables:
  59. $betweenShortAuthorsDelim = " & "; // used to connect individual author names in the short author list
  60. $betweenFullAuthorsDelim = ", "; // used to connect multiple authors in the full author list
  61. $betweenLastAuthorsDelim = " & "; // used to connect the second-to-last author with the last author in the full author list
  62. $etalIdentifierString = " et~al."; // appended to the first author's last name if the record's number of authors exceeds the number given in '$maxAuthorCountInShortAuthorList'
  63. $maxAuthorCountInShortAuthorList = 2; // maximum number of authors to be used in the short author list
  64. // -- placeholder strings:
  65. // (please see the refbase online documentation for more info about placeholders and their syntax: <http://placeholders.refbase.net/>)
  66. $year = "<:year[4]:>"; // year format, e.g. "1990"
  67. $shortAuthorList = "<:authors[" . $maxAuthorCountInShortAuthorList . "|" . $betweenShortAuthorsDelim . "|" . $etalIdentifierString . "]:>"; // format of the short author list which is used as text citation, e.g. "Jones", "Jones and Baker" or "Jones et al."
  68. $fullAuthorList = "<:authors[0|_#_�_~_|]:>"; // format of the full author list, e.g. "Jones, Baker, and Williams"
  69. // NOTE: in the above placeholder string, we use the string "_#_�_~_" as author delimiter which allows us to uniquely identify that delimiter again below, and replace it with the contents of either '$betweenFullAuthorsDelim' or '$betweenLastAuthorsDelim' (depending on its position)
  70. // 4) Define \bibitem option format:
  71. // -- extended natbib 5.3 style:
  72. $bibItemOptionShort = "[{" . $shortAuthorList . "}(" . $year . ")" . "]"; // e.g.: \bibitem[{Jones et al.}(1990)]{key}...
  73. $bibItemOptionFull = "[{" . $shortAuthorList . "}(" . $year . "){" . $fullAuthorList . "}]"; // e.g.: \bibitem[{Jones et al.}(1990){Jones, Baker, and Williams}]{key}...
  74. // NOTE: since the author lists may contain parentheses, we need to enclose them in curly brackets to work around a natbib limitation
  75. // -- native natbib style:
  76. // $bibItemOptionShort = "[" . $shortAuthorList . "(" . $year . ")" . "]"; // e.g.: \bibitem[Jones et al.(1990)]{key}...
  77. // $bibItemOptionFull = "";
  78. // -- apalike style:
  79. // $bibItemOptionShort = "[" . $shortAuthorList . ", " . $year . "]"; // e.g.: \bibitem[Jones et al., 1990]{key}...
  80. // $bibItemOptionFull = "";
  81. // -- newapa or chicago styles:
  82. // $bibItemOptionShort = "[\\protect\\citeauthoryear{" . $fullAuthorList . "}{" . $shortAuthorList . "}{" . $year . "}" . "]"; // this assumes that it's not allowed to omit the full author list, though I dunno)
  83. // $bibItemOptionFull = "[\\protect\\citeauthoryear{" . $fullAuthorList . "}{" . $shortAuthorList . "}{" . $year . "}" . "]"; // e.g.: \bibitem[\protect\citeauthoryear{Jones, Baker, and Williams}{Jones et al.}{1990}]{key}...
  84. // -- named style:
  85. // $bibItemOptionShort = "[\\protect\\citeauthoryear{" . $shortAuthorList . "}{" . $year . "}" . "]"; // e.g.: \bibitem[\protect\citeauthoryear{Jones et al.}{1990}]{key}...
  86. // $bibItemOptionFull = "";
  87. // -- astron style:
  88. // $bibItemOptionShort = "[\\protect\\astroncite{" . $shortAuthorList . "}{" . $year . "}" . "]"; // e.g.: \bibitem[\protect\astroncite{Jones et al.}{1990}]{key}...
  89. // $bibItemOptionFull = "";
  90. // -- authordate style:
  91. // $bibItemOptionShort = "[\\protect\\citename{" . $shortAuthorList . ", }" . $year . "]"; // e.g.: \bibitem[\protect\citename{Jones et al., }1990]{key}...
  92. // $bibItemOptionFull = "";
  93. // -- harvard style:
  94. // $bibItemOptionShort = "[" . $shortAuthorList . "]{" . $fullAuthorList . "}{" . $year . "}"; // this assumes that it's not allowed to omit the full author list, though I dunno)
  95. // $bibItemOptionFull = "[" . $shortAuthorList . "]{" . $fullAuthorList . "}{" . $year . "}"; // e.g.: \harvarditem[Jones et al.]{Jones, Baker, and Williams}{1990}{key}...
  96. // 5) Define deduplication prefix and suffix strings:
  97. // Identical text citation strings get uniquified by appending letters to the year, e.g. duplicate occurrences
  98. // of "Jones et al. 1990" should become "Jones et al. 1990a" and "Jones et al. 1990b" in the final output.
  99. // These prefix/suffix strings will be inserted before/after the deduplication letter:
  100. $dedupPrefix = "{\\natexlab{"; // these prefix/suffix strings are required by natbib; if any custom command (such as '\natexlab') is used, make sure that the command is also defined above in variable '$bblHeader'
  101. $dedupSuffix = "}}";
  102. // --- End style-specific settings ------------------------------------
  103. // Initialize array variables:
  104. $yearsArray = array();
  105. $typeTitlesArray = array();
  106. $bibItemsArray = array();
  107. // Define inline text markup to be used by the 'citeRecord()' function:
  108. $markupPatternsArray = array("bold-prefix" => "\\textbf{",
  109. "bold-suffix" => "}",
  110. "italic-prefix" => "\\textit{",
  111. "italic-suffix" => "}",
  112. "underline-prefix" => "\\ul{", // the '\ul' command requires '\usepackage{soul}'
  113. "underline-suffix" => "}",
  114. "endash" => "--", // or use '{\\textendash}'
  115. "emdash" => "---", // or use '{\\textemdash}'
  116. "ampersand" => "&", // conversion of ampersands is done below, after the citation has been generated
  117. "double-quote" => '"',
  118. "double-quote-left" => "{\\textquotedblleft}",
  119. "double-quote-right" => "{\\textquotedblright}",
  120. "single-quote" => "'", // same as for ampersands
  121. "single-quote-left" => "{\\textquoteleft}",
  122. "single-quote-right" => "{\\textquoteright}",
  123. "less-than" => "<",
  124. "greater-than" => ">",
  125. "newline" => "\n\n"
  126. );
  127. // Defines search & replace 'actions' that will be applied upon LaTeX output to all those refbase fields that are listed
  128. // in the corresponding 'fields' element:
  129. $latexSearchReplaceActionsArray = array(
  130. array('fields' => array("title", "publication", "abbrev_journal", "address", "keywords", "abstract", "orig_title", "series_title", "abbrev_series_title", "notes"),
  131. 'actions' => $transtab_refbase_latex
  132. )
  133. );
  134. // For CLI queries, we'll allow paging thru the result set, i.e. we honour the values of the CLI options '-S|--start' ('$rowOffset')
  135. // and '-R|--rows' ('$showRows') ('$rowOffset' and '$showRows' are re-assigned in function 'seekInMySQLResultsToOffset()' in 'include.inc.php')
  136. if (preg_match("/^cli/i", $client)) // if the query originated from a command line client such as the "refbase" CLI client ("cli-refbase-1.0")
  137. $showMaxRows = $showRows; // show only rows up to the value given in '$showRows'
  138. else
  139. $showMaxRows = $rowsFound; // otherwise show all rows
  140. // Setup the basic .bbl document structure:
  141. $latexData = "\\begin{thebibliography}{" . $showMaxRows . "}\n\n";
  142. // Add header with .bst style-specific declarations:
  143. $latexData .= $bblHeader;
  144. // LOOP OVER EACH RECORD:
  145. // Fetch one page of results (or less if on the last page)
  146. // (i.e., upto the limit specified in $showMaxRows) fetch a row into the $row array and ...
  147. for ($rowCounter=0; (($rowCounter < $showMaxRows) && ($row = @ mysqli_fetch_array($result))); $rowCounter++)
  148. {
  149. foreach ($row as $rowFieldName => $rowFieldValue)
  150. // Apply search & replace 'actions' to all fields that are listed in the 'fields' element of the arrays contained in '$latexSearchReplaceActionsArray':
  151. foreach ($latexSearchReplaceActionsArray as $fieldActionsArray)
  152. if (in_array($rowFieldName, $fieldActionsArray['fields']))
  153. $row[$rowFieldName] = searchReplaceText($fieldActionsArray['actions'], $row[$rowFieldName], true); // function 'searchReplaceText()' is defined in 'include.inc.php'
  154. // Order attributes according to the chosen output style & record type:
  155. $record = citeRecord($row, $citeStyle, $citeType, $markupPatternsArray, false); // function 'citeRecord()' is defined in the citation style file given in '$citeStyleFile' (which, in turn, must reside in the 'cite' directory of the refbase root directory), see function 'generateCitations()'
  156. // Print out the current record:
  157. if (!empty($record)) // unless the record buffer is empty...
  158. {
  159. // This is a stupid hack that maps the names of the '$row' array keys to those used
  160. // by the '$formVars' array (which is required by function 'generateCiteKey()')
  161. // (eventually, the '$formVars' array should use the MySQL field names as names for its array keys)
  162. $formVars = buildFormVarsArray($row); // function 'buildFormVarsArray()' is defined in 'include.inc.php'
  163. // Generate or extract the cite key for this record:
  164. // NOTE: currently, the following placeholders are not available for citation output:
  165. // <:keywords:>, <:issn:>, <:area:>, <:notes:>, <:userKeys:>
  166. // if the cite key specification uses one of these placeholders, it will get ignored
  167. $citeKey = generateCiteKey($formVars); // function 'generateCiteKey()' is defined in 'include.inc.php'
  168. // The '\bibitem' command requires a cite key, which is why we'll include the record's serial number
  169. // even when the user's export options specify 'export_cite_keys=no' or 'autogenerate_cite_keys=no':
  170. // NOTE: maybe we should always generate a cite key here, even if 'export_cite_keys=no' or 'autogenerate_cite_keys=no'??
  171. if (empty($citeKey))
  172. $citeKey = $row['serial'];
  173. // Generate the \bibitem option string that's used to build the proper text citation:
  174. $bibItemOptionShortString = parsePlaceholderString($formVars, $bibItemOptionShort, "<:authors[2| and | et~al.]:><(:year:)>"); // function 'parsePlaceholderString()' is defined in 'include.inc.php'
  175. if (!empty($bibItemOptionFull) AND preg_match("/^[^;]+(;[^;]+){" . $maxAuthorCountInShortAuthorList . "}/", $row['author'])) // if the 'author' field does contain more authors as specified in '$maxAuthorCountInShortAuthorList'
  176. {
  177. // include full author list:
  178. $bibItemOptionString = parsePlaceholderString($formVars, $bibItemOptionFull, "<:authors[2| and | et~al.]:><(:year:)><:authors[0|, |]:>");
  179. $bibItemOptionString = preg_replace("/_#_�_~_(?!.*?_#_�_~_)/", $betweenLastAuthorsDelim, $bibItemOptionString); // replace last occurrence of "_#_�_~_"
  180. $bibItemOptionString = preg_replace("/_#_�_~_/", $betweenFullAuthorsDelim, $bibItemOptionString); // replace all other occurrences of "_#_�_~_"
  181. }
  182. else // only include short author list:
  183. $bibItemOptionString = $bibItemOptionShortString;
  184. // In case of duplicate text citation strings, append a letter to the year in the \bibitem option string as well as in the formatted reference:
  185. // NOTE: - this is not fool proof and currently only works if there are no more than 26 duplicate text citations (letters a-z)
  186. // - the below replace actions get triggered on the first four-digit number that happens to be identical to the record's year, but depending on the reference format, this may not be the actual year of the reference but some other number!
  187. for ($i=0; isset($bibItemsArray[$bibItemOptionShortString]); $i++)
  188. {
  189. // Update the existing \bibitem entry (that has an identical text citation string) and append an "a" to it's year items:
  190. if ($i == 0)
  191. {
  192. $dedupPrefixQuoted = preg_quote($dedupPrefix, "/"); // escape meta characters (including '/' that is used as delimiter for the PCRE replace functions below and which gets passed as second argument)
  193. $dedupSuffixQuoted = preg_quote($dedupSuffix, "/");
  194. $oldBibItemOptionShortString = preg_replace("/^\[(.+)\]$/", "\\1", $bibItemOptionShortString); // remove square brackets from short \bibitem option string (which is required for the subsequent 'str_replace()' action to work with short and full option strings)
  195. $newBibItemOptionShortString = preg_replace("/(" . $row['year'] . ")(" . $dedupPrefixQuoted . "[a-z]*" . $dedupSuffixQuoted . ")*/", "\\1" . $dedupPrefix . "a" . $dedupSuffix, $oldBibItemOptionShortString, 1);
  196. $bibItemsArray[$bibItemOptionShortString] = str_replace($oldBibItemOptionShortString, $newBibItemOptionShortString, $bibItemsArray[$bibItemOptionShortString]);
  197. $bibItemsArray[$bibItemOptionShortString] = preg_replace("/(" . $row['year'] . ")(" . $dedupPrefixQuoted . "[a-z]*" . $dedupSuffixQuoted . ")*(?!.*?" . $row['year'] . ")/", "\\1" . $dedupPrefix . "a" . $dedupSuffix, $bibItemsArray[$bibItemOptionShortString], 1); // note that this will fail if the formatted reference contains a number after the year that is identical to the year!
  198. }
  199. // Append a letter to the year items of the current \bibitem entry:
  200. $bibItemOptionShortString = preg_replace("/(" . $row['year'] . ")(" . $dedupPrefixQuoted . "[a-z]*" . $dedupSuffixQuoted . ")*/e", "'\\1$dedupPrefix'.chr(98 + $i).'$dedupSuffix'", $bibItemOptionShortString, 1); // the 'e' modifier allows to execute PHP code within the replacement pattern
  201. $bibItemOptionString = preg_replace("/(" . $row['year'] . ")(" . $dedupPrefixQuoted . "[a-z]*" . $dedupSuffixQuoted . ")*/e", "'\\1$dedupPrefix'.chr(98 + $i).'$dedupSuffix'", $bibItemOptionString, 1);
  202. $record = preg_replace("/(" . $row['year'] . ")(" . $dedupPrefixQuoted . "[a-z]*" . $dedupSuffixQuoted . ")*/e", "'\\1$dedupPrefix'.chr(98 + $i).'$dedupSuffix'", $record, 1);
  203. }
  204. // Attempt to convert higher ASCII chars (i.e., characters with an ASCII value of >= 128) to their corresponding LaTeX entities:
  205. if ($contentTypeCharset == "UTF-8")
  206. $recordEncoded = searchReplaceText($transtab_unicode_latex, $record, false); // function 'searchReplaceText()' is defined in 'include.inc.php'
  207. else
  208. $recordEncoded = searchReplaceText($transtab_latin1_latex, $record, false);
  209. // We have to make sure that the \bibitem option string also contains proper LaTeX entities:
  210. if ($contentTypeCharset == "UTF-8")
  211. $bibItemOptionStringEncoded = searchReplaceText($transtab_unicode_latex, $bibItemOptionString, false);
  212. else
  213. $bibItemOptionStringEncoded = searchReplaceText($transtab_latin1_latex, $bibItemOptionString, false);
  214. // Copy the current \bibitem entry to the array of generated \bibitem entries:
  215. $bibItemsArray[$bibItemOptionShortString] = $bibItemCommand . $bibItemOptionStringEncoded . "{" . $citeKey . "} " . $recordEncoded;
  216. }
  217. }
  218. $latexData .= implode("\n\n", $bibItemsArray) . "\n\n";
  219. $latexData .= "\\end{thebibliography}\n\n";
  220. return $latexData;
  221. }
  222. // --- END CITATION FORMAT ---
  223. ?>