window.relearn = window.relearn || {}; window.relearn.runInitialSearch = function () { if (window.relearn.isSearchInit && window.relearn.isLunrInit) { var input = document.querySelector('#R-search-by-detail'); if (!input) { return; } var value = input.value; searchDetail(value); } }; var lunrIndex, pagesIndex; function initLunrIndex(index) { pagesIndex = index; // Set up Lunr by declaring the fields we use // Also provide their boost level for the ranking lunrIndex = lunr(function () { this.use(lunr.multiLanguage.apply(null, contentLangs)); this.ref('index'); this.field('title', { boost: 15, }); this.field('tags', { boost: 10, }); this.field('content', { boost: 5, }); this.pipeline.remove(lunr.stemmer); this.searchPipeline.remove(lunr.stemmer); // Feed Lunr with each file and let LUnr actually index them pagesIndex.forEach(function (page, idx) { page.index = idx; this.add(page); }, this); }); window.relearn.isLunrInit = true; window.relearn.runInitialSearch(); } function triggerSearch() { var input = document.querySelector('#R-search-by-detail'); if (!input) { return; } var value = input.value; searchDetail(value); // add a new entry to the history after the user // changed the term; this does not reload the page // but will add to the history and update the address bar URL var url = new URL(window.location); var oldValue = url.searchParams.get('search-by'); if (value != oldValue) { var state = window.history.state || {}; state = Object.assign({}, typeof state === 'object' ? state : {}); url.searchParams.set('search-by', value); state.search = url.toString(); // with normal pages, this is handled by the 'pagehide' event, but this // doesn't fire in case of pushState, so we have to do the same thing // here, too state.contentScrollTop = +elc.scrollTop; window.history.pushState(state, '', url); } } window.addEventListener('popstate', function (event) { // restart search if browsed through history if (event.state) { var state = window.history.state || {}; state = Object.assign({}, typeof state === 'object' ? state : {}); if (state.search) { var url = new URL(state.search); if (url.searchParams.has('search-by')) { var search = url.searchParams.get('search-by'); // we have to insert the old search term into the inputs var inputs = document.querySelectorAll('input.search-by'); inputs.forEach(function (e) { e.value = search; var event = document.createEvent('Event'); event.initEvent('input', false, false); e.dispatchEvent(event); }); // recreate the last search results and eventually // restore the previous scrolling position searchDetail(search); } } } }); var input = document.querySelector('#R-search-by-detail'); if (input) { input.addEventListener('keydown', function (event) { // if we are pressing ESC in the searchdetail our focus will // be stolen by the other event handlers, so we have to refocus // here after a short while if (event.key == 'Escape') { setTimeout(function () { input.focus(); }, 0); } }); } function initLunrJs() { // new way to load our search index if (window.index_js_url) { var js = document.createElement('script'); js.src = index_js_url; js.setAttribute('async', ''); js.onload = function () { initLunrIndex(relearn_searchindex); }; js.onerror = function (e) { console.error('Error getting Hugo index file'); }; document.head.appendChild(js); } } /** * Trigger a search in Lunr and transform the result * * @param {String} term * @return {Array} results */ function search(term) { // Find the item in our index corresponding to the Lunr one to have more info // Remove Lunr special search characters: https://lunrjs.com/guides/searching.html term = term.replace(/[*:^~+-]/g, ' '); var searchTerm = lunr .tokenizer(term) .reduce(function (a, token) { return a.concat(searchPatterns(token.str)); }, []) .join(' '); return !searchTerm || !lunrIndex ? [] : lunrIndex.search(searchTerm).map(function (result) { return { index: result.ref, matches: Object.keys(result.matchData.metadata) }; }); } function searchPatterns(word) { // for short words high amounts of typos doesn't make sense // for long words we allow less typos because this largly increases search time var typos = [ { len: -1, typos: 1 }, { len: 60, typos: 2 }, { len: 40, typos: 3 }, { len: 20, typos: 4 }, { len: 16, typos: 3 }, { len: 12, typos: 2 }, { len: 8, typos: 1 }, { len: 4, typos: 0 }, ]; return [ word + '^100', word + '*^10', '*' + word + '^10', word + '~' + typos.reduce(function (a, c, i) { return word.length < c.len ? c : a; }).typos + '^1', ]; } function resolvePlaceholders(s, args) { var args = args || []; // use replace to iterate over the string // select the match and check if the related argument is present // if yes, replace the match with the argument return s.replace(/{([0-9]+)}/g, function (match, index) { // check if the argument is present return typeof args[index] == 'undefined' ? match : args[index]; }); } function searchDetail(value) { var results = document.querySelector('#R-searchresults'); var hint = document.querySelector('.searchhint'); hint.innerText = ''; results.textContent = ''; var a = search(value); if (a.length) { hint.innerText = resolvePlaceholders(window.T_N_results_found, [value, a.length]); a.forEach(function (item) { var page = pagesIndex[item.index]; var numContextWords = 10; var contextPattern = '(?:\\S+ +){0,' + numContextWords + '}\\S*\\b(?:' + item.matches .map(function (match) { return match.replace(/\W/g, '\\$&'); }) .join('|') + ')\\b\\S*(?: +\\S+){0,' + numContextWords + '}'; var context = page.content.match(new RegExp(contextPattern, 'i')); var divsuggestion = document.createElement('a'); divsuggestion.className = 'autocomplete-suggestion'; divsuggestion.setAttribute('data-term', value); divsuggestion.setAttribute('data-title', page.title); divsuggestion.setAttribute('href', window.relearn.relBaseUri + page.uri); divsuggestion.setAttribute('data-context', context); var divtitle = document.createElement('div'); divtitle.className = 'title'; divtitle.innerText = '» ' + page.title; divsuggestion.appendChild(divtitle); var divbreadcrumb = document.createElement('div'); divbreadcrumb.className = 'breadcrumbs'; divbreadcrumb.innerText = page.breadcrumb || ''; divsuggestion.appendChild(divbreadcrumb); if (context) { var divcontext = document.createElement('div'); divcontext.className = 'context'; divcontext.innerText = context || ''; divsuggestion.appendChild(divcontext); } results.appendChild(divsuggestion); }); window.relearn.markSearch(); } else if (value.length) { hint.innerText = resolvePlaceholders(window.T_No_results_found, [value]); } input.focus(); setTimeout(adjustContentWidth, 0); // if we are initiating search because of a browser history // operation, we have to restore the scrolling postion the // user previously has used; if this search isn't initiated // by a browser history operation, it simply does nothing var state = window.history.state || {}; state = Object.assign({}, typeof state === 'object' ? state : {}); if (state.hasOwnProperty('contentScrollTop')) { window.setTimeout(function () { elc.scrollTop = +state.contentScrollTop; }, 10); return; } } initLunrJs(); function startSearch() { var input = document.querySelector('#R-search-by-detail'); if (input) { var state = window.history.state || {}; state = Object.assign({}, typeof state === 'object' ? state : {}); state.search = window.location.toString(); window.history.replaceState(state, '', window.location); } var searchList = new autoComplete({ /* selector for the search box element */ selectorToInsert: 'search:has(.searchbox)', selector: '#R-search-by', /* source is the callback to perform the search */ source: function (term, response) { response(search(term)); }, /* renderItem displays individual search results */ renderItem: function (item, term) { var page = pagesIndex[item.index]; var numContextWords = 2; var contextPattern = '(?:\\S+ +){0,' + numContextWords + '}\\S*\\b(?:' + item.matches .map(function (match) { return match.replace(/\W/g, '\\$&'); }) .join('|') + ')\\b\\S*(?: +\\S+){0,' + numContextWords + '}'; var context = page.content.match(new RegExp(contextPattern, 'i')); var divsuggestion = document.createElement('div'); divsuggestion.className = 'autocomplete-suggestion'; divsuggestion.setAttribute('data-term', term); divsuggestion.setAttribute('data-title', page.title); divsuggestion.setAttribute('data-uri', window.relearn.relBaseUri + page.uri); divsuggestion.setAttribute('data-context', context); var divtitle = document.createElement('div'); divtitle.className = 'title'; divtitle.innerText = '» ' + page.title; divsuggestion.appendChild(divtitle); if (context) { var divcontext = document.createElement('div'); divcontext.className = 'context'; divcontext.innerText = context || ''; divsuggestion.appendChild(divcontext); } return divsuggestion.outerHTML; }, /* onSelect callback fires when a search suggestion is chosen */ onSelect: function (e, term, item) { location.href = item.getAttribute('data-uri'); e.preventDefault(); }, }); } function ready(fn) { if (document.readyState == 'complete') { fn(); } else { document.addEventListener('DOMContentLoaded', fn); } } ready(startSearch);