Add data-driven filter-bar partial (todo §1); GET form: search/segmented/select/chips/daterange + applied pills
This commit is contained in:
84
src/filter-bar.test.ts
Normal file
84
src/filter-bar.test.ts
Normal file
@@ -0,0 +1,84 @@
|
||||
import assert from "node:assert/strict";
|
||||
import { dirname, join } from "node:path";
|
||||
import { test } from "node:test";
|
||||
import { fileURLToPath } from "node:url";
|
||||
import * as ejs from "ejs";
|
||||
|
||||
const filterBar = join(dirname(fileURLToPath(import.meta.url)), "..", "views", "partials", "filter-bar.ejs");
|
||||
const render = (data: Record<string, unknown> = {}): Promise<string> => ejs.renderFile(filterBar, data);
|
||||
const flat = (s: string): string => s.replace(/>\s+</g, "><").replace(/\s+/g, " ").trim();
|
||||
|
||||
const config = {
|
||||
label: "Filter people",
|
||||
rows: [
|
||||
[
|
||||
{ type: "search", name: "q", placeholder: "Search people…", value: "ann", label: "Search people" },
|
||||
{
|
||||
type: "segmented",
|
||||
name: "status",
|
||||
legend: "Status",
|
||||
value: "active",
|
||||
options: [
|
||||
{ value: "all", label: "All", count: "1,284" },
|
||||
{ value: "active", label: "Active" },
|
||||
{ value: "archived", label: "Archived" },
|
||||
],
|
||||
},
|
||||
{ type: "select", name: "team", label: "Team", value: "design", options: [{ value: "", label: "All teams" }, { value: "design", label: "Design" }] },
|
||||
{ type: "spacer" },
|
||||
],
|
||||
[
|
||||
{
|
||||
type: "chips",
|
||||
name: "tag",
|
||||
legend: "Tags",
|
||||
value: ["engineering", "oncall"],
|
||||
options: [{ value: "engineering", label: "Engineering" }, { value: "design", label: "Design" }, { value: "oncall", label: "On-call" }],
|
||||
},
|
||||
{ type: "daterange", legend: "Joined", from: { name: "joined_from", value: "2026-01-01", label: "Joined from" }, to: { name: "joined_to", value: "2026-06-14", label: "Joined to" } },
|
||||
],
|
||||
],
|
||||
pills: [{ label: "Team", value: "Engineering", remove: "?tag=oncall" }],
|
||||
clearHref: "?",
|
||||
};
|
||||
|
||||
test("filter-bar renders a GET form with every control type, reflecting current values", async () => {
|
||||
const html = flat(await render(config));
|
||||
|
||||
// GET form (server-side filtering, zero-JS).
|
||||
assert.match(html, /<form class="filters" method="get" aria-label="Filter people">/);
|
||||
|
||||
// search — icon + value reflected.
|
||||
assert.match(html, /<label class="search"><span class="sr-only">Search people<\/span><svg class="ico ico-sm" aria-hidden="true"><use href="#i-search"\s*\/?><\/svg><input type="search" name="q" placeholder="Search people…" value="ann"><\/label>/);
|
||||
|
||||
// segmented — current value checked, others not, optional count badge.
|
||||
assert.match(html, /<input type="radio" name="status" value="all"><span>All<\/span><span class="seg-count">1,284<\/span>/);
|
||||
assert.match(html, /<input type="radio" name="status" value="active" checked><span>Active<\/span>/);
|
||||
|
||||
// select — matching option selected.
|
||||
assert.match(html, /<select id="f-team" name="team"><option value="">All teams<\/option><option value="design" selected>Design<\/option><\/select>/);
|
||||
|
||||
// chips — checkbox group, only current values checked.
|
||||
assert.match(html, /<input type="checkbox" name="tag" value="engineering" checked>Engineering/);
|
||||
assert.match(html, /<input type="checkbox" name="tag" value="oncall" checked>On-call/);
|
||||
assert.match(html, /<input type="checkbox" name="tag" value="design">Design/);
|
||||
|
||||
// daterange — calendar icon + two date inputs with values.
|
||||
assert.match(html, /<div class="daterange"><svg class="ico ico-sm" aria-hidden="true"><use href="#i-cal"\s*\/?><\/svg>.*?<input type="date" id="f-joined_from" name="joined_from" value="2026-01-01">.*?<span class="to" aria-hidden="true">to<\/span>.*?<input type="date" id="f-joined_to" name="joined_to" value="2026-06-14">/);
|
||||
|
||||
// spacer.
|
||||
assert.match(html, /<div class="spacer"><\/div>/);
|
||||
|
||||
// applied pills + clear-all + Reset/Apply actions.
|
||||
assert.match(html, /<div class="active-pills" aria-label="Applied filters"><span class="filter-legend">Applied<\/span><span class="pill"><b>Team:<\/b> Engineering <a class="pill-x" href="\?tag=oncall" aria-label="Remove Team filter">/);
|
||||
assert.match(html, /<a class="pill-clear" href="\?">Clear all<\/a>/);
|
||||
assert.match(html, /<button type="reset" class="btn">Reset<\/button>/);
|
||||
assert.match(html, /<button type="submit" class="btn btn-primary"><svg class="ico ico-sm" aria-hidden="true"><use href="#i-search"\s*\/?><\/svg>Apply filters<\/button>/);
|
||||
});
|
||||
|
||||
test("filter-bar renders with defaults: form + actions, no pills, never throws", async () => {
|
||||
const html = flat(await render());
|
||||
assert.match(html, /<form class="filters" method="get"/);
|
||||
assert.match(html, /<button type="submit" class="btn btn-primary"/);
|
||||
assert.doesNotMatch(html, /active-pills/);
|
||||
});
|
||||
Reference in New Issue
Block a user