mirror of
https://github.com/matrix-org/matrix-spec
synced 2026-04-07 20:04:09 +02:00
Add loading spinner while the search is running
Signed-off-by: Johannes Marbach <n0-0ne+github@mailbox.org>
This commit is contained in:
parent
6f05c2c78e
commit
92b7e714e7
|
|
@ -37,7 +37,7 @@ search backend.
|
||||||
});
|
});
|
||||||
|
|
||||||
//
|
//
|
||||||
// Register handler
|
// Set up search input handler.
|
||||||
//
|
//
|
||||||
|
|
||||||
$searchInput.on("change", (event) => {
|
$searchInput.on("change", (event) => {
|
||||||
|
|
@ -53,12 +53,12 @@ search backend.
|
||||||
});
|
});
|
||||||
|
|
||||||
//
|
//
|
||||||
// Pagefind
|
// Callback for searching and rendering the results.
|
||||||
//
|
//
|
||||||
|
|
||||||
const render = async ($targetSearchInput) => {
|
const render = async ($targetSearchInput) => {
|
||||||
//
|
//
|
||||||
// Dispose existing popover
|
// Dispose any existing popover.
|
||||||
//
|
//
|
||||||
|
|
||||||
{
|
{
|
||||||
|
|
@ -69,7 +69,7 @@ search backend.
|
||||||
}
|
}
|
||||||
|
|
||||||
//
|
//
|
||||||
// Search
|
// Kick off the search and collect the results.
|
||||||
//
|
//
|
||||||
|
|
||||||
const searchQuery = $targetSearchInput.val();
|
const searchQuery = $targetSearchInput.val();
|
||||||
|
|
@ -77,34 +77,61 @@ search backend.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Show the results popover with a spinner while we're busy.
|
||||||
|
const $spinner = $("<div>")
|
||||||
|
.addClass("spinner-container")
|
||||||
|
.append($("<div>")
|
||||||
|
.addClass("spinner-border")
|
||||||
|
.attr("role", "status")
|
||||||
|
.append($("<div>")
|
||||||
|
.addClass("visually-hidden")
|
||||||
|
.text("Loading...")))
|
||||||
|
.append($("<p>")
|
||||||
|
.text("Loading..."));
|
||||||
|
const popover = new bootstrap.Popover($targetSearchInput, {
|
||||||
|
content: $spinner[0],
|
||||||
|
html: true,
|
||||||
|
customClass: "td-offline-search-results",
|
||||||
|
placement: "bottom",
|
||||||
|
});
|
||||||
|
popover.show();
|
||||||
|
|
||||||
|
// Kick off the search.
|
||||||
const search = await pagefind.debouncedSearch(searchQuery);
|
const search = await pagefind.debouncedSearch(searchQuery);
|
||||||
if (search === null) {
|
if (search === null) {
|
||||||
// A more recent search call has been made, nothing to do.
|
// A more recent search call has been made, nothing to do.
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const results = await Promise.all(search.results.slice(0, 20).map(r => r.data()));
|
|
||||||
|
// Load all result details. We may want to switch to a paged UI in future for better performance.
|
||||||
|
const results = await Promise.all(search.results.slice(0, 100).map(r => r.data()));
|
||||||
|
|
||||||
//
|
//
|
||||||
// Make result html
|
// Construct the search results html.
|
||||||
//
|
//
|
||||||
|
|
||||||
const $html = $("<div>");
|
const $html = $("<div>");
|
||||||
|
|
||||||
$html.append(
|
// Add the header and close button.
|
||||||
$("<div>")
|
$html.append($("<div>")
|
||||||
.css({
|
.css({
|
||||||
display: "flex",
|
display: "flex",
|
||||||
justifyContent: "space-between",
|
justifyContent: "space-between",
|
||||||
marginBottom: "1em",
|
marginBottom: "1em",
|
||||||
|
})
|
||||||
|
.append($("<span>")
|
||||||
|
.text("Search results")
|
||||||
|
.css({ fontWeight: "bold" }))
|
||||||
|
.append($("<span>")
|
||||||
|
.addClass("td-offline-search-results__close-button")
|
||||||
|
.on("click", () => {
|
||||||
|
$targetSearchInput.val("");
|
||||||
|
$targetSearchInput.trigger("change");
|
||||||
})
|
})
|
||||||
.append(
|
)
|
||||||
$("<span>").text("Search results").css({ fontWeight: "bold" })
|
|
||||||
)
|
|
||||||
.append(
|
|
||||||
$("<span>").addClass("td-offline-search-results__close-button")
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
// Add the main search results body.
|
||||||
const $searchResultBody = $("<div>").css({
|
const $searchResultBody = $("<div>").css({
|
||||||
maxHeight: `calc(100vh - ${
|
maxHeight: `calc(100vh - ${
|
||||||
$targetSearchInput.offset().top - $(window).scrollTop() + 180
|
$targetSearchInput.offset().top - $(window).scrollTop() + 180
|
||||||
|
|
@ -119,7 +146,7 @@ search backend.
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
results.forEach((r, index_r) => {
|
results.forEach((r, index_r) => {
|
||||||
// Add the main result"s page title.
|
// Add the main result's page title.
|
||||||
$searchResultBody.append(
|
$searchResultBody.append(
|
||||||
$("<a>")
|
$("<a>")
|
||||||
.addClass("d-block")
|
.addClass("d-block")
|
||||||
|
|
@ -136,8 +163,6 @@ search backend.
|
||||||
let $wrapper = null;
|
let $wrapper = null;
|
||||||
|
|
||||||
r.sub_results.forEach((s, index_s) => {
|
r.sub_results.forEach((s, index_s) => {
|
||||||
|
|
||||||
|
|
||||||
if (index_s === LIMIT) {
|
if (index_s === LIMIT) {
|
||||||
const wrapper_id = `collapssible-subresults-${index_r}`;
|
const wrapper_id = `collapssible-subresults-${index_r}`;
|
||||||
const $expander = $("<a>")
|
const $expander = $("<a>")
|
||||||
|
|
@ -179,20 +204,16 @@ search backend.
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
$targetSearchInput.one("shown.bs.popover", () => {
|
// Finally, show the search results.
|
||||||
$(".td-offline-search-results__close-button").on("click", () => {
|
//
|
||||||
$targetSearchInput.val("");
|
// Ideally we would just call setContent but there appears to be a bug in Bootstrap
|
||||||
$targetSearchInput.trigger("change");
|
// that causes the popover to be hidden when setContent is called after the popover
|
||||||
});
|
// has been shown. To work around this, we use the hack from [1] to inject the HTML
|
||||||
});
|
// content manually.
|
||||||
|
//
|
||||||
const popover = new bootstrap.Popover($targetSearchInput, {
|
// [1]: https://github.com/twbs/bootstrap/issues/37206#issuecomment-1259541205
|
||||||
content: $html[0],
|
$(popover.tip.querySelector('.popover-body')).html($html[0]);
|
||||||
html: true,
|
popover.update();
|
||||||
customClass: "td-offline-search-results",
|
|
||||||
placement: "bottom",
|
|
||||||
});
|
|
||||||
popover.show();
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
})(jQuery);
|
})(jQuery);
|
||||||
|
|
|
||||||
|
|
@ -664,5 +664,10 @@ dd {
|
||||||
|
|
||||||
/* Style for the page search widget */
|
/* Style for the page search widget */
|
||||||
.td-offline-search-results {
|
.td-offline-search-results {
|
||||||
|
width: 100%;
|
||||||
max-width: 460px;
|
max-width: 460px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.td-offline-search-results .spinner-container {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue