交易明细页面添加操作按钮
This commit is contained in:
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "lanhu-admin",
|
"name": "lanhu-admin",
|
||||||
"version": "1.9.0",
|
"version": "1.10.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "next dev -H 0.0.0.0 -p 3001 --turbopack",
|
"dev": "next dev -H 0.0.0.0 -p 3001 --turbopack",
|
||||||
|
|||||||
@@ -30,3 +30,22 @@ export async function getTrade(params: {
|
|||||||
}) {
|
}) {
|
||||||
return callByUser<PageRecord<Trade>>("/api/admin/trade/page/of-user", 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)
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,15 +1,29 @@
|
|||||||
"use client"
|
"use client"
|
||||||
import { zodResolver } from "@hookform/resolvers/zod"
|
import { zodResolver } from "@hookform/resolvers/zod"
|
||||||
import { format } from "date-fns"
|
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 Link from "next/link"
|
||||||
import { Suspense, useCallback, useState } from "react"
|
import { Suspense, useCallback, useState } from "react"
|
||||||
import { Controller, useForm } from "react-hook-form"
|
import { Controller, useForm } from "react-hook-form"
|
||||||
|
import { toast } from "sonner"
|
||||||
import { z } from "zod"
|
import { z } from "zod"
|
||||||
import { getPageTrade } from "@/actions/trade"
|
import {
|
||||||
|
getPageTrade,
|
||||||
|
getTradeCheckk,
|
||||||
|
updateTradeRemark,
|
||||||
|
} from "@/actions/trade"
|
||||||
import { DataTable, useDataTable } from "@/components/data-table"
|
import { DataTable, useDataTable } from "@/components/data-table"
|
||||||
import { Page } from "@/components/page"
|
import { Page } from "@/components/page"
|
||||||
import { Button } from "@/components/ui/button"
|
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 { Field, FieldError, FieldLabel } from "@/components/ui/field"
|
||||||
import { Input } from "@/components/ui/input"
|
import { Input } from "@/components/ui/input"
|
||||||
import {
|
import {
|
||||||
@@ -105,6 +119,7 @@ export default function TradePage() {
|
|||||||
setFilters(result)
|
setFilters(result)
|
||||||
table.pagination.onPageChange(1)
|
table.pagination.onPageChange(1)
|
||||||
})
|
})
|
||||||
|
console.log(...table.data.map(i => i.remark), "tabletabletabletable")
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Page>
|
<Page>
|
||||||
@@ -365,9 +380,173 @@ export default function TradePage() {
|
|||||||
},
|
},
|
||||||
},
|
},
|
||||||
{ header: "渠道订单号", accessorKey: "outer_no" },
|
{ 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>
|
</Suspense>
|
||||||
</Page>
|
</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>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|||||||
@@ -15,4 +15,5 @@ export type Trade = {
|
|||||||
canceled_at: Date
|
canceled_at: Date
|
||||||
updated_at: Date
|
updated_at: Date
|
||||||
user?: User
|
user?: User
|
||||||
|
remark: string
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user