|
@@ -0,0 +1,256 @@
|
|
|
+<template>
|
|
|
+ <Dialog :title="dialogTitle" v-model="dialogVisible">
|
|
|
+ <el-form
|
|
|
+ ref="formRef"
|
|
|
+ :model="formData"
|
|
|
+ :rules="formRules"
|
|
|
+ label-width="120px"
|
|
|
+ v-loading="formLoading"
|
|
|
+ >
|
|
|
+ <el-form-item label="转账凭证附件" prop="attachment">
|
|
|
+ <!-- <el-input v-model="formData.attachment" placeholder="请输入转账凭证附件" /> -->
|
|
|
+ <UploadImg v-model="formData.attachment" :limit="1">
|
|
|
+ <!-- <template #tip>建议宽度 750px</template> -->
|
|
|
+ </UploadImg>
|
|
|
+ </el-form-item>
|
|
|
+ <el-form-item label="用户" prop="userId">
|
|
|
+ <el-select
|
|
|
+ v-model="formData.userId"
|
|
|
+ clearable
|
|
|
+ filterable
|
|
|
+ remote
|
|
|
+ reserve-keyword
|
|
|
+ placeholder="请输入用户名称"
|
|
|
+ :remote-method="debouncedRemoteMethod"
|
|
|
+ :loading="UserSearchLoading"
|
|
|
+ style="width: 240px"
|
|
|
+ >
|
|
|
+ <el-option
|
|
|
+ v-for="item in UserSearchOptions"
|
|
|
+ :key="item.value"
|
|
|
+ :label="item.label"
|
|
|
+ :value="item.value"
|
|
|
+ >
|
|
|
+ <div class="flex items-center">
|
|
|
+ <el-avatar :size="28" :src="item.avatar" />
|
|
|
+ <span class="m-l-2">{{ item.username }}({{ item.mobile }})</span>
|
|
|
+ </div>
|
|
|
+ </el-option>
|
|
|
+ </el-select>
|
|
|
+ </el-form-item>
|
|
|
+
|
|
|
+ <el-form-item label="充值金额" prop="consumptionPoints">
|
|
|
+ <el-input v-model.number="formData.consumptionPoints" placeholder="请输入充值金额" :disabled="!formData.userId" @input="(val) => formData.consumptionPoints = val.replace(/[^\d]/g, '')" :maxlength='9'>
|
|
|
+ <template #suffix>
|
|
|
+ <template v-if="!state.hide">
|
|
|
+ <span class="text-red-500" v-if="state.isFirstConsumption">该用户第一次充值至少需要{{state.minimumConsumptionPoints}}</span>
|
|
|
+ <span class="text-red-500" v-else>单次充值不低于{{state.minimumConsumptionPoints}}</span>
|
|
|
+ </template>
|
|
|
+ </template>
|
|
|
+ </el-input>
|
|
|
+ </el-form-item>
|
|
|
+ <el-form-item label="消费分" prop="">
|
|
|
+ {{consumptionPoints}}
|
|
|
+ </el-form-item>
|
|
|
+ <el-form-item label="手机号" prop="mobile">
|
|
|
+ <el-input v-model="formData.smsCodeUseReqDTO.mobile" placeholder="请输入手机号" :disabled="true">
|
|
|
+ <template #append>
|
|
|
+ <span
|
|
|
+ v-if="mobileCodeTimer <= 0"
|
|
|
+ class="getMobileCode"
|
|
|
+ style="cursor: pointer"
|
|
|
+ @click="getSmsCode"
|
|
|
+ >
|
|
|
+ {{ t('login.getSmsCode') }}
|
|
|
+ </span>
|
|
|
+ <span v-if="mobileCodeTimer > 0" class="getMobileCode" style="cursor: pointer">
|
|
|
+ {{ mobileCodeTimer }}秒后可重新获取
|
|
|
+ </span>
|
|
|
+ </template>
|
|
|
+ </el-input>
|
|
|
+ </el-form-item>
|
|
|
+ <el-form-item label="验证码" prop="smsCodeUseReqDTO.code">
|
|
|
+ <el-input v-model.number="formData.smsCodeUseReqDTO.code" placeholder="请输入验证码" max="4"/>
|
|
|
+ </el-form-item>
|
|
|
+
|
|
|
+ </el-form>
|
|
|
+ <template #footer>
|
|
|
+ <el-button @click="submitForm" type="primary" :disabled="formLoading">确定</el-button>
|
|
|
+ </template>
|
|
|
+ </Dialog>
|
|
|
+</template>
|
|
|
+<script setup lang="ts">
|
|
|
+import { ConsumptionTopUpLogApi, ConsumptionTopUpLogVO } from '@/api/system/distri/consumption'
|
|
|
+import * as UserApi from '@/api/member/user'
|
|
|
+import { getUserProfile } from '@/api/system/user/profile'
|
|
|
+import { debounce } from 'lodash-es'
|
|
|
+import { sendSmsCode } from '@/api/login'
|
|
|
+
|
|
|
+/** 平台消费分充值记录 表单 */
|
|
|
+defineOptions({ name: 'ConsumptionTopUpLogForm' })
|
|
|
+
|
|
|
+const { t } = useI18n() // 国际化
|
|
|
+const message = useMessage() // 消息弹窗
|
|
|
+
|
|
|
+const dialogVisible = ref(false) // 弹窗的是否展示
|
|
|
+const dialogTitle = ref('') // 弹窗的标题
|
|
|
+const formLoading = ref(false) // 表单的加载中:1)修改时的数据加载;2)提交的按钮禁用
|
|
|
+const UserSearchLoading = ref(false)
|
|
|
+const UserSearchList = ref([])
|
|
|
+const UserSearchOptions = ref([])
|
|
|
+const UserSearchValue = ref([])
|
|
|
+const formType = ref('') // 表单的类型:create - 新增;update - 修改
|
|
|
+const formData = reactive({
|
|
|
+ id: undefined,
|
|
|
+ userId: undefined,
|
|
|
+ attachment: undefined,
|
|
|
+ consumptionPoints: undefined,
|
|
|
+ practicalConsumptionPoints: undefined,
|
|
|
+ userName: undefined,
|
|
|
+ smsCodeUseReqDTO: {
|
|
|
+ mobile: '',
|
|
|
+ scene: '10',
|
|
|
+ code: '',
|
|
|
+ }
|
|
|
+})
|
|
|
+// 初始化或重置表单数据的函数
|
|
|
+const resetFormData = () => {
|
|
|
+ formData.id = undefined;
|
|
|
+ // formData.userId = undefined;
|
|
|
+ formData.attachment = undefined;
|
|
|
+ formData.consumptionPoints = undefined;
|
|
|
+ formData.practicalConsumptionPoints = undefined;
|
|
|
+ formData.userName = undefined;
|
|
|
+ formData.smsCodeUseReqDTO = {
|
|
|
+ mobile: '',
|
|
|
+ scene: '10', // 默认值
|
|
|
+ code: '',
|
|
|
+ }
|
|
|
+}
|
|
|
+const formRules = reactive({
|
|
|
+ consumptionPoints: [{ required: true, message: '消费分不能为空', trigger: 'blur' }],
|
|
|
+ userId: [{ required: true, message: '用户不能为空', trigger: 'blur' }],
|
|
|
+ attachment: [{ required: true, message: '转账凭证附件不能为空', trigger: 'blur' }],
|
|
|
+ "smsCodeUseReqDTO.code": [{ required: true, message: '验证码不能为空', trigger: 'blur' }]
|
|
|
+})
|
|
|
+const formRef = ref() // 表单 Ref
|
|
|
+
|
|
|
+/** 打开弹窗 */
|
|
|
+const open = async (type: string) => {
|
|
|
+ dialogVisible.value = true
|
|
|
+ dialogTitle.value = t('action.' + type)
|
|
|
+ formType.value = type
|
|
|
+ resetFormData()
|
|
|
+ await getUserInfo()
|
|
|
+
|
|
|
+}
|
|
|
+defineExpose({ open }) // 提供 open 方法,用于打开弹窗
|
|
|
+
|
|
|
+/** 提交表单 */
|
|
|
+const emit = defineEmits(['success']) // 定义 success 事件,用于操作成功后的回调
|
|
|
+const submitForm = async () => {
|
|
|
+ console.log(formData)
|
|
|
+ // 校验表单
|
|
|
+ await formRef.value.validate()
|
|
|
+ // 提交请求
|
|
|
+ formLoading.value = true
|
|
|
+ try {
|
|
|
+ await ConsumptionTopUpLogApi.createConsumptionTopUpLog(formData)
|
|
|
+ message.success(t('common.createSuccess'))
|
|
|
+
|
|
|
+ dialogVisible.value = false
|
|
|
+ // 发送操作成功的事件
|
|
|
+ emit('success')
|
|
|
+ } finally {
|
|
|
+ formLoading.value = false
|
|
|
+ }
|
|
|
+}
|
|
|
+
|
|
|
+// 实际请求方法
|
|
|
+const remoteMethod = async (query: string) => {
|
|
|
+ if (query) {
|
|
|
+ UserSearchLoading.value = true
|
|
|
+ try {
|
|
|
+ const response = await UserApi.getUserListByUsername({ username: query })
|
|
|
+ // console.log(response)
|
|
|
+ UserSearchOptions.value = response.map((item: any) => ({
|
|
|
+ value: item.id,
|
|
|
+ label: item.username,
|
|
|
+ mobile: item.mobile,
|
|
|
+ avatar: item.avatar,
|
|
|
+ username: item.username
|
|
|
+ }))
|
|
|
+ } catch (error) {
|
|
|
+ console.error('Error fetching data:', error)
|
|
|
+ } finally {
|
|
|
+ UserSearchLoading.value = false
|
|
|
+ }
|
|
|
+ } else {
|
|
|
+ }
|
|
|
+}
|
|
|
+// 使用 lodash 防抖,延迟 200ms 触发请求
|
|
|
+const debouncedRemoteMethod = debounce(remoteMethod, 200)
|
|
|
+
|
|
|
+const state = reactive({
|
|
|
+ hide: true,
|
|
|
+ isFirstConsumption:false,
|
|
|
+ minimumConsumptionPoints:0,
|
|
|
+ consumptionMagnification:''
|
|
|
+})
|
|
|
+// 根据 value 查找 username
|
|
|
+const findUsernameByValue = (value) => {
|
|
|
+ const user = UserSearchOptions.value.find(item => item.value === value);
|
|
|
+ return user ? user.username : null;
|
|
|
+};
|
|
|
+async function isUserFirstRecharge(userId: number) {
|
|
|
+ state.hide = false
|
|
|
+ const data = await ConsumptionTopUpLogApi.isUserFirstRecharge({ userId })
|
|
|
+ state.isFirstConsumption = data.isFirstConsumption
|
|
|
+ state.minimumConsumptionPoints = data.minimumConsumptionPoints
|
|
|
+ state.consumptionMagnification = data.consumptionMagnification
|
|
|
+}
|
|
|
+// 监听 el-select 选项值的变化
|
|
|
+watch(
|
|
|
+ () => formData.userId,
|
|
|
+ (newValue) => {
|
|
|
+ isUserFirstRecharge(newValue);
|
|
|
+ formData.userName = findUsernameByValue(newValue);
|
|
|
+ }
|
|
|
+)
|
|
|
+const consumptionPoints = computed(() => {
|
|
|
+ if(!formData.consumptionPoints){
|
|
|
+ return 0
|
|
|
+ }
|
|
|
+ const point = formData.consumptionPoints * parseFloat(state.consumptionMagnification)
|
|
|
+ console.log(formData.consumptionPoints,state.consumptionMagnification,point)
|
|
|
+ return point.toFixed(2)
|
|
|
+})
|
|
|
+
|
|
|
+const smsVO = reactive({
|
|
|
+ smsCode: {
|
|
|
+ mobile: '',
|
|
|
+ scene: 10
|
|
|
+ }
|
|
|
+})
|
|
|
+const mobileCodeTimer = ref(0)
|
|
|
+const getSmsCode = async () => {
|
|
|
+ smsVO.smsCode.mobile = formData.smsCodeUseReqDTO.mobile
|
|
|
+ await sendSmsCode(smsVO.smsCode).then(async () => {
|
|
|
+ message.success(t('login.SmsSendMsg'))
|
|
|
+ // 设置倒计时
|
|
|
+ mobileCodeTimer.value = 60
|
|
|
+ let msgTimer = setInterval(() => {
|
|
|
+ mobileCodeTimer.value = mobileCodeTimer.value - 1
|
|
|
+ if (mobileCodeTimer.value <= 0) {
|
|
|
+ clearInterval(msgTimer)
|
|
|
+ }
|
|
|
+ }, 1000)
|
|
|
+ })
|
|
|
+}
|
|
|
+
|
|
|
+// 获取当前登录用户的手机号并且回填
|
|
|
+const getUserInfo = async () => {
|
|
|
+ const users = await getUserProfile()
|
|
|
+ formData.smsCodeUseReqDTO.mobile = users.mobile
|
|
|
+}
|
|
|
+</script>
|