FUJI/src/pages/repair/RepairAdmin.tsx
2025-05-24 13:54:06 +08:00

446 lines
13 KiB
TypeScript

import {
Table,
TableHeader,
TableColumn,
TableBody,
TableRow,
TableCell,
User,
Pagination,
Spinner,
Popover,
PopoverTrigger,
PopoverContent,
Button,
CheckboxGroup,
Checkbox,
Drawer,
DrawerContent,
DrawerHeader,
DrawerBody,
DrawerFooter,
useDisclosure,
Chip,
} from "@heroui/react"
import { useCallback, useEffect, useMemo, useRef, useState } from "react"
import { useAsyncList } from "@react-stately/data"
import type { components } from "../../types/saturday"
import { saturdayApiBaseUrl, saturdayClient } from "../../utils/client"
import EventDetail, { EventStatusChip, type EventDetailRef } from "./EventDetail"
import dayjs from "dayjs"
import { UserEventStatus } from "../../types/event"
import { makeLogtoClient } from "../../utils/auth"
import type { PublicMember } from "../../store/member"
import type { UserInfoResponse } from "@logto/browser"
import { getAvailableEventActions, type EventAction, type IdentityContext } from "./EventAction"
type PublicEvent = components["schemas"]["PublicEvent"]
export const EyeIcon = (props) => {
return (
<svg
aria-hidden="true"
fill="none"
focusable="false"
height="1em"
role="presentation"
viewBox="0 0 20 20"
width="1em"
{...props}
>
<path
d="M12.9833 10C12.9833 11.65 11.65 12.9833 10 12.9833C8.35 12.9833 7.01666 11.65 7.01666 10C7.01666 8.35 8.35 7.01666 10 7.01666C11.65 7.01666 12.9833 8.35 12.9833 10Z"
stroke="currentColor"
strokeLinecap="round"
strokeLinejoin="round"
strokeWidth={1.5}
/>
<path
d="M9.99999 16.8916C12.9417 16.8916 15.6833 15.1583 17.5917 12.1583C18.3417 10.9833 18.3417 9.00831 17.5917 7.83331C15.6833 4.83331 12.9417 3.09998 9.99999 3.09998C7.05833 3.09998 4.31666 4.83331 2.40833 7.83331C1.65833 9.00831 1.65833 10.9833 2.40833 12.1583C4.31666 15.1583 7.05833 16.8916 9.99999 16.8916Z"
stroke="currentColor"
strokeLinecap="round"
strokeLinejoin="round"
strokeWidth={1.5}
/>
</svg>
)
}
function CheckboxPopover(props: {
value: string[]
onValueChange: (value: string[]) => void
}) {
return (
<Popover placement="bottom">
<PopoverTrigger>
<Button size="sm" isIconOnly variant="bordered">
<svg xmlns="http://www.w3.org/2000/svg" fill="none" viewBox="0 0 24 24" strokeWidth={1.5} stroke="currentColor" className="size-6">
<path strokeLinecap="round" strokeLinejoin="round" d="M12 6.75a.75.75 0 1 1 0-1.5.75.75 0 0 1 0 1.5ZM12 12.75a.75.75 0 1 1 0-1.5.75.75 0 0 1 0 1.5ZM12 18.75a.75.75 0 1 1 0-1.5.75.75 0 0 1 0 1.5Z" />
</svg>
</Button>
</PopoverTrigger>
<PopoverContent>
<div className="p-2">
<CheckboxGroup
value={props.value}
onValueChange={props.onValueChange}
orientation="vertical"
>
{
UserEventStatus.map((status) => {
return (
<Checkbox key={status.status} value={status.status}>
<EventStatusChip size="sm" status={status.status} />
</Checkbox>
)
})
}
</CheckboxGroup>
</div>
</PopoverContent>
</Popover>
)
}
function TicketDetailDrawer(props: {
event: PublicEvent
identity: IdentityContext
isOpen: boolean
onEventUpdated: (event: PublicEvent) => void
onOpenChange: (isOpen: boolean) => void
onClose: () => void
onDelete: () => void
onEdit: () => void
}) {
const { isOpen, onOpenChange, onClose } = props
const [isLoading, setIsLoading] = useState("")
const eventDetailRef = useRef<EventDetailRef>(null)
const [availableActions, setAvailableActions] = useState<EventAction[]>([])
useEffect(() => {
if (!props.event || !props.identity?.member || !props.identity?.userInfo?.roles) {
return
}
setAvailableActions(getAvailableEventActions(props.event, props.identity))
}, [props.event, props.identity])
const onEventUpdated = async (event: PublicEvent) => {
props.onEventUpdated(event)
const res = await eventDetailRef.current?.refresh()
console.log("onEventUpdated", res)
if (!res || !props.identity?.member || !props.identity?.userInfo?.roles) {
return
}
setAvailableActions(getAvailableEventActions(res, props.identity))
}
return (
<Drawer isOpen={isOpen} onOpenChange={onOpenChange}>
<DrawerContent>
<DrawerHeader>
<h2 className="text-2xl font-bold"></h2>
{isLoading}
</DrawerHeader>
<DrawerBody>
<EventDetail ref={eventDetailRef} eventId={props.event?.eventId}>
{
event => (
<div className="mb-12 flex flex-col gap-2">
{
availableActions?.map((action) => {
return (
<action.jsxHandler
key={action.action}
event={event}
isLoading={isLoading}
identityContext={props.identity}
onUpdated={onEventUpdated}
onLoading={(action) => {
setIsLoading(action)
}}
>
</action.jsxHandler>
)
}) || <></>
}
</div>
)
}
</EventDetail>
</DrawerBody>
<DrawerFooter>
<Button variant="flat" onPress={onClose}>
</Button>
</DrawerFooter>
</DrawerContent>
</Drawer>
)
}
export const validateRepairRole = (roles: string[]) => {
const acceptableRoles = ["repair admin", "repair member"]
return roles.some(role => acceptableRoles.includes(role.toLowerCase()))
}
export default function App() {
const [isLoading, setIsLoading] = useState(true)
const [page, setPage] = useState(1)
const rowsPerPage = 10
const [statusFilter, setStatusFilter] = useState<string[]>([])
const { isOpen, onOpen, onOpenChange } = useDisclosure()
const [userInfo, setUserInfo] = useState<UserInfoResponse>()
const [currentMember, setCurrentMember] = useState<PublicMember>()
const [token, setToken] = useState<string>()
useEffect(() => {
const check = async () => {
const adminPath = "/repair/admin"
const authenticated = await makeLogtoClient().isAuthenticated()
if (!authenticated) {
window.location.href = `/repair/login-hint?redirectUrl=${adminPath}`
return
}
const res = await makeLogtoClient().getIdTokenClaims()
const token = await makeLogtoClient().getAccessToken()
setToken(token)
const hasRole = validateRepairRole(res.roles)
if (!hasRole) {
window.location.href = `/repair/login-hint?redirectUrl=${adminPath}`
return
}
setUserInfo(res)
const currentMember = await fetch(`${saturdayApiBaseUrl}/member`, {
method: "GET",
headers: {
Authorization: `Bearer ${token}`,
},
}).then(res => res.json())
setCurrentMember(currentMember)
}
check()
}, [])
const list = useAsyncList<PublicEvent>({
async load() {
const { data } = await saturdayClient.GET("/events", {
params: {
query: {
order: "DESC",
offset: 1,
limit: 1000,
},
},
})
setIsLoading(false)
return {
items: data,
}
},
async sort({ items, sortDescriptor }) {
return {
items: items.sort((a, b) => {
const first = a[sortDescriptor.column]
const second = b[sortDescriptor.column]
let cmp = (parseInt(first) || first) < (parseInt(second) || second) ? -1 : 1
if (sortDescriptor.direction === "descending") {
cmp *= -1
}
return cmp
}),
}
},
})
const filteredList = useMemo(() => {
if (statusFilter.length > 0) {
return list.items.filter(item => statusFilter.includes(item.status))
}
return list.items
}, [list, statusFilter])
const items = useMemo(() => {
const start = (page - 1) * rowsPerPage
const end = start + rowsPerPage
return filteredList.slice(start, end)
}, [filteredList, page, rowsPerPage])
const pages = useMemo(() => {
return Math.ceil(filteredList.length / rowsPerPage)
}, [filteredList, rowsPerPage])
useEffect(() => {
setPage(1)
}, [statusFilter])
const columns: {
key: string
label: string
allowSorting?: boolean
content?: JSX.Element
}[] = [
{
key: "eventId",
label: "单号",
},
{
key: "problem",
label: "问题描述",
},
{
key: "model",
label: "型号",
},
{
key: "size",
label: "工作量",
},
{
key: "memberId",
label: "处理人",
},
{
key: "gmtCreate",
label: "创建时间",
},
{
key: "status",
label: "状态",
content: (
<div className="flex items-center gap-2">
<CheckboxPopover value={statusFilter} onValueChange={setStatusFilter}></CheckboxPopover>
</div>
),
},
{
key: "actions",
label: "操作",
},
]
const [activeEvent, setActiveEvent] = useState<PublicEvent>()
const onOpenEventDetail = (event: PublicEvent) => {
setActiveEvent(event)
onOpen()
}
const renderCell = useCallback((event: PublicEvent, columnKey: string | number) => {
const cellValue = event[columnKey]
switch (columnKey) {
case "problem":
return (
<div className="max-w-40 line-clamp-2 overflow-hidden text-ellipsis">
{cellValue}
</div>
)
case "memberId":
return (
event.member
? (
<User
avatarProps={{ radius: "full", src: event.member.avatar, size: "sm" }}
name={event.member.alias}
description={event.member.memberId}
>
{event.member.alias}
</User>
)
: <></>
)
case "size":
return (
cellValue ? <Chip size="sm">{"size:" + cellValue}</Chip> : <></>
)
case "gmtCreate":
return (
<span>
{dayjs(cellValue).format("YYYY-MM-DD HH:mm")}
</span>
)
case "status":
return EventStatusChip({
status: cellValue,
size: "sm",
})
case "actions":
return (
<Button onPress={() => onOpenEventDetail(event)} size="sm" isIconOnly variant="light">
<span className="text-lg text-default-400 cursor-pointer active:opacity-50">
<EyeIcon />
</span>
</Button>
)
default:
return cellValue
}
}, [])
return (
<section className="box-border mb-24">
<div className="section-content mt-6">
<h2 className="text-2xl font-bold"></h2>
</div>
<div className="section-content my-8 flex flex-col gap-4">
<Table
aria-label="Example table with dynamic content"
sortDescriptor={list.sortDescriptor}
onSortChange={list.sort}
bottomContent={(
<div className="flex w-full justify-center">
<Pagination
isCompact
showControls
showShadow
color="secondary"
page={page}
total={pages}
onChange={page => setPage(page)}
/>
</div>
)}
>
<TableHeader columns={columns}>
{column => <TableColumn key={column.key} allowsSorting={column.allowSorting} children={column.content ?? <div>{column.label}</div>}></TableColumn>}
</TableHeader>
<TableBody isLoading={isLoading} items={items} loadingContent={<Spinner label="Loading..." />}>
{item => (
<TableRow key={item.eventId}>
{columnKey => <TableCell>{renderCell(item, columnKey)}</TableCell>}
</TableRow>
)}
</TableBody>
</Table>
</div>
<TicketDetailDrawer
event={activeEvent}
onEventUpdated={list.reload}
identity={
{
member: currentMember,
userInfo: userInfo,
token: token,
}
}
isOpen={isOpen}
onOpenChange={onOpenChange}
onClose={() => {
onOpenChange()
}}
onDelete={() => {}}
onEdit={() => {}}
>
</TicketDetailDrawer>
</section>
)
}