mirror of
https://github.com/matrix-org/matrix-spec
synced 2026-04-08 12:24:09 +02:00
Merge 6f05c2c78e into 3c9ba4a06d
This commit is contained in:
commit
907dac6be0
4
.github/workflows/main.yml
vendored
4
.github/workflows/main.yml
vendored
|
|
@ -236,6 +236,10 @@ jobs:
|
||||||
run: |
|
run: |
|
||||||
tar -C "spec${baseURL}" --strip-components=1 -xzf openapi.tar.gz
|
tar -C "spec${baseURL}" --strip-components=1 -xzf openapi.tar.gz
|
||||||
|
|
||||||
|
- name: "🔍 pagefind indexing"
|
||||||
|
run: |
|
||||||
|
npx -y pagefind --site "spec${baseURL}"
|
||||||
|
|
||||||
- name: "📦 Tarball creation"
|
- name: "📦 Tarball creation"
|
||||||
run: |
|
run: |
|
||||||
cd spec
|
cd spec
|
||||||
|
|
|
||||||
198
assets/js/offline-search.js
Normal file
198
assets/js/offline-search.js
Normal file
|
|
@ -0,0 +1,198 @@
|
||||||
|
/*
|
||||||
|
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.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
Adapted from [1] to combine Docsy"s built-in search UI with the Pagefind
|
||||||
|
search backend.
|
||||||
|
|
||||||
|
[1]: https://github.com/matrix-org/docsy/blob/71d103ebb20ace3d528178c4b6d92b6cc4f7fd53/assets/js/offline-search.js
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function ($) {
|
||||||
|
"use strict";
|
||||||
|
|
||||||
|
$(document).ready(async function () {
|
||||||
|
const pagefind = await import("/pagefind/pagefind.js");
|
||||||
|
const $searchInput = $(".td-search input");
|
||||||
|
|
||||||
|
//
|
||||||
|
// Lazily initialise Pagefind only when the user is about to start a search.
|
||||||
|
//
|
||||||
|
|
||||||
|
$searchInput.focus(() => {
|
||||||
|
pagefind.init();
|
||||||
|
});
|
||||||
|
|
||||||
|
//
|
||||||
|
// Register handler
|
||||||
|
//
|
||||||
|
|
||||||
|
$searchInput.on("change", (event) => {
|
||||||
|
render($(event.target));
|
||||||
|
|
||||||
|
// Hide keyboard on mobile browser
|
||||||
|
$searchInput.blur();
|
||||||
|
});
|
||||||
|
|
||||||
|
// Prevent reloading page by enter key on sidebar search.
|
||||||
|
$searchInput.closest("form").on("submit", () => {
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
//
|
||||||
|
// Pagefind
|
||||||
|
//
|
||||||
|
|
||||||
|
const render = async ($targetSearchInput) => {
|
||||||
|
//
|
||||||
|
// Dispose existing popover
|
||||||
|
//
|
||||||
|
|
||||||
|
{
|
||||||
|
let popover = bootstrap.Popover.getInstance($targetSearchInput[0]);
|
||||||
|
if (popover !== null) {
|
||||||
|
popover.dispose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
// Search
|
||||||
|
//
|
||||||
|
|
||||||
|
const searchQuery = $targetSearchInput.val();
|
||||||
|
if (searchQuery === "") {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const search = await pagefind.debouncedSearch(searchQuery);
|
||||||
|
if (search === null) {
|
||||||
|
// A more recent search call has been made, nothing to do.
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const results = await Promise.all(search.results.slice(0, 20).map(r => r.data()));
|
||||||
|
|
||||||
|
//
|
||||||
|
// Make result html
|
||||||
|
//
|
||||||
|
|
||||||
|
const $html = $("<div>");
|
||||||
|
|
||||||
|
$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")
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
const $searchResultBody = $("<div>").css({
|
||||||
|
maxHeight: `calc(100vh - ${
|
||||||
|
$targetSearchInput.offset().top - $(window).scrollTop() + 180
|
||||||
|
}px)`,
|
||||||
|
overflowY: "auto",
|
||||||
|
});
|
||||||
|
$html.append($searchResultBody);
|
||||||
|
|
||||||
|
if (results.length === 0) {
|
||||||
|
$searchResultBody.append(
|
||||||
|
$("<p>").text(`No results found for query "${searchQuery}"`)
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
results.forEach((r, index_r) => {
|
||||||
|
// Add the main result"s page title.
|
||||||
|
$searchResultBody.append(
|
||||||
|
$("<a>")
|
||||||
|
.addClass("d-block")
|
||||||
|
.css({
|
||||||
|
fontSize: "1.2rem",
|
||||||
|
})
|
||||||
|
.attr("href", r.url)
|
||||||
|
.text(r.meta.title)
|
||||||
|
);
|
||||||
|
|
||||||
|
// 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}`;
|
||||||
|
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)
|
||||||
|
.css("margin-left", "0.5rem")
|
||||||
|
.text(`${r.sub_results.length - index_s} more result(s) from ${r.meta.title}`);
|
||||||
|
|
||||||
|
$searchResultBody.append($("<p>").append($expander));
|
||||||
|
$wrapper = $("<div>")
|
||||||
|
.addClass("collapse")
|
||||||
|
.attr("id", wrapper_id);
|
||||||
|
$searchResultBody.append($wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
const $entry = $("<div>")
|
||||||
|
.css("margin-top", "0.5rem")
|
||||||
|
.css("margin-left", "0.5rem");
|
||||||
|
|
||||||
|
$entry.append(
|
||||||
|
$("<a>")
|
||||||
|
.addClass("d-block")
|
||||||
|
.attr("href", s.url)
|
||||||
|
.text(s.title)
|
||||||
|
);
|
||||||
|
|
||||||
|
$entry.append($("<p>").html(s.excerpt));
|
||||||
|
|
||||||
|
if (index_s < LIMIT) {
|
||||||
|
$searchResultBody.append($entry);
|
||||||
|
} else {
|
||||||
|
$wrapper.append($entry);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
$targetSearchInput.one("shown.bs.popover", () => {
|
||||||
|
$(".td-offline-search-results__close-button").on("click", () => {
|
||||||
|
$targetSearchInput.val("");
|
||||||
|
$targetSearchInput.trigger("change");
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
const popover = new bootstrap.Popover($targetSearchInput, {
|
||||||
|
content: $html[0],
|
||||||
|
html: true,
|
||||||
|
customClass: "td-offline-search-results",
|
||||||
|
placement: "bottom",
|
||||||
|
});
|
||||||
|
popover.show();
|
||||||
|
};
|
||||||
|
});
|
||||||
|
})(jQuery);
|
||||||
|
|
@ -661,3 +661,8 @@ dd {
|
||||||
margin: 0;
|
margin: 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Style for the page search widget */
|
||||||
|
.td-offline-search-results {
|
||||||
|
max-width: 460px;
|
||||||
|
}
|
||||||
|
|
|
||||||
1
changelogs/internal/newsfragments/2331.feature
Normal file
1
changelogs/internal/newsfragments/2331.feature
Normal file
|
|
@ -0,0 +1 @@
|
||||||
|
Add page search widget.
|
||||||
|
|
@ -66,6 +66,7 @@ description = "Home of the Matrix specification for decentralised communication"
|
||||||
|
|
||||||
[params]
|
[params]
|
||||||
copyright = "The Matrix.org Foundation C.I.C."
|
copyright = "The Matrix.org Foundation C.I.C."
|
||||||
|
offlineSearch = true
|
||||||
|
|
||||||
[params.version]
|
[params.version]
|
||||||
# must be one of "unstable", "current", "historical"
|
# must be one of "unstable", "current", "historical"
|
||||||
|
|
@ -151,7 +152,8 @@ sidebar_menu_compact = true
|
||||||
[server.headers.values]
|
[server.headers.values]
|
||||||
# `style-src 'unsafe-inline'` is needed to correctly render the maths in the Olm spec:
|
# `style-src 'unsafe-inline'` is needed to correctly render the maths in the Olm spec:
|
||||||
# https://github.com/KaTeX/KaTeX/issues/4096
|
# https://github.com/KaTeX/KaTeX/issues/4096
|
||||||
Content-Security-Policy = "default-src 'self'; style-src 'self' 'unsafe-inline'; script-src 'self'; img-src 'self' data:; connect-src 'self'; font-src 'self' data:; media-src 'self'; child-src 'self'; form-action 'self'; object-src 'self'"
|
# TODO: Figure out CSP to allow loading the Pagefind Wasm
|
||||||
|
#Content-Security-Policy = "default-src 'self'; style-src 'self' 'unsafe-inline'; script-src 'self'; img-src 'self' data:; connect-src 'self'; font-src 'self' data:; media-src 'self'; child-src 'self'; form-action 'self'; object-src 'self'"
|
||||||
X-XSS-Protection = "1; mode=block"
|
X-XSS-Protection = "1; mode=block"
|
||||||
X-Content-Type-Options = "nosniff"
|
X-Content-Type-Options = "nosniff"
|
||||||
# Strict-Transport-Security = "max-age=31536000; includeSubDomains; preload"
|
# Strict-Transport-Security = "max-age=31536000; includeSubDomains; preload"
|
||||||
|
|
|
||||||
10
layouts/_partials/search-input.html
Normal file
10
layouts/_partials/search-input.html
Normal file
|
|
@ -0,0 +1,10 @@
|
||||||
|
<div class="td-search td-search--offline">
|
||||||
|
<div class="td-search__icon"></div>
|
||||||
|
<input
|
||||||
|
type="search"
|
||||||
|
class="td-search__input form-control"
|
||||||
|
placeholder="Search the spec"
|
||||||
|
aria-label="Search the spec"
|
||||||
|
autocomplete="off"
|
||||||
|
>
|
||||||
|
</div>
|
||||||
Loading…
Reference in a new issue