mirror of
https://github.com/searxng/searxng.git
synced 2025-12-27 14:10:00 +00:00
* [mod] client/simple: client plugins
Defines a new interface for client side *"plugins"* that coexist with server
side plugin system. Each plugin (e.g., `InfiniteScroll`) extends the base
`ts Plugin`. Client side plugins are independent and lazy‑loaded via `router.ts`
when their `load()` conditions are met. On each navigation request, all
applicable plugins are instanced.
Since these are client side plugins, we can only invoke them once DOM is fully
loaded. E.g. `Calculator` will not render a new `answer` block until fully
loaded and executed.
For some plugins, we might want to handle its availability in `settings.yml`
and toggle in UI, like we do for server side plugins. In that case, we extend
`py Plugin` instancing only the information and then checking client side if
[`settings.plugins`](1ad832b1dc/client/simple/src/js/toolkit.ts (L134))
array has the plugin id.
* [mod] client/simple: rebuild static
133 lines
4.0 KiB
TypeScript
133 lines
4.0 KiB
TypeScript
// SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
import { http, listen, settings } from "../toolkit.ts";
|
|
import { assertElement } from "../util/assertElement.ts";
|
|
|
|
const fetchResults = async (qInput: HTMLInputElement, query: string): Promise<void> => {
|
|
try {
|
|
let res: Response;
|
|
|
|
if (settings.method === "GET") {
|
|
res = await http("GET", `./autocompleter?q=${query}`);
|
|
} else {
|
|
res = await http("POST", "./autocompleter", { body: new URLSearchParams({ q: query }) });
|
|
}
|
|
|
|
const results = await res.json();
|
|
|
|
const autocomplete = document.querySelector<HTMLElement>(".autocomplete");
|
|
assertElement(autocomplete);
|
|
|
|
const autocompleteList = document.querySelector<HTMLUListElement>(".autocomplete ul");
|
|
assertElement(autocompleteList);
|
|
|
|
autocomplete.classList.add("open");
|
|
autocompleteList.replaceChildren();
|
|
|
|
// show an error message that no result was found
|
|
if (results?.[1]?.length === 0) {
|
|
const noItemFoundMessage = Object.assign(document.createElement("li"), {
|
|
className: "no-item-found",
|
|
textContent: settings.translations?.no_item_found ?? "No results found"
|
|
});
|
|
autocompleteList.append(noItemFoundMessage);
|
|
return;
|
|
}
|
|
|
|
const fragment = new DocumentFragment();
|
|
|
|
for (const result of results[1]) {
|
|
const li = Object.assign(document.createElement("li"), { textContent: result });
|
|
|
|
listen("mousedown", li, () => {
|
|
qInput.value = result;
|
|
|
|
const form = document.querySelector<HTMLFormElement>("#search");
|
|
form?.submit();
|
|
|
|
autocomplete.classList.remove("open");
|
|
});
|
|
|
|
fragment.append(li);
|
|
}
|
|
|
|
autocompleteList.append(fragment);
|
|
} catch (error) {
|
|
console.error("Error fetching autocomplete results:", error);
|
|
}
|
|
};
|
|
|
|
const qInput = document.getElementById("q") as HTMLInputElement | null;
|
|
assertElement(qInput);
|
|
|
|
let timeoutId: number;
|
|
|
|
listen("input", qInput, () => {
|
|
clearTimeout(timeoutId);
|
|
|
|
const query = qInput.value;
|
|
const minLength = settings.autocomplete_min ?? 2;
|
|
|
|
if (query.length < minLength) return;
|
|
|
|
timeoutId = window.setTimeout(async () => {
|
|
if (query === qInput.value) {
|
|
await fetchResults(qInput, query);
|
|
}
|
|
}, 300);
|
|
});
|
|
|
|
const autocomplete: HTMLElement | null = document.querySelector<HTMLElement>(".autocomplete");
|
|
const autocompleteList: HTMLUListElement | null = document.querySelector<HTMLUListElement>(".autocomplete ul");
|
|
if (autocompleteList) {
|
|
listen("keyup", qInput, (event: KeyboardEvent) => {
|
|
const listItems = [...autocompleteList.children] as HTMLElement[];
|
|
|
|
const currentIndex = listItems.findIndex((item) => item.classList.contains("active"));
|
|
let newCurrentIndex = -1;
|
|
|
|
switch (event.key) {
|
|
case "ArrowUp": {
|
|
const currentItem = listItems[currentIndex];
|
|
if (currentItem && currentIndex >= 0) {
|
|
currentItem.classList.remove("active");
|
|
}
|
|
// we need to add listItems.length to the index calculation here because the JavaScript modulos
|
|
// operator doesn't work with negative numbers
|
|
newCurrentIndex = (currentIndex - 1 + listItems.length) % listItems.length;
|
|
break;
|
|
}
|
|
case "ArrowDown": {
|
|
const currentItem = listItems[currentIndex];
|
|
if (currentItem && currentIndex >= 0) {
|
|
currentItem.classList.remove("active");
|
|
}
|
|
newCurrentIndex = (currentIndex + 1) % listItems.length;
|
|
break;
|
|
}
|
|
case "Tab":
|
|
case "Enter":
|
|
if (autocomplete) {
|
|
autocomplete.classList.remove("open");
|
|
}
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
if (newCurrentIndex !== -1) {
|
|
const selectedItem = listItems[newCurrentIndex];
|
|
if (selectedItem) {
|
|
selectedItem.classList.add("active");
|
|
|
|
if (!selectedItem.classList.contains("no-item-found")) {
|
|
const qInput = document.getElementById("q") as HTMLInputElement | null;
|
|
if (qInput) {
|
|
qInput.value = selectedItem.textContent ?? "";
|
|
}
|
|
}
|
|
}
|
|
}
|
|
});
|
|
}
|