2026-03-17 13:04:43 +01:00
|
|
|
|
/*
|
|
|
|
|
|
Copyright 2026 The Matrix.org Foundation C.I.C.
|
|
|
|
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
|
|
limitations under the License.
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
2026-03-19 08:50:45 +01:00
|
|
|
|
Adapted from [1] to combine Docsy"s built-in search UI with the Pagefind
|
2026-03-17 13:04:43 +01:00
|
|
|
|
search backend.
|
|
|
|
|
|
|
|
|
|
|
|
[1]: https://github.com/matrix-org/docsy/blob/71d103ebb20ace3d528178c4b6d92b6cc4f7fd53/assets/js/offline-search.js
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
(function ($) {
|
2026-03-19 08:50:45 +01:00
|
|
|
|
"use strict";
|
2026-03-17 13:04:43 +01:00
|
|
|
|
|
|
|
|
|
|
$(document).ready(async function () {
|
|
|
|
|
|
const pagefind = await import("/pagefind/pagefind.js");
|
2026-03-19 08:50:45 +01:00
|
|
|
|
const $searchInput = $(".td-search input");
|
2026-03-17 13:04:43 +01:00
|
|
|
|
|
|
|
|
|
|
//
|
|
|
|
|
|
// Lazily initialise Pagefind only when the user is about to start a search.
|
|
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
$searchInput.focus(() => {
|
|
|
|
|
|
pagefind.init();
|
2026-03-17 13:21:42 +01:00
|
|
|
|
});
|
2026-03-17 13:04:43 +01:00
|
|
|
|
|
|
|
|
|
|
//
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Set up search input handler.
|
2026-03-17 13:04:43 +01:00
|
|
|
|
//
|
|
|
|
|
|
|
2026-03-19 08:50:45 +01:00
|
|
|
|
$searchInput.on("change", (event) => {
|
2026-03-17 13:04:43 +01:00
|
|
|
|
render($(event.target));
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
// Prevent reloading page by enter key on sidebar search.
|
2026-03-19 08:50:45 +01:00
|
|
|
|
$searchInput.closest("form").on("submit", () => {
|
2026-03-17 13:04:43 +01:00
|
|
|
|
return false;
|
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
|
|
//
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Callback for searching and rendering the results.
|
2026-03-17 13:04:43 +01:00
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
const render = async ($targetSearchInput) => {
|
|
|
|
|
|
//
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Dispose any existing popover.
|
2026-03-17 13:04:43 +01:00
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
let popover = bootstrap.Popover.getInstance($targetSearchInput[0]);
|
|
|
|
|
|
if (popover !== null) {
|
|
|
|
|
|
popover.dispose();
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Kick off the search and collect the results.
|
2026-03-17 13:04:43 +01:00
|
|
|
|
//
|
|
|
|
|
|
|
|
|
|
|
|
const searchQuery = $targetSearchInput.val();
|
2026-03-19 08:50:45 +01:00
|
|
|
|
if (searchQuery === "") {
|
2026-03-17 13:04:43 +01:00
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// 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.
|
2026-03-17 13:04:43 +01:00
|
|
|
|
const search = await pagefind.debouncedSearch(searchQuery);
|
|
|
|
|
|
if (search === null) {
|
|
|
|
|
|
// A more recent search call has been made, nothing to do.
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
2026-03-20 12:52:01 +01:00
|
|
|
|
|
|
|
|
|
|
// 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()));
|
2026-03-17 13:04:43 +01:00
|
|
|
|
|
|
|
|
|
|
//
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Construct the search results html.
|
2026-03-17 13:04:43 +01:00
|
|
|
|
//
|
|
|
|
|
|
|
2026-03-19 08:50:45 +01:00
|
|
|
|
const $html = $("<div>");
|
2026-03-17 13:04:43 +01:00
|
|
|
|
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Add the header and close button.
|
|
|
|
|
|
$html.append($("<div>")
|
|
|
|
|
|
.css({
|
|
|
|
|
|
display: "flex",
|
|
|
|
|
|
justifyContent: "space-between",
|
|
|
|
|
|
marginBottom: "1em",
|
|
|
|
|
|
})
|
|
|
|
|
|
.append($("<span>")
|
|
|
|
|
|
.text("Search results")
|
|
|
|
|
|
.css({ fontWeight: "bold" }))
|
|
|
|
|
|
.append($("<span>")
|
|
|
|
|
|
.addClass("td-offline-search-results__close-button")
|
2026-03-20 13:13:05 +01:00
|
|
|
|
.attr("role", "button")
|
|
|
|
|
|
.attr("aria-label", "Close")
|
2026-03-20 12:52:01 +01:00
|
|
|
|
.on("click", () => {
|
|
|
|
|
|
$targetSearchInput.val("");
|
|
|
|
|
|
$targetSearchInput.trigger("change");
|
2026-03-17 13:04:43 +01:00
|
|
|
|
})
|
2026-03-20 12:52:01 +01:00
|
|
|
|
)
|
2026-03-17 13:04:43 +01:00
|
|
|
|
);
|
|
|
|
|
|
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Add the main search results body.
|
2026-03-19 08:50:45 +01:00
|
|
|
|
const $searchResultBody = $("<div>").css({
|
2026-03-17 13:04:43 +01:00
|
|
|
|
maxHeight: `calc(100vh - ${
|
|
|
|
|
|
$targetSearchInput.offset().top - $(window).scrollTop() + 180
|
|
|
|
|
|
}px)`,
|
2026-03-19 08:50:45 +01:00
|
|
|
|
overflowY: "auto",
|
2026-03-17 13:04:43 +01:00
|
|
|
|
});
|
|
|
|
|
|
$html.append($searchResultBody);
|
|
|
|
|
|
|
|
|
|
|
|
if (results.length === 0) {
|
|
|
|
|
|
$searchResultBody.append(
|
2026-03-19 08:50:45 +01:00
|
|
|
|
$("<p>").text(`No results found for query "${searchQuery}"`)
|
2026-03-17 13:04:43 +01:00
|
|
|
|
);
|
|
|
|
|
|
} else {
|
2026-03-19 08:50:45 +01:00
|
|
|
|
results.forEach((r, index_r) => {
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Add the main result's page title.
|
2026-03-20 13:37:41 +01:00
|
|
|
|
$searchResultBody.append($("<div>")
|
|
|
|
|
|
.append($("<a>")
|
2026-03-19 08:50:45 +01:00
|
|
|
|
.css({
|
|
|
|
|
|
fontSize: "1.2rem",
|
|
|
|
|
|
})
|
|
|
|
|
|
.attr("href", r.url)
|
2026-03-20 13:37:41 +01:00
|
|
|
|
.text(r.meta.title))
|
|
|
|
|
|
.append($("<span>")
|
|
|
|
|
|
.addClass("text-body-secondary")
|
|
|
|
|
|
.text(` – ${r.sub_results.length} result(s)`)));
|
2026-03-19 08:50:45 +01:00
|
|
|
|
|
|
|
|
|
|
// Render the first 3 subresults per page and wrap the rest
|
|
|
|
|
|
// in a collapsed container.
|
|
|
|
|
|
const LIMIT = 3;
|
|
|
|
|
|
let $wrapper = null;
|
|
|
|
|
|
|
|
|
|
|
|
r.sub_results.forEach((s, index_s) => {
|
|
|
|
|
|
if (index_s === LIMIT) {
|
|
|
|
|
|
const wrapper_id = `collapssible-subresults-${index_r}`;
|
2026-03-20 13:38:12 +01:00
|
|
|
|
const $action = $("<span>").text("▶ Show");
|
2026-03-19 08:50:45 +01:00
|
|
|
|
const $expander = $("<a>")
|
|
|
|
|
|
.attr("data-bs-toggle", "collapse")
|
|
|
|
|
|
.attr("data-bs-target", `#${wrapper_id}`)
|
|
|
|
|
|
.attr("href", "#")
|
|
|
|
|
|
.attr("role", "button")
|
|
|
|
|
|
.attr("aria-expanded", "false")
|
|
|
|
|
|
.attr("aria-controls", wrapper_id)
|
2026-03-20 13:08:53 +01:00
|
|
|
|
.append($action)
|
|
|
|
|
|
.append($("<span>").text(` ${r.sub_results.length - index_s} more result(s) from ${r.meta.title}`));
|
2026-03-19 08:50:45 +01:00
|
|
|
|
|
|
|
|
|
|
$searchResultBody.append($("<p>").append($expander));
|
|
|
|
|
|
$wrapper = $("<div>")
|
|
|
|
|
|
.addClass("collapse")
|
2026-03-20 13:08:53 +01:00
|
|
|
|
.attr("id", wrapper_id)
|
2026-03-20 13:38:12 +01:00
|
|
|
|
.on("hide.bs.collapse", _ => $action.text("▶ Show"))
|
|
|
|
|
|
.on("show.bs.collapse", _ => $action.text("▼ Hide"));
|
2026-03-19 08:50:45 +01:00
|
|
|
|
$searchResultBody.append($wrapper);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
const $entry = $("<div>")
|
2026-03-20 13:38:12 +01:00
|
|
|
|
.css("margin-top", "0.5rem");
|
2026-03-17 13:04:43 +01:00
|
|
|
|
|
|
|
|
|
|
$entry.append(
|
2026-03-19 08:50:45 +01:00
|
|
|
|
$("<a>")
|
|
|
|
|
|
.addClass("d-block")
|
|
|
|
|
|
.attr("href", s.url)
|
2026-03-17 13:04:43 +01:00
|
|
|
|
.text(s.title)
|
|
|
|
|
|
);
|
|
|
|
|
|
|
2026-03-19 08:50:45 +01:00
|
|
|
|
$entry.append($("<p>").html(s.excerpt));
|
2026-03-17 13:04:43 +01:00
|
|
|
|
|
2026-03-19 08:50:45 +01:00
|
|
|
|
if (index_s < LIMIT) {
|
|
|
|
|
|
$searchResultBody.append($entry);
|
|
|
|
|
|
} else {
|
|
|
|
|
|
$wrapper.append($entry);
|
|
|
|
|
|
}
|
2026-03-17 13:04:43 +01:00
|
|
|
|
});
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-03-20 12:52:01 +01:00
|
|
|
|
// Finally, show the search results.
|
|
|
|
|
|
//
|
|
|
|
|
|
// Ideally we would just call setContent but there appears to be a bug in Bootstrap
|
|
|
|
|
|
// 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.
|
|
|
|
|
|
//
|
|
|
|
|
|
// [1]: https://github.com/twbs/bootstrap/issues/37206#issuecomment-1259541205
|
|
|
|
|
|
$(popover.tip.querySelector('.popover-body')).html($html[0]);
|
|
|
|
|
|
popover.update();
|
2026-03-17 13:04:43 +01:00
|
|
|
|
};
|
|
|
|
|
|
});
|
|
|
|
|
|
})(jQuery);
|