functions.php 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254
  1. <?php
  2. /* Copyright (c) 2019 Geert Bergman (geert@scrivo.nl), highlight.php
  3. *
  4. * Redistribution and use in source and binary forms, with or without
  5. * modification, are permitted provided that the following conditions are met:
  6. *
  7. * 1. Redistributions of source code must retain the above copyright notice,
  8. * this list of conditions and the following disclaimer.
  9. * 2. Redistributions in binary form must reproduce the above copyright notice,
  10. * this list of conditions and the following disclaimer in the documentation
  11. * and/or other materials provided with the distribution.
  12. * 3. Neither the name of "highlight.js", "highlight.php", nor the names of its
  13. * contributors may be used to endorse or promote products derived from this
  14. * software without specific prior written permission.
  15. *
  16. * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  17. * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  18. * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
  19. * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
  20. * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
  21. * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
  22. * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
  23. * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
  24. * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
  25. * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
  26. * POSSIBILITY OF SUCH DAMAGE.
  27. */
  28. namespace HighlightUtilities;
  29. require_once __DIR__ . '/_internals.php';
  30. require_once __DIR__ . '/_themeColors.php';
  31. /**
  32. * Get a list of available stylesheets.
  33. *
  34. * By default, a list of filenames without the `.css` extension will be returned.
  35. * This can be configured with the `$filePaths` argument.
  36. *
  37. * @api
  38. *
  39. * @since 9.15.8.1
  40. *
  41. * @param bool $filePaths Return absolute paths to stylesheets instead
  42. *
  43. * @return string[]
  44. */
  45. function getAvailableStyleSheets($filePaths = false)
  46. {
  47. $results = array();
  48. $folder = getStyleSheetFolder();
  49. $dh = @dir($folder);
  50. if ($dh) {
  51. while (($entry = $dh->read()) !== false) {
  52. if (substr($entry, -4, 4) !== ".css") {
  53. continue;
  54. }
  55. if ($filePaths) {
  56. $results[] = implode(DIRECTORY_SEPARATOR, array($folder, $entry));
  57. } else {
  58. $results[] = basename($entry, ".css");
  59. }
  60. }
  61. $dh->close();
  62. }
  63. return $results;
  64. }
  65. /**
  66. * Get the RGB representation used for the background of a given theme as an
  67. * array of three numbers.
  68. *
  69. * @api
  70. *
  71. * @since 9.18.1.1
  72. *
  73. * @param string $name The stylesheet name (with or without the extension)
  74. *
  75. * @throws \DomainException when no stylesheet with this name exists
  76. *
  77. * @return float[] An array representing RGB numerical values
  78. */
  79. function getThemeBackgroundColor($name)
  80. {
  81. return _getThemeBackgroundColor(_getNoCssExtension($name));
  82. }
  83. /**
  84. * Get the contents of the given stylesheet.
  85. *
  86. * @api
  87. *
  88. * @since 9.15.8.1
  89. *
  90. * @param string $name The stylesheet name (with or without the extension)
  91. *
  92. * @throws \DomainException when the no stylesheet with this name exists
  93. *
  94. * @return false|string The CSS content of the stylesheet or FALSE when
  95. * the stylesheet content could be read
  96. */
  97. function getStyleSheet($name)
  98. {
  99. $path = getStyleSheetPath($name);
  100. return file_get_contents($path);
  101. }
  102. /**
  103. * Get the absolute path to the folder containing the stylesheets distributed in this package.
  104. *
  105. * @api
  106. *
  107. * @since 9.15.8.1
  108. *
  109. * @return string An absolute path to the folder
  110. */
  111. function getStyleSheetFolder()
  112. {
  113. $paths = array(__DIR__, '..', 'styles');
  114. return implode(DIRECTORY_SEPARATOR, $paths);
  115. }
  116. /**
  117. * Get the absolute path to a given stylesheet distributed in this package.
  118. *
  119. * @api
  120. *
  121. * @since 9.15.8.1
  122. *
  123. * @param string $name The stylesheet name (with or without the extension)
  124. *
  125. * @throws \DomainException when the no stylesheet with this name exists
  126. *
  127. * @return string The absolute path to the stylesheet with the given name
  128. */
  129. function getStyleSheetPath($name)
  130. {
  131. $name = _getNoCssExtension($name);
  132. $path = implode(DIRECTORY_SEPARATOR, array(getStyleSheetFolder(), $name)) . ".css";
  133. if (!file_exists($path)) {
  134. throw new \DomainException("There is no stylesheet with by the name of '$name'.");
  135. }
  136. return $path;
  137. }
  138. /**
  139. * Get the directory path for the bundled languages folder.
  140. *
  141. * @api
  142. *
  143. * @since 9.18.1.4
  144. *
  145. * @return string An absolute path to the bundled languages folder
  146. */
  147. function getLanguagesFolder()
  148. {
  149. return __DIR__ . '/../Highlight/languages';
  150. }
  151. /**
  152. * Get the file path for the specified bundled language definition.
  153. *
  154. * @api
  155. *
  156. * @since 9.18.1.4
  157. *
  158. * @param string $name The slug of the language to look for
  159. *
  160. * @throws \DomainException when the no definition for this language exists
  161. *
  162. * @return string
  163. */
  164. function getLanguageDefinitionPath($name)
  165. {
  166. $path = getLanguagesFolder() . '/' . $name . '.json';
  167. if (!file_exists($path)) {
  168. throw new \DomainException("There is no language definition for $name");
  169. }
  170. return $path;
  171. }
  172. /**
  173. * Convert the HTML generated by Highlighter and split it up into an array of lines.
  174. *
  175. * @api
  176. *
  177. * @since 9.18.1.6 `RuntimeException` and `UnexpectedValueException` can no longer be thrown.
  178. * @since 9.15.6.1
  179. *
  180. * @param string $html An HTML string generated by `Highlighter::highlight()`
  181. *
  182. * @return string[]|false An array of lines of code as strings. False if an error occurred in splitting up by lines
  183. */
  184. function splitCodeIntoArray($html)
  185. {
  186. if (trim($html) === "") {
  187. return array();
  188. }
  189. $queuedPrefix = '';
  190. $regexWorkspace = array();
  191. $rawLines = preg_split('/\R/u', $html);
  192. if ($rawLines === false) {
  193. return false;
  194. }
  195. foreach ($rawLines as &$rawLine) {
  196. // If the previous line has been marked as "open", then we'll have something
  197. // in our queue
  198. if ($queuedPrefix !== '') {
  199. $rawLine = $queuedPrefix . $rawLine;
  200. $queuedPrefix = '';
  201. }
  202. // Find how many opening `<span>` tags exist on this line
  203. preg_match_all('/<span[^>]*+>/u', $rawLine, $regexWorkspace);
  204. $openingTags = count($regexWorkspace[0]);
  205. // Find all of the closing `</span>` tags that exist on this line
  206. preg_match_all('/<\/span>/u', $rawLine, $regexWorkspace);
  207. $closingTags = count($regexWorkspace[0]);
  208. // If the number of opening tags matches the number of closing tags, then
  209. // we don't have any new tags that span multiple lines
  210. if ($openingTags === $closingTags) {
  211. continue;
  212. }
  213. // Find all of the complete `<span>` tags and remove them from a working
  214. // copy of the line. Then we'll be left with just opening tags.
  215. $workingLine = preg_replace('/<span[^>]*+>[^<]*+<\/span>/u', '', $rawLine);
  216. preg_match_all('/<span[^>]*+>/u', $workingLine, $regexWorkspace);
  217. $queuedPrefix = implode('', $regexWorkspace[0]);
  218. // Close all of the remaining open tags on this line
  219. $diff = str_repeat('</span>', $openingTags - $closingTags);
  220. $rawLine .= $diff;
  221. }
  222. return $rawLines;
  223. }