mirror of
https://github.com/msberends/a_new_instrument_for_microbial_epidemiology.git
synced 2025-01-28 06:34:37 +01:00
271 lines
8.9 KiB
JavaScript
271 lines
8.9 KiB
JavaScript
gitbook.require(["gitbook", "lodash", "jQuery"], function(gitbook, _, $) {
|
|
var index = null;
|
|
var fuse = null;
|
|
var _search = {engine: 'lunr', opts: {}};
|
|
var $searchInput, $searchLabel, $searchForm;
|
|
var $highlighted = [], hi, hiOpts = { className: 'search-highlight' };
|
|
var collapse = false, toc_visible = [];
|
|
|
|
function init(config) {
|
|
// Instantiate search settings
|
|
_search = gitbook.storage.get("search", {
|
|
engine: config.search.engine || 'lunr',
|
|
opts: config.search.options || {},
|
|
});
|
|
};
|
|
|
|
// Save current search settings
|
|
function saveSearchSettings() {
|
|
gitbook.storage.set("search", _search);
|
|
}
|
|
|
|
// Use a specific index
|
|
function loadIndex(data) {
|
|
// [Yihui] In bookdown, I use a character matrix to store the chapter
|
|
// content, and the index is dynamically built on the client side.
|
|
// Gitbook prebuilds the index data instead: https://github.com/GitbookIO/plugin-search
|
|
// We can certainly do that via R packages V8 and jsonlite, but let's
|
|
// see how slow it really is before improving it. On the other hand,
|
|
// lunr cannot handle non-English text very well, e.g. the default
|
|
// tokenizer cannot deal with Chinese text, so we may want to replace
|
|
// lunr with a dumb simple text matching approach.
|
|
if (_search.engine === 'lunr') {
|
|
index = lunr(function () {
|
|
this.ref('url');
|
|
this.field('title', { boost: 10 });
|
|
this.field('body');
|
|
});
|
|
data.map(function(item) {
|
|
index.add({
|
|
url: item[0],
|
|
title: item[1],
|
|
body: item[2]
|
|
});
|
|
});
|
|
return;
|
|
}
|
|
fuse = new Fuse(data.map((_data => {
|
|
return {
|
|
url: _data[0],
|
|
title: _data[1],
|
|
body: _data[2]
|
|
};
|
|
})), Object.assign(
|
|
{
|
|
includeScore: true,
|
|
threshold: 0.1,
|
|
ignoreLocation: true,
|
|
keys: ["title", "body"]
|
|
},
|
|
_search.opts
|
|
));
|
|
}
|
|
|
|
// Fetch the search index
|
|
function fetchIndex() {
|
|
return $.getJSON(gitbook.state.basePath+"/search_index.json")
|
|
.then(loadIndex); // [Yihui] we need to use this object later
|
|
}
|
|
|
|
// Search for a term and return results
|
|
function search(q) {
|
|
let results = [];
|
|
switch (_search.engine) {
|
|
case 'fuse':
|
|
if (!fuse) return;
|
|
results = fuse.search(q).map(function(result) {
|
|
var parts = result.item.url.split('#');
|
|
return {
|
|
path: parts[0],
|
|
hash: parts[1]
|
|
};
|
|
});
|
|
break;
|
|
case 'lunr':
|
|
default:
|
|
if (!index) return;
|
|
results = _.chain(index.search(q)).map(function(result) {
|
|
var parts = result.ref.split("#");
|
|
return {
|
|
path: parts[0],
|
|
hash: parts[1]
|
|
};
|
|
})
|
|
.value();
|
|
}
|
|
|
|
// [Yihui] Highlight the search keyword on current page
|
|
$highlighted = $('.page-inner')
|
|
.unhighlight(hiOpts).highlight(q, hiOpts).find('span.search-highlight');
|
|
scrollToHighlighted(0);
|
|
|
|
return results;
|
|
}
|
|
|
|
// [Yihui] Scroll the chapter body to the i-th highlighted string
|
|
function scrollToHighlighted(d) {
|
|
var n = $highlighted.length;
|
|
hi = hi === undefined ? 0 : hi + d;
|
|
// navignate to the previous/next page in the search results if reached the top/bottom
|
|
var b = hi < 0;
|
|
if (d !== 0 && (b || hi >= n)) {
|
|
var path = currentPath(), n2 = toc_visible.length;
|
|
if (n2 === 0) return;
|
|
for (var i = b ? 0 : n2; (b && i < n2) || (!b && i >= 0); i += b ? 1 : -1) {
|
|
if (toc_visible.eq(i).data('path') === path) break;
|
|
}
|
|
i += b ? -1 : 1;
|
|
if (i < 0) i = n2 - 1;
|
|
if (i >= n2) i = 0;
|
|
var lnk = toc_visible.eq(i).find('a[href$=".html"]');
|
|
if (lnk.length) lnk[0].click();
|
|
return;
|
|
}
|
|
if (n === 0) return;
|
|
var $p = $highlighted.eq(hi);
|
|
$p[0].scrollIntoView();
|
|
$highlighted.css('background-color', '');
|
|
// an orange background color on the current item and removed later
|
|
$p.css('background-color', 'orange');
|
|
setTimeout(function() {
|
|
$p.css('background-color', '');
|
|
}, 2000);
|
|
}
|
|
|
|
function currentPath() {
|
|
var href = window.location.pathname;
|
|
href = href.substr(href.lastIndexOf('/') + 1);
|
|
return href === '' ? 'index.html' : href;
|
|
}
|
|
|
|
// Create search form
|
|
function createForm(value) {
|
|
if ($searchForm) $searchForm.remove();
|
|
if ($searchLabel) $searchLabel.remove();
|
|
if ($searchInput) $searchInput.remove();
|
|
|
|
$searchForm = $('<div>', {
|
|
'class': 'book-search',
|
|
'role': 'search'
|
|
});
|
|
|
|
$searchLabel = $('<label>', {
|
|
'for': 'search-box',
|
|
'aria-hidden': 'false',
|
|
'hidden': ''
|
|
});
|
|
|
|
$searchInput = $('<input>', {
|
|
'id': 'search-box',
|
|
'type': 'search',
|
|
'class': 'form-control',
|
|
'val': value,
|
|
'placeholder': 'Type to search (Enter for navigation)',
|
|
'title': 'Use Enter or the <Down> key to navigate to the next match, or the <Up> key to the previous match'
|
|
});
|
|
|
|
$searchLabel.append("Type to search");
|
|
$searchLabel.appendTo($searchForm);
|
|
$searchInput.appendTo($searchForm);
|
|
$searchForm.prependTo(gitbook.state.$book.find('.book-summary'));
|
|
}
|
|
|
|
// Return true if search is open
|
|
function isSearchOpen() {
|
|
return gitbook.state.$book.hasClass("with-search");
|
|
}
|
|
|
|
// Toggle the search
|
|
function toggleSearch(_state) {
|
|
if (isSearchOpen() === _state) return;
|
|
if (!$searchInput) return;
|
|
|
|
gitbook.state.$book.toggleClass("with-search", _state);
|
|
|
|
// If search bar is open: focus input
|
|
if (isSearchOpen()) {
|
|
gitbook.sidebar.toggle(true);
|
|
$searchInput.focus();
|
|
} else {
|
|
$searchInput.blur();
|
|
$searchInput.val("");
|
|
gitbook.storage.remove("keyword");
|
|
gitbook.sidebar.filter(null);
|
|
$('.page-inner').unhighlight(hiOpts);
|
|
}
|
|
}
|
|
|
|
function sidebarFilter(results) {
|
|
gitbook.sidebar.filter(_.pluck(results, "path"));
|
|
toc_visible = $('ul.summary').find('li:visible');
|
|
}
|
|
|
|
// Recover current search when page changed
|
|
function recoverSearch() {
|
|
var keyword = gitbook.storage.get("keyword", "");
|
|
|
|
createForm(keyword);
|
|
|
|
if (keyword.length > 0) {
|
|
if(!isSearchOpen()) {
|
|
toggleSearch(true); // [Yihui] open the search box
|
|
}
|
|
sidebarFilter(search(keyword));
|
|
}
|
|
}
|
|
|
|
|
|
gitbook.events.bind("start", function(e, config) {
|
|
// [Yihui] disable search
|
|
if (config.search === false) return;
|
|
init(config);
|
|
collapse = !config.toc || config.toc.collapse === 'section' ||
|
|
config.toc.collapse === 'subsection';
|
|
|
|
// Pre-fetch search index and create the form
|
|
fetchIndex()
|
|
// [Yihui] recover search after the page is loaded
|
|
.then(recoverSearch);
|
|
|
|
|
|
// Type in search bar
|
|
$(document).on("keyup", ".book-search input", function(e) {
|
|
var key = (e.keyCode ? e.keyCode : e.which);
|
|
// [Yihui] Escape -> close search box; Up/Down/Enter: previous/next highlighted
|
|
if (key == 27) {
|
|
e.preventDefault();
|
|
toggleSearch(false);
|
|
} else if (key == 38) {
|
|
scrollToHighlighted(-1);
|
|
} else if (key == 40 || key == 13) {
|
|
scrollToHighlighted(1);
|
|
}
|
|
}).on("input", ".book-search input", function(e) {
|
|
var q = $(this).val().trim();
|
|
if (q.length === 0) {
|
|
gitbook.sidebar.filter(null);
|
|
gitbook.storage.remove("keyword");
|
|
$('.page-inner').unhighlight(hiOpts);
|
|
} else {
|
|
var results = search(q);
|
|
sidebarFilter(results);
|
|
gitbook.storage.set("keyword", q);
|
|
}
|
|
});
|
|
|
|
// Create the toggle search button
|
|
gitbook.toolbar.createButton({
|
|
icon: 'fa fa-search',
|
|
label: 'Search',
|
|
position: 'left',
|
|
onClick: toggleSearch
|
|
});
|
|
|
|
// Bind keyboard to toggle search
|
|
gitbook.keyboard.bind(['f'], toggleSearch);
|
|
});
|
|
|
|
// [Yihui] do not try to recover search; always start fresh
|
|
// gitbook.events.bind("page.change", recoverSearch);
|
|
});
|