Merge pull request 'Tree layout toggles + fan + card->profile + server search' (#16) from phase2-tree-toggles into main
build-frontend / build (push) Successful in 1m26s
build-frontend / build (push) Successful in 1m26s
This commit was merged in pull request #16.
This commit is contained in:
@@ -34,6 +34,7 @@ export default function FamilyViewPage() {
|
|||||||
const [ready, setReady] = useState(false);
|
const [ready, setReady] = useState(false);
|
||||||
const [focusId, setFocusId] = useState<string | null>(null);
|
const [focusId, setFocusId] = useState<string | null>(null);
|
||||||
const [search, setSearch] = useState("");
|
const [search, setSearch] = useState("");
|
||||||
|
const [results, setResults] = useState<Person[] | null>(null); // server fuzzy search
|
||||||
const [firstName, setFirstName] = useState("");
|
const [firstName, setFirstName] = useState("");
|
||||||
// Inline add-relative form: which anchor + kind is open, and the typed name.
|
// Inline add-relative form: which anchor + kind is open, and the typed name.
|
||||||
// `key` keeps each empty slot's inline form independent (a person has 2
|
// `key` keeps each empty slot's inline form independent (a person has 2
|
||||||
@@ -65,6 +66,22 @@ export default function FamilyViewPage() {
|
|||||||
load();
|
load();
|
||||||
}, [load]);
|
}, [load]);
|
||||||
|
|
||||||
|
// Debounced server-side fuzzy search (pg_trgm) across the whole tree.
|
||||||
|
useEffect(() => {
|
||||||
|
const q = search.trim();
|
||||||
|
if (!q) {
|
||||||
|
setResults(null);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const t = setTimeout(async () => {
|
||||||
|
const { data } = await api.GET("/api/v1/trees/{tree_id}/persons", {
|
||||||
|
params: { path: { tree_id: treeId }, query: { q } },
|
||||||
|
});
|
||||||
|
setResults(data ?? []);
|
||||||
|
}, 250);
|
||||||
|
return () => clearTimeout(t);
|
||||||
|
}, [search, treeId]);
|
||||||
|
|
||||||
const byId = useMemo(() => new Map(people.map((p) => [p.id, p])), [people]);
|
const byId = useMemo(() => new Map(people.map((p) => [p.id, p])), [people]);
|
||||||
const parentsOf = (id: string) =>
|
const parentsOf = (id: string) =>
|
||||||
rels.filter((r) => r.type === "parent_child" && r.person_to_id === id).map((r) => r.person_from_id);
|
rels.filter((r) => r.type === "parent_child" && r.person_to_id === id).map((r) => r.person_from_id);
|
||||||
@@ -265,10 +282,9 @@ export default function FamilyViewPage() {
|
|||||||
const sorted = [...people].sort((a, b) =>
|
const sorted = [...people].sort((a, b) =>
|
||||||
(a.primary_name ?? "").localeCompare(b.primary_name ?? ""),
|
(a.primary_name ?? "").localeCompare(b.primary_name ?? ""),
|
||||||
);
|
);
|
||||||
const matches = search
|
// Server fuzzy results when searching; otherwise the loaded set.
|
||||||
? sorted.filter((p) => (p.primary_name ?? "").toLowerCase().includes(search.toLowerCase()))
|
const directory = results ?? sorted;
|
||||||
: sorted;
|
const shown = directory.slice(0, 200); // cap DOM nodes; refine search to narrow
|
||||||
const shown = matches.slice(0, 200); // cap DOM nodes; refine search to narrow
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="space-y-8">
|
<div className="space-y-8">
|
||||||
@@ -358,9 +374,9 @@ export default function FamilyViewPage() {
|
|||||||
))
|
))
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
{matches.length > shown.length && (
|
{directory.length > shown.length && (
|
||||||
<div className="border-t border-[var(--border)] bg-[var(--surface)] px-4 py-2 text-xs text-[var(--muted)]">
|
<div className="border-t border-[var(--border)] bg-[var(--surface)] px-4 py-2 text-xs text-[var(--muted)]">
|
||||||
Showing {shown.length} of {matches.length} — refine your search to narrow.
|
Showing {shown.length} of {directory.length} — refine your search to narrow.
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</Card>
|
</Card>
|
||||||
|
|||||||
@@ -3,14 +3,19 @@
|
|||||||
// Vendored from family-chart/dist/styles (the package blocks the CSS subpath export).
|
// Vendored from family-chart/dist/styles (the package blocks the CSS subpath export).
|
||||||
import "./chart.css";
|
import "./chart.css";
|
||||||
|
|
||||||
|
import Link from "next/link";
|
||||||
import { useParams, useRouter } from "next/navigation";
|
import { useParams, useRouter } from "next/navigation";
|
||||||
import { useEffect, useRef, useState } from "react";
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||||
|
|
||||||
import { api } from "@/lib/api/client";
|
import { api } from "@/lib/api/client";
|
||||||
import type { components } from "@/lib/api/schema";
|
import type { components } from "@/lib/api/schema";
|
||||||
|
import { Button } from "@/components/ui/button";
|
||||||
|
import { FanChart } from "@/components/fan-chart";
|
||||||
|
|
||||||
|
type Person = components["schemas"]["PersonRead"];
|
||||||
type Relationship = components["schemas"]["RelationshipRead"];
|
type Relationship = components["schemas"]["RelationshipRead"];
|
||||||
type Event = components["schemas"]["EventRead"];
|
type Event = components["schemas"]["EventRead"];
|
||||||
|
type Mode = "landscape" | "portrait" | "fan";
|
||||||
|
|
||||||
function splitName(name: string | null | undefined): [string, string] {
|
function splitName(name: string | null | undefined): [string, string] {
|
||||||
const t = (name ?? "").trim().split(/\s+/).filter(Boolean);
|
const t = (name ?? "").trim().split(/\s+/).filter(Boolean);
|
||||||
@@ -23,11 +28,16 @@ export default function TreePage() {
|
|||||||
const params = useParams<{ id: string }>();
|
const params = useParams<{ id: string }>();
|
||||||
const treeId = params.id;
|
const treeId = params.id;
|
||||||
const containerRef = useRef<HTMLDivElement>(null);
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
|
const [people, setPeople] = useState<Person[]>([]);
|
||||||
|
const [rels, setRels] = useState<Relationship[]>([]);
|
||||||
|
const [events, setEvents] = useState<Event[]>([]);
|
||||||
const [status, setStatus] = useState<"loading" | "empty" | "ready" | "error">("loading");
|
const [status, setStatus] = useState<"loading" | "empty" | "ready" | "error">("loading");
|
||||||
|
const [focusId, setFocusId] = useState<string | null>(null);
|
||||||
|
const [mode, setMode] = useState<Mode>("landscape");
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
let cancelled = false;
|
let cancelled = false;
|
||||||
|
|
||||||
(async () => {
|
(async () => {
|
||||||
const p = await api.GET("/api/v1/trees/{tree_id}/persons", {
|
const p = await api.GET("/api/v1/trees/{tree_id}/persons", {
|
||||||
params: { path: { tree_id: treeId } },
|
params: { path: { tree_id: treeId } },
|
||||||
@@ -40,31 +50,56 @@ export default function TreePage() {
|
|||||||
api.GET("/api/v1/trees/{tree_id}/relationships", { params: { path: { tree_id: treeId } } }),
|
api.GET("/api/v1/trees/{tree_id}/relationships", { params: { path: { tree_id: treeId } } }),
|
||||||
api.GET("/api/v1/trees/{tree_id}/events", { params: { path: { tree_id: treeId } } }),
|
api.GET("/api/v1/trees/{tree_id}/events", { params: { path: { tree_id: treeId } } }),
|
||||||
]);
|
]);
|
||||||
const people = p.data ?? [];
|
if (cancelled) return;
|
||||||
const rels: Relationship[] = r.data ?? [];
|
const ppl = p.data ?? [];
|
||||||
const events: Event[] = e.data ?? [];
|
setPeople(ppl);
|
||||||
if (people.length === 0) {
|
setRels(r.data ?? []);
|
||||||
if (!cancelled) setStatus("empty");
|
setEvents(e.data ?? []);
|
||||||
return;
|
setFocusId((cur) => cur ?? ppl[0]?.id ?? null);
|
||||||
}
|
setStatus(ppl.length ? "ready" : "empty");
|
||||||
|
})().catch(() => !cancelled && setStatus("error"));
|
||||||
const parentsOf = (id: string) =>
|
return () => {
|
||||||
rels.filter((x) => x.type === "parent_child" && x.person_to_id === id).map((x) => x.person_from_id);
|
cancelled = true;
|
||||||
const childrenOf = (id: string) =>
|
};
|
||||||
rels.filter((x) => x.type === "parent_child" && x.person_from_id === id).map((x) => x.person_to_id);
|
}, [router, treeId]);
|
||||||
const partnersOf = (id: string) =>
|
|
||||||
rels
|
const byId = useMemo(() => new Map(people.map((p) => [p.id, p])), [people]);
|
||||||
.filter((x) => x.type === "partnership" && (x.person_from_id === id || x.person_to_id === id))
|
const parentsOf = useCallback(
|
||||||
.map((x) => (x.person_from_id === id ? x.person_to_id : x.person_from_id));
|
(id: string) =>
|
||||||
|
rels.filter((x) => x.type === "parent_child" && x.person_to_id === id).map((x) => x.person_from_id),
|
||||||
const birthYear = new Map<string, string>();
|
[rels],
|
||||||
for (const ev of events) {
|
);
|
||||||
if (ev.person_id && ev.event_type === "birth" && !birthYear.has(ev.person_id)) {
|
const childrenOf = useCallback(
|
||||||
const y = ev.date_start ? ev.date_start.slice(0, 4) : ev.date_value ?? "";
|
(id: string) =>
|
||||||
if (y) birthYear.set(ev.person_id, y);
|
rels.filter((x) => x.type === "parent_child" && x.person_from_id === id).map((x) => x.person_to_id),
|
||||||
}
|
[rels],
|
||||||
|
);
|
||||||
|
const partnersOf = useCallback(
|
||||||
|
(id: string) =>
|
||||||
|
rels
|
||||||
|
.filter((x) => x.type === "partnership" && (x.person_from_id === id || x.person_to_id === id))
|
||||||
|
.map((x) => (x.person_from_id === id ? x.person_to_id : x.person_from_id)),
|
||||||
|
[rels],
|
||||||
|
);
|
||||||
|
const years = useMemo(() => {
|
||||||
|
const m = new Map<string, string>();
|
||||||
|
for (const ev of events) {
|
||||||
|
if (ev.person_id && ev.event_type === "birth" && !m.has(ev.person_id)) {
|
||||||
|
const y = ev.date_start ? ev.date_start.slice(0, 4) : ev.date_value ?? "";
|
||||||
|
if (y) m.set(ev.person_id, y);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
return m;
|
||||||
|
}, [events]);
|
||||||
|
const nameOf = useCallback((id: string) => byId.get(id)?.primary_name ?? "Unknown", [byId]);
|
||||||
|
const yearOf = useCallback((id: string) => years.get(id) ?? "", [years]);
|
||||||
|
|
||||||
|
// family-chart for landscape/portrait. Intentionally not keyed on focusId —
|
||||||
|
// card clicks recenter via updateMainId without rebuilding the chart.
|
||||||
|
useEffect(() => {
|
||||||
|
if (status !== "ready" || mode === "fan" || !containerRef.current) return;
|
||||||
|
let cancelled = false;
|
||||||
|
(async () => {
|
||||||
const data = people.map((pp) => {
|
const data = people.map((pp) => {
|
||||||
const [fn, ln] = splitName(pp.primary_name);
|
const [fn, ln] = splitName(pp.primary_name);
|
||||||
return {
|
return {
|
||||||
@@ -72,56 +107,98 @@ export default function TreePage() {
|
|||||||
data: {
|
data: {
|
||||||
"first name": fn || "Unnamed",
|
"first name": fn || "Unnamed",
|
||||||
"last name": ln,
|
"last name": ln,
|
||||||
birthday: birthYear.get(pp.id) ?? "",
|
birthday: years.get(pp.id) ?? "",
|
||||||
gender: pp.gender === "female" ? "F" : "M",
|
gender: pp.gender === "female" ? "F" : "M",
|
||||||
},
|
},
|
||||||
rels: {
|
rels: { spouses: partnersOf(pp.id), parents: parentsOf(pp.id), children: childrenOf(pp.id) },
|
||||||
spouses: partnersOf(pp.id),
|
|
||||||
parents: parentsOf(pp.id),
|
|
||||||
children: childrenOf(pp.id),
|
|
||||||
},
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
const f3 = await import("family-chart");
|
||||||
if (cancelled || !containerRef.current) return;
|
if (cancelled || !containerRef.current) return;
|
||||||
try {
|
containerRef.current.innerHTML = "";
|
||||||
const f3 = await import("family-chart");
|
const chart = f3.createChart(containerRef.current, data);
|
||||||
containerRef.current.innerHTML = "";
|
chart
|
||||||
const chart = f3.createChart(containerRef.current, data);
|
.setCardHtml()
|
||||||
chart.setCardHtml().setCardDisplay([["first name", "last name"], ["birthday"]]);
|
.setCardDisplay([["first name", "last name"], ["birthday"]])
|
||||||
chart.updateTree({ initial: true });
|
.setOnCardClick((_e: unknown, d: { data?: { id?: string } }) => {
|
||||||
if (!cancelled) setStatus("ready");
|
const id = d?.data?.id;
|
||||||
} catch {
|
if (id) {
|
||||||
if (!cancelled) setStatus("error");
|
setFocusId(id);
|
||||||
}
|
chart.updateMainId(id);
|
||||||
})().catch(() => {
|
chart.updateTree();
|
||||||
if (!cancelled) setStatus("error");
|
}
|
||||||
});
|
});
|
||||||
|
if (mode === "portrait") chart.setOrientationVertical();
|
||||||
|
else chart.setOrientationHorizontal();
|
||||||
|
if (focusId) chart.updateMainId(focusId);
|
||||||
|
chart.updateTree({ initial: true });
|
||||||
|
})();
|
||||||
return () => {
|
return () => {
|
||||||
cancelled = true;
|
cancelled = true;
|
||||||
};
|
};
|
||||||
}, [router, treeId]);
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [status, mode, people, rels, events]);
|
||||||
|
|
||||||
|
const ModeButton = ({ m, label }: { m: Mode; label: string }) => (
|
||||||
|
<button
|
||||||
|
onClick={() => setMode(m)}
|
||||||
|
className={`rounded-md px-3 py-1.5 text-sm transition-colors ${
|
||||||
|
mode === m ? "bg-bronze text-paper" : "text-[var(--muted)] hover:text-[var(--foreground)]"
|
||||||
|
}`}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</button>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="space-y-4">
|
<div className="space-y-4">
|
||||||
<div className="flex flex-wrap items-center justify-between gap-2">
|
<div className="flex flex-wrap items-center justify-between gap-3">
|
||||||
<h1 className="text-2xl font-semibold">Tree</h1>
|
<h1 className="text-2xl font-semibold">Tree</h1>
|
||||||
<span className="text-sm text-[var(--muted)]">
|
<div className="flex items-center gap-3">
|
||||||
Drag to pan · scroll to zoom · click a person to recenter
|
<div className="flex items-center rounded-lg border border-[var(--border)] p-0.5">
|
||||||
</span>
|
<ModeButton m="landscape" label="Landscape" />
|
||||||
|
<ModeButton m="portrait" label="Portrait" />
|
||||||
|
<ModeButton m="fan" label="Fan" />
|
||||||
|
</div>
|
||||||
|
{focusId && (
|
||||||
|
<Link
|
||||||
|
href={`/trees/${treeId}/persons/${focusId}`}
|
||||||
|
className="text-sm text-bronze hover:underline"
|
||||||
|
>
|
||||||
|
Open {nameOf(focusId)} →
|
||||||
|
</Link>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{status === "empty" && (
|
{status === "empty" && (
|
||||||
<p className="text-[var(--muted)]">
|
<p className="text-[var(--muted)]">No people yet — add some under People, or import a GEDCOM.</p>
|
||||||
No people yet — add some under People, or import a GEDCOM.
|
|
||||||
</p>
|
|
||||||
)}
|
)}
|
||||||
{status === "error" && <p className="text-[var(--muted)]">Could not render the tree.</p>}
|
{status === "error" && <p className="text-[var(--muted)]">Could not render the tree.</p>}
|
||||||
<div
|
|
||||||
ref={containerRef}
|
{status === "ready" && mode === "fan" && focusId ? (
|
||||||
className="f3 rounded-xl border border-[var(--border)]"
|
<div className="rounded-xl border border-[var(--border)] bg-[var(--surface)] p-4">
|
||||||
style={{ width: "100%", height: "74vh", background: "var(--surface)" }}
|
<FanChart
|
||||||
/>
|
focusId={focusId}
|
||||||
|
parentsOf={parentsOf}
|
||||||
|
nameOf={nameOf}
|
||||||
|
yearOf={yearOf}
|
||||||
|
onSelect={setFocusId}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div
|
||||||
|
ref={containerRef}
|
||||||
|
className="f3 rounded-xl border border-[var(--border)]"
|
||||||
|
style={{ width: "100%", height: "74vh", background: "var(--surface)" }}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
|
||||||
|
<p className="text-sm text-[var(--muted)]">
|
||||||
|
{mode === "fan"
|
||||||
|
? "Click an ancestor to recenter the fan."
|
||||||
|
: "Drag to pan · scroll to zoom · click a person to recenter."}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -0,0 +1,128 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
// Radial fan chart of a focus person's ancestors (family-chart has no fan).
|
||||||
|
// Each generation is a ring; slot p in generation g descends from slot floor(p/2)
|
||||||
|
// in g-1. Click a wedge to refocus.
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
focusId: string;
|
||||||
|
parentsOf: (id: string) => string[];
|
||||||
|
nameOf: (id: string) => string;
|
||||||
|
yearOf: (id: string) => string;
|
||||||
|
onSelect: (id: string) => void;
|
||||||
|
generations?: number;
|
||||||
|
};
|
||||||
|
|
||||||
|
const SIZE = 720;
|
||||||
|
const CENTER = SIZE / 2;
|
||||||
|
const FOCUS_R = 46;
|
||||||
|
const SPAN = Math.PI * 1.6; // 288° fan
|
||||||
|
|
||||||
|
function polar(r: number, a: number): [number, number] {
|
||||||
|
// a = 0 points up, increasing clockwise.
|
||||||
|
return [CENTER + r * Math.sin(a), CENTER - r * Math.cos(a)];
|
||||||
|
}
|
||||||
|
|
||||||
|
function sector(r0: number, r1: number, a0: number, a1: number): string {
|
||||||
|
const [x0, y0] = polar(r1, a0);
|
||||||
|
const [x1, y1] = polar(r1, a1);
|
||||||
|
const [x2, y2] = polar(r0, a1);
|
||||||
|
const [x3, y3] = polar(r0, a0);
|
||||||
|
const large = a1 - a0 > Math.PI ? 1 : 0;
|
||||||
|
return `M${x0} ${y0} A${r1} ${r1} 0 ${large} 1 ${x1} ${y1} L${x2} ${y2} A${r0} ${r0} 0 ${large} 0 ${x3} ${y3} Z`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function clip(s: string, n: number): string {
|
||||||
|
return s.length > n ? s.slice(0, n - 1) + "…" : s;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function FanChart({
|
||||||
|
focusId,
|
||||||
|
parentsOf,
|
||||||
|
nameOf,
|
||||||
|
yearOf,
|
||||||
|
onSelect,
|
||||||
|
generations = 4,
|
||||||
|
}: Props) {
|
||||||
|
const gens: (string | null)[][] = [[focusId]];
|
||||||
|
for (let g = 1; g <= generations; g++) {
|
||||||
|
const row: (string | null)[] = [];
|
||||||
|
for (const slot of gens[g - 1]) {
|
||||||
|
const ps = slot ? parentsOf(slot) : [];
|
||||||
|
row.push(ps[0] ?? null, ps[1] ?? null);
|
||||||
|
}
|
||||||
|
gens.push(row);
|
||||||
|
}
|
||||||
|
|
||||||
|
const ringT = (CENTER - 60 - FOCUS_R) / generations;
|
||||||
|
const start = -SPAN / 2;
|
||||||
|
const wedges: React.ReactNode[] = [];
|
||||||
|
|
||||||
|
for (let g = 1; g <= generations; g++) {
|
||||||
|
const row = gens[g];
|
||||||
|
const w = SPAN / row.length;
|
||||||
|
const r0 = FOCUS_R + (g - 1) * ringT;
|
||||||
|
const r1 = FOCUS_R + g * ringT;
|
||||||
|
row.forEach((id, i) => {
|
||||||
|
const a0 = start + i * w;
|
||||||
|
const a1 = start + (i + 1) * w;
|
||||||
|
const mid = (a0 + a1) / 2;
|
||||||
|
const [tx, ty] = polar((r0 + r1) / 2, mid);
|
||||||
|
let deg = (mid * 180) / Math.PI;
|
||||||
|
if (deg > 90 || deg < -90) deg += 180; // keep text upright
|
||||||
|
wedges.push(
|
||||||
|
<g
|
||||||
|
key={`${g}-${i}`}
|
||||||
|
onClick={() => id && onSelect(id)}
|
||||||
|
style={{ cursor: id ? "pointer" : "default" }}
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
d={sector(r0 + 1, r1 - 1, a0 + 0.004, a1 - 0.004)}
|
||||||
|
fill={id ? "var(--surface)" : "transparent"}
|
||||||
|
stroke="var(--border)"
|
||||||
|
/>
|
||||||
|
{id && (
|
||||||
|
<text
|
||||||
|
x={tx}
|
||||||
|
y={ty}
|
||||||
|
transform={`rotate(${deg} ${tx} ${ty})`}
|
||||||
|
textAnchor="middle"
|
||||||
|
dominantBaseline="middle"
|
||||||
|
style={{ fontSize: g >= 3 ? 9 : 11, fill: "var(--foreground)" }}
|
||||||
|
>
|
||||||
|
{clip(nameOf(id), g >= 3 ? 12 : 18)}
|
||||||
|
</text>
|
||||||
|
)}
|
||||||
|
</g>,
|
||||||
|
);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const [fx, fy] = [CENTER, CENTER];
|
||||||
|
return (
|
||||||
|
<div className="overflow-auto">
|
||||||
|
<svg viewBox={`0 0 ${SIZE} ${SIZE}`} className="mx-auto block w-full max-w-3xl">
|
||||||
|
{wedges}
|
||||||
|
<circle cx={fx} cy={fy} r={FOCUS_R} fill="var(--color-bronze)" />
|
||||||
|
<text
|
||||||
|
x={fx}
|
||||||
|
y={fy - 4}
|
||||||
|
textAnchor="middle"
|
||||||
|
dominantBaseline="middle"
|
||||||
|
style={{ fontSize: 12, fill: "var(--color-paper)", fontWeight: 600 }}
|
||||||
|
>
|
||||||
|
{clip(nameOf(focusId), 12)}
|
||||||
|
</text>
|
||||||
|
<text
|
||||||
|
x={fx}
|
||||||
|
y={fy + 12}
|
||||||
|
textAnchor="middle"
|
||||||
|
dominantBaseline="middle"
|
||||||
|
style={{ fontSize: 10, fill: "var(--color-paper)" }}
|
||||||
|
>
|
||||||
|
{yearOf(focusId)}
|
||||||
|
</text>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
Vendored
+1
@@ -1412,6 +1412,7 @@ export interface operations {
|
|||||||
parameters: {
|
parameters: {
|
||||||
query?: {
|
query?: {
|
||||||
deleted?: boolean;
|
deleted?: boolean;
|
||||||
|
q?: string | null;
|
||||||
};
|
};
|
||||||
header?: never;
|
header?: never;
|
||||||
path: {
|
path: {
|
||||||
|
|||||||
@@ -564,6 +564,22 @@
|
|||||||
"default": false,
|
"default": false,
|
||||||
"title": "Deleted"
|
"title": "Deleted"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "q",
|
||||||
|
"in": "query",
|
||||||
|
"required": false,
|
||||||
|
"schema": {
|
||||||
|
"anyOf": [
|
||||||
|
{
|
||||||
|
"type": "string"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"type": "null"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"title": "Q"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"responses": {
|
"responses": {
|
||||||
|
|||||||
Reference in New Issue
Block a user