
Co-authored-by: John Wang <takatost@gmail.com> Co-authored-by: Jyong <718720800@qq.com> Co-authored-by: 金伟强 <iamjoel007@gmail.com>
255 lines
11 KiB
TypeScript
255 lines
11 KiB
TypeScript
'use client'
|
|
import type { FC } from 'react'
|
|
import React, { useState } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import cn from 'classnames'
|
|
import copy from 'copy-to-clipboard'
|
|
import { HandThumbDownIcon, HandThumbUpIcon } from '@heroicons/react/24/outline'
|
|
import { useBoolean } from 'ahooks'
|
|
import { Markdown } from '@/app/components/base/markdown'
|
|
import Loading from '@/app/components/base/loading'
|
|
import Toast from '@/app/components/base/toast'
|
|
import type { Feedbacktype } from '@/app/components/app/chat'
|
|
import { fetchMoreLikeThis, updateFeedback } from '@/service/share'
|
|
|
|
const MAX_DEPTH = 3
|
|
export type IGenerationItemProps = {
|
|
className?: string
|
|
content: string
|
|
messageId?: string | null
|
|
isLoading?: boolean
|
|
isInWebApp?: boolean
|
|
moreLikeThis?: boolean
|
|
depth?: number
|
|
feedback?: Feedbacktype
|
|
onFeedback?: (feedback: Feedbacktype) => void
|
|
onSave?: (messageId: string) => void
|
|
isMobile?: boolean
|
|
isInstalledApp: boolean
|
|
installedAppId?: string
|
|
}
|
|
|
|
export const SimpleBtn = ({ className, onClick, children }: {
|
|
className?: string
|
|
onClick?: () => void
|
|
children: React.ReactNode
|
|
}) => (
|
|
<div
|
|
className={cn(className, 'flex items-center h-7 px-3 rounded-md border border-gray-200 text-xs text-gray-700 font-medium cursor-pointer hover:shadow-sm hover:border-gray-300')}
|
|
onClick={() => onClick?.()}
|
|
>
|
|
{children}
|
|
</div>
|
|
)
|
|
|
|
export const copyIcon = (
|
|
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
<path d="M9.3335 2.33341C9.87598 2.33341 10.1472 2.33341 10.3698 2.39304C10.9737 2.55486 11.4454 3.02657 11.6072 3.63048C11.6668 3.85302 11.6668 4.12426 11.6668 4.66675V10.0334C11.6668 11.0135 11.6668 11.5036 11.4761 11.8779C11.3083 12.2072 11.0406 12.4749 10.7113 12.6427C10.337 12.8334 9.84692 12.8334 8.86683 12.8334H5.1335C4.1534 12.8334 3.66336 12.8334 3.28901 12.6427C2.95973 12.4749 2.69201 12.2072 2.52423 11.8779C2.3335 11.5036 2.3335 11.0135 2.3335 10.0334V4.66675C2.3335 4.12426 2.3335 3.85302 2.39313 3.63048C2.55494 3.02657 3.02665 2.55486 3.63056 2.39304C3.8531 2.33341 4.12435 2.33341 4.66683 2.33341M5.60016 3.50008H8.40016C8.72686 3.50008 8.89021 3.50008 9.01499 3.4365C9.12475 3.38058 9.21399 3.29134 9.26992 3.18158C9.3335 3.05679 9.3335 2.89345 9.3335 2.56675V2.10008C9.3335 1.77338 9.3335 1.61004 9.26992 1.48525C9.21399 1.37549 9.12475 1.28625 9.01499 1.23033C8.89021 1.16675 8.72686 1.16675 8.40016 1.16675H5.60016C5.27347 1.16675 5.11012 1.16675 4.98534 1.23033C4.87557 1.28625 4.78634 1.37549 4.73041 1.48525C4.66683 1.61004 4.66683 1.77338 4.66683 2.10008V2.56675C4.66683 2.89345 4.66683 3.05679 4.73041 3.18158C4.78634 3.29134 4.87557 3.38058 4.98534 3.4365C5.11012 3.50008 5.27347 3.50008 5.60016 3.50008Z" stroke="#344054" strokeWidth="1.25" strokeLinecap="round" strokeLinejoin="round" />
|
|
</svg>
|
|
)
|
|
|
|
const moreLikeThisIcon = (
|
|
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
<g clipPath="url(#clip0_4138_1944)">
|
|
<path d="M2.62533 12.8337V9.91699M2.62533 4.08366V1.16699M1.16699 2.62533H4.08366M1.16699 11.3753H4.08366M7.58366 1.75033L6.57206 4.38049C6.40755 4.80821 6.32529 5.02207 6.19738 5.20196C6.08402 5.36139 5.94472 5.50069 5.78529 5.61405C5.6054 5.74196 5.39155 5.82421 4.96383 5.98872L2.33366 7.00033L4.96383 8.01193C5.39155 8.17644 5.60541 8.25869 5.78529 8.3866C5.94472 8.49996 6.08402 8.63926 6.19738 8.79869C6.32529 8.97858 6.40755 9.19244 6.57206 9.62016L7.58366 12.2503L8.59526 9.62015C8.75977 9.19244 8.84202 8.97858 8.96993 8.79869C9.0833 8.63926 9.22259 8.49996 9.38203 8.3866C9.56191 8.25869 9.77577 8.17644 10.2035 8.01193L12.8337 7.00033L10.2035 5.98872C9.77577 5.82421 9.56191 5.74196 9.38203 5.61405C9.22259 5.50069 9.0833 5.36139 8.96993 5.20196C8.84202 5.02207 8.75977 4.80821 8.59526 4.38049L7.58366 1.75033Z" stroke="#344054" strokeWidth="1.25" strokeLinecap="round" strokeLinejoin="round" />
|
|
</g>
|
|
<defs>
|
|
<clipPath id="clip0_4138_1944">
|
|
<rect width="14" height="14" fill="white" />
|
|
</clipPath>
|
|
</defs>
|
|
</svg>
|
|
)
|
|
|
|
const saveIcon = (
|
|
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
<path d="M11.0837 12.25L7.00033 9.33333L2.91699 12.25V2.91667C2.91699 2.60725 3.03991 2.3105 3.2587 2.09171C3.47749 1.87292 3.77424 1.75 4.08366 1.75H9.91699C10.2264 1.75 10.5232 1.87292 10.7419 2.09171C10.9607 2.3105 11.0837 2.60725 11.0837 2.91667V12.25Z" stroke="#344054" strokeWidth="1.25" strokeLinecap="round" strokeLinejoin="round" />
|
|
</svg>
|
|
)
|
|
|
|
const GenerationItem: FC<IGenerationItemProps> = ({
|
|
className,
|
|
content,
|
|
messageId,
|
|
isLoading,
|
|
moreLikeThis,
|
|
isInWebApp = false,
|
|
feedback,
|
|
onFeedback,
|
|
onSave,
|
|
depth = 1,
|
|
isMobile,
|
|
isInstalledApp,
|
|
installedAppId,
|
|
}) => {
|
|
const { t } = useTranslation()
|
|
const isTop = depth === 1
|
|
|
|
const [completionRes, setCompletionRes] = useState('')
|
|
const [childMessageId, setChildMessageId] = useState<string | null>(null)
|
|
const hasChild = !!childMessageId
|
|
const [childFeedback, setChildFeedback] = useState<Feedbacktype>({
|
|
rating: null,
|
|
})
|
|
|
|
const handleFeedback = async (childFeedback: Feedbacktype) => {
|
|
await updateFeedback({ url: `/messages/${childMessageId}/feedbacks`, body: { rating: childFeedback.rating } }, isInstalledApp, installedAppId)
|
|
setChildFeedback(childFeedback)
|
|
}
|
|
|
|
const [isQuerying, { setTrue: startQuerying, setFalse: stopQuerying }] = useBoolean(false)
|
|
|
|
const childProps = {
|
|
isInWebApp: true,
|
|
content: completionRes,
|
|
messageId: childMessageId,
|
|
depth: depth + 1,
|
|
moreLikeThis: true,
|
|
onFeedback: handleFeedback,
|
|
isLoading: isQuerying,
|
|
feedback: childFeedback,
|
|
onSave,
|
|
isMobile,
|
|
isInstalledApp,
|
|
installedAppId,
|
|
}
|
|
|
|
const handleMoreLikeThis = async () => {
|
|
if (isQuerying || !messageId) {
|
|
Toast.notify({ type: 'warning', message: t('appDebug.errorMessage.waitForResponse') })
|
|
return
|
|
}
|
|
startQuerying()
|
|
const res: any = await fetchMoreLikeThis(messageId as string, isInstalledApp, installedAppId)
|
|
setCompletionRes(res.answer)
|
|
setChildMessageId(res.id)
|
|
stopQuerying()
|
|
}
|
|
|
|
const mainStyle = (() => {
|
|
const res: any = !isTop
|
|
? {
|
|
background: depth % 2 === 0 ? 'linear-gradient(90.07deg, #F9FAFB 0.05%, rgba(249, 250, 251, 0) 99.93%)' : '#fff',
|
|
}
|
|
: {}
|
|
|
|
if (hasChild)
|
|
res.boxShadow = '0px 1px 2px rgba(16, 24, 40, 0.05)'
|
|
|
|
return res
|
|
})()
|
|
return (
|
|
<div className={cn(className, isTop ? 'rounded-xl border border-gray-200 bg-white' : 'rounded-br-xl !mt-0')}
|
|
style={isTop
|
|
? {
|
|
boxShadow: '0px 1px 2px rgba(16, 24, 40, 0.05)',
|
|
}
|
|
: {}}
|
|
>
|
|
{isLoading
|
|
? (
|
|
<div className='flex items-center h-10'><Loading type='area' /></div>
|
|
)
|
|
: (
|
|
<div
|
|
className={cn(!isTop && 'rounded-br-xl border-l-2 border-primary-400', 'p-4')}
|
|
style={mainStyle}
|
|
>
|
|
<Markdown content={content} />
|
|
{messageId && (
|
|
<div className='flex items-center justify-between mt-3'>
|
|
<div className='flex items-center'>
|
|
<SimpleBtn
|
|
className={cn(isMobile && '!px-1.5', 'space-x-1')}
|
|
onClick={() => {
|
|
copy(content)
|
|
Toast.notify({ type: 'success', message: t('common.actionMsg.copySuccessfully') })
|
|
}}>
|
|
{copyIcon}
|
|
{!isMobile && <div>{t('common.operation.copy')}</div>}
|
|
</SimpleBtn>
|
|
{isInWebApp && (
|
|
<>
|
|
<SimpleBtn
|
|
className={cn(isMobile && '!px-1.5', 'ml-2 space-x-1')}
|
|
onClick={() => { onSave?.(messageId as string) }}
|
|
>
|
|
{saveIcon}
|
|
{!isMobile && <div>{t('common.operation.save')}</div>}
|
|
</SimpleBtn>
|
|
{(moreLikeThis && depth < MAX_DEPTH) && (
|
|
<SimpleBtn
|
|
className={cn(isMobile && '!px-1.5', 'ml-2 space-x-1')}
|
|
onClick={handleMoreLikeThis}
|
|
>
|
|
{moreLikeThisIcon}
|
|
{!isMobile && <div>{t('appDebug.feature.moreLikeThis.title')}</div>}
|
|
</SimpleBtn>)}
|
|
<div className="mx-3 w-[1px] h-[14px] bg-gray-200"></div>
|
|
{!feedback?.rating && (
|
|
<SimpleBtn className="!px-0">
|
|
<>
|
|
<div
|
|
onClick={() => {
|
|
onFeedback?.({
|
|
rating: 'like',
|
|
})
|
|
}}
|
|
className='flex w-6 h-6 items-center justify-center rounded-md cursor-pointer hover:bg-gray-100'>
|
|
<HandThumbUpIcon width={16} height={16} />
|
|
</div>
|
|
<div
|
|
onClick={() => {
|
|
onFeedback?.({
|
|
rating: 'dislike',
|
|
})
|
|
}}
|
|
className='flex w-6 h-6 items-center justify-center rounded-md cursor-pointer hover:bg-gray-100'>
|
|
<HandThumbDownIcon width={16} height={16} />
|
|
</div>
|
|
</>
|
|
</SimpleBtn>
|
|
)}
|
|
{feedback?.rating === 'like' && (
|
|
<div
|
|
onClick={() => {
|
|
onFeedback?.({
|
|
rating: null,
|
|
})
|
|
}}
|
|
className='flex w-7 h-7 items-center justify-center rounded-md cursor-pointer !text-primary-600 border border-primary-200 bg-primary-100 hover:border-primary-300 hover:bg-primary-200'>
|
|
<HandThumbUpIcon width={16} height={16} />
|
|
</div>
|
|
)}
|
|
{feedback?.rating === 'dislike' && (
|
|
<div
|
|
onClick={() => {
|
|
onFeedback?.({
|
|
rating: null,
|
|
})
|
|
}}
|
|
className='flex w-7 h-7 items-center justify-center rounded-md cursor-pointer !text-red-600 border border-red-200 bg-red-100 hover:border-red-300 hover:bg-red-200'>
|
|
<HandThumbDownIcon width={16} height={16} />
|
|
</div>
|
|
)}
|
|
</>
|
|
)}
|
|
</div>
|
|
<div className='text-xs text-gray-500'>{content?.length} {t('common.unit.char')}</div>
|
|
</div>
|
|
)}
|
|
|
|
</div>
|
|
)}
|
|
|
|
{((childMessageId || isQuerying) && depth < 3) && (
|
|
<div className='pl-4'>
|
|
<GenerationItem {...childProps} />
|
|
</div>
|
|
)}
|
|
|
|
</div>
|
|
)
|
|
}
|
|
export default React.memo(GenerationItem)
|