3 changed files with 281 additions and 87 deletions
			
			
		| @ -1,75 +1,164 @@ | |||||
| <template> | <template> | ||||
|     <div class="flex"> |   <div class="phone-code-input"> | ||||
|       <GuipInput |      <!-- 手机号输入框 + 发送按钮 --> | ||||
|         ref="phoneNumber" |      <GuipFormItem column="column" addClass="w388"> | ||||
|         width="277px" |       <div slot="formLeft" class="form-top-icon"> | ||||
|         height="38px" |         <img src="../assets/franchise1/storeNum.svg" alt=""> | ||||
|         placeholder="请输入手机号码"/> |         <span>手机号码</span> | ||||
|       <GuipButton type="reverseMild" @click="sendCode" :btnstyle="{width:'99px',height:'38px'}" :disabled="isCounting">{{ countdownText }}</GuipButton> |       </div> | ||||
|     </div> |       <el-form-item slot="formDom" :prop="phoneProp" :rules="phoneRules" style="display: flex;justify-content: space-between;"> | ||||
|   </template> |         <el-input v-model="phone"     @input="validateInput" | ||||
|  |         placeholder="请输入手机号" style="width: 277px;" clearable @blur="validatePhone"> | ||||
|  |         </el-input> | ||||
|  |         <GuipButton type="reverseMild" :btnstyle="{width:'99px',height:'38px'}" :disabled="!canSendCode" @click="sendCode" class="send-btn"> | ||||
|  |           {{ countdown > 0 ? `${countdown}s后重试` : '获取验证码' }}</GuipButton> | ||||
|  |       </el-form-item> | ||||
|  |     </GuipFormItem> | ||||
|  |     <!-- 验证码输入框 --> | ||||
|  |     <GuipFormItem column="column" addClass="w388"> | ||||
|  |       <div slot="formLeft" class="form-top-icon"> | ||||
|  |         <img src="../assets/franchise1/codeNumber.svg" alt=""> | ||||
|  |         <span>验证码</span> | ||||
|  |       </div> | ||||
|  |       <el-form-item slot="formDom" :prop="codeProp" :rules="codeRules"> | ||||
|  |         <div class="code-input-wrapper"> | ||||
|  |           <el-input v-model="code" placeholder="请输入验证码" clearable @blur="validateCode"></el-input> | ||||
|  |         </div> | ||||
|  |       </el-form-item> | ||||
|  |     </GuipFormItem> | ||||
|  |   </div> | ||||
|  | </template> | ||||
| 
 | 
 | ||||
|   <script> | <script> | ||||
| import axios from 'axios'; | import GuipFormItem from '@/components/GuipFormItem.vue'; | ||||
| import GuipButton from '@/components/GuipButton.vue'; | import GuipButton from '@/components/GuipButton.vue'; | ||||
| import GuipInput from '@/components/GuipInput.vue'; |  | ||||
| 
 | 
 | ||||
|   export default { | export default { | ||||
|     data() { |   name: "PhoneCodeInput", | ||||
|       return { |   components:{ | ||||
|         phoneValue: '', |     GuipButton, | ||||
|         countdown: 60, // 倒计时时间(秒) |     GuipFormItem, | ||||
|         isCounting: false, // 是否正在倒计时 |   }, | ||||
|       }; |   props: { | ||||
|  |     // 用于 Form 校验的 prop(如 'phone' 和 'code') | ||||
|  |     phoneProp: { | ||||
|  |       type: String, | ||||
|  |       default: "phone", | ||||
|     }, |     }, | ||||
|     components: { |     codeProp: { | ||||
|         GuipButton, |       type: String, | ||||
|         GuipInput, |       default: "code", | ||||
|     }, |     }, | ||||
|     computed: { |     // 初始值(v-model) | ||||
|       countdownText() { |     value: { | ||||
|         return this.isCounting ? `${this.countdown}秒后重试` : '获取验证码'; |       type: Object, | ||||
|       } |       default: () => ({ | ||||
|  |         phone: "", | ||||
|  |         code: "", | ||||
|  |       }), | ||||
|  |     }, | ||||
|  |     // 自定义校验规则(可选) | ||||
|  |     phoneRules: { | ||||
|  |       type: Array, | ||||
|  |       default: () => [ | ||||
|  |         { required: true, message: "请输入手机号", trigger: "blur" }, | ||||
|  |         { pattern: /^1[3-9]\d{9}$/, message: "手机号格式错误", trigger: "blur" }, | ||||
|  |       ], | ||||
|  |     }, | ||||
|  |     codeRules: { | ||||
|  |       type: Array, | ||||
|  |       default: () => [ | ||||
|  |         { required: true, message: "请输入验证码", trigger: "blur" }, | ||||
|  |         { pattern: /^\d{6}$/, message: "验证码必须是6位数字", trigger: "blur" }, | ||||
|  |       ], | ||||
|  |     }, | ||||
|  |   }, | ||||
|  |   data() { | ||||
|  |     return { | ||||
|  |       phone: this.value.phone, | ||||
|  |       code: this.value.code, | ||||
|  |       countdown: 0, // 倒计时 | ||||
|  |       timer: null, // 定时器 | ||||
|  |     }; | ||||
|  |   }, | ||||
|  |   computed: { | ||||
|  |     // 是否可以发送验证码(手机号格式正确且未在倒计时) | ||||
|  |     canSendCode() { | ||||
|  |       return /^1[3-9]\d{9}$/.test(this.phone) && this.countdown === 0; | ||||
|  |     }, | ||||
|  |   }, | ||||
|  |   watch: { | ||||
|  |     // 监听内部数据变化,同步到父组件 | ||||
|  |     phone(newVal) { | ||||
|  |       this.$emit("input", { ...this.value, phone: newVal }); | ||||
|  |     }, | ||||
|  |     code(newVal) { | ||||
|  |       this.$emit("input", { ...this.value, code: newVal }); | ||||
|  |     }, | ||||
|  |     // 监听外部 v-model 变化,更新内部数据 | ||||
|  |     value: { | ||||
|  |       deep: true, | ||||
|  |       handler(newVal) { | ||||
|  |         this.phone = newVal.phone; | ||||
|  |         this.code = newVal.code; | ||||
|  |       }, | ||||
|  |     }, | ||||
|  |   }, | ||||
|  |   methods: { | ||||
|  |     // 校验手机号 | ||||
|  |     validatePhone() { | ||||
|  |       console.log('ooooo'); | ||||
|  |       // formRef  固定名称 | ||||
|  |       this.$refs.formRef?.validateField(this.phoneProp); | ||||
|     }, |     }, | ||||
|     methods: { |     validateInput() { | ||||
|         async sendCode() { |       // formRef  固定名称 | ||||
|             if (!/^1\d{10}$/.test(this.$refs.phoneNumber.inputValue)) { |       // this.$refs.formRef?.validateField(this.phoneProp); | ||||
|                 this.$message.warning('请输入有效的手机号码'); |       if (/^1[3-9]\d{9}$/.test(this.phone)) { | ||||
|                 return; |         this.$refs.formRef?.validateField(this.phoneProp, (errorMsg) => { | ||||
|             } |           if (!errorMsg) { | ||||
|             this.phoneValue = this.$refs.phoneNumber.inputValue; |             console.log('-----00000'); | ||||
|             // 根据实际接口请求 |             // 校验通过,错误提示会自动消失 | ||||
|             try { |           }else{ | ||||
|                 const response = await axios.post('/agentnew/ajax_send_verify_code', { phoneNumber: this.$refs.phoneNumber.value }); // 替换为实际API地址和参数名 |             console.log('776767676'); | ||||
|                 if (response.data.status) { // 根据后端返回的结构调整这里的判断条件 |  | ||||
|                     this.$message.success('验证码已发送'); |  | ||||
|                     this.startCountdown(); |  | ||||
|                 } else { |  | ||||
|                     throw new Error(response.data.info || '发送失败'); // 使用后端返回的错误信息或者默认信息 |  | ||||
|                 } |  | ||||
|             } catch (error) { |  | ||||
|                 console.error('发送验证码失败:', error); // 处理错误信息,可以在这里做更多的错误处理逻辑,比如重试机制等。 |  | ||||
|                 this.$message.warning('发送验证码失败,请稍后再试'); |  | ||||
|             } |  | ||||
|         }, |  | ||||
|       startCountdown() { |  | ||||
|         this.isCounting = true; |  | ||||
|         const interval = setInterval(() => { |  | ||||
|           this.countdown--; |  | ||||
|           if (this.countdown <= 0) { |  | ||||
|             clearInterval(interval); |  | ||||
|             this.isCounting = false; |  | ||||
|             this.countdown = 60; // 重置倒计时时间 |  | ||||
|           } |           } | ||||
|         }, 1000); |         }); | ||||
|       } |       } | ||||
|     } |     }, | ||||
|   }; |     // 校验验证码 | ||||
|   </script> |     validateCode() { | ||||
|  |       this.$refs.formRef?.validateField(this.codeProp); | ||||
|  |     }, | ||||
|  |     // 发送验证码 | ||||
|  |     sendCode() { | ||||
|  |       if (!this.canSendCode) return; | ||||
|  | 
 | ||||
|  |       // 模拟发送验证码(实际调用 API) | ||||
|        |        | ||||
|   <style lang="scss" scoped> |       this.startCountdown() | ||||
|     .flex{ | 
 | ||||
|         display: flex; |       this.$emit("send-code", this.phone); | ||||
|         justify-content: space-between; |     }, | ||||
|  |     startCountdown() { | ||||
|  |       this.countdown = 60; | ||||
|  |       this.timer = setInterval(() => { | ||||
|  |         this.countdown--; | ||||
|  |         if (this.countdown <= 0) { | ||||
|  |           clearInterval(this.timer); | ||||
|  |         } | ||||
|  |       }, 1000); | ||||
|     } |     } | ||||
|     </style> |   }, | ||||
|  |   beforeDestroy() { | ||||
|  |     if (this.timer) clearInterval(this.timer); | ||||
|  |   }, | ||||
|  | }; | ||||
|  | </script> | ||||
|  | 
 | ||||
|  | <style scoped> | ||||
|  | .code-input-wrapper { | ||||
|  |   display: flex; | ||||
|  |   gap: 10px; | ||||
|  |   flex: 1; | ||||
|  | } | ||||
|  | </style> | ||||
					Loading…
					
					
				
		Reference in new issue