2019-07-29 15:43:53 +02:00
|
|
|
const markJsUtils = {};
|
2019-01-18 17:56:56 +00:00
|
|
|
|
2023-04-04 21:09:17 +02:00
|
|
|
const isInsideContainer = (node, tagName) => {
|
|
|
|
if (!node) return false;
|
|
|
|
|
|
|
|
tagName = tagName.toLowerCase();
|
|
|
|
|
|
|
|
while (node) {
|
|
|
|
if (node.tagName && node.tagName.toLowerCase() === tagName) return true;
|
|
|
|
node = node.parentNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
|
2019-01-18 17:56:56 +00:00
|
|
|
markJsUtils.markKeyword = (mark, keyword, stringUtils, extraOptions = null) => {
|
|
|
|
if (typeof keyword === 'string') {
|
|
|
|
keyword = {
|
|
|
|
type: 'text',
|
|
|
|
value: keyword,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
const isBasicSearch = ['ja', 'zh', 'ko'].indexOf(keyword.scriptType) >= 0;
|
|
|
|
|
|
|
|
let value = keyword.value;
|
2019-07-29 15:43:53 +02:00
|
|
|
let accuracy = keyword.accuracy ? keyword.accuracy : { value: 'exactly', limiters: ':;.,-–—‒_(){}[]!\'"+='.split('') };
|
2019-01-18 17:56:56 +00:00
|
|
|
if (isBasicSearch) accuracy = 'partially';
|
|
|
|
if (keyword.type === 'regex') {
|
|
|
|
accuracy = 'complementary';
|
2023-04-04 21:09:17 +02:00
|
|
|
// Remove the trailing wildcard and "accuracy = complementary" will take
|
|
|
|
// care of highlighting the relevant keywords.
|
2019-07-29 15:43:53 +02:00
|
|
|
|
2023-04-04 21:09:17 +02:00
|
|
|
// Known bug: it will also highlight word that contain the term as a
|
|
|
|
// suffix for example for "ent*", it will highlight "present" which is
|
|
|
|
// incorrect (it should only highlight what starts with "ent") but for
|
|
|
|
// now will do. Mark.js doesn't have an option to tweak this behaviour.
|
2019-01-18 18:31:07 +00:00
|
|
|
value = keyword.value.substr(0, keyword.value.length - 1);
|
2019-01-18 17:56:56 +00:00
|
|
|
}
|
|
|
|
|
2019-07-29 15:43:53 +02:00
|
|
|
mark.mark(
|
|
|
|
[value],
|
2023-06-01 12:02:36 +01:00
|
|
|
{
|
|
|
|
|
|
|
|
accuracy: accuracy,
|
|
|
|
filter: (node, _term, _totalCounter, _counter) => {
|
|
|
|
// We exclude SVG because it creates a "<mark>" tag inside
|
|
|
|
// the document, which is not a valid SVG tag. As a result
|
|
|
|
// the content within that tag disappears.
|
|
|
|
//
|
|
|
|
// mark.js has an "exclude" parameter, but it doesn't work
|
|
|
|
// so we use "filter" instead.
|
|
|
|
//
|
|
|
|
// https://github.com/joplin/plugin-abc-sheet-music
|
|
|
|
if (isInsideContainer(node, 'SVG')) return false;
|
|
|
|
return true;
|
2019-07-29 15:43:53 +02:00
|
|
|
},
|
2023-06-01 12:02:36 +01:00
|
|
|
...extraOptions,
|
2023-08-22 11:58:53 +01:00
|
|
|
},
|
2019-07-29 15:43:53 +02:00
|
|
|
);
|
|
|
|
};
|
2019-01-18 17:56:56 +00:00
|
|
|
|
|
|
|
if (typeof module !== 'undefined') {
|
|
|
|
module.exports = markJsUtils;
|
2019-07-29 15:43:53 +02:00
|
|
|
}
|