mirror of
https://github.com/McShelby/hugo-theme-relearn.git
synced 2025-01-18 19:00:24 +00:00
243 lines
9.9 KiB
JavaScript
243 lines
9.9 KiB
JavaScript
//import { init, search } from './lunr-adapter.js';
|
|
import { init, search } from './orama-adapter.js';
|
|
|
|
(function(){
|
|
|
|
function escapeRegex( string ){
|
|
return string.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
|
}
|
|
|
|
window.relearn = window.relearn || {};
|
|
|
|
window.relearn.executeInitialSearch =
|
|
function executeInitialSearch(){
|
|
if( window.relearn.isSearchInterfaceReady && window.relearn.isSearchEngineReady ){
|
|
var input = document.querySelector('#R-search-by-detail');
|
|
if( !input ){
|
|
return;
|
|
}
|
|
var value = input.value;
|
|
executeSearch( value );
|
|
}
|
|
}
|
|
|
|
window.relearn.executeTriggeredSearch =
|
|
function executeTriggeredSearch(){
|
|
var input = document.querySelector('#R-search-by-detail');
|
|
if( !input ){
|
|
return;
|
|
}
|
|
var value = input.value;
|
|
executeSearch( 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 );
|
|
}
|
|
}
|
|
|
|
function executeHistorySearch( 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
|
|
executeSearch( search );
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function executeSearch( value ) {
|
|
var input = document.querySelector('#R-search-by-detail');
|
|
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];
|
|
});
|
|
};
|
|
|
|
var results = document.querySelector('#R-searchresults');
|
|
var hint = document.querySelector('.searchhint');
|
|
hint.innerText = '';
|
|
results.textContent = '';
|
|
(async function(){
|
|
var a = await search( value );
|
|
if( a.length ){
|
|
hint.innerText = resolvePlaceholders( window.T_N_results_found, [ value, a.length ] );
|
|
a.forEach( function(item){
|
|
var page = item.page;
|
|
var context = [];
|
|
if( item.matches && item.matches.length ){
|
|
var numContextWords = 10;
|
|
var contextPattern = '(?:\\S+ +){0,' + numContextWords + '}\\S*\\b(?:' +
|
|
escapeRegex( item.matches[0] ) +
|
|
')\\b\\S*(?: +\\S+){0,' + numContextWords + '}';
|
|
context = page.content.match(new RegExp(contextPattern, 'i'));
|
|
if( !context ){
|
|
item.matches.shift();
|
|
var contextPattern = '(?:\\S+ +){0,' + numContextWords + '}\\S*\\b(?:' +
|
|
item.matches.map( escapeRegex ).join('|') +
|
|
')\\b\\S*(?: +\\S+){0,' + numContextWords + '}';
|
|
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;
|
|
}
|
|
})();
|
|
}
|
|
|
|
function initSearchAfterLoad(){
|
|
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 );
|
|
}
|
|
|
|
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: async function( term, response ) {
|
|
let a = await search( term )
|
|
response( a );
|
|
},
|
|
/* renderItem displays individual search results */
|
|
renderItem: function( item, term ) {
|
|
var page = item.page;
|
|
var context = [];
|
|
if( item.matches && item.matches.length ){
|
|
var numContextWords = 2;
|
|
var contextPattern = '(?:\\S+ +){0,' + numContextWords + '}\\S*\\b(?:' +
|
|
escapeRegex( item.matches[0] ) +
|
|
')\\b\\S*(?: +\\S+){0,' + numContextWords + '}';
|
|
context = page.content.match(new RegExp(contextPattern, 'i'));
|
|
if( !context ){
|
|
item.matches.shift();
|
|
var contextPattern = '(?:\\S+ +){0,' + numContextWords + '}\\S*\\b(?:' +
|
|
item.matches.map( escapeRegex ).join('|') +
|
|
')\\b\\S*(?: +\\S+){0,' + numContextWords + '}';
|
|
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 initSearch(){
|
|
init();
|
|
|
|
window.addEventListener( 'popstate', executeHistorySearch );
|
|
|
|
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 );
|
|
}
|
|
});
|
|
}
|
|
ready( initSearchAfterLoad );
|
|
}
|
|
|
|
initSearch();
|
|
})();
|