2019-10-27 11:58:58 +01:00
|
|
|
'use strict';
|
2019-07-16 14:34:02 +02:00
|
|
|
|
2021-05-19 20:26:21 +02:00
|
|
|
{{ $searchDataFile := printf "%s.search-data.json" .Language.Lang }}
|
|
|
|
{{ $searchData := resources.Get "search-data.json" | resources.ExecuteAsTemplate $searchDataFile . | resources.Minify | resources.Fingerprint }}
|
|
|
|
{{ $searchConfig := i18n "bookSearchConfig" | default "{}" }}
|
2019-11-10 14:35:07 +01:00
|
|
|
|
2020-07-06 22:17:36 +02:00
|
|
|
(function () {
|
2021-05-19 20:26:21 +02:00
|
|
|
const searchDataURL = '{{ $searchData.RelPermalink }}';
|
|
|
|
const indexConfig = Object.assign({{ $searchConfig }}, {
|
|
|
|
doc: {
|
|
|
|
id: 'id',
|
|
|
|
field: ['title', 'content'],
|
|
|
|
store: ['title', 'href', 'section']
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-10-20 19:02:32 +02:00
|
|
|
const input = document.querySelector('#book-search-input');
|
|
|
|
const results = document.querySelector('#book-search-results');
|
2019-07-16 14:34:02 +02:00
|
|
|
|
2020-04-15 23:24:51 +02:00
|
|
|
if (!input) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-10-20 19:02:32 +02:00
|
|
|
input.addEventListener('focus', init);
|
|
|
|
input.addEventListener('keyup', search);
|
2019-07-16 14:34:02 +02:00
|
|
|
|
2020-01-19 21:57:53 +01:00
|
|
|
document.addEventListener('keypress', focusSearchFieldOnKeyPress);
|
|
|
|
|
2020-04-12 20:50:03 +02:00
|
|
|
/**
|
|
|
|
* @param {Event} event
|
|
|
|
*/
|
|
|
|
function focusSearchFieldOnKeyPress(event) {
|
2021-11-19 14:24:28 +01:00
|
|
|
if (event.target.value !== undefined) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-01-19 21:57:53 +01:00
|
|
|
if (input === document.activeElement) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2020-04-12 20:50:03 +02:00
|
|
|
const characterPressed = String.fromCharCode(event.charCode);
|
2020-01-19 21:57:53 +01:00
|
|
|
if (!isHotkey(characterPressed)) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
input.focus();
|
2020-04-12 20:50:03 +02:00
|
|
|
event.preventDefault();
|
2020-01-19 21:57:53 +01:00
|
|
|
}
|
|
|
|
|
2020-04-12 20:50:03 +02:00
|
|
|
/**
|
|
|
|
* @param {String} character
|
|
|
|
* @returns {Boolean}
|
|
|
|
*/
|
2020-01-19 21:57:53 +01:00
|
|
|
function isHotkey(character) {
|
|
|
|
const dataHotkeys = input.getAttribute('data-hotkeys') || '';
|
|
|
|
return dataHotkeys.indexOf(character) >= 0;
|
|
|
|
}
|
|
|
|
|
2019-07-16 14:34:02 +02:00
|
|
|
function init() {
|
2019-10-27 11:58:58 +01:00
|
|
|
input.removeEventListener('focus', init); // init once
|
2019-07-17 17:56:12 +02:00
|
|
|
input.required = true;
|
|
|
|
|
2021-05-19 20:26:21 +02:00
|
|
|
fetch(searchDataURL)
|
|
|
|
.then(pages => pages.json())
|
|
|
|
.then(pages => {
|
|
|
|
window.bookSearchIndex = FlexSearch.create('balance', indexConfig);
|
|
|
|
window.bookSearchIndex.add(pages);
|
|
|
|
})
|
|
|
|
.then(() => input.required = false)
|
|
|
|
.then(search);
|
2019-07-15 18:25:21 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
function search() {
|
2019-07-16 14:34:02 +02:00
|
|
|
while (results.firstChild) {
|
|
|
|
results.removeChild(results.firstChild);
|
|
|
|
}
|
|
|
|
|
2019-07-17 17:56:12 +02:00
|
|
|
if (!input.value) {
|
|
|
|
return;
|
2019-07-17 13:42:39 +02:00
|
|
|
}
|
2019-07-16 14:34:02 +02:00
|
|
|
|
2019-10-27 11:58:58 +01:00
|
|
|
const searchHits = window.bookSearchIndex.search(input.value, 10);
|
2020-07-06 22:17:36 +02:00
|
|
|
searchHits.forEach(function (page) {
|
2024-01-30 14:31:20 +01:00
|
|
|
const li = element('<li><a href></a><small contenteditable="false" ></small></li>');
|
|
|
|
const a = li.querySelector('a'), small = li.querySelector('small');
|
2019-07-17 13:42:39 +02:00
|
|
|
|
|
|
|
a.href = page.href;
|
|
|
|
a.textContent = page.title;
|
2024-01-30 14:31:20 +01:00
|
|
|
|
|
|
|
fetch(page.href)
|
|
|
|
.then((response) => response.text())
|
|
|
|
.then((content) => {
|
|
|
|
// Initialize the DOM parser
|
|
|
|
var parser = new DOMParser();
|
|
|
|
// Parse the text
|
|
|
|
var doc = parser.parseFromString(content, "text/html");
|
|
|
|
// You can now even select part of that html as you would in the regular DOM
|
|
|
|
// Example:
|
|
|
|
// var docArticle = doc.querySelector('article').innerHTML;
|
|
|
|
content = doc.querySelector('#content').querySelector('article').textContent
|
|
|
|
let contentArray = lengthSplit(content, 20);
|
2024-01-29 16:50:15 +01:00
|
|
|
contentArray.forEach(line => {
|
|
|
|
if (line.match(/^\s+$/) || line.match(/^[ ]+$/) || line.match(/^[ ]*$/) || line.match(/^\s*$/)) {
|
|
|
|
return;
|
|
|
|
}
|
2024-01-30 14:31:20 +01:00
|
|
|
if (line.search(input.value) != -1 && small.innerHTML == '') {
|
|
|
|
small.innerHTML = line.replaceAll(input.value, String.raw`<strong style="color:red" >${input.value}</strong>`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
})
|
|
|
|
if (small.innerHTML == '') {
|
|
|
|
contentArray.forEach(line => {
|
|
|
|
if (line.match(/^\s+$/) || line.match(/^[ ]+$/) || line.match(/^[ ]*$/) || line.match(/^\s*$/)) {
|
2024-01-29 16:50:15 +01:00
|
|
|
return;
|
|
|
|
}
|
2024-01-30 14:31:20 +01:00
|
|
|
input.value.split('').forEach(s => {
|
|
|
|
if (line.search(s) != -1 && small.innerHTML == '') {
|
|
|
|
small.innerHTML = line.replaceAll(s, String.raw`<strong style="color:red" >${s}</strong>`);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
})
|
2024-01-29 16:50:15 +01:00
|
|
|
})
|
2024-01-30 14:31:20 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2019-07-17 13:42:39 +02:00
|
|
|
results.appendChild(li);
|
|
|
|
});
|
2019-07-15 18:25:21 +02:00
|
|
|
}
|
2020-07-06 22:17:36 +02:00
|
|
|
|
2024-01-30 14:31:20 +01:00
|
|
|
function lengthSplit(str, num) {
|
2024-01-29 16:50:15 +01:00
|
|
|
let strArr = [];
|
|
|
|
for (let i = 0; i < str.length; i += num) strArr.push(str.slice(i, i + num));
|
|
|
|
return strArr;
|
|
|
|
}
|
|
|
|
|
2020-07-06 22:17:36 +02:00
|
|
|
/**
|
|
|
|
* @param {String} content
|
|
|
|
* @returns {Node}
|
|
|
|
*/
|
|
|
|
function element(content) {
|
|
|
|
const div = document.createElement('div');
|
|
|
|
div.innerHTML = content;
|
|
|
|
return div.firstChild;
|
|
|
|
}
|
2019-07-16 14:34:02 +02:00
|
|
|
})();
|