Files
paliad/frontend/src/client/project-form.ts
mAi 2cfd54f0cd wip(projects): t-paliad-222 — backend + frontend changes (pre-merge checkpoint)
Backend: mig 110/111 (will be renumbered after merging main),
validators + helpers widened, BuildProjectCode helper + projection
populator wired into List/GetByID/ListAncestors/GetTree/CCR. All
internal Go tests pass.

Frontend: ProjectFormFields conditional render — opponent_code on
litigation, our_side renamed to Client Role on case with grouped
optgroups. i18n keys for both DE and EN. fristenrechner perspective
mapping widened. project-form.ts payload reader/writer + showFieldsForType
toggle for new litigation block.

Migration slots about to be bumped (mig 110 was claimed by euler's
project_type_other on main).
2026-05-20 14:45:33 +02:00

256 lines
9.4 KiB
TypeScript

import { t, tDyn } from "./i18n";
// Shared logic for the Project form rendered by ProjectFormFields.tsx.
// Used by /projects/new and the edit modal on /projects/{id}.
export interface ProjectMini {
id: string;
title: string;
type: string;
reference?: string | null;
}
export interface ProjectFormState {
type: string;
parentID: string;
parentTitle: string;
title: string;
reference: string;
description: string;
status: string;
clientNumber: string;
matterNumber: string;
billingReference: string;
netDocumentsURL: string;
industry: string;
country: string;
patentNumber: string;
filingDate: string;
grantDate: string;
court: string;
caseNumber: string;
ourSide: string;
}
let parentCandidates: ProjectMini[] = [];
function $(id: string): HTMLElement {
const el = document.getElementById(id);
if (!el) throw new Error("missing form element: " + id);
return el;
}
function tryGet(id: string): HTMLElement | null {
return document.getElementById(id);
}
// showFieldsForType toggles parent-picker + type-specific blocks.
export function showFieldsForType(typeSel: string) {
const parentWrap = tryGet("projekt-parent-wrap") as HTMLDivElement | null;
const clientFields = tryGet("fields-client") as HTMLDivElement | null;
const litigationFields = tryGet("fields-litigation") as HTMLDivElement | null;
const patentFields = tryGet("fields-patent") as HTMLDivElement | null;
const caseFields = tryGet("fields-case") as HTMLDivElement | null;
if (clientFields) clientFields.style.display = typeSel === "client" ? "block" : "none";
if (litigationFields) litigationFields.style.display = typeSel === "litigation" ? "block" : "none";
if (patentFields) patentFields.style.display = typeSel === "patent" ? "block" : "none";
if (caseFields) caseFields.style.display = typeSel === "case" ? "block" : "none";
if (parentWrap) parentWrap.style.display = typeSel === "client" ? "none" : "block";
}
export async function loadParentCandidates(excludeID?: string) {
try {
const resp = await fetch("/api/projects");
if (!resp.ok) return;
const all = (await resp.json()) as ProjectMini[];
parentCandidates = excludeID ? all.filter((p) => p.id !== excludeID) : all;
} catch {
/* network — leave empty */
}
}
function esc(s: string): string {
const d = document.createElement("div");
d.textContent = s;
return d.innerHTML;
}
export function initParentPicker() {
const input = tryGet("projekt-parent-input") as HTMLInputElement | null;
const hidden = tryGet("projekt-parent-id") as HTMLInputElement | null;
const sugs = tryGet("projekt-parent-suggestions") as HTMLDivElement | null;
if (!input || !hidden || !sugs) return;
input.addEventListener("input", () => {
const q = input.value.trim().toLowerCase();
hidden.value = "";
if (!q) {
sugs.innerHTML = "";
return;
}
const matches = parentCandidates
.filter((p) => {
const hay = (p.title + " " + (p.reference || "")).toLowerCase();
return hay.includes(q);
})
.slice(0, 8);
sugs.innerHTML = matches
.map(
(p) =>
`<div class="collab-suggestion" data-id="${esc(p.id)}" data-title="${esc(p.title)}">
<strong>${esc(p.title)}</strong>
<span class="entity-type-chip entity-type-${esc(p.type)}">${esc(tDyn("projects.type." + p.type) || p.type)}</span>
</div>`,
)
.join("");
sugs.querySelectorAll<HTMLDivElement>(".collab-suggestion").forEach((el) => {
el.addEventListener("click", () => {
hidden.value = el.dataset.id!;
input.value = el.dataset.title!;
sugs.innerHTML = "";
});
});
});
}
// wireTypeChange wires the <select id="project-type"> change handler and runs
// the visibility pass once with the current value.
export function wireTypeChange() {
const typeSel = $("project-type") as HTMLSelectElement;
showFieldsForType(typeSel.value);
typeSel.addEventListener("change", () => showFieldsForType(typeSel.value));
}
// readPayload collects the form's current values into a CreateProjectInput /
// UpdateProjectInput compatible JSON payload. Returns null + sets msg when
// title is missing.
//
// `omitEmpty` controls whether empty-string fields are sent. For Create we
// drop them (server treats absent as default). For Update we send them as
// `""` so the server can clear the column — except for ParentID (handled
// specially because client→non-client requires structural changes the user
// shouldn't trigger from an edit form).
export function readPayload(
msg: HTMLElement,
opts: { omitEmpty: boolean; mode: "create" | "edit" },
): Record<string, unknown> | null {
const type = ($("project-type") as HTMLSelectElement).value;
const title = ($("project-title") as HTMLInputElement).value.trim();
if (!title) {
msg.textContent = t("projects.error.title_required") || "Title required";
msg.className = "form-msg form-msg-error";
return null;
}
const payload: Record<string, unknown> = {
type,
title,
status: ($("project-status") as HTMLSelectElement).value,
};
const parentID = ($("projekt-parent-id") as HTMLInputElement).value;
if (type !== "client" && parentID) {
payload.parent_id = parentID;
}
const stringField = (id: string, key: string) => {
const v = ($(id) as HTMLInputElement).value.trim();
if (v) payload[key] = v;
else if (!opts.omitEmpty) payload[key] = "";
};
stringField("project-ref", "reference");
stringField("project-client-number", "client_number");
stringField("project-matter-number", "matter_number");
stringField("project-billing-ref", "billing_reference");
stringField("project-netdocs", "netdocuments_url");
if (type === "client") {
stringField("project-industry", "industry");
stringField("project-country", "country");
}
if (type === "patent") {
stringField("project-patent-number", "patent_number");
const fd = ($("project-filing-date") as HTMLInputElement).value;
if (fd) payload.filing_date = fd + "T00:00:00Z";
const gd = ($("project-grant-date") as HTMLInputElement).value;
if (gd) payload.grant_date = gd + "T00:00:00Z";
}
if (type === "litigation") {
// opponent_code is the litigation-only short slug used as the
// middle segment when BuildProjectCode auto-derives a project
// code from the ancestor tree (t-paliad-222 / m/paliad#50).
// Uppercased on submit so the user can type lowercase comfortably
// — the DB CHECK enforces the [A-Z0-9-]{1,16} pattern.
const ocEl = tryGet("project-opponent-code") as HTMLInputElement | null;
if (ocEl) {
const v = ocEl.value.trim().toUpperCase();
if (v) payload.opponent_code = v;
else if (!opts.omitEmpty) payload.opponent_code = "";
}
}
if (type === "case") {
stringField("project-court", "court");
stringField("project-case-number", "case_number");
// Client Role (DB column: our_side) — case-only after t-paliad-222.
// The select uses "" for the unset option; the service maps empty
// string to NULL via nullableOurSide.
const osSel = tryGet("project-our-side") as HTMLSelectElement | null;
if (osSel) {
const v = osSel.value.trim();
if (v) payload.our_side = v;
else if (!opts.omitEmpty) payload.our_side = "";
}
}
const desc = ($("project-description") as HTMLTextAreaElement).value.trim();
if (desc) payload.description = desc;
else if (!opts.omitEmpty) payload.description = "";
// The edit modal currently doesn't expose reparenting; keep parent_id off
// the payload unless the user is creating.
if (opts.mode === "edit") {
delete payload.parent_id;
}
return payload;
}
// prefillForm hydrates the form fields from an existing Project record.
export function prefillForm(p: Record<string, unknown>) {
const get = (id: string) => $(id) as HTMLInputElement;
const getSel = (id: string) => $(id) as HTMLSelectElement;
const getTA = (id: string) => $(id) as HTMLTextAreaElement;
const type = String(p.type ?? "project");
getSel("project-type").value = type;
showFieldsForType(type);
get("project-title").value = String(p.title ?? "");
get("project-ref").value = String(p.reference ?? "");
get("project-client-number").value = String(p.client_number ?? "");
get("project-matter-number").value = String(p.matter_number ?? "");
get("project-billing-ref").value = String(p.billing_reference ?? "");
get("project-netdocs").value = String(p.netdocuments_url ?? "");
get("project-industry").value = String(p.industry ?? "");
get("project-country").value = String(p.country ?? "");
get("project-patent-number").value = String(p.patent_number ?? "");
get("project-filing-date").value = isoToDate(p.filing_date as string | null | undefined);
get("project-grant-date").value = isoToDate(p.grant_date as string | null | undefined);
get("project-court").value = String(p.court ?? "");
get("project-case-number").value = String(p.case_number ?? "");
const osSel = tryGet("project-our-side") as HTMLSelectElement | null;
if (osSel) osSel.value = String(p.our_side ?? "");
const ocEl = tryGet("project-opponent-code") as HTMLInputElement | null;
if (ocEl) ocEl.value = String(p.opponent_code ?? "");
getTA("project-description").value = String(p.description ?? "");
getSel("project-status").value = String(p.status ?? "active");
}
function isoToDate(iso: string | null | undefined): string {
if (!iso) return "";
// Accept YYYY-MM-DD or full ISO; slice to date.
return iso.slice(0, 10);
}