component.tsx 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990
  1. import type { FC } from 'react'
  2. import { useState } from 'react'
  3. import { useTranslation } from 'react-i18next'
  4. import { useSelectOrDelete, useTrigger } from '../../hooks'
  5. import { UPDATE_HISTORY_EVENT_EMITTER } from '../../constants'
  6. import type { RoleName } from './index'
  7. import { DELETE_HISTORY_BLOCK_COMMAND } from './index'
  8. import { DotsHorizontal } from '@/app/components/base/icons/src/vender/line/general'
  9. import { MessageClockCircle } from '@/app/components/base/icons/src/vender/solid/general'
  10. import {
  11. PortalToFollowElem,
  12. PortalToFollowElemContent,
  13. PortalToFollowElemTrigger,
  14. } from '@/app/components/base/portal-to-follow-elem'
  15. import { useEventEmitterContextContext } from '@/context/event-emitter'
  16. type HistoryBlockComponentProps = {
  17. nodeKey: string
  18. roleName?: RoleName
  19. onEditRole: () => void
  20. }
  21. const HistoryBlockComponent: FC<HistoryBlockComponentProps> = ({
  22. nodeKey,
  23. roleName = { user: '', assistant: '' },
  24. onEditRole,
  25. }) => {
  26. const { t } = useTranslation()
  27. const [ref, isSelected] = useSelectOrDelete(nodeKey, DELETE_HISTORY_BLOCK_COMMAND)
  28. const [triggerRef, open, setOpen] = useTrigger()
  29. const { eventEmitter } = useEventEmitterContextContext()
  30. const [localRoleName, setLocalRoleName] = useState<RoleName>(roleName)
  31. eventEmitter?.useSubscription((v: any) => {
  32. if (v?.type === UPDATE_HISTORY_EVENT_EMITTER)
  33. setLocalRoleName(v.payload)
  34. })
  35. return (
  36. <div className={`
  37. group inline-flex items-center pl-1 pr-0.5 h-6 border border-transparent text-[#DD2590] rounded-[5px] hover:bg-[#FCE7F6]
  38. ${open ? 'bg-[#FCE7F6]' : 'bg-[#FDF2FA]'}
  39. ${isSelected && '!border-[#F670C7]'}
  40. `} ref={ref}>
  41. <MessageClockCircle className='mr-1 w-[14px] h-[14px]' />
  42. <div className='mr-1 text-xs font-medium'>{t('common.promptEditor.history.item.title')}</div>
  43. <PortalToFollowElem
  44. open={open}
  45. onOpenChange={setOpen}
  46. placement='top-end'
  47. offset={{
  48. mainAxis: 4,
  49. alignmentAxis: -148,
  50. }}
  51. >
  52. <PortalToFollowElemTrigger ref={triggerRef}>
  53. <div className={`
  54. flex items-center justify-center w-[18px] h-[18px] rounded cursor-pointer
  55. ${open ? 'bg-[#DD2590] text-white' : 'bg-white/50 group-hover:bg-white group-hover:shadow-xs'}
  56. `}>
  57. <DotsHorizontal className='w-3 h-3' />
  58. </div>
  59. </PortalToFollowElemTrigger>
  60. <PortalToFollowElemContent style={{ zIndex: 100 }}>
  61. <div className='w-[360px] bg-white rounded-xl shadow-lg'>
  62. <div className='p-4'>
  63. <div className='mb-2 text-xs font-medium text-gray-500'>{t('common.promptEditor.history.modal.title')}</div>
  64. <div className='flex items-center text-sm text-gray-700'>
  65. <div className='mr-1 w-20 text-xs font-semibold'>{localRoleName?.user}</div>
  66. {t('common.promptEditor.history.modal.user')}
  67. </div>
  68. <div className='flex items-center text-sm text-gray-700'>
  69. <div className='mr-1 w-20 text-xs font-semibold'>{localRoleName?.assistant}</div>
  70. {t('common.promptEditor.history.modal.assistant')}
  71. </div>
  72. </div>
  73. <div
  74. className='px-4 py-3 text-xs text-[#155EEF] border-t border-black/5 rounded-b-xl cursor-pointer'
  75. onClick={onEditRole}
  76. >
  77. {t('common.promptEditor.history.modal.edit')}
  78. </div>
  79. </div>
  80. </PortalToFollowElemContent>
  81. </PortalToFollowElem>
  82. </div>
  83. )
  84. }
  85. export default HistoryBlockComponent