child-segment-detail.tsx 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135
  1. import React, { type FC, useMemo, useState } from 'react'
  2. import { useTranslation } from 'react-i18next'
  3. import {
  4. RiCloseLine,
  5. RiCollapseDiagonalLine,
  6. RiExpandDiagonalLine,
  7. } from '@remixicon/react'
  8. import ActionButtons from './common/action-buttons'
  9. import ChunkContent from './common/chunk-content'
  10. import Dot from './common/dot'
  11. import { SegmentIndexTag } from './common/segment-index-tag'
  12. import { useSegmentListContext } from './index'
  13. import type { ChildChunkDetail, ChunkingMode } from '@/models/datasets'
  14. import { useEventEmitterContextContext } from '@/context/event-emitter'
  15. import { formatNumber } from '@/utils/format'
  16. import classNames from '@/utils/classnames'
  17. import Divider from '@/app/components/base/divider'
  18. import { formatTime } from '@/utils/time'
  19. type IChildSegmentDetailProps = {
  20. chunkId: string
  21. childChunkInfo?: Partial<ChildChunkDetail> & { id: string }
  22. onUpdate: (segmentId: string, childChunkId: string, content: string) => void
  23. onCancel: () => void
  24. docForm: ChunkingMode
  25. }
  26. /**
  27. * Show all the contents of the segment
  28. */
  29. const ChildSegmentDetail: FC<IChildSegmentDetailProps> = ({
  30. chunkId,
  31. childChunkInfo,
  32. onUpdate,
  33. onCancel,
  34. docForm,
  35. }) => {
  36. const { t } = useTranslation()
  37. const [content, setContent] = useState(childChunkInfo?.content || '')
  38. const { eventEmitter } = useEventEmitterContextContext()
  39. const [loading, setLoading] = useState(false)
  40. const fullScreen = useSegmentListContext(s => s.fullScreen)
  41. const toggleFullScreen = useSegmentListContext(s => s.toggleFullScreen)
  42. eventEmitter?.useSubscription((v) => {
  43. if (v === 'update-child-segment')
  44. setLoading(true)
  45. if (v === 'update-child-segment-done')
  46. setLoading(false)
  47. })
  48. const handleCancel = () => {
  49. onCancel()
  50. }
  51. const handleSave = () => {
  52. onUpdate(chunkId, childChunkInfo?.id || '', content)
  53. }
  54. const wordCountText = useMemo(() => {
  55. const count = content.length
  56. return `${formatNumber(count)} ${t('datasetDocuments.segment.characters', { count })}`
  57. // eslint-disable-next-line react-hooks/exhaustive-deps
  58. }, [content.length])
  59. const EditTimeText = useMemo(() => {
  60. const timeText = formatTime({
  61. date: (childChunkInfo?.updated_at ?? 0) * 1000,
  62. dateFormat: 'MM/DD/YYYY h:mm:ss',
  63. })
  64. return `${t('datasetDocuments.segment.editedAt')} ${timeText}`
  65. // eslint-disable-next-line react-hooks/exhaustive-deps
  66. }, [childChunkInfo?.updated_at])
  67. return (
  68. <div className={'flex flex-col h-full'}>
  69. <div className={classNames('flex items-center justify-between', fullScreen ? 'py-3 pr-4 pl-6 border border-divider-subtle' : 'pt-3 pr-3 pl-4')}>
  70. <div className='flex flex-col'>
  71. <div className='text-text-primary system-xl-semibold'>{t('datasetDocuments.segment.editChildChunk')}</div>
  72. <div className='flex items-center gap-x-2'>
  73. <SegmentIndexTag positionId={childChunkInfo?.position || ''} labelPrefix={t('datasetDocuments.segment.childChunk') as string} />
  74. <Dot />
  75. <span className='text-text-tertiary system-xs-medium'>{wordCountText}</span>
  76. <Dot />
  77. <span className='text-text-tertiary system-xs-medium'>
  78. {EditTimeText}
  79. </span>
  80. </div>
  81. </div>
  82. <div className='flex items-center'>
  83. {fullScreen && (
  84. <>
  85. <ActionButtons
  86. handleCancel={handleCancel}
  87. handleSave={handleSave}
  88. loading={loading}
  89. isChildChunk={true}
  90. />
  91. <Divider type='vertical' className='h-3.5 bg-divider-regular ml-4 mr-2' />
  92. </>
  93. )}
  94. <div className='w-8 h-8 flex justify-center items-center p-1.5 cursor-pointer mr-1' onClick={toggleFullScreen}>
  95. {fullScreen ? <RiCollapseDiagonalLine className='w-4 h-4 text-text-tertiary' /> : <RiExpandDiagonalLine className='w-4 h-4 text-text-tertiary' />}
  96. </div>
  97. <div className='w-8 h-8 flex justify-center items-center p-1.5 cursor-pointer' onClick={onCancel}>
  98. <RiCloseLine className='w-4 h-4 text-text-tertiary' />
  99. </div>
  100. </div>
  101. </div>
  102. <div className={classNames('flex grow w-full', fullScreen ? 'flex-row justify-center px-6 pt-6' : 'py-3 px-4')}>
  103. <div className={classNames('break-all overflow-hidden whitespace-pre-line h-full', fullScreen ? 'w-1/2' : 'w-full')}>
  104. <ChunkContent
  105. docForm={docForm}
  106. question={content}
  107. onQuestionChange={content => setContent(content)}
  108. isEditMode={true}
  109. />
  110. </div>
  111. </div>
  112. {!fullScreen && (
  113. <div className='flex items-center justify-end p-4 pt-3 border-t-[1px] border-t-divider-subtle'>
  114. <ActionButtons
  115. handleCancel={handleCancel}
  116. handleSave={handleSave}
  117. loading={loading}
  118. isChildChunk={true}
  119. />
  120. </div>
  121. )}
  122. </div>
  123. )
  124. }
  125. export default React.memo(ChildSegmentDetail)