EditForm.vue 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. <script setup lang="ts">
  2. import { ElDialog, ElForm, ElFormItem, ElInput, ElDatePicker, ElButton, ElMessage, ElRadio } from 'element-plus'
  3. import type { FormInstance, FormRules } from 'element-plus'
  4. import FaImageUpload from '@/ui/components/FaImageUpload/index.vue'
  5. import VoiceSelector from '@/components/VoiceSelector.vue'
  6. import LangSelector from '@/components/LangSelector.vue'
  7. import NationalitySelector from '@/components/NationalitySelector.vue'
  8. import { saveRole } from '@/api/modules/anycallService'
  9. import type { TSaveRoleParams } from '@/api/modules/anycallService'
  10. // 定义表单数据类型
  11. type IFormData = {
  12. id?: string
  13. name: string
  14. photo: string
  15. prompt: string
  16. language: string
  17. languageCode: string
  18. area: string
  19. areaCode: string
  20. gender: number
  21. description: string
  22. voiceId: string
  23. voiceName: string
  24. }
  25. // 定义组件的属性
  26. interface Props {
  27. visible: boolean
  28. modelValue?: IFormData | null
  29. mode?: 'create' | 'edit'
  30. }
  31. // 定义组件的事件
  32. interface Emits {
  33. (e: 'update:visible', value: boolean): void
  34. (e: 'cancel'): void
  35. (e: 'refresh'): void
  36. }
  37. // 接收属性和事件
  38. const props = withDefaults(defineProps<Props>(), {
  39. visible: false,
  40. modelValue: null,
  41. mode: 'create'
  42. })
  43. const avatars = ref<string[]>([])
  44. const emit = defineEmits<Emits>()
  45. // 创建本地响应式变量用于 v-model 绑定
  46. const dialogVisible = ref(props.visible)
  47. // 监听 props.visible 变化,更新本地变量
  48. watch(() => props.visible, (newVisible) => {
  49. dialogVisible.value = newVisible
  50. })
  51. // 监听本地变量变化,发出 update 事件
  52. watch(dialogVisible, (newVisible) => {
  53. emit('update:visible', newVisible)
  54. })
  55. // 表单引用
  56. const editFormRef = ref<FormInstance>();
  57. const currentSelected = ref<any>(null)
  58. // 响应式数据 - 直接定义所有必需字段
  59. const formData = ref<Partial<IFormData> & {voiceId: string, voiceName: string}>({
  60. name: '',
  61. photo: '',
  62. prompt: '',
  63. language: '',
  64. languageCode: '',
  65. area: '',
  66. areaCode: '',
  67. description: '',
  68. gender: 1,
  69. voiceId: '',
  70. voiceName: '',
  71. });
  72. // 监听 props.modelValue 变化
  73. watch(() => props.modelValue, (newModelValue) => {
  74. if (newModelValue) {
  75. formData.value = {
  76. ...formData.value,
  77. ...newModelValue
  78. };
  79. avatars.value = newModelValue.photo ? [newModelValue.photo] : []
  80. }
  81. });
  82. // 表单验证规则
  83. const formRules = ref<FormRules>({
  84. name: [
  85. { required: true, message: '请输入名称', trigger: 'blur' },
  86. ],
  87. // prompt: [
  88. // { required: true, message: '请输入 prompt', trigger: 'blur' },
  89. // ],
  90. description: [
  91. { required: true, message: '请输入描述', trigger: 'blur' },
  92. ],
  93. // language: [
  94. // { required: true, message: '请输入语言', trigger: 'blur' },
  95. // ],
  96. })
  97. // 监听可见性变化
  98. watch(() => props.visible, (newVisible) => {
  99. if(!newVisible){
  100. currentSelected.value = null
  101. resetForm()
  102. }
  103. })
  104. // 重置表单
  105. function resetForm() {
  106. console.log('reset')
  107. if (editFormRef.value) {
  108. editFormRef.value.resetFields()
  109. }
  110. avatars.value = [],
  111. formData.value = {
  112. photo: '',
  113. name: '',
  114. prompt: '',
  115. description: '',
  116. language: '',
  117. languageCode: '',
  118. area: '',
  119. areaCode: '',
  120. gender: 1,
  121. voiceId: '',
  122. voiceName: '',
  123. }
  124. }
  125. // 处理确认
  126. async function handleConfirm() {
  127. if (!editFormRef.value) return
  128. try {
  129. await editFormRef.value.validate()
  130. const defaultData = {
  131. // Matches TSaveRoleParams
  132. id: formData.value.id,
  133. name: formData.value.name ?? '',
  134. voiceId: formData.value.voiceId ?? '',
  135. language: formData.value.language ?? '',
  136. languageCode: formData.value.languageCode ?? '',
  137. area: formData.value.area ?? '',
  138. areaCode: formData.value.areaCode ?? '',
  139. description: formData.value.description ?? '',
  140. prompt: formData.value.prompt ?? '',
  141. photo: '',
  142. gender: formData.value.gender ?? 1,
  143. }
  144. const avatar = avatars.value?.[0] ?? ''
  145. // 字段全后再对齐字段
  146. //@ts-ignore
  147. const d: TSaveRoleParams = { ...defaultData, photo: avatar }
  148. console.log(d)
  149. const { code } = await saveRole(d)
  150. if (code === 0) {
  151. ElMessage.success('创建成功')
  152. emit('refresh')
  153. }
  154. // 关闭对话框
  155. emit('update:visible', false)
  156. } catch (error) {
  157. // 验证失败,不做处理
  158. }
  159. }
  160. function handleCancel() {
  161. emit('cancel')
  162. emit('update:visible', false)
  163. }
  164. function handleClose() {
  165. emit('update:visible', false)
  166. }
  167. const handleSelectorConfirm = (selected: any) => {
  168. console.log(selected)
  169. if (selected) {
  170. }
  171. }
  172. const handleRemoveAgent = ()=> {
  173. currentSelected.value = null
  174. }
  175. </script>
  176. <template>
  177. <div>
  178. <ElDialog :title="mode === 'create' ? '创建角色' : '编辑角色'" v-model="dialogVisible" align-center @close="handleClose"
  179. width="800" :z-index="2000">
  180. <ElForm ref="editFormRef" :model="formData" :rules="formRules" label-width="120px" class="mt-4 space-y-4 w-full">
  181. <ElFormItem label="name" prop="name" label-width="120">
  182. <ElInput v-model="formData.name" placeholder="Enter name" />
  183. </ElFormItem>
  184. <ElFormItem label="性别" prop="gender">
  185. <ElRadioGroup v-model="formData.gender">
  186. <ElRadio :value="1">男</ElRadio>
  187. <ElRadio :value="2">女</ElRadio>
  188. <ElRadio :value="3">其他</ElRadio>
  189. </ElRadioGroup>
  190. </ElFormItem>
  191. <ElFormItem label="prompt" prop="prompt" label-width="120">
  192. <ElInput type="textarea" v-model="formData.prompt" placeholder="Enter prompt" />
  193. </ElFormItem>
  194. <ElFormItem label="description" prop="description" label-width="120">
  195. <ElInput type="textarea" v-model="formData.description" placeholder="Enter description" />
  196. </ElFormItem>
  197. <ElFormItem label="音色" prop="voice" label-width="120">
  198. <VoiceSelector :gender="formData.gender ?? 1" v-model="formData.voiceId" v-model:voice-name="formData.voiceName" />
  199. </ElFormItem>
  200. <ElFormItem label="Avatar" prop="photo" label-width="120">
  201. <FaImageUpload
  202. v-model="avatars"
  203. :max-count="1"
  204. list-type="avatar"
  205. />
  206. </ElFormItem>
  207. <ElFormItem label="区域" prop="区域">
  208. <NationalitySelector v-model="formData.areaCode" v-model:key="formData.area" placeholder="区域"></NationalitySelector>
  209. </ElFormItem>
  210. <ElFormItem label="语言" prop="language">
  211. <LangSelector v-model="formData.languageCode" v-model:key="formData.language" placeholder="请选择语言"></LangSelector>
  212. </ElFormItem>
  213. </ElForm>
  214. <template #footer>
  215. <div class="flex justify-end space-x-2">
  216. <ElButton @click="handleCancel">取消</ElButton>
  217. <ElButton type="primary" @click="handleConfirm">确定</ElButton>
  218. </div>
  219. </template>
  220. </ElDialog>
  221. </div>
  222. </template>