component.tsx 4.1 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697
  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_DATASETS_EVENT_EMITTER } from '../../constants'
  6. import type { Dataset } from './index'
  7. import { DELETE_CONTEXT_BLOCK_COMMAND } from './index'
  8. import { File05, Folder } from '@/app/components/base/icons/src/vender/solid/files'
  9. import { Plus } from '@/app/components/base/icons/src/vender/line/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 ContextBlockComponentProps = {
  17. nodeKey: string
  18. datasets?: Dataset[]
  19. onAddContext: () => void
  20. }
  21. const ContextBlockComponent: FC<ContextBlockComponentProps> = ({
  22. nodeKey,
  23. datasets = [],
  24. onAddContext,
  25. }) => {
  26. const { t } = useTranslation()
  27. const [ref, isSelected] = useSelectOrDelete(nodeKey, DELETE_CONTEXT_BLOCK_COMMAND)
  28. const [triggerRef, open, setOpen] = useTrigger()
  29. const { eventEmitter } = useEventEmitterContextContext()
  30. const [localDatasets, setLocalDatasets] = useState<Dataset[]>(datasets)
  31. eventEmitter?.useSubscription((v: any) => {
  32. if (v?.type === UPDATE_DATASETS_EVENT_EMITTER)
  33. setLocalDatasets(v.payload)
  34. })
  35. return (
  36. <div className={`
  37. group inline-flex items-center pl-1 pr-0.5 h-6 border border-transparent bg-[#F4F3FF] text-[#6938EF] rounded-[5px] hover:bg-[#EBE9FE]
  38. ${open ? 'bg-[#EBE9FE]' : 'bg-[#F4F3FF]'}
  39. ${isSelected && '!border-[#9B8AFB]'}
  40. `} ref={ref}>
  41. <File05 className='mr-1 w-[14px] h-[14px]' />
  42. <div className='mr-1 text-xs font-medium'>{t('common.promptEditor.context.item.title')}</div>
  43. <PortalToFollowElem
  44. open={open}
  45. onOpenChange={setOpen}
  46. placement='bottom-end'
  47. offset={{
  48. mainAxis: 3,
  49. alignmentAxis: -147,
  50. }}
  51. >
  52. <PortalToFollowElemTrigger ref={triggerRef}>
  53. <div className={`
  54. flex items-center justify-center w-[18px] h-[18px] text-[11px] font-semibold rounded cursor-pointer
  55. ${open ? 'bg-[#6938EF] text-white' : 'bg-white/50 group-hover:bg-white group-hover:shadow-xs'}
  56. `}>{localDatasets.length}</div>
  57. </PortalToFollowElemTrigger>
  58. <PortalToFollowElemContent style={{ zIndex: 100 }}>
  59. <div className='w-[360px] bg-white rounded-xl shadow-lg'>
  60. <div className='p-4'>
  61. <div className='mb-2 text-xs font-medium text-gray-500'>
  62. {t('common.promptEditor.context.modal.title', { num: localDatasets.length })}
  63. </div>
  64. <div className='max-h-[270px] overflow-y-auto'>
  65. {
  66. localDatasets.map(dataset => (
  67. <div key={dataset.id} className='flex items-center h-8'>
  68. <div className='flex items-center justify-center shrink-0 mr-2 w-6 h-6 bg-[#F5F8FF] rounded-md border-[0.5px] border-[#EAECF5]'>
  69. <Folder className='w-4 h-4 text-[#444CE7]' />
  70. </div>
  71. <div className='text-sm text-gray-800 truncate' title=''>{dataset.name}</div>
  72. </div>
  73. ))
  74. }
  75. </div>
  76. <div className='flex items-center h-8 text-[#155EEF] cursor-pointer' onClick={onAddContext}>
  77. <div className='shrink-0 flex justify-center items-center mr-2 w-6 h-6 rounded-md border-[0.5px] border-gray-100'>
  78. <Plus className='w-[14px] h-[14px]' />
  79. </div>
  80. <div className='text-[13px] font-medium' title=''>{t('common.promptEditor.context.modal.add')}</div>
  81. </div>
  82. </div>
  83. <div className='px-4 py-3 text-xs text-gray-500 bg-gray-50 border-t-[0.5px] border-gray-50 rounded-b-xl'>
  84. {t('common.promptEditor.context.modal.footer')}
  85. </div>
  86. </div>
  87. </PortalToFollowElemContent>
  88. </PortalToFollowElem>
  89. </div>
  90. )
  91. }
  92. export default ContextBlockComponent