NapCatQQ/napcat.webui/src/components/onebot/display_card/render.tsx
2025-01-24 21:13:44 +08:00

152 lines
4.3 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

import { Button } from '@heroui/button'
import { Card, CardBody, CardHeader } from '@heroui/card'
import { Popover, PopoverContent, PopoverTrigger } from '@heroui/popover'
import { Snippet } from '@heroui/snippet'
import { motion } from 'motion/react'
import { IoCode } from 'react-icons/io5'
import OneBotDisplayMeta from '@/components/onebot/display_card/meta'
import { getEventName, isOB11Event } from '@/utils/onebot'
import { timestampToDateString } from '@/utils/time'
import type {
AllOB11WsResponse,
OB11AllEvent,
OB11Request
} from '@/types/onebot'
import OneBotMessage from './message'
import OneBotNotice from './notice'
import OneBotDisplayResponse from './response'
const itemVariants = {
hidden: { opacity: 0, scale: 0.8, y: 50 },
visible: {
opacity: 1,
scale: 1,
y: 0,
transition: { type: 'spring', stiffness: 300, damping: 20 }
}
}
function RequestComponent({ data: _ }: { data: OB11Request }) {
return <div>Request消息</div>
}
export interface OneBotItemRenderProps {
data: AllOB11WsResponse[]
index: number
style: React.CSSProperties
}
export const getItemSize = (event: OB11AllEvent['post_type']) => {
if (event === 'meta_event') {
return 100
}
if (event === 'message') {
return 180
}
if (event === 'request') {
return 100
}
if (event === 'notice') {
return 100
}
if (event === 'message_sent') {
return 250
}
return 100
}
const renderDetail = (data: AllOB11WsResponse) => {
if (isOB11Event(data)) {
switch (data.post_type) {
case 'meta_event':
return <OneBotDisplayMeta data={data} />
case 'message':
return <OneBotMessage data={data} />
case 'request':
return <RequestComponent data={data} />
case 'notice':
return <OneBotNotice data={data} />
case 'message_sent':
return <OneBotMessage data={data} />
default:
return <div></div>
}
}
return <OneBotDisplayResponse data={data} />
}
const OneBotItemRender = ({ data, index, style }: OneBotItemRenderProps) => {
const msg = data[index]
const isEvent = isOB11Event(msg)
return (
<div style={style} className="p-1 overflow-visible w-full h-full">
<motion.div
variants={itemVariants}
initial="hidden"
animate="visible"
className="h-full px-2"
>
<Card className="w-full h-full py-2 bg-opacity-50 backdrop-blur-sm">
<CardHeader className="py-0 text-default-500 flex-row gap-2">
<div className="font-bold">
{isEvent ? getEventName(msg.post_type) : '请求响应'}
</div>
<div className="text-sm">
{isEvent && timestampToDateString(msg.time)}
</div>
<div className="ml-auto">
<Popover
placement="left"
showArrow
classNames={{
content: 'max-h-96 max-w-96 overflow-hidden p-0'
}}
>
<PopoverTrigger>
<Button
size="sm"
color="danger"
variant="flat"
radius="full"
isIconOnly
className="text-medium"
>
<IoCode />
</Button>
</PopoverTrigger>
<PopoverContent>
<Snippet
hideSymbol
tooltipProps={{
content: '点击复制'
}}
classNames={{
copyButton: 'self-start sticky top-0 right-0'
}}
className="bg-content1 h-full overflow-y-scroll items-start"
>
{JSON.stringify(msg, null, 2)
.split('\n')
.map((line, i) => (
<span key={i} className="whitespace-pre-wrap break-all">
{line}
</span>
))}
</Snippet>
</PopoverContent>
</Popover>
</div>
</CardHeader>
<CardBody className="py-0">{renderDetail(msg)}</CardBody>
</Card>
</motion.div>
</div>
)
}
export default OneBotItemRender