| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 | import type { FC } from 'react'import React, { useState } from 'react'import cn from 'classnames'import { ArrowUpRightIcon } from '@heroicons/react/24/outline'import { useTranslation } from 'react-i18next'import { StatusItem } from '../../list'import { DocumentTitle } from '../index'import s from './style.module.css'import { SegmentIndexTag } from './index'import Modal from '@/app/components/base/modal'import Button from '@/app/components/base/button'import Switch from '@/app/components/base/switch'import Divider from '@/app/components/base/divider'import Indicator from '@/app/components/header/indicator'import { formatNumber } from '@/utils/format'import type { SegmentDetailModel } from '@/models/datasets'import { AlertCircle } from '@/app/components/base/icons/src/vender/solid/alertsAndFeedback'import { Trash03 } from '@/app/components/base/icons/src/vender/line/general'const ProgressBar: FC<{ percent: number; loading: boolean }> = ({ percent, loading }) => {  return (    <div className={s.progressWrapper}>      <div className={cn(s.progress, loading ? s.progressLoading : '')}>        <div          className={s.progressInner}          style={{ width: `${loading ? 0 : (percent * 100).toFixed(2)}%` }}        />      </div>      <div className={loading ? s.progressTextLoading : s.progressText}>{loading ? null : percent.toFixed(2)}</div>    </div>  )}export type UsageScene = 'doc' | 'hitTesting'type ISegmentCardProps = {  loading: boolean  detail?: SegmentDetailModel & { document: { name: string } }  score?: number  onClick?: () => void  onChangeSwitch?: (segId: string, enabled: boolean) => Promise<void>  onDelete?: (segId: string) => Promise<void>  scene?: UsageScene  className?: string  archived?: boolean  embeddingAvailable?: boolean}const SegmentCard: FC<ISegmentCardProps> = ({  detail = {},  score,  onClick,  onChangeSwitch,  onDelete,  loading = true,  scene = 'doc',  className = '',  archived,  embeddingAvailable,}) => {  const { t } = useTranslation()  const {    id,    position,    enabled,    content,    word_count,    hit_count,    index_node_hash,    answer,  } = detail as Required<ISegmentCardProps>['detail']  const isDocScene = scene === 'doc'  const [showModal, setShowModal] = useState(false)  const renderContent = () => {    if (answer) {      return (        <>          <div className='flex mb-2'>            <div className='mr-2 text-[13px] font-semibold text-gray-400'>Q</div>            <div className='text-[13px]'>{content}</div>          </div>          <div className='flex'>            <div className='mr-2 text-[13px] font-semibold text-gray-400'>A</div>            <div className='text-[13px]'>{answer}</div>          </div>        </>      )    }    return content  }  return (    <div      className={cn(        s.segWrapper,        (isDocScene && !enabled) ? 'bg-gray-25' : '',        'group',        !loading ? 'pb-4 hover:pb-[10px]' : '',        className,      )}      onClick={() => onClick?.()}    >      <div className={s.segTitleWrapper}>        {isDocScene          ? <>            <SegmentIndexTag positionId={position} className={cn('w-fit group-hover:opacity-100', (isDocScene && !enabled) ? 'opacity-50' : '')} />            <div className={s.segStatusWrapper}>              {loading                ? (                  <Indicator                    color="gray"                    className="bg-gray-200 border-gray-300 shadow-none"                  />                )                : (                  <>                    <StatusItem status={enabled ? 'enabled' : 'disabled'} reverse textCls="text-gray-500 text-xs" />                    {embeddingAvailable && (                      <div className="hidden group-hover:inline-flex items-center">                        <Divider type="vertical" className="!h-2" />                        <div                          onClick={(e: React.MouseEvent<HTMLDivElement, MouseEvent>) =>                            e.stopPropagation()                          }                          className="inline-flex items-center"                        >                          <Switch                            size='md'                            disabled={archived || detail.status !== 'completed'}                            defaultValue={enabled}                            onChange={async (val) => {                              await onChangeSwitch?.(id, val)                            }}                          />                        </div>                      </div>                    )}                  </>                )}            </div>          </>          : (            score !== null              ? (                <div className={s.hitTitleWrapper}>                  <div className={cn(s.commonIcon, s.targetIcon, loading ? '!bg-gray-300' : '', '!w-3.5 !h-3.5')} />                  <ProgressBar percent={score ?? 0} loading={loading} />                </div>              )              : null          )}      </div>      {loading        ? (          <div className={cn(s.cardLoadingWrapper, s.cardLoadingIcon)}>            <div className={cn(s.cardLoadingBg)} />          </div>        )        : (          isDocScene            ? <>              <div                className={cn(                  s.segContent,                  enabled ? '' : 'opacity-50',                  'group-hover:text-transparent group-hover:bg-clip-text group-hover:bg-gradient-to-b',                )}              >                {renderContent()}              </div>              <div className={cn('group-hover:flex', s.segData)}>                <div className="flex items-center mr-6">                  <div className={cn(s.commonIcon, s.typeSquareIcon)}></div>                  <div className={s.segDataText}>{formatNumber(word_count)}</div>                </div>                <div className="flex items-center mr-6">                  <div className={cn(s.commonIcon, s.targetIcon)} />                  <div className={s.segDataText}>{formatNumber(hit_count)}</div>                </div>                <div className="grow flex items-center">                  <div className={cn(s.commonIcon, s.bezierCurveIcon)} />                  <div className={s.segDataText}>{index_node_hash}</div>                </div>                {!archived && embeddingAvailable && (                  <div className='shrink-0 w-6 h-6 flex items-center justify-center rounded-md hover:bg-red-100 hover:text-red-600 cursor-pointer group/delete' onClick={(e) => {                    e.stopPropagation()                    setShowModal(true)                  }}>                    <Trash03 className='w-[14px] h-[14px] text-gray-500 group-hover/delete:text-red-600' />                  </div>                )}              </div>            </>            : <>              <div className="h-[140px] overflow-hidden text-ellipsis text-sm font-normal text-gray-800">                {renderContent()}              </div>              <div className={cn('w-full bg-gray-50 group-hover:bg-white')}>                <Divider />                <div className="relative flex items-center w-full">                  <DocumentTitle                    name={detail?.document?.name || ''}                    extension={(detail?.document?.name || '').split('.').pop() || 'txt'}                    wrapperCls='w-full'                    iconCls="!h-4 !w-4 !bg-contain"                    textCls="text-xs text-gray-700 !font-normal overflow-hidden whitespace-nowrap text-ellipsis"                  />                  <div className={cn(s.chartLinkText, 'group-hover:inline-flex')}>                    {t('datasetHitTesting.viewChart')}                    <ArrowUpRightIcon className="w-3 h-3 ml-1 stroke-current stroke-2" />                  </div>                </div>              </div>            </>        )}      {showModal && <Modal isShow={showModal} onClose={() => setShowModal(false)} className={s.delModal} closable>        <div>          <div className={s.warningWrapper}>            <AlertCircle className='w-6 h-6 text-red-600' />          </div>          <div className='text-xl font-semibold text-gray-900 mb-1'>{t('datasetDocuments.segment.delete')}</div>          <div className='flex gap-2 justify-end'>            <Button onClick={() => setShowModal(false)}>{t('common.operation.cancel')}</Button>            <Button              type='warning'              onClick={async () => {                await onDelete?.(id)              }}              className='border-red-700 border-[0.5px]'            >              {t('common.operation.sure')}            </Button>          </div>        </div>      </Modal>}    </div>  )}export default SegmentCard
 |