search.js 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273
  1. (function (compodoc) {
  2. var usePushState = typeof history.pushState !== 'undefined',
  3. // DOM Elements
  4. $body = document.querySelector('body'),
  5. $searchResults,
  6. $searchInput,
  7. $searchList,
  8. $searchTitle,
  9. $searchResultsCount,
  10. $searchQuery,
  11. $mainContainer,
  12. $xsMenu;
  13. // Throttle search
  14. function throttle(fn, wait) {
  15. var timeout;
  16. return function () {
  17. var ctx = this,
  18. args = arguments;
  19. if (!timeout) {
  20. timeout = setTimeout(function () {
  21. timeout = undefined;
  22. fn.apply(ctx, args);
  23. }, wait);
  24. }
  25. };
  26. }
  27. function displayResults(res) {
  28. var noResults = res.count == 0;
  29. var groups = {};
  30. if (noResults) {
  31. $searchResults.classList.add('no-results');
  32. } else {
  33. $searchResults.classList.remove('no-results');
  34. }
  35. // Clear old results
  36. $searchList.innerText = '';
  37. // Display title for research
  38. $searchResultsCount.innerText = res.count;
  39. $searchQuery.innerText = res.query;
  40. // Group result by context
  41. res.results.forEach(function (res) {
  42. var context = res.title.split(' - ')[0];
  43. if (typeof groups[context] === 'undefined') {
  44. groups[context] = {
  45. results: [res]
  46. };
  47. } else {
  48. groups[context].results.push(res);
  49. }
  50. });
  51. var sortedGroups = Object.keys(groups).sort();
  52. for (var i = 0; i < sortedGroups.length; i++) {
  53. var property = sortedGroups[i];
  54. var $li = document.createElement('li');
  55. $li.classList.add('search-results-group');
  56. var finalPropertyLabel = '';
  57. var propertyLabels = property.split('-');
  58. if (
  59. propertyLabels.length === 2 &&
  60. propertyLabels[0] !== 'miscellaneous' &&
  61. propertyLabels[0] !== 'additional'
  62. ) {
  63. finalPropertyLabel =
  64. propertyLabels[0].charAt(0).toUpperCase() +
  65. propertyLabels[0].substring(1) +
  66. ' - ' +
  67. propertyLabels[1].charAt(0).toUpperCase() +
  68. propertyLabels[1].substring(1) +
  69. ' (' +
  70. groups[property].results.length +
  71. ')';
  72. } else if (propertyLabels[0] === 'additional') {
  73. finalPropertyLabel =
  74. 'Additional pages' + ' (' + groups[property].results.length + ')';
  75. } else {
  76. finalPropertyLabel =
  77. propertyLabels[0].charAt(0).toUpperCase() +
  78. propertyLabels[0].substring(1) +
  79. ' (' +
  80. groups[property].results.length +
  81. ')';
  82. }
  83. var $groupTitle = document.createElement('h3');
  84. $groupTitle.innerText = finalPropertyLabel;
  85. $li.appendChild($groupTitle);
  86. var $ulResults = document.createElement('ul');
  87. $ulResults.classList.add('search-results-list');
  88. groups[property].results.forEach(function (res) {
  89. var link = '';
  90. var $liResult = document.createElement('li');
  91. $liResult.classList.add('search-results-item');
  92. switch (COMPODOC_CURRENT_PAGE_DEPTH) {
  93. case 0:
  94. link = './';
  95. break;
  96. case 1:
  97. case 2:
  98. case 3:
  99. case 4:
  100. case 5:
  101. link = '../'.repeat(COMPODOC_CURRENT_PAGE_DEPTH);
  102. break;
  103. }
  104. var finalResLabel =
  105. res.title.split(' - ')[1].charAt(0).toUpperCase() +
  106. res.title.split(' - ')[1].substring(1);
  107. var $link = document.createElement('a');
  108. $link.innerText = finalResLabel;
  109. $link.href = link + res.url;
  110. $liResult.appendChild($link);
  111. $ulResults.appendChild($liResult);
  112. });
  113. $li.appendChild($ulResults);
  114. $searchList.appendChild($li);
  115. }
  116. }
  117. function launchSearch(q) {
  118. $body.classList.add('with-search');
  119. if ($xsMenu.style.display === 'block') {
  120. $mainContainer.style.height = 'calc(100% - 100px)';
  121. $mainContainer.style.marginTop = '100px';
  122. }
  123. throttle(
  124. compodoc.search.query(q, 0, MAX_SEARCH_RESULTS).then(function (results) {
  125. displayResults(results);
  126. }),
  127. 1000
  128. );
  129. }
  130. function closeSearch() {
  131. $body.classList.remove('with-search');
  132. if ($xsMenu.style.display === 'block') {
  133. $mainContainer.style.height = 'calc(100% - 50px)';
  134. }
  135. }
  136. function bindMenuButton() {
  137. document.getElementById('btn-menu').addEventListener('click', function () {
  138. if ($xsMenu.style.display === 'none') {
  139. $body.classList.remove('with-search');
  140. $mainContainer.style.height = 'calc(100% - 50px)';
  141. }
  142. $searchInputs.forEach((item, index) => {
  143. item.value = '';
  144. });
  145. });
  146. }
  147. function bindSearch() {
  148. // Bind DOM
  149. $searchInputs = document.querySelectorAll('#book-search-input input');
  150. $searchResults = document.querySelector('.search-results');
  151. $searchList = $searchResults.querySelector('.search-results-list');
  152. $searchTitle = $searchResults.querySelector('.search-results-title');
  153. $searchResultsCount = $searchTitle.querySelector('.search-results-count');
  154. $searchQuery = $searchTitle.querySelector('.search-query');
  155. $mainContainer = document.querySelector('.container-fluid');
  156. $xsMenu = document.querySelector('.xs-menu');
  157. // Launch query based on input content
  158. function handleUpdate(item) {
  159. var q = item.value;
  160. if (q.length == 0) {
  161. closeSearch();
  162. window.location.href = window.location.href.replace(window.location.search, '');
  163. } else {
  164. launchSearch(q);
  165. }
  166. }
  167. // Detect true content change in search input
  168. var propertyChangeUnbound = false;
  169. $searchInputs.forEach((item, index) => {
  170. // HTML5 (IE9 & others)
  171. item.addEventListener('input', function (e) {
  172. handleUpdate(this);
  173. });
  174. // Workaround for IE < 9
  175. item.addEventListener('propertychange', function (e) {
  176. if (e.originalEvent.propertyName == 'value') {
  177. handleUpdate(this);
  178. }
  179. });
  180. // Push to history on blur
  181. item.addEventListener('blur', function (e) {
  182. // Update history state
  183. if (usePushState) {
  184. var uri = updateQueryString('q', this.value);
  185. if (this.value !== '') {
  186. history.pushState({ path: uri }, null, uri);
  187. }
  188. }
  189. });
  190. });
  191. }
  192. function launchSearchFromQueryString() {
  193. var q = getParameterByName('q');
  194. if (q && q.length > 0) {
  195. // Update search inputs
  196. $searchInputs.forEach((item, index) => {
  197. item.value = q;
  198. });
  199. // Launch search
  200. launchSearch(q);
  201. }
  202. }
  203. compodoc.addEventListener(compodoc.EVENTS.SEARCH_READY, function (event) {
  204. bindSearch();
  205. bindMenuButton();
  206. launchSearchFromQueryString();
  207. });
  208. function getParameterByName(name) {
  209. var url = window.location.href;
  210. name = name.replace(/[\[\]]/g, '\\$&');
  211. var regex = new RegExp('[?&]' + name + '(=([^&#]*)|&|#|$)', 'i'),
  212. results = regex.exec(url);
  213. if (!results) return null;
  214. if (!results[2]) return '';
  215. return decodeURIComponent(results[2].replace(/\+/g, ' '));
  216. }
  217. function updateQueryString(key, value) {
  218. value = encodeURIComponent(value);
  219. var url = window.location.href;
  220. var re = new RegExp('([?&])' + key + '=.*?(&|#|$)(.*)', 'gi'),
  221. hash;
  222. if (re.test(url)) {
  223. if (typeof value !== 'undefined' && value !== null)
  224. return url.replace(re, '$1' + key + '=' + value + '$2$3');
  225. else {
  226. hash = url.split('#');
  227. url = hash[0].replace(re, '$1$3').replace(/(&|\?)$/, '');
  228. if (typeof hash[1] !== 'undefined' && hash[1] !== null) url += '#' + hash[1];
  229. return url;
  230. }
  231. } else {
  232. if (typeof value !== 'undefined' && value !== null) {
  233. var separator = url.indexOf('?') !== -1 ? '&' : '?';
  234. hash = url.split('#');
  235. url = hash[0] + separator + key + '=' + value;
  236. if (typeof hash[1] !== 'undefined' && hash[1] !== null) url += '#' + hash[1];
  237. return url;
  238. } else return url;
  239. }
  240. }
  241. })(compodoc);