mirror of
https://github.com/McShelby/hugo-theme-relearn.git
synced 2025-05-28 20:44:02 +00:00
deploy: 348702d8c2
This commit is contained in:
parent
6fccdd8852
commit
caf112c63a
1093 changed files with 29143 additions and 28952 deletions
518
js/search.js
518
js/search.js
|
@ -1,128 +1,130 @@
|
|||
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 );
|
||||
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);
|
||||
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,
|
||||
});
|
||||
|
||||
window.relearn.isLunrInit = true;
|
||||
window.relearn.runInitialSearch();
|
||||
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 );
|
||||
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 );
|
||||
}
|
||||
// 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' );
|
||||
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 );
|
||||
});
|
||||
// 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 );
|
||||
}
|
||||
}
|
||||
// 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 );
|
||||
}
|
||||
});
|
||||
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);
|
||||
}
|
||||
// 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);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -132,158 +134,186 @@ function initLunrJs() {
|
|||
* @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) }
|
||||
});
|
||||
// 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'
|
||||
];
|
||||
// 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 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);
|
||||
|
||||
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;
|
||||
}
|
||||
// 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 );
|
||||
}
|
||||
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();
|
||||
}
|
||||
});
|
||||
};
|
||||
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();
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
ready( startSearch );
|
||||
ready(startSearch);
|
||||
|
|
3220
js/theme.js
3220
js/theme.js
File diff suppressed because it is too large
Load diff
1280
js/variant.js
1280
js/variant.js
File diff suppressed because it is too large
Load diff
Loading…
Add table
Add a link
Reference in a new issue