Files
yudao-ui-admin-vben/apps/web-antd/src/views/ai/write/index/components/Left.vue
2025-06-25 18:48:16 +08:00

248 lines
6.7 KiB
Vue
Raw Blame History

This file contains ambiguous Unicode characters
This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.
<script setup lang="ts">
import type { AiWriteApi } from '#/api/ai/write';
import { ref } from 'vue';
import { IconifyIcon } from '@vben/icons';
import { createReusableTemplate } from '@vueuse/core';
import { Button, message, Textarea } from 'ant-design-vue';
import {
AiWriteTypeEnum,
DICT_TYPE,
getDictOptions,
WriteExample,
} from '#/utils';
import Tag from './Tag.vue';
type TabType = AiWriteApi.Write['type'];
defineProps<{
isWriting: boolean;
}>();
const emit = defineEmits<{
(e: 'example', param: 'reply' | 'write'): void;
(e: 'reset'): void;
(e: 'submit', params: Partial<AiWriteApi.Write>): void;
}>();
function omit(obj: Record<string, any>, keysToOmit: string[]) {
const result: Record<string, any> = {};
for (const key in obj) {
if (!keysToOmit.includes(key)) {
result[key] = obj[key];
}
}
return result;
}
/** 点击示例的时候,将定义好的文章作为示例展示出来 */
function example(type: 'reply' | 'write') {
formData.value = {
...initData,
...omit(WriteExample[type], ['data']),
};
emit('example', type);
}
/** 重置,将表单值作为初选值 */
function reset() {
formData.value = { ...initData };
emit('reset');
}
const selectedTab = ref<TabType>(AiWriteTypeEnum.WRITING);
const tabs: {
text: string;
value: TabType;
}[] = [
{ text: '撰写', value: AiWriteTypeEnum.WRITING },
{ text: '回复', value: AiWriteTypeEnum.REPLY },
];
const [DefineTab, ReuseTab] = createReusableTemplate<{
active?: boolean;
itemClick: () => void;
text: string;
}>();
const [DefineLabel, ReuseLabel] = createReusableTemplate<{
class?: string;
hint?: string;
hintClick?: () => void;
label: string;
}>();
const initData: AiWriteApi.Write = {
type: 1,
prompt: '',
originalContent: '',
tone: 1,
language: 1,
length: 1,
format: 1,
};
const formData = ref<AiWriteApi.Write>({ ...initData });
/** 用来记录切换之前所填写的数据,切换的时候给赋值回来 */
const recordFormData = {} as Record<AiWriteTypeEnum, AiWriteApi.Write>;
/** 切换tab */
function switchTab(value: TabType) {
if (value !== selectedTab.value) {
// 保存之前的久数据
recordFormData[selectedTab.value] = formData.value;
selectedTab.value = value;
// 将之前的旧数据赋值回来
formData.value = { ...initData, ...recordFormData[value] };
}
}
/** 提交写作 */
function submit() {
if (selectedTab.value === 2 && !formData.value.originalContent) {
message.warning('请输入原文');
return;
}
if (!formData.value.prompt) {
message.warning(`请输入${selectedTab.value === 1 ? '写作' : '回复'}内容`);
return;
}
emit('submit', {
/** 撰写的时候没有 originalContent 字段*/
...(selectedTab.value === 1
? omit(formData.value, ['originalContent'])
: formData.value),
/** 使用选中 tab 值覆盖当前的 type 类型 */
type: selectedTab.value,
});
}
</script>
<template>
<DefineTab v-slot="{ active, text, itemClick }">
<span
:class="active ? 'text-black shadow-md' : 'hover:bg-gray-200'"
class="relative z-10 inline-block w-1/2 cursor-pointer rounded-full text-center leading-7 text-gray-400 hover:text-black"
@click="itemClick"
>
{{ text }}
</span>
</DefineTab>
<!-- 定义 label 组件长度/格式/语气/语言等 -->
<DefineLabel v-slot="{ label, hint, hintClick }">
<h3 class="mb-3 mt-5 flex items-center justify-between text-sm">
<span>{{ label }}</span>
<span
v-if="hint"
class="flex cursor-pointer select-none items-center text-xs text-purple-500"
@click="hintClick"
>
<IconifyIcon icon="lucide:circle-help" />
{{ hint }}
</span>
</h3>
</DefineLabel>
<div class="flex flex-col" v-bind="$attrs">
<div class="flex w-full justify-center bg-gray-50 pt-2">
<div class="z-10 w-72 rounded-full bg-gray-200 p-1">
<div
:class="
selectedTab === AiWriteTypeEnum.REPLY &&
'after:translate-x-[100%] after:transform'
"
class="relative flex items-center after:absolute after:left-0 after:top-0 after:block after:h-7 after:w-1/2 after:rounded-full after:bg-white after:transition-transform after:content-['']"
>
<ReuseTab
v-for="tab in tabs"
:key="tab.value"
:active="tab.value === selectedTab"
:item-click="() => switchTab(tab.value)"
:text="tab.text"
class="relative z-20"
/>
</div>
</div>
</div>
<div
class="box-border h-full w-96 flex-grow overflow-y-auto bg-gray-50 px-7 pb-2 lg:block"
>
<div>
<template v-if="selectedTab === 1">
<ReuseLabel
:hint-click="() => example('write')"
hint="示例"
label="写作内容"
/>
<Textarea
v-model:value="formData.prompt"
:maxlength="500"
:rows="5"
placeholder="请输入写作内容"
show-count
/>
</template>
<template v-else>
<ReuseLabel
:hint-click="() => example('reply')"
hint="示例"
label="原文"
/>
<Textarea
v-model:value="formData.originalContent"
:maxlength="500"
:rows="5"
placeholder="请输入原文"
show-count
/>
<ReuseLabel label="回复内容" />
<Textarea
v-model:value="formData.prompt"
:maxlength="500"
:rows="5"
placeholder="请输入回复内容"
show-count
/>
</template>
<ReuseLabel label="长度" />
<Tag
v-model="formData.length"
:tags="getDictOptions(DICT_TYPE.AI_WRITE_LENGTH, 'number')"
/>
<ReuseLabel label="格式" />
<Tag
v-model="formData.format"
:tags="getDictOptions(DICT_TYPE.AI_WRITE_FORMAT, 'number')"
/>
<ReuseLabel label="语气" />
<Tag
v-model="formData.tone"
:tags="getDictOptions(DICT_TYPE.AI_WRITE_TONE, 'number')"
/>
<ReuseLabel label="语言" />
<Tag
v-model="formData.language"
:tags="getDictOptions(DICT_TYPE.AI_WRITE_LANGUAGE, 'number')"
/>
<div class="mt-3 flex items-center justify-center">
<Button :disabled="isWriting" class="mr-2" @click="reset">
重置
</Button>
<Button
:loading="isWriting"
class="bg-purple-500 text-white"
@click="submit"
>
生成
</Button>
</div>
</div>
</div>
</div>
</template>