2018-09-18 13:14:39 +02:00
|
|
|
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
(function(){
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
var scope;
|
|
|
|
if ((typeof exports) !== "undefined"){
|
|
|
|
scope = exports;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
window.scope.search = {};
|
|
|
|
scope = window.scope.search;
|
|
|
|
}
|
|
|
|
|
|
|
|
const texts = [];
|
|
|
|
|
|
|
|
scope.register = function(text, content){
|
|
|
|
texts.push({text: text.toLowerCase(), content});
|
|
|
|
};
|
|
|
|
scope.search = function(search){
|
2018-09-19 09:15:19 +02:00
|
|
|
const resultSets = search.toLowerCase().split(/\s+/).filter(function(term){
|
|
|
|
return term.trim();
|
|
|
|
}).map(function(term){
|
|
|
|
return new RegExp(term);
|
|
|
|
}).map(function(term){
|
|
|
|
const matching = new Set();
|
|
|
|
texts.forEach(function(text){
|
|
|
|
if (term.test(text.text)){
|
|
|
|
matching.add(text.content);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return matching;
|
2018-09-18 13:14:39 +02:00
|
|
|
});
|
2018-09-19 09:15:19 +02:00
|
|
|
if (resultSets.length){
|
|
|
|
return Array.from(
|
|
|
|
resultSets.reduce(function(previousSet, set){
|
|
|
|
var andSet = new Set();
|
|
|
|
set.forEach(function(entry){
|
|
|
|
if (previousSet.has(entry)){
|
|
|
|
andSet.add(entry);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return andSet;
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return [];
|
|
|
|
}
|
2018-09-18 13:14:39 +02:00
|
|
|
};
|
|
|
|
const searchListeners = [];
|
|
|
|
scope.init = function(){
|
|
|
|
const node = document.createElement("input");
|
|
|
|
node.id = "search";
|
|
|
|
node.placeholder = browser.i18n.getMessage("search");
|
|
|
|
window.setTimeout(() => node.focus(), 1);
|
|
|
|
let lastResults = [];
|
|
|
|
node.addEventListener("input", function(){
|
|
|
|
const search = this.value;
|
|
|
|
const results = search? scope.search(search): [];
|
|
|
|
searchListeners.forEach(function(callback){
|
|
|
|
callback({search, results, lastResults});
|
|
|
|
});
|
|
|
|
lastResults = results;
|
|
|
|
});
|
|
|
|
return node;
|
|
|
|
};
|
|
|
|
scope.on = function(callback){
|
|
|
|
searchListeners.push(callback);
|
|
|
|
};
|
|
|
|
}());
|