<script setup lang='ts'>
|
import { computed, ref, watch } from 'vue'
|
import { NDropdown, useMessage } from 'naive-ui'
|
import AvatarComponent from './Avatar.vue'
|
import TextComponent from './Text.vue'
|
import { SvgIcon } from '@/components/common'
|
import { useIconRender } from '@/hooks/useIconRender'
|
import { t } from '@/locales'
|
import { useBasicLayout } from '@/hooks/useBasicLayout'
|
import { copyToClip } from '@/utils/copy'
|
import { homeStore } from '@/store'
|
import { getSeed, mlog} from '@/api'
|
|
interface Props {
|
dateTime?: string
|
text?: string
|
inversion?: boolean
|
error?: boolean
|
loading?: boolean
|
chat:Chat.Chat
|
index:number
|
}
|
|
interface Emit {
|
(ev: 'regenerate'): void
|
(ev: 'delete'): void
|
(ev: 'edit'): void
|
}
|
|
const props = defineProps<Props>()
|
|
const emit = defineEmits<Emit>()
|
|
const { isMobile } = useBasicLayout()
|
|
const { iconRender } = useIconRender()
|
|
const message = useMessage()
|
|
const textRef = ref<HTMLElement>()
|
|
const asRawText = ref(props.inversion && homeStore.myData.session.isCloseMdPreview)
|
|
const messageRef = ref<HTMLElement>()
|
|
const options = computed(() => {
|
const common = [
|
{
|
label: t('chat.copy'),
|
key: 'copyText',
|
icon: iconRender({ icon: 'ri:file-copy-2-line' }),
|
},
|
{
|
label: t('common.delete'),
|
key: 'delete',
|
icon: iconRender({ icon: 'ri:delete-bin-line' }),
|
},
|
// {
|
// label: t('common.edit'),
|
// key: 'edit',
|
// icon: iconRender({ icon: 'ri:edit-2-line' }),
|
// },
|
]
|
|
if (!props.inversion) {
|
common.unshift({
|
label: asRawText.value ? t('chat.preview') : t('chat.showRawText'),
|
key: 'toggleRenderType',
|
icon: iconRender({ icon: asRawText.value ? 'ic:outline-code-off' : 'ic:outline-code' }),
|
});
|
common.unshift({
|
label: t('mj.tts'),
|
key: 'tts',
|
icon: iconRender({ icon:'mdi:tts' }),
|
})
|
}
|
|
return common
|
})
|
|
function handleSelect(key: 'copyText' | 'delete' | 'edit' | 'toggleRenderType' | 'tts') {
|
switch (key) {
|
case 'tts':
|
homeStore.setMyData({act:'gpt.ttsv2', actData:{ index:props.index , uuid:props.chat.uuid, text:props.text } });
|
return;
|
case 'copyText':
|
handleCopy()
|
return
|
case 'toggleRenderType':
|
asRawText.value = !asRawText.value
|
return
|
case 'delete':
|
emit('delete')
|
return
|
case 'edit':
|
emit('edit')
|
}
|
}
|
|
function handleRegenerate() {
|
messageRef.value?.scrollIntoView()
|
emit('regenerate')
|
}
|
|
|
async function handleCopy(txt?:string) {
|
try {
|
await copyToClip( txt|| props.text || '')
|
message.success( t('chat.copied'))
|
}
|
catch {
|
message.error( t('mj.copyFail') )
|
}
|
}
|
|
const sendReload = () => {
|
homeStore.setMyData({act:'mjReload', actData:{mjID:props.chat.mjID} })
|
}
|
|
function handleRegenerate2() {
|
messageRef.value?.scrollIntoView()
|
//emit('regenerate')
|
mlog('重新发送!');
|
homeStore.setMyData({act:'gpt.resubmit', actData:{ index:props.index , uuid:props.chat.uuid } });
|
}
|
|
</script>
|
|
<template>
|
<div
|
ref="messageRef"
|
class="flex w-full mb-8 overflow-hidden"
|
:class="[{ 'flex-row-reverse': inversion }]"
|
>
|
<div
|
class="flex items-center justify-center flex-shrink-0 h-8 overflow-hidden rounded-full basis-8"
|
:class="[inversion ? 'ml-2' : 'mr-2']" v-if="!inversion">
|
<AvatarComponent :image="inversion" :logo="chat.logo"/>
|
</div>
|
<div class="overflow-hidden text-sm " :class="[inversion ? 'items-end' : 'items-start']">
|
<p class="text-xs group text-[#b4bbc4] flex items-center space-x-2 " :class="[inversion ? 'justify-end' : 'justify-start']">
|
<!-- <span>{{ dateTime }}</span>
|
<span v-if="chat.model" class="text-[#b4bbc4]/50">{{ chat.model }}</span> -->
|
<!-- <span>{{ chat.opt?.progress }}</span> -->
|
<template v-if="chat.opt?.status=='SUCCESS'">
|
<SvgIcon icon="ri:restart-line" @click="sendReload" class="cursor-pointer text-neutral-300 hover:text-neutral-800 dark:hover:text-neutral-300 " ></SvgIcon>
|
|
<div @click="getSeed(chat, message )" class="cursor-pointer">
|
<span v-if="chat.opt?.seed">Seed:{{ chat.opt?.seed }}</span>
|
<span v-else>Seed</span>
|
</div>
|
<!-- <a :href=" mjImgUrl(chat.opt?.imageUrl)" class="hidden group-hover:block active cursor-pointer underline " target="_blank">{{ $t('mj.ulink') }}</a> -->
|
</template>
|
</p>
|
|
<div class="flex items-end gap-1"
|
:class="[inversion ? 'flex-row-reverse' : 'flex-row']" >
|
<TextComponent
|
ref="textRef"
|
:inversion="inversion"
|
:error="error"
|
:text="text"
|
:loading="loading"
|
:as-raw-text="asRawText"
|
:chat="chat"
|
/>
|
<!-- <div class="flex flex-col" v-if="!chat.mjID && chat.model!='dall-e-3' && chat.model!='dall-e-2' "> -->
|
<div class="flex flex-col" v-if="!chat.mjID">
|
<!-- <button
|
v-if="!inversion "
|
class="mb-2 transition text-neutral-300 hover:text-neutral-800 dark:hover:text-neutral-300"
|
@click="handleRegenerate"
|
>
|
<SvgIcon icon="ri:restart-line" />
|
</button> -->
|
<button
|
v-if="!inversion "
|
class="mb-2 transition text-neutral-300 hover:text-neutral-800 dark:hover:text-neutral-300"
|
@click="handleRegenerate2"
|
>
|
<SvgIcon icon="ri:restart-line" />
|
</button>
|
<NDropdown
|
:trigger="isMobile ? 'click' : 'hover'"
|
:placement="!inversion ? 'right' : 'left'"
|
:options="options"
|
@select="handleSelect"
|
>
|
<button class="transition text-neutral-300 hover:text-neutral-800 dark:hover:text-neutral-200">
|
<SvgIcon icon="ri:more-2-fill" />
|
</button>
|
</NDropdown>
|
</div>
|
</div>
|
</div>
|
</div>
|
</template>
|