mirror of
https://github.com/gethomepage/homepage.git
synced 2026-02-08 00:40:52 +08:00
Chore: homepage tests (#6278)
This commit is contained in:
390
src/components/quicklaunch.test.jsx
Normal file
390
src/components/quicklaunch.test.jsx
Normal file
@@ -0,0 +1,390 @@
|
||||
// @vitest-environment jsdom
|
||||
|
||||
import { act, fireEvent, screen, waitFor } from "@testing-library/react";
|
||||
import { useState } from "react";
|
||||
import { beforeEach, describe, expect, it, vi } from "vitest";
|
||||
|
||||
import { renderWithProviders } from "test-utils/render-with-providers";
|
||||
|
||||
const { state, useSWR, getStoredProvider } = vi.hoisted(() => ({
|
||||
state: {
|
||||
widgets: {},
|
||||
},
|
||||
useSWR: vi.fn((key) => {
|
||||
if (key === "/api/widgets") return { data: state.widgets, error: undefined };
|
||||
return { data: undefined, error: undefined };
|
||||
}),
|
||||
getStoredProvider: vi.fn(() => null),
|
||||
}));
|
||||
|
||||
vi.mock("swr", () => ({
|
||||
default: useSWR,
|
||||
}));
|
||||
|
||||
vi.mock("./resolvedicon", () => ({
|
||||
default: function ResolvedIconMock() {
|
||||
return <div data-testid="resolved-icon" />;
|
||||
},
|
||||
}));
|
||||
|
||||
vi.mock("./widgets/search/search", () => ({
|
||||
getStoredProvider,
|
||||
searchProviders: {
|
||||
duckduckgo: {
|
||||
name: "DuckDuckGo",
|
||||
url: "https://duckduckgo.example/?q=",
|
||||
suggestionUrl: "https://duckduckgo.example/ac/?q=",
|
||||
target: "_self",
|
||||
},
|
||||
},
|
||||
}));
|
||||
|
||||
import QuickLaunch from "./quicklaunch";
|
||||
|
||||
function Wrapper({ servicesAndBookmarks = [], initialOpen = true } = {}) {
|
||||
const [searchString, setSearchString] = useState("");
|
||||
const [isOpen, setSearching] = useState(initialOpen);
|
||||
|
||||
return (
|
||||
<QuickLaunch
|
||||
servicesAndBookmarks={servicesAndBookmarks}
|
||||
searchString={searchString}
|
||||
setSearchString={setSearchString}
|
||||
isOpen={isOpen}
|
||||
setSearching={setSearching}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
describe("components/quicklaunch", () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
state.widgets = {};
|
||||
});
|
||||
|
||||
it("uses a custom provider from quicklaunch settings when configured", async () => {
|
||||
renderWithProviders(<Wrapper />, {
|
||||
settings: {
|
||||
quicklaunch: {
|
||||
provider: "custom",
|
||||
name: "MySearch",
|
||||
url: "https://custom.example/?q=",
|
||||
showSearchSuggestions: false,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "abc" } });
|
||||
|
||||
expect(await screen.findByText("MySearch quicklaunch.search")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("uses the search widget's custom provider configuration when quicklaunch settings are not provided", async () => {
|
||||
state.widgets = {
|
||||
w: {
|
||||
type: "search",
|
||||
options: { provider: "custom", name: "WidgetSearch", url: "https://widget.example/?q=" },
|
||||
},
|
||||
};
|
||||
|
||||
renderWithProviders(<Wrapper />, { settings: { quicklaunch: { showSearchSuggestions: false } } });
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "abc" } });
|
||||
|
||||
expect(await screen.findByText("WidgetSearch quicklaunch.search")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("uses the search widget's provider setting when quicklaunch settings are not provided", async () => {
|
||||
state.widgets = {
|
||||
w: {
|
||||
type: "search",
|
||||
options: { provider: "duckduckgo" },
|
||||
},
|
||||
};
|
||||
|
||||
renderWithProviders(<Wrapper />, { settings: { quicklaunch: { showSearchSuggestions: false } } });
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "abc" } });
|
||||
|
||||
expect(await screen.findByText("DuckDuckGo quicklaunch.search")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("renders results for urls and opens the selected result on Enter", async () => {
|
||||
const openSpy = vi.spyOn(window, "open").mockImplementation(() => null);
|
||||
|
||||
renderWithProviders(<Wrapper />, {
|
||||
settings: {
|
||||
target: "_self",
|
||||
quicklaunch: {
|
||||
provider: "duckduckgo",
|
||||
showSearchSuggestions: false,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "example.com" } });
|
||||
|
||||
expect(await screen.findByText("quicklaunch.visit URL")).toBeInTheDocument();
|
||||
expect(screen.getByText("DuckDuckGo quicklaunch.search")).toBeInTheDocument();
|
||||
|
||||
fireEvent.keyDown(input, { key: "Enter" });
|
||||
|
||||
await act(async () => {
|
||||
// Close/reset schedules timeouts (200ms + 300ms); flush them to avoid state updates after cleanup.
|
||||
await new Promise((r) => setTimeout(r, 350));
|
||||
});
|
||||
|
||||
expect(openSpy).toHaveBeenCalledWith("https://example.com/", "_self", "noreferrer");
|
||||
|
||||
openSpy.mockRestore();
|
||||
});
|
||||
|
||||
it("closes on Escape and clears the search string after the timeout", async () => {
|
||||
renderWithProviders(<Wrapper />, {
|
||||
settings: {
|
||||
quicklaunch: {
|
||||
provider: "duckduckgo",
|
||||
showSearchSuggestions: false,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "abc" } });
|
||||
expect(input).toHaveValue("abc");
|
||||
|
||||
fireEvent.keyDown(input, { key: "Escape" });
|
||||
|
||||
await act(async () => {
|
||||
await new Promise((r) => setTimeout(r, 350));
|
||||
});
|
||||
|
||||
expect(input).toHaveValue("");
|
||||
});
|
||||
|
||||
it("supports ArrowUp/ArrowDown navigation and opens a result on click", async () => {
|
||||
const openSpy = vi.spyOn(window, "open").mockImplementation(() => null);
|
||||
|
||||
renderWithProviders(
|
||||
<Wrapper
|
||||
servicesAndBookmarks={[
|
||||
{ name: "Alpha", href: "https://alpha.example", icon: "mdi:test" },
|
||||
{ name: "Alpine", href: "https://alpine.example" },
|
||||
]}
|
||||
/>,
|
||||
{ settings: { target: "_self", quicklaunch: { showSearchSuggestions: false } } },
|
||||
);
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "al" } });
|
||||
|
||||
await waitFor(() => {
|
||||
expect(document.querySelector('button[data-index="0"]')).toBeTruthy();
|
||||
expect(document.querySelector('button[data-index="1"]')).toBeTruthy();
|
||||
});
|
||||
|
||||
// Icon/abbr container renders when icon is present.
|
||||
expect(screen.getByTestId("resolved-icon")).toBeInTheDocument();
|
||||
|
||||
const button0 = document.querySelector('button[data-index="0"]');
|
||||
const button1 = document.querySelector('button[data-index="1"]');
|
||||
expect(button0.className).toContain("bg-theme-300/50");
|
||||
|
||||
fireEvent.keyDown(input, { key: "ArrowDown" });
|
||||
expect(button1.className).toContain("bg-theme-300/50");
|
||||
|
||||
fireEvent.keyDown(input, { key: "ArrowUp" });
|
||||
expect(button0.className).toContain("bg-theme-300/50");
|
||||
|
||||
fireEvent.click(button0);
|
||||
|
||||
await act(async () => {
|
||||
await new Promise((r) => setTimeout(r, 350));
|
||||
});
|
||||
|
||||
expect(openSpy).toHaveBeenCalledWith("https://alpha.example", "_self", "noreferrer");
|
||||
openSpy.mockRestore();
|
||||
});
|
||||
|
||||
it("handles Escape on a result button (not just the input)", async () => {
|
||||
renderWithProviders(<Wrapper servicesAndBookmarks={[{ name: "Alpha", href: "https://alpha.example" }]} />, {
|
||||
settings: { quicklaunch: { showSearchSuggestions: false } },
|
||||
});
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "al" } });
|
||||
await waitFor(() => expect(document.querySelector('button[data-index="0"]')).toBeTruthy());
|
||||
const button0 = document.querySelector('button[data-index="0"]');
|
||||
|
||||
button0.focus();
|
||||
fireEvent.keyDown(button0, { key: "Escape" });
|
||||
|
||||
await act(async () => {
|
||||
await new Promise((r) => setTimeout(r, 350));
|
||||
});
|
||||
|
||||
expect(input).toHaveValue("");
|
||||
});
|
||||
|
||||
it("highlights matching description text when searchDescriptions is enabled", async () => {
|
||||
renderWithProviders(
|
||||
<Wrapper
|
||||
servicesAndBookmarks={[
|
||||
{ name: "Unrelated", description: "This has MatchMe inside", href: "https://example.com" },
|
||||
]}
|
||||
/>,
|
||||
{
|
||||
settings: {
|
||||
quicklaunch: {
|
||||
provider: "duckduckgo",
|
||||
searchDescriptions: true,
|
||||
showSearchSuggestions: false,
|
||||
},
|
||||
},
|
||||
},
|
||||
);
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "matchme" } });
|
||||
|
||||
// A description-only match uses highlightText (bg-theme-300/10).
|
||||
const highlight = await screen.findByText(/matchme/i);
|
||||
expect(highlight.closest("span")?.className).toContain("bg-theme-300/10");
|
||||
});
|
||||
|
||||
it("fetches search suggestions and ArrowRight autocompletes the selected suggestion", async () => {
|
||||
const originalFetch = globalThis.fetch;
|
||||
const fetchSpy = vi.fn(async () => ({
|
||||
json: async () => ["test", ["test 1", "test 2", "test 3", "test 4", "test 5"]],
|
||||
}));
|
||||
// eslint-disable-next-line no-global-assign
|
||||
fetch = fetchSpy;
|
||||
|
||||
renderWithProviders(<Wrapper />, {
|
||||
settings: {
|
||||
quicklaunch: {
|
||||
provider: "duckduckgo",
|
||||
showSearchSuggestions: true,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "test" } });
|
||||
|
||||
// Suggestions are fetched via the API route.
|
||||
await waitFor(() => {
|
||||
expect(fetchSpy).toHaveBeenCalledWith(
|
||||
expect.stringContaining("/api/search/searchSuggestion?query=test"),
|
||||
expect.objectContaining({ signal: expect.any(AbortSignal) }),
|
||||
);
|
||||
});
|
||||
|
||||
await waitFor(() => {
|
||||
expect(screen.getAllByText("quicklaunch.searchsuggestion").length).toBeGreaterThan(0);
|
||||
});
|
||||
|
||||
const suggestionButton = Array.from(document.querySelectorAll("button")).find((btn) =>
|
||||
btn.textContent?.includes("test 1"),
|
||||
);
|
||||
expect(suggestionButton).toBeTruthy();
|
||||
fireEvent.mouseEnter(suggestionButton);
|
||||
fireEvent.keyDown(input, { key: "ArrowRight" });
|
||||
|
||||
expect(input).toHaveValue("test 1");
|
||||
|
||||
// eslint-disable-next-line no-global-assign
|
||||
fetch = originalFetch;
|
||||
});
|
||||
|
||||
it("uses the stored provider when the search widget provides a provider list", async () => {
|
||||
state.widgets = {
|
||||
w: {
|
||||
type: "search",
|
||||
options: { provider: ["duckduckgo"] },
|
||||
},
|
||||
};
|
||||
getStoredProvider.mockReturnValue({
|
||||
name: "StoredProvider",
|
||||
url: "https://stored.example/?q=",
|
||||
suggestionUrl: "https://stored.example/ac/?q=",
|
||||
});
|
||||
|
||||
renderWithProviders(<Wrapper />, { settings: { quicklaunch: { showSearchSuggestions: false } } });
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "abc" } });
|
||||
|
||||
expect(await screen.findByText("StoredProvider quicklaunch.search")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("renders the mobile button when configured and opens the dialog when clicked", async () => {
|
||||
renderWithProviders(<Wrapper initialOpen={false} />, {
|
||||
settings: {
|
||||
quicklaunch: {
|
||||
mobileButtonPosition: "top-right",
|
||||
provider: "duckduckgo",
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const mobileButton = screen.getByRole("button", { name: "" });
|
||||
expect(mobileButton.className).toContain("top-4 right-4");
|
||||
|
||||
fireEvent.click(mobileButton);
|
||||
const input = await screen.findByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
});
|
||||
|
||||
it("closes when the backdrop is clicked and clears the search string after the timeout", async () => {
|
||||
renderWithProviders(<Wrapper />, {
|
||||
settings: {
|
||||
quicklaunch: {
|
||||
provider: "duckduckgo",
|
||||
showSearchSuggestions: false,
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
const input = screen.getByPlaceholderText("Search");
|
||||
await waitFor(() => expect(input).toHaveFocus());
|
||||
|
||||
fireEvent.change(input, { target: { value: "example.com" } });
|
||||
expect(input).toHaveValue("example.com");
|
||||
|
||||
// The backdrop is a DIV; clicking it should close and schedule a reset.
|
||||
const backdrop = document.querySelector(".fixed.inset-0.bg-gray-500.opacity-50");
|
||||
expect(backdrop).toBeTruthy();
|
||||
fireEvent.click(backdrop);
|
||||
|
||||
await act(async () => {
|
||||
await new Promise((r) => setTimeout(r, 350));
|
||||
});
|
||||
|
||||
expect(input).toHaveValue("");
|
||||
});
|
||||
});
|
||||
Reference in New Issue
Block a user