local-search.js 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278
  1. /* global CONFIG */
  2. window.addEventListener('DOMContentLoaded', () => {
  3. // Popup Window
  4. let isfetched = false;
  5. let datas;
  6. let isXml = true;
  7. // Search DB path
  8. let searchPath = CONFIG.path;
  9. if (searchPath.length === 0) {
  10. searchPath = 'search.xml';
  11. } else if (searchPath.endsWith('json')) {
  12. isXml = false;
  13. }
  14. const input = document.querySelector('.search-input');
  15. const resultContent = document.getElementById('search-result');
  16. const getIndexByWord = (word, text, caseSensitive) => {
  17. if (CONFIG.localsearch.unescape) {
  18. let div = document.createElement('div');
  19. div.innerText = word;
  20. word = div.innerHTML;
  21. }
  22. let wordLen = word.length;
  23. if (wordLen === 0) return [];
  24. let startPosition = 0;
  25. let position = [];
  26. let index = [];
  27. if (!caseSensitive) {
  28. text = text.toLowerCase();
  29. word = word.toLowerCase();
  30. }
  31. while ((position = text.indexOf(word, startPosition)) > -1) {
  32. index.push({ position, word });
  33. startPosition = position + wordLen;
  34. }
  35. return index;
  36. };
  37. // Merge hits into slices
  38. const mergeIntoSlice = (start, end, index, searchText) => {
  39. let item = index[index.length - 1];
  40. let { position, word } = item;
  41. let hits = [];
  42. let searchTextCountInSlice = 0;
  43. while (position + word.length <= end && index.length !== 0) {
  44. if (word === searchText) {
  45. searchTextCountInSlice++;
  46. }
  47. hits.push({
  48. position,
  49. length: word.length
  50. });
  51. let wordEnd = position + word.length;
  52. // Move to next position of hit
  53. index.pop();
  54. while (index.length !== 0) {
  55. item = index[index.length - 1];
  56. position = item.position;
  57. word = item.word;
  58. if (wordEnd > position) {
  59. index.pop();
  60. } else {
  61. break;
  62. }
  63. }
  64. }
  65. return {
  66. hits,
  67. start,
  68. end,
  69. searchTextCount: searchTextCountInSlice
  70. };
  71. };
  72. // Highlight title and content
  73. const highlightKeyword = (text, slice) => {
  74. let result = '';
  75. let prevEnd = slice.start;
  76. slice.hits.forEach(hit => {
  77. result += text.substring(prevEnd, hit.position);
  78. let end = hit.position + hit.length;
  79. result += `<b class="search-keyword">${text.substring(hit.position, end)}</b>`;
  80. prevEnd = end;
  81. });
  82. result += text.substring(prevEnd, slice.end);
  83. return result;
  84. };
  85. const inputEventFunction = () => {
  86. if (!isfetched) return;
  87. let searchText = input.value.trim().toLowerCase();
  88. let keywords = searchText.split(/[-\s]+/);
  89. if (keywords.length > 1) {
  90. keywords.push(searchText);
  91. }
  92. let resultItems = [];
  93. if (searchText.length > 0) {
  94. // Perform local searching
  95. datas.forEach(({ title, content, url }) => {
  96. let titleInLowerCase = title.toLowerCase();
  97. let contentInLowerCase = content.toLowerCase();
  98. let indexOfTitle = [];
  99. let indexOfContent = [];
  100. let searchTextCount = 0;
  101. keywords.forEach(keyword => {
  102. indexOfTitle = indexOfTitle.concat(getIndexByWord(keyword, titleInLowerCase, false));
  103. indexOfContent = indexOfContent.concat(getIndexByWord(keyword, contentInLowerCase, false));
  104. });
  105. // Show search results
  106. if (indexOfTitle.length > 0 || indexOfContent.length > 0) {
  107. let hitCount = indexOfTitle.length + indexOfContent.length;
  108. // Sort index by position of keyword
  109. [indexOfTitle, indexOfContent].forEach(index => {
  110. index.sort((itemLeft, itemRight) => {
  111. if (itemRight.position !== itemLeft.position) {
  112. return itemRight.position - itemLeft.position;
  113. }
  114. return itemLeft.word.length - itemRight.word.length;
  115. });
  116. });
  117. let slicesOfTitle = [];
  118. if (indexOfTitle.length !== 0) {
  119. let tmp = mergeIntoSlice(0, title.length, indexOfTitle, searchText);
  120. searchTextCount += tmp.searchTextCountInSlice;
  121. slicesOfTitle.push(tmp);
  122. }
  123. let slicesOfContent = [];
  124. while (indexOfContent.length !== 0) {
  125. let item = indexOfContent[indexOfContent.length - 1];
  126. let { position, word } = item;
  127. // Cut out 100 characters
  128. let start = position - 20;
  129. let end = position + 80;
  130. if (start < 0) {
  131. start = 0;
  132. }
  133. if (end < position + word.length) {
  134. end = position + word.length;
  135. }
  136. if (end > content.length) {
  137. end = content.length;
  138. }
  139. let tmp = mergeIntoSlice(start, end, indexOfContent, searchText);
  140. searchTextCount += tmp.searchTextCountInSlice;
  141. slicesOfContent.push(tmp);
  142. }
  143. // Sort slices in content by search text's count and hits' count
  144. slicesOfContent.sort((sliceLeft, sliceRight) => {
  145. if (sliceLeft.searchTextCount !== sliceRight.searchTextCount) {
  146. return sliceRight.searchTextCount - sliceLeft.searchTextCount;
  147. } else if (sliceLeft.hits.length !== sliceRight.hits.length) {
  148. return sliceRight.hits.length - sliceLeft.hits.length;
  149. }
  150. return sliceLeft.start - sliceRight.start;
  151. });
  152. // Select top N slices in content
  153. let upperBound = parseInt(CONFIG.localsearch.top_n_per_article, 10);
  154. if (upperBound >= 0) {
  155. slicesOfContent = slicesOfContent.slice(0, upperBound);
  156. }
  157. let resultItem = '';
  158. if (slicesOfTitle.length !== 0) {
  159. resultItem += `<li><a href="${url}" class="search-result-title">${highlightKeyword(title, slicesOfTitle[0])}</a>`;
  160. } else {
  161. resultItem += `<li><a href="${url}" class="search-result-title">${title}</a>`;
  162. }
  163. slicesOfContent.forEach(slice => {
  164. resultItem += `<a href="${url}"><p class="search-result">${highlightKeyword(content, slice)}...</p></a>`;
  165. });
  166. resultItem += '</li>';
  167. resultItems.push({
  168. item: resultItem,
  169. id : resultItems.length,
  170. hitCount,
  171. searchTextCount
  172. });
  173. }
  174. });
  175. }
  176. if (keywords.length === 1 && keywords[0] === '') {
  177. resultContent.innerHTML = '<div id="no-result"><i class="fa fa-search fa-5x"></i></div>';
  178. } else if (resultItems.length === 0) {
  179. resultContent.innerHTML = '<div id="no-result"><i class="fa fa-frown-o fa-5x"></i></div>';
  180. } else {
  181. resultItems.sort((resultLeft, resultRight) => {
  182. if (resultLeft.searchTextCount !== resultRight.searchTextCount) {
  183. return resultRight.searchTextCount - resultLeft.searchTextCount;
  184. } else if (resultLeft.hitCount !== resultRight.hitCount) {
  185. return resultRight.hitCount - resultLeft.hitCount;
  186. }
  187. return resultRight.id - resultLeft.id;
  188. });
  189. resultContent.innerHTML = `<ul class="search-result-list">${resultItems.map(result => result.item).join('')}</ul>`;
  190. window.pjax && window.pjax.refresh(resultContent);
  191. }
  192. };
  193. const fetchData = () => {
  194. fetch(CONFIG.root + searchPath)
  195. .then(response => response.text())
  196. .then(res => {
  197. // Get the contents from search data
  198. isfetched = true;
  199. datas = isXml ? [...new DOMParser().parseFromString(res, 'text/xml').querySelectorAll('entry')].map(element => {
  200. return {
  201. title : element.querySelector('title').textContent,
  202. content: element.querySelector('content').textContent,
  203. url : element.querySelector('url').textContent
  204. };
  205. }) : JSON.parse(res);
  206. // Only match articles with not empty titles
  207. datas = datas.filter(data => data.title).map(data => {
  208. data.title = data.title.trim();
  209. data.content = data.content ? data.content.trim().replace(/<[^>]+>/g, '') : '';
  210. data.url = decodeURIComponent(data.url).replace(/\/{2,}/g, '/');
  211. return data;
  212. });
  213. // Remove loading animation
  214. document.getElementById('no-result').innerHTML = '<i class="fa fa-search fa-5x"></i>';
  215. inputEventFunction();
  216. });
  217. };
  218. if (CONFIG.localsearch.preload) {
  219. fetchData();
  220. }
  221. if (CONFIG.localsearch.trigger === 'auto') {
  222. input.addEventListener('input', inputEventFunction);
  223. } else {
  224. document.querySelector('.search-icon').addEventListener('click', inputEventFunction);
  225. input.addEventListener('keypress', event => {
  226. if (event.key === 'Enter') {
  227. inputEventFunction();
  228. }
  229. });
  230. }
  231. // Handle and trigger popup window
  232. document.querySelectorAll('.popup-trigger').forEach(element => {
  233. element.addEventListener('click', () => {
  234. document.body.style.overflow = 'hidden';
  235. document.querySelector('.search-pop-overlay').classList.add('search-active');
  236. input.focus();
  237. if (!isfetched) fetchData();
  238. });
  239. });
  240. // Monitor main search box
  241. const onPopupClose = () => {
  242. document.body.style.overflow = '';
  243. document.querySelector('.search-pop-overlay').classList.remove('search-active');
  244. };
  245. document.querySelector('.search-pop-overlay').addEventListener('click', event => {
  246. if (event.target === document.querySelector('.search-pop-overlay')) {
  247. onPopupClose();
  248. }
  249. });
  250. document.querySelector('.popup-btn-close').addEventListener('click', onPopupClose);
  251. window.addEventListener('pjax:success', onPopupClose);
  252. window.addEventListener('keyup', event => {
  253. if (event.key === 'Escape') {
  254. onPopupClose();
  255. }
  256. });
  257. });