123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101 |
- 'use client'
- import type { FC } from 'react'
- import React from 'react'
- import { useTranslation } from 'react-i18next'
- import { RiArrowDownSLine, RiArrowRightSLine } from '@remixicon/react'
- import { useBoolean } from 'ahooks'
- import ChildChunkItem from './child-chunks-item'
- import ChunkDetailModal from './chunk-detail-modal'
- import ResultItemMeta from './result-item-meta'
- import ResultItemFooter from './result-item-footer'
- import type { HitTesting } from '@/models/datasets'
- import cn from '@/utils/classnames'
- import type { FileAppearanceTypeEnum } from '@/app/components/base/file-uploader/types'
- import Tag from '@/app/components/datasets/documents/detail/completed/common/tag'
- import { extensionToFileType } from '@/app/components/datasets/hit-testing/utils/extension-to-file-type'
- import { Markdown } from '@/app/components/base/markdown'
- const i18nPrefix = 'datasetHitTesting'
- type Props = {
- payload: HitTesting
- }
- const ResultItem: FC<Props> = ({
- payload,
- }) => {
- const { t } = useTranslation()
- const { segment, score, child_chunks } = payload
- const data = segment
- const { position, word_count, content, sign_content, keywords, document } = data
- const isParentChildRetrieval = !!(child_chunks && child_chunks.length > 0)
- const extension = document.name.split('.').slice(-1)[0] as FileAppearanceTypeEnum
- const fileType = extensionToFileType(extension)
- const [isFold, {
- toggle: toggleFold,
- }] = useBoolean(false)
- const Icon = isFold ? RiArrowRightSLine : RiArrowDownSLine
- const [isShowDetailModal, {
- setTrue: showDetailModal,
- setFalse: hideDetailModal,
- }] = useBoolean(false)
- return (
- <div className={cn('pt-3 bg-chat-bubble-bg rounded-xl hover:shadow-lg cursor-pointer')} onClick={showDetailModal}>
- {/* Meta info */}
- <ResultItemMeta className='px-3' labelPrefix={`${isParentChildRetrieval ? 'Parent-' : ''}Chunk`} positionId={position} wordCount={word_count} score={score} />
- {/* Main */}
- <div className='mt-1 px-3'>
- <Markdown
- className='line-clamp-2'
- content={sign_content || content}
- customDisallowedElements={['input']}
- />
- {isParentChildRetrieval && (
- <div className='mt-1'>
- <div
- className={cn('inline-flex items-center h-6 space-x-0.5 text-text-secondary select-none rounded-lg cursor-pointer', isFold && 'pl-1 bg-[linear-gradient(90deg,_rgba(200,_206,_218,_0.20)_0%,_rgba(200,_206,_218,_0.04)_100%)]')}
- onClick={(e) => {
- e.stopPropagation()
- toggleFold()
- }}
- >
- <Icon className={cn('w-4 h-4', isFold && 'opacity-50')} />
- <div className='text-xs font-semibold uppercase'>{t(`${i18nPrefix}.hitChunks`, { num: child_chunks.length })}</div>
- </div>
- {!isFold && (
- <div className='space-y-2'>
- {child_chunks.map(item => (
- <div key={item.id} className='ml-[7px] pl-[7px] border-l-[2px] border-text-accent-secondary'>
- <ChildChunkItem payload={item} isShowAll={false} />
- </div>
- ))}
- </div>
- )}
- </div>
- )}
- {!isParentChildRetrieval && keywords && keywords.length > 0 && (
- <div className='mt-2 flex flex-wrap'>
- {keywords.map(keyword => (
- <Tag key={keyword} text={keyword} className='mr-2' />
- ))}
- </div>
- )}
- </div>
- {/* Foot */}
- <ResultItemFooter docType={fileType} docTitle={document.name} showDetailModal={showDetailModal} />
- {
- isShowDetailModal && (
- <ChunkDetailModal
- payload={payload}
- onHide={hideDetailModal}
- />
- )
- }
- </div >
- )
- }
- export default React.memo(ResultItem)
|