refactor: simplify payment settings UI

This commit is contained in:
JetSprow
2026-04-30 17:24:14 +10:00
parent 1e194aabdb
commit 9d99590338
2 changed files with 229 additions and 99 deletions

View File

@@ -1,12 +1,25 @@
"use client"; "use client";
import { useState } from "react"; import { useMemo, useState, type FormEvent } from "react";
import { useRouter } from "next/navigation";
import { Check, CreditCard, Pencil, ShieldCheck } from "lucide-react";
import { savePaymentConfig } from "@/actions/admin/payments";
import { ActiveStatusBadge, StatusBadge } from "@/components/shared/status-badge";
import { BooleanToggle } from "@/components/ui/boolean-toggle"; import { BooleanToggle } from "@/components/ui/boolean-toggle";
import { Button } from "@/components/ui/button"; import { Button } from "@/components/ui/button";
import {
Dialog,
DialogContent,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "@/components/ui/dialog";
import { Input } from "@/components/ui/input"; import { Input } from "@/components/ui/input";
import { Label } from "@/components/ui/label"; import { Label } from "@/components/ui/label";
import { savePaymentConfig } from "@/actions/admin/payments";
import { getErrorMessage } from "@/lib/errors"; import { getErrorMessage } from "@/lib/errors";
import { cn } from "@/lib/utils";
import { toast } from "sonner"; import { toast } from "sonner";
interface Field { interface Field {
@@ -20,48 +33,102 @@ interface Field {
interface Props { interface Props {
provider: string; provider: string;
providerName: string;
providerDescription: string;
fields: Field[]; fields: Field[];
currentConfig?: Record<string, string>; currentConfig?: Record<string, string>;
secretConfigured?: Record<string, boolean>; secretConfigured?: Record<string, boolean>;
enabled: boolean; enabled: boolean;
} }
export function PaymentConfigForm({ function selectedOptionLabels(field: Field, rawValue: string | undefined) {
const selected = new Set((rawValue ?? "").split(",").map((item) => item.trim()).filter(Boolean));
return (field.options ?? [])
.filter((option) => selected.has(option.value))
.map((option) => option.label);
}
function buildInitialCheckboxValues(fields: Field[], currentConfig?: Record<string, string>) {
const values: Record<string, Set<string>> = {};
for (const field of fields) {
if (field.type !== "checkboxes") continue;
values[field.key] = new Set(
(currentConfig?.[field.key] ?? "")
.split(",")
.map((item) => item.trim())
.filter(Boolean),
);
}
return values;
}
function configCompleteness(fields: Field[], currentConfig: Record<string, string> | undefined, secretConfigured: Record<string, boolean>) {
let configured = 0;
for (const field of fields) {
if (field.secret) {
if (secretConfigured[field.key]) configured += 1;
continue;
}
if (field.type === "checkboxes") {
if (selectedOptionLabels(field, currentConfig?.[field.key]).length > 0) configured += 1;
continue;
}
if (currentConfig?.[field.key]?.trim()) configured += 1;
}
return { configured, total: fields.length };
}
export function PaymentConfigItem({
provider, provider,
providerName,
providerDescription,
fields, fields,
currentConfig, currentConfig,
secretConfigured = {}, secretConfigured = {},
enabled: initialEnabled, enabled: initialEnabled,
}: Props) { }: Props) {
const router = useRouter();
const [open, setOpen] = useState(false);
const [enabled, setEnabled] = useState(initialEnabled); const [enabled, setEnabled] = useState(initialEnabled);
const [saving, setSaving] = useState(false); const [saving, setSaving] = useState(false);
const [checkboxValues, setCheckboxValues] = useState<Record<string, Set<string>>>(() =>
// Track checkbox field values (comma-separated strings) buildInitialCheckboxValues(fields, currentConfig),
const [checkboxValues, setCheckboxValues] = useState<Record<string, Set<string>>>(() => { );
const init: Record<string, Set<string>> = {}; const completeness = useMemo(
for (const field of fields) { () => configCompleteness(fields, currentConfig, secretConfigured),
if (field.type === "checkboxes") { [currentConfig, fields, secretConfigured],
const raw = currentConfig?.[field.key] || ""; );
init[field.key] = new Set(raw.split(",").map((s) => s.trim()).filter(Boolean)); const secretFields = fields.filter((field) => field.secret);
} const configuredSecretCount = secretFields.filter((field) => secretConfigured[field.key]).length;
} const displayName = currentConfig?.displayName?.trim();
return init; const checkboxSummaries = fields
}); .filter((field) => field.type === "checkboxes")
.flatMap((field) => selectedOptionLabels(field, currentConfig?.[field.key]));
function toggleCheckbox(fieldKey: string, value: string) { function toggleCheckbox(fieldKey: string, value: string) {
setCheckboxValues((prev) => { setCheckboxValues((current) => {
const next = new Set(prev[fieldKey]); const next = new Set(current[fieldKey] ?? []);
if (next.has(value)) next.delete(value); if (next.has(value)) {
else next.add(value); next.delete(value);
return { ...prev, [fieldKey]: next }; } else {
next.add(value);
}
return { ...current, [fieldKey]: next };
}); });
} }
async function handleSubmit(e: React.FormEvent<HTMLFormElement>) { async function handleSubmit(event: FormEvent<HTMLFormElement>) {
e.preventDefault(); event.preventDefault();
setSaving(true); if (saving) return;
const formData = new FormData(e.currentTarget);
const form = event.currentTarget;
const formData = new FormData(form);
const config: Record<string, string> = {}; const config: Record<string, string> = {};
for (const field of fields) { for (const field of fields) {
if (field.type === "checkboxes") { if (field.type === "checkboxes") {
config[field.key] = Array.from(checkboxValues[field.key] ?? []).join(","); config[field.key] = Array.from(checkboxValues[field.key] ?? []).join(",");
@@ -70,71 +137,147 @@ export function PaymentConfigForm({
} }
} }
setSaving(true);
try { try {
await savePaymentConfig(provider, config, enabled); await savePaymentConfig(provider, config, enabled);
for (const field of fields) { for (const field of fields) {
if (!field.secret) continue; if (!field.secret) continue;
const input = e.currentTarget.elements.namedItem(field.key); const input = form.elements.namedItem(field.key);
if (input instanceof HTMLInputElement) { if (input instanceof HTMLInputElement) {
input.value = ""; input.value = "";
} }
} }
toast.success("保存成功"); toast.success("支付配置已保存");
setOpen(false);
router.refresh();
} catch (error) { } catch (error) {
toast.error(getErrorMessage(error, "保存失败")); toast.error(getErrorMessage(error, "保存支付配置失败"));
} finally {
setSaving(false);
} }
setSaving(false);
} }
return ( return (
<form onSubmit={handleSubmit} className="form-panel space-y-5"> <section className="grid gap-4 border-t border-border/60 px-4 py-4 first:border-t-0 lg:grid-cols-[minmax(0,1fr)_auto_auto] lg:items-center">
<div className="grid gap-4 sm:grid-cols-2"> <div className="flex min-w-0 items-start gap-3">
{fields.map((field) => <span className="flex size-9 shrink-0 items-center justify-center rounded-lg border border-primary/15 bg-primary/10 text-primary">
field.type === "checkboxes" ? ( <CreditCard className="size-4" />
<div key={field.key} className="sm:col-span-2"> </span>
<Label>{field.label}</Label> <div className="min-w-0">
<div className="mt-3 flex flex-wrap gap-3"> <div className="flex flex-wrap items-center gap-2">
{field.options?.map((opt) => ( <h3 className="text-base font-semibold tracking-tight">{providerName}</h3>
<label key={opt.value} className="choice-card flex cursor-pointer items-center gap-2 px-3 py-2"> {displayName && <StatusBadge tone="neutral">{displayName}</StatusBadge>}
<input </div>
type="checkbox" <p className="mt-1 line-clamp-2 text-sm leading-6 text-muted-foreground">{providerDescription}</p>
className="size-4 rounded border-border accent-primary" </div>
checked={checkboxValues[field.key]?.has(opt.value) ?? false} </div>
onChange={() => toggleCheckbox(field.key, opt.value)}
<div className="flex flex-wrap gap-2 lg:justify-end">
<ActiveStatusBadge active={enabled} activeLabel="已启用" inactiveLabel="未启用" />
<StatusBadge tone={completeness.configured === completeness.total ? "success" : "neutral"}>
{completeness.configured}/{completeness.total}
</StatusBadge>
{secretFields.length > 0 && (
<StatusBadge tone={configuredSecretCount === secretFields.length ? "success" : "warning"}>
{configuredSecretCount}/{secretFields.length}
</StatusBadge>
)}
{checkboxSummaries.slice(0, 2).map((label) => (
<StatusBadge key={label} tone="info">{label}</StatusBadge>
))}
</div>
<Dialog open={open} onOpenChange={(nextOpen) => !saving && setOpen(nextOpen)}>
<DialogTrigger render={<Button variant="outline" size="sm" className="w-full lg:w-auto" />}>
<Pencil className="size-3.5" />
</DialogTrigger>
<DialogContent className="sm:max-w-3xl">
<DialogHeader>
<div className="flex size-9 items-center justify-center rounded-lg border border-primary/15 bg-primary/10 text-primary">
<ShieldCheck className="size-4" />
</div>
<DialogTitle>{providerName}</DialogTitle>
<DialogDescription>{providerDescription}</DialogDescription>
</DialogHeader>
<form onSubmit={handleSubmit} className="space-y-5">
<div className="grid gap-4 sm:grid-cols-2">
{fields.map((field) =>
field.type === "checkboxes" ? (
<div key={field.key} className="sm:col-span-2">
<Label>{field.label}</Label>
<div className="mt-3 grid gap-2 sm:grid-cols-2">
{field.options?.map((option) => {
const selected = checkboxValues[field.key]?.has(option.value) ?? false;
return (
<button
key={option.value}
type="button"
aria-pressed={selected}
onClick={() => toggleCheckbox(field.key, option.value)}
className={cn(
"flex min-h-10 items-center justify-between gap-3 rounded-lg border px-3 py-2 text-left text-sm font-medium transition-colors focus-visible:outline-none focus-visible:ring-[3px] focus-visible:ring-ring/20",
selected
? "border-primary/35 bg-primary/10 text-primary"
: "border-border bg-muted/20 text-muted-foreground hover:bg-muted/45 hover:text-foreground",
)}
>
<span className="truncate">{option.label}</span>
{selected && <Check className="size-4 shrink-0" />}
</button>
);
})}
</div>
</div>
) : (
<div key={field.key} className="space-y-2">
<Label htmlFor={`${provider}-${field.key}`}>{field.label}</Label>
<Input
id={`${provider}-${field.key}`}
name={field.key}
type={field.secret ? "password" : "text"}
placeholder={field.secret && secretConfigured[field.key] ? "留空保持不变" : field.placeholder}
defaultValue={field.secret ? "" : currentConfig?.[field.key] || ""}
/> />
<span className="text-sm">{opt.label}</span> {field.secret && secretConfigured[field.key] && (
</label> <p className="text-xs leading-5 text-muted-foreground"></p>
))} )}
</div>
),
)}
</div>
<div className="rounded-lg border border-border bg-muted/20 p-3">
<div className="flex flex-col gap-3 sm:flex-row sm:items-center sm:justify-between">
<div>
<Label className="text-sm font-semibold"></Label>
<p className="mt-1 text-xs leading-5 text-muted-foreground"></p>
</div>
<div className="w-full sm:w-56">
<BooleanToggle
value={enabled}
onChange={setEnabled}
trueLabel="启用"
falseLabel="停用"
ariaLabel="支付通道状态"
disabled={saving}
/>
</div>
</div> </div>
</div> </div>
) : (
<div key={field.key}> <DialogFooter>
<Label>{field.label}</Label> <Button type="button" variant="outline" onClick={() => setOpen(false)} disabled={saving}>
<Input
name={field.key} </Button>
type={field.secret ? "password" : "text"} <Button type="submit" disabled={saving}>
placeholder={field.secret && secretConfigured[field.key] ? "留空保持不变" : field.placeholder} {saving ? "保存中..." : "保存配置"}
defaultValue={field.secret ? "" : currentConfig?.[field.key] || ""} </Button>
/> </DialogFooter>
</div> </form>
), </DialogContent>
)} </Dialog>
</div> </section>
<div className="flex flex-col gap-3 border-t border-border/50 pt-4 sm:flex-row sm:items-center sm:justify-between">
<div className="w-full sm:w-56">
<Label className="mb-2 block text-xs text-muted-foreground"></Label>
<BooleanToggle
value={enabled}
onChange={setEnabled}
trueLabel="启用"
falseLabel="停用"
ariaLabel="支付通道状态"
/>
</div>
<Button type="submit" size="sm" disabled={saving}>
{saving ? "保存中..." : "保存配置"}
</Button>
</div>
</form>
); );
} }

View File

@@ -1,8 +1,6 @@
import type { Metadata } from "next"; import type { Metadata } from "next";
import { CreditCard } from "lucide-react";
import { PageHeader, PageShell } from "@/components/shared/page-shell"; import { PageHeader, PageShell } from "@/components/shared/page-shell";
import { ActiveStatusBadge } from "@/components/shared/status-badge"; import { PaymentConfigItem } from "./config-form";
import { PaymentConfigForm } from "./config-form";
import { getPaymentProviderConfigs } from "./payments-data"; import { getPaymentProviderConfigs } from "./payments-data";
export const metadata: Metadata = { export const metadata: Metadata = {
@@ -19,29 +17,18 @@ export default async function PaymentsPage() {
eyebrow="系统" eyebrow="系统"
title="支付配置" title="支付配置"
/> />
<div className="grid gap-5"> <div className="overflow-hidden rounded-lg border border-border bg-card">
{providerConfigs.map(({ provider, config, secretConfigured }) => ( {providerConfigs.map(({ provider, config, secretConfigured }) => (
<section key={provider.id} className="surface-card overflow-hidden rounded-xl p-4"> <PaymentConfigItem
<div className="mb-4 flex flex-col gap-4 lg:flex-row lg:items-start lg:justify-between"> key={provider.id}
<div className="flex items-start gap-3"> providerName={provider.name}
<span className="flex size-9 shrink-0 items-center justify-center rounded-lg bg-primary/10 text-primary"> providerDescription={provider.description}
<CreditCard className="size-4" /> provider={provider.id}
</span> fields={provider.fields}
<div> currentConfig={config?.config}
<h3 className="text-lg font-semibold tracking-tight">{provider.name}</h3> secretConfigured={secretConfigured}
<p className="mt-1 text-sm leading-6 text-muted-foreground text-pretty">{provider.description}</p> enabled={config?.enabled ?? false}
</div> />
</div>
<ActiveStatusBadge active={config?.enabled ?? false} activeLabel="已启用" inactiveLabel="未启用" />
</div>
<PaymentConfigForm
provider={provider.id}
fields={provider.fields}
currentConfig={config?.config}
secretConfigured={secretConfigured}
enabled={config?.enabled ?? false}
/>
</section>
))} ))}
</div> </div>
</PageShell> </PageShell>