mirror of
https://github.com/simon987/sist2.git
synced 2025-04-04 07:52:59 +00:00
354 lines
8.9 KiB
JavaScript
354 lines
8.9 KiB
JavaScript
const SIZE = 20;
|
|
let mimeMap = [];
|
|
let tree;
|
|
|
|
let searchBar = document.getElementById("searchBar");
|
|
let pathBar = document.getElementById("pathBar");
|
|
let scroll_id = null;
|
|
let docCount = 0;
|
|
let coolingDown = false;
|
|
let searchBusy = true;
|
|
let selectedIndices = [];
|
|
|
|
jQuery["jsonPost"] = function (url, data) {
|
|
return jQuery.ajax({
|
|
url: url,
|
|
type: "post",
|
|
data: JSON.stringify(data),
|
|
contentType: "application/json"
|
|
}).fail(err => {
|
|
console.log(err);
|
|
});
|
|
};
|
|
|
|
function toggleSearchBar() {
|
|
searchDebounced();
|
|
}
|
|
|
|
$.jsonPost("i").then(resp => {
|
|
resp["indices"].forEach(idx => {
|
|
$("#indices").append($("<option>")
|
|
.attr("value", idx.id)
|
|
.attr("selected", true)
|
|
.append(idx.name)
|
|
);
|
|
selectedIndices.push(idx.id);
|
|
});
|
|
});
|
|
|
|
$.jsonPost("es", {
|
|
aggs: {
|
|
mimeTypes: {
|
|
terms: {
|
|
field: "mime",
|
|
size: 10000
|
|
}
|
|
}
|
|
},
|
|
size: 0,
|
|
}).then(resp => {
|
|
resp["aggregations"]["mimeTypes"]["buckets"].forEach(bucket => {
|
|
let tmp = bucket["key"].split("/");
|
|
let category = tmp[0];
|
|
let mime = tmp[1];
|
|
|
|
let category_exists = false;
|
|
|
|
let child = {
|
|
"id": bucket["key"],
|
|
"text": `${mime} (${bucket["doc_count"]})`
|
|
};
|
|
|
|
mimeMap.forEach(node => {
|
|
if (node.text === category) {
|
|
node.children.push(child);
|
|
category_exists = true;
|
|
}
|
|
});
|
|
|
|
if (!category_exists) {
|
|
mimeMap.push({"text": category, children: [child]});
|
|
}
|
|
});
|
|
mimeMap.push({"text": "All", "id": "any"});
|
|
|
|
tree = new InspireTree({
|
|
selection: {
|
|
mode: 'checkbox'
|
|
},
|
|
data: mimeMap
|
|
});
|
|
new InspireTreeDOM(tree, {
|
|
target: '.tree'
|
|
});
|
|
tree.on("node.click", function (event, node, handler) {
|
|
event.preventTreeDefault();
|
|
|
|
if (node.id === "any") {
|
|
if (!node.itree.state.checked) {
|
|
tree.deselect();
|
|
}
|
|
} else {
|
|
tree.node("any").deselect();
|
|
}
|
|
|
|
handler();
|
|
searchDebounced();
|
|
});
|
|
tree.select();
|
|
tree.node("any").deselect();
|
|
searchBusy = false;
|
|
});
|
|
|
|
new autoComplete({
|
|
selector: '#pathBar',
|
|
minChars: 1,
|
|
delay: 75,
|
|
renderItem: function (item) {
|
|
return '<div class="autocomplete-suggestion" data-val="' + item + '">' + item + '</div>';
|
|
},
|
|
source: async function (term, suggest) {
|
|
term = term.toLowerCase();
|
|
|
|
const choices = await getPathChoices();
|
|
|
|
let matches = [];
|
|
for (let i = 0; i < choices.length; i++) {
|
|
if (~choices[i].toLowerCase().indexOf(term)) {
|
|
matches.push(choices[i]);
|
|
}
|
|
}
|
|
suggest(matches);
|
|
},
|
|
onSelect: function () {
|
|
searchDebounced();
|
|
}
|
|
});
|
|
|
|
function insertHits(resultContainer, hits) {
|
|
for (let i = 0; i < hits.length; i++) {
|
|
resultContainer.appendChild(createDocCard(hits[i]));
|
|
docCount++;
|
|
}
|
|
}
|
|
|
|
window.addEventListener("scroll", function () {
|
|
if (!coolingDown && !searchBusy) {
|
|
let threshold = 400;
|
|
|
|
if ((window.innerHeight + window.scrollY) >= document.body.offsetHeight - threshold) {
|
|
coolingDown = true;
|
|
doScroll();
|
|
}
|
|
}
|
|
});
|
|
|
|
function doScroll() {
|
|
$.get("scroll", {scroll_id: scroll_id})
|
|
.then(searchResult => {
|
|
let searchResults = document.getElementById("searchResults");
|
|
let hits = searchResult["hits"]["hits"];
|
|
|
|
//Page indicator
|
|
let pageIndicator = makePageIndicator(searchResult);
|
|
searchResults.appendChild(pageIndicator);
|
|
|
|
//Result container
|
|
let resultContainer = makeResultContainer();
|
|
searchResults.appendChild(resultContainer);
|
|
|
|
insertHits(resultContainer, hits);
|
|
|
|
if (hits.length === SIZE) {
|
|
coolingDown = false;
|
|
}
|
|
})
|
|
.fail(() => {
|
|
window.location.reload();
|
|
})
|
|
}
|
|
|
|
function getSelectedMimeTypes() {
|
|
let mimeTypes = [];
|
|
|
|
let selected = tree.selected();
|
|
|
|
for (let i = 0; i < selected.length; i++) {
|
|
|
|
if (selected[i].id === "any") {
|
|
return ["any"]
|
|
}
|
|
|
|
//Only get children
|
|
if (selected[i].text.indexOf("(") !== -1) {
|
|
mimeTypes.push(selected[i].id);
|
|
}
|
|
}
|
|
|
|
return mimeTypes
|
|
}
|
|
|
|
function search() {
|
|
if (searchBusy) {
|
|
return;
|
|
}
|
|
searchBusy = true;
|
|
//Clear old search results
|
|
let searchResults = document.getElementById("searchResults");
|
|
while (searchResults.firstChild) {
|
|
searchResults.removeChild(searchResults.firstChild);
|
|
}
|
|
|
|
let query = searchBar.value;
|
|
let condition = $("#barToggle").prop("checked") ? "must" : "should";
|
|
let filters = [
|
|
{range: {size: {gte: size_min, lte: size_max}}},
|
|
{terms: {index: selectedIndices}}
|
|
];
|
|
|
|
let path = pathBar.value.replace(/\/$/, "").toLowerCase(); //remove trailing slashes
|
|
if (path !== "") {
|
|
filters.push([{term: {path: path}}])
|
|
}
|
|
let mimeTypes = getSelectedMimeTypes();
|
|
if (!mimeTypes.includes("any")) {
|
|
filters.push([{terms: {"mime": mimeTypes}}]);
|
|
}
|
|
|
|
$.jsonPost("es?scroll=1", {
|
|
"_source": {
|
|
excludes: ["content"]
|
|
},
|
|
query: {
|
|
bool: {
|
|
[condition]: {
|
|
multi_match: {
|
|
query: query,
|
|
type: "most_fields",
|
|
fields: [
|
|
"name^8", "name.nGram^3", "content^3",
|
|
"content.nGram",
|
|
"album^8", "artist^8", "title^8", "genre^2", "album_artist^8",
|
|
"font_name^6"
|
|
],
|
|
operator: "and"
|
|
}
|
|
},
|
|
filter: filters
|
|
}
|
|
},
|
|
sort: [
|
|
"_score"
|
|
],
|
|
highlight: {
|
|
pre_tags: ["<mark>"],
|
|
post_tags: ["</mark>"],
|
|
fields: {
|
|
content: {},
|
|
name: {},
|
|
"name.nGram": {},
|
|
// font_name: {},
|
|
}
|
|
},
|
|
aggs: {
|
|
total_size: {"sum": {"field": "size"}}
|
|
},
|
|
size: SIZE,
|
|
}).then(searchResult => {
|
|
scroll_id = searchResult["_scroll_id"];
|
|
|
|
//Search stats
|
|
searchResults.appendChild(makeStatsCard(searchResult));
|
|
|
|
//Autocomplete
|
|
if (searchResult.hasOwnProperty("suggest") && searchResult["suggest"].hasOwnProperty("path")) {
|
|
pathAutoComplete = [];
|
|
for (let i = 0; i < searchResult["suggest"]["path"][0]["options"].length; i++) {
|
|
pathAutoComplete.push(searchResult["suggest"]["path"][0]["options"][i].text)
|
|
}
|
|
}
|
|
|
|
//Setup page
|
|
let resultContainer = makeResultContainer();
|
|
searchResults.appendChild(resultContainer);
|
|
|
|
docCount = 0;
|
|
insertHits(resultContainer, searchResult["hits"]["hits"]);
|
|
|
|
searchBusy = false;
|
|
});
|
|
}
|
|
|
|
let pathAutoComplete = [];
|
|
let size_min = 0;
|
|
let size_max = 10000000000000;
|
|
|
|
let searchDebounced = _.debounce(function () {
|
|
coolingDown = false;
|
|
search()
|
|
}, 500);
|
|
searchBar.addEventListener("keyup", searchDebounced);
|
|
document.getElementById("pathBar").addEventListener("keyup", searchDebounced);
|
|
|
|
//Size slider
|
|
$("#sizeSlider").ionRangeSlider({
|
|
type: "double",
|
|
grid: false,
|
|
force_edges: true,
|
|
min: 0,
|
|
max: 3684.03149864,
|
|
from: 0,
|
|
to: 3684.03149864,
|
|
min_interval: 5,
|
|
drag_interval: true,
|
|
prettify: function (num) {
|
|
|
|
if (num === 0) {
|
|
return "0 B"
|
|
} else if (num >= 3684) {
|
|
return humanFileSize(num * num * num) + "+";
|
|
}
|
|
|
|
return humanFileSize(num * num * num)
|
|
},
|
|
onChange: function (e) {
|
|
size_min = (e.from * e.from * e.from);
|
|
size_max = (e.to * e.to * e.to);
|
|
|
|
if (e.to >= 3684) {
|
|
size_max = 10000000000000;
|
|
}
|
|
|
|
searchDebounced();
|
|
}
|
|
});
|
|
|
|
function updateIndices() {
|
|
let selected = $('#indices').find('option:selected');
|
|
selectedIndices = [];
|
|
$(selected).each(function () {
|
|
selectedIndices.push($(this).val());
|
|
});
|
|
|
|
searchDebounced();
|
|
}
|
|
|
|
document.getElementById("indices").addEventListener("change", updateIndices);
|
|
updateIndices();
|
|
|
|
//Suggest
|
|
function getPathChoices() {
|
|
return new Promise(getPaths => {
|
|
|
|
let xhttp = new XMLHttpRequest();
|
|
xhttp.onreadystatechange = function () {
|
|
if (this.readyState === 4 && this.status === 200) {
|
|
getPaths(JSON.parse(xhttp.responseText))
|
|
}
|
|
};
|
|
xhttp.open("GET", "suggest?prefix=" + pathBar.value, true);
|
|
xhttp.send();
|
|
});
|
|
}
|
|
|