Files
MemoryBear/web/src/views/UserMemoryDetail/components/EndUserProfile.tsx
2026-04-17 19:51:01 +08:00

119 lines
3.8 KiB
TypeScript

/*
* @Author: ZhaoYing
* @Date: 2026-02-03 18:33:30
* @Last Modified by: ZhaoYing
* @Last Modified time: 2026-04-17 17:57:15
*/
/**
* End User Profile Component
* Displays and manages end user profile information
*/
import { forwardRef, useImperativeHandle, useEffect, useState, useRef } from 'react'
import { useTranslation } from 'react-i18next'
import { useParams } from 'react-router-dom'
import { Skeleton, Flex } from 'antd';
import dayjs from 'dayjs'
import clsx from 'clsx'
import RbCard from '@/components/RbCard/Card'
import {
getEndUserInfo,
} from '@/api/memory'
import EndUserProfileModal from './EndUserProfileModal'
import type { EndUser, EndUserProfileModalRef, EndUserProfileRef } from '../types'
/**
* Component props
*/
interface EndUserProfileProps {
onDataLoaded?: (data?: EndUser) => void;
className?: string;
}
const EndUserProfile = forwardRef<EndUserProfileRef, EndUserProfileProps>(({ className, onDataLoaded }, ref) => {
const { t } = useTranslation()
const { id } = useParams()
const endUserProfileModalRef = useRef<EndUserProfileModalRef>(null)
const [loading, setLoading] = useState<boolean>(false)
const [data, setData] = useState<EndUser | null>(null)
useEffect(() => {
if (!id) return
getData()
}, [id])
/** Fetch profile data */
const getData = () => {
if (!id) return
setLoading(true)
getEndUserInfo(id).then((res) => {
const userData = res as EndUser
setData(userData)
setLoading(false)
onDataLoaded?.(userData as EndUser)
})
.finally(() => {
setLoading(false)
})
}
/** Open edit modal */
const handleEdit = () => {
if (!data) return
endUserProfileModalRef.current?.handleOpen(data)
}
useImperativeHandle(ref, () => ({
data
}));
return (
<RbCard
title={t('userMemory.endUserProfile')}
extra={
<div
className="rb:w-5 rb:h-5 rb:cursor-pointer rb:bg-cover rb:bg-[url('@/assets/images/edit.svg')] rb:hover:bg-[url('@/assets/images/edit_hover.svg')]"
onClick={handleEdit}
></div>
}
headerClassName="rb:min-h-[46px]!! rb:font-medium!"
className={clsx("rb:bg-[#FFFFFF]! rb:shadow-[0px_2px_6px_0px_rgba(33,35,50,0.13)]! rb:absolute! rb:w-80 rb:top-29 rb:left-26", className)}
bodyClassName="rb:px-5! rb:pb-5! rb:pt-3.75! rb:max-h-[calc(100vh-186px)] rb:overflow-auto"
>
{loading
? <Skeleton />
: <Flex vertical gap={20} className="rb:leading-5">
<div>
<div className="rb:text-[#7B8085]">{t('userMemory.other_name')}</div>
<div className="rb:mt-0.5">{data?.other_name || '-'}</div>
</div>
<div>
<div className="rb:text-[#7B8085]">{t('userMemory.role')}</div>
<div className="rb:mt-0.5">{data?.profile?.role?.join(' | ') || '-'}</div>
</div>
<div>
<div className="rb:text-[#7B8085]">{t('userMemory.domain')}</div>
<div className="rb:mt-0.5">{data?.profile?.domain?.join(' | ') || '-'}</div>
</div>
<div>
<div className="rb:text-[#7B8085]">{t('userMemory.expertise')}</div>
<div className="rb:mt-0.5">{data?.profile?.expertise?.join(' | ') || '-'}</div>
</div>
<div>
<div className="rb:text-[#7B8085]">{t('userMemory.interests')}</div>
<div className="rb:mt-0.5">{data?.profile?.interests?.join(' | ') || '-'}</div>
</div>
<div className="rb:text-[#7B8085] rb:text-[12px] rb:leading-4.5">
{t('userMemory.updated_at')}: {data?.updated_at ? dayjs(data?.updated_at).format('YYYY/MM/DD HH:mm:ss') : ''}
</div>
</Flex>
}
<EndUserProfileModal
ref={endUserProfileModalRef}
refresh={getData}
/>
</RbCard>
)
})
export default EndUserProfile