交易明细页面添加操作按钮

This commit is contained in:
Eamon
2026-06-18 18:16:42 +08:00
parent 01f4f5343d
commit e70cfbd9a8
4 changed files with 202 additions and 3 deletions

View File

@@ -1,6 +1,6 @@
{
"name": "lanhu-admin",
"version": "1.9.0",
"version": "1.10.0",
"private": true,
"scripts": {
"dev": "next dev -H 0.0.0.0 -p 3001 --turbopack",

View File

@@ -30,3 +30,22 @@ export async function getTrade(params: {
}) {
return callByUser<PageRecord<Trade>>("/api/admin/trade/page/of-user", params)
}
export async function updateTradeRemark(params: {
trade_no: string
remark: string
}) {
return callByUser<PageRecord<Trade>>("/api/admin/trade/update/remark", params)
}
type PayCloseData = {
status: 0 | 1 | 2
TransId: string
}
export async function getTradeCheckk(params: {
trade_no: string
method: number
}) {
return callByUser<PayCloseData>("/api/admin/trade/check", params)
}

View File

@@ -1,15 +1,29 @@
"use client"
import { zodResolver } from "@hookform/resolvers/zod"
import { format } from "date-fns"
import { CheckCircle, Clock, XCircle } from "lucide-react"
import { CheckCircle, Clock, Loader2, XCircle } from "lucide-react"
import Link from "next/link"
import { Suspense, useCallback, useState } from "react"
import { Controller, useForm } from "react-hook-form"
import { toast } from "sonner"
import { z } from "zod"
import { getPageTrade } from "@/actions/trade"
import {
getPageTrade,
getTradeCheckk,
updateTradeRemark,
} from "@/actions/trade"
import { DataTable, useDataTable } from "@/components/data-table"
import { Page } from "@/components/page"
import { Button } from "@/components/ui/button"
import {
Dialog,
DialogClose,
DialogContent,
DialogFooter,
DialogHeader,
DialogTitle,
DialogTrigger,
} from "@/components/ui/dialog"
import { Field, FieldError, FieldLabel } from "@/components/ui/field"
import { Input } from "@/components/ui/input"
import {
@@ -105,6 +119,7 @@ export default function TradePage() {
setFilters(result)
table.pagination.onPageChange(1)
})
console.log(...table.data.map(i => i.remark), "tabletabletabletable")
return (
<Page>
@@ -365,9 +380,173 @@ export default function TradePage() {
},
},
{ header: "渠道订单号", accessorKey: "outer_no" },
{
header: "备注信息",
accessorKey: "remark",
},
{
header: "操作",
id: "actions",
meta: { pin: "right" },
cell: ({ row }) => (
<div className="flex gap-2">
<CheckOrder trade={row.original} />
<UpdateRole trade={row.original} onSuccess={table.refresh} />
</div>
),
},
]}
/>
</Suspense>
</Page>
)
}
type PayCloseData = {
status: 0 | 1 | 2
TransId?: string
}
function CheckOrder(props: { trade: Trade }) {
const [open, setOpen] = useState(false)
const [loading, setLoading] = useState(false)
const [data, setData] = useState<PayCloseData | null>(null)
const handleCheck = useCallback(async () => {
setLoading(true)
try {
const res = await getTradeCheckk({
trade_no: props.trade.inner_no,
method: Number(props.trade.method),
})
console.log(res, "res")
if (res.success) {
setData(res.data)
} else {
toast.error(res.message)
}
setOpen(true)
} catch (error) {
const message = error instanceof Error ? error.message : error
toast.error(`检查失败,请重试: ${message}`)
} finally {
setLoading(false)
}
}, [props.trade.inner_no, props.trade.method])
return (
<>
<Button size="sm" onClick={handleCheck} disabled={loading}>
{loading ? <Loader2 className="h-4 w-4 animate-spin" /> : "检查订单"}
</Button>
<Dialog open={open} onOpenChange={setOpen}>
<DialogContent>
<DialogHeader>
<DialogTitle></DialogTitle>
</DialogHeader>
{data && (
<div className="py-4">
<div
className={`p-4 rounded-lg ${data.status === 1 ? "bg-green-50" : "bg-red-50"}`}
>
<p className="font-medium">:{data?.TransId}</p>
<p className="mt-2">
{data?.status === 1
? "已支付"
: data?.status === 0
? "待支付"
: "已取消"}
</p>
</div>
</div>
)}
<DialogFooter>
<DialogClose asChild>
<Button variant="outline"></Button>
</DialogClose>
</DialogFooter>
</DialogContent>
</Dialog>
</>
)
}
function UpdateRole(props: { trade: Trade; onSuccess?: () => void }) {
const [open, setOpen] = useState(false)
const [loading, setLoading] = useState(false)
const { register, handleSubmit, reset } = useForm<{
remark: string
}>({
defaultValues: { remark: "" },
})
const handleOpenChange = (isOpen: boolean) => {
setOpen(isOpen)
if (isOpen) {
reset({ remark: props.trade.remark ?? "" })
}
}
const submitEditRemark = async (data: { remark: string }) => {
try {
setLoading(true)
const response = await updateTradeRemark({
trade_no: props.trade.inner_no || "",
remark: data.remark,
})
if (!response.success) {
throw new Error(response.message)
}
toast.success("备注修改成功")
setOpen(false)
props.onSuccess?.()
} catch (error) {
toast.error(
`修改备注失败:${error instanceof Error ? error.message : String(error)}`,
)
} finally {
setLoading(false)
}
}
return (
<Dialog open={open} onOpenChange={handleOpenChange}>
<DialogTrigger asChild>
<Button size="sm"></Button>
</DialogTrigger>
<DialogContent>
<form onSubmit={handleSubmit(submitEditRemark)} id="edit-remark-form">
<DialogHeader>
<DialogTitle></DialogTitle>
</DialogHeader>
<div className="py-4">
<Field className="flex flex-col gap-1">
<FieldLabel></FieldLabel>
<Input
{...register("remark")}
placeholder="请输入备注信息"
clearable
/>
</Field>
</div>
<DialogFooter>
<DialogClose asChild>
<Button variant="ghost" disabled={loading}>
</Button>
</DialogClose>
<Button type="submit" form="edit-remark-form" disabled={loading}>
{loading ? "保存中..." : "保存"}
</Button>
</DialogFooter>
</form>
</DialogContent>
</Dialog>
)
}

View File

@@ -15,4 +15,5 @@ export type Trade = {
canceled_at: Date
updated_at: Date
user?: User
remark: string
}