|
@@ -0,0 +1,353 @@
|
|
|
+<template>
|
|
|
+ <div class="graphic-verification-code-container">
|
|
|
+ <div style="display: flex;align-items: center;justify-content: space-between;">
|
|
|
+ <el-input
|
|
|
+ :value="inputCode" maxlength="6" type="text" auto-complete="off"
|
|
|
+ style="width: 63%;"
|
|
|
+ placeholder="请输入验证码" @input="handleCodeInput" @keyup.enter.native="$emit('enter-down')"
|
|
|
+ >
|
|
|
+ <svg-icon slot="prefix" icon-class="password" class="el-input__icon input-icon" />
|
|
|
+ </el-input>
|
|
|
+ <div>
|
|
|
+ <el-button type="primary" :loading="codeloading" @click="handleValidateImageInit">
|
|
|
+ <span v-if="!codeloading">获取验证码</span>
|
|
|
+ <span v-else>{{ count }} s</span>
|
|
|
+ </el-button>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <el-dialog
|
|
|
+ :visible.sync="verificationVisible" v-bind="{ closeOnClickModal: false, width: '600px', title: '滑动码验证' }"
|
|
|
+ append-to-body
|
|
|
+ >
|
|
|
+ <div>
|
|
|
+ <div class="comImageValidate rightValidate">
|
|
|
+ <div ref="imgBg_2" class="imgBg_2"></div>
|
|
|
+ <div ref="imgBg_3" class="imgBg_3"></div>
|
|
|
+ <div style="width: 100%; height: 30px;"></div>
|
|
|
+ <div ref="imgBg" class="imgBg" @mouseover="handleImgBgMouseover" @mouseleave="handleImgBgMouseleave">
|
|
|
+ <div ref="imgBtn" class="imgBtn">
|
|
|
+ <img ref="imgBtnImg" alt="" src="">
|
|
|
+ </div>
|
|
|
+ <span ref="refresh" class="refresh" @click="handleValidateImageInit">
|
|
|
+ <img alt="" src="">
|
|
|
+ </span>
|
|
|
+ </div>
|
|
|
+ <div
|
|
|
+ ref="hkinnerWrap" class="hkinnerWrap" style="height:30px; position: relative"
|
|
|
+ @mouseover="handleHkinnerWrapMouseover" @mouseleave="handleHkinnerWrapMouseleave"
|
|
|
+ @mousemove="handleHkinnerWrapMousemove" @mouseup="handleHkinnerWrapMouseup"
|
|
|
+ >
|
|
|
+ <span
|
|
|
+ ref="v_rightBtn" class="v_rightBtn " @mousedown="handleVRightBtnMousedown"
|
|
|
+ @mouseup="handleVRightBtnMouseup"
|
|
|
+ >
|
|
|
+ <em class="notSel">→</em>
|
|
|
+ </span>
|
|
|
+ <span
|
|
|
+ ref="huakuai" class="huakuai"
|
|
|
+ style="font-size: 12px;line-height: 33px;color: #A9A9A9;"
|
|
|
+ >
|
|
|
+ 向右滑动滑块填充拼图完成验证
|
|
|
+ </span>
|
|
|
+ <input type="hidden" name="validX" />
|
|
|
+ </div>
|
|
|
+
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ </el-dialog>
|
|
|
+ </div>
|
|
|
+</template>
|
|
|
+
|
|
|
+<script>
|
|
|
+const TIME_COUNT = 60 // 更改倒计时时间
|
|
|
+import { getCode, getVerificationImageCaptchaApi } from '@/api/account'
|
|
|
+
|
|
|
+export default {
|
|
|
+ name: 'GraphicVerificationCode',
|
|
|
+ props: {
|
|
|
+ phone: {
|
|
|
+ type: [String, Number],
|
|
|
+ default: ''
|
|
|
+ }
|
|
|
+ },
|
|
|
+ data() {
|
|
|
+ return {
|
|
|
+ inputCode: '',
|
|
|
+ codeloading: false,
|
|
|
+ count: '',
|
|
|
+ timer: null,
|
|
|
+ verificationVisible: false,
|
|
|
+ isOpenHkinnerWrapMouse: false,
|
|
|
+ dx: 0,
|
|
|
+ slideOptions: {
|
|
|
+ sliderKey: '',
|
|
|
+ sliderX: '',
|
|
|
+ sliderY: ''
|
|
|
+ }
|
|
|
+ }
|
|
|
+ },
|
|
|
+ mounted() { },
|
|
|
+ methods: {
|
|
|
+ handleCodeInput(e) {
|
|
|
+ console.log(e)
|
|
|
+ this.inputCode = e
|
|
|
+ this.$emit('input', this.inputCode)
|
|
|
+ },
|
|
|
+ // 获取验证码
|
|
|
+ async getVerificationCode() {
|
|
|
+ if (!this.phone) {
|
|
|
+ this.$message.error('请填写电话号码')
|
|
|
+ return
|
|
|
+ }
|
|
|
+ try {
|
|
|
+ const res = await getCode({ phone: this.phone, x: this.slideOptions.sliderX, y: this.slideOptions.sliderY })
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.addClass('green')
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('red')
|
|
|
+ setTimeout(() => {
|
|
|
+ this.verificationVisible = false
|
|
|
+ this.$message({
|
|
|
+ message: '发送成功,请注意查看手机短信',
|
|
|
+ type: 'success'
|
|
|
+ })
|
|
|
+ if (!this.timer) {
|
|
|
+ this.codeloading = true
|
|
|
+ this.count = TIME_COUNT
|
|
|
+ this.timer = setInterval(() => {
|
|
|
+ if (this.count > 1 && this.count <= TIME_COUNT) {
|
|
|
+ this.count--
|
|
|
+ } else {
|
|
|
+ clearInterval(this.timer) // 清除定时器
|
|
|
+ this.timer = null
|
|
|
+ this.codeloading = false
|
|
|
+ }
|
|
|
+ }, 1000)
|
|
|
+ }
|
|
|
+ }, 1000)
|
|
|
+ } catch (e) {
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.addClass('green')
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('red')
|
|
|
+ setTimeout(() => {
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('red')
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('green')
|
|
|
+ this.$refs.v_rightBtn.style.left = 0
|
|
|
+ this.$refs.imgBtn.style.left = 0
|
|
|
+ this.handleValidateImageInit()
|
|
|
+ }, 500)
|
|
|
+ }
|
|
|
+ },
|
|
|
+
|
|
|
+ handleHkinnerWrapMouseover() {
|
|
|
+ this.$refs.imgBg && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ this.$refs.refresh && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ },
|
|
|
+ handleHkinnerWrapMouseleave() {
|
|
|
+ this.$refs.imgBg && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ this.$refs.refresh && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ },
|
|
|
+ handleHkinnerWrapMousemove(e) {
|
|
|
+ if (this.isOpenHkinnerWrapMouse) {
|
|
|
+ const newLeft = e.pageX - this.dx
|
|
|
+ this.$refs.v_rightBtn.offsetLeft = newLeft
|
|
|
+ let newL = parseInt(this.$refs.v_rightBtn.style.left)
|
|
|
+ if (newL <= 0) {
|
|
|
+ newL = 0
|
|
|
+ } else if (newL >= 240) {
|
|
|
+ newL = 240
|
|
|
+ }
|
|
|
+ this.$refs.v_rightBtn.style.left = newL + 'px'
|
|
|
+ this.$refs.imgBtn.offsetLeft = newLeft
|
|
|
+ this.$refs.imgBtn.style.left = newL + 'px'
|
|
|
+ }
|
|
|
+ },
|
|
|
+ handleHkinnerWrapMouseup() {
|
|
|
+ if (this.isOpenHkinnerWrapMouse) {
|
|
|
+ this.isOpenHkinnerWrapMouse = false
|
|
|
+ }
|
|
|
+ },
|
|
|
+ handleImgBgMouseover() {
|
|
|
+ this.$refs.imgBg && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ this.$refs.refresh && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ },
|
|
|
+ handleImgBgMouseleave() {
|
|
|
+ this.$refs.imgBg && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ this.$refs.refresh && (this.$refs.imgBg.style.display = 'block')
|
|
|
+ },
|
|
|
+ handleVRightBtnMousedown(e) {
|
|
|
+ this.$refs.huakuai && (this.$refs.huakuai.innerHTML = '')
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('red')
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('green')
|
|
|
+ this.dx = e.pageX - this.$refs.v_rightBtn.offsetLeft
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('green')
|
|
|
+ this.isOpenHkinnerWrapMouse = true
|
|
|
+ },
|
|
|
+ handleVRightBtnMouseup() {
|
|
|
+ this.isOpenHkinnerWrapMouse = false
|
|
|
+ this.slideOptions.sliderX = this.$refs.v_rightBtn.style.left.replace('px', '')
|
|
|
+ this.getVerificationCode()
|
|
|
+ },
|
|
|
+
|
|
|
+ async handleValidateImageInit() {
|
|
|
+ if (!this.phone) {
|
|
|
+ this.$message.error('请填写电话号码')
|
|
|
+ return
|
|
|
+ }
|
|
|
+ // try {
|
|
|
+ this.verificationVisible = true
|
|
|
+ // const res = await getVerificationImageCaptchaApi({ type: 'slide' })
|
|
|
+ const res = {
|
|
|
+ data: {
|
|
|
+ 'type': 'slide',
|
|
|
+ 'charImage': '',
|
|
|
+ 'operationImage': '',
|
|
|
+ 'originImage': '/Z',
|
|
|
+ 'shadeImage': 'iVBmCC',
|
|
|
+ 'cutoutImage': 'iVgg==',
|
|
|
+ 'x': 0,
|
|
|
+ 'y': 67
|
|
|
+ }
|
|
|
+ }
|
|
|
+ switch (res.data.type) {
|
|
|
+ case 'operation': this.initOperationVerificationCode(res.data); break
|
|
|
+ case 'char': this.initCharVerificationCode(res.data); break
|
|
|
+ case 'slide': this.initSlideVerificationCode(res.data); break
|
|
|
+ default: console.log('验证码错误'); break
|
|
|
+ }
|
|
|
+ // } catch (e) {
|
|
|
+ // console.log(e)
|
|
|
+ // }
|
|
|
+ },
|
|
|
+ initOperationVerificationCode(data) {
|
|
|
+ this.$refs.imgBg_3 && (this.$refs.imgBg_3.style.background = '#fff url("data:image/jpg;base64,' + data.operationImage + '")')
|
|
|
+ },
|
|
|
+ initCharVerificationCode(data) {
|
|
|
+ this.$refs.imgBg_2 && (this.$refs.imgBg_2.style.background = '#fff url("data:image/jpg;base64,' + data.charImage + '")')
|
|
|
+ this.$refs.imgBg_3 && (this.$refs.imgBg_3.style.display = 'none')
|
|
|
+ },
|
|
|
+ initSlideVerificationCode(data) {
|
|
|
+ this.slideOptions.sliderY = data.y
|
|
|
+ this.$refs.huakuai && (this.$refs.huakuai.innerHTML = '向右滑动滑块填充拼图')
|
|
|
+ this.$refs.imgBg && (this.$refs.imgBg.style.background = '#fff url("data:image/jpg;base64,' + data.shadeImage + '")')
|
|
|
+ this.$refs.imgBtn && (this.$refs.imgBtn.style.top = data.y + 'px')
|
|
|
+ this.$refs.imgBtnImg && (this.$refs.imgBtnImg.src = 'data:image/png;base64,' + data.cutoutImage)
|
|
|
+ console.dir(this.$refs.hkinnerWrap)
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('red')
|
|
|
+ this.$refs.hkinnerWrap && this.$refs.hkinnerWrap.$el.removeClass('green')
|
|
|
+ this.$refs.v_rightBtn && (this.$refs.v_rightBtn.style.left = 0)
|
|
|
+ this.$refs.imgBtn && (this.$refs.imgBtn.style.left = 0)
|
|
|
+ }
|
|
|
+ }
|
|
|
+}
|
|
|
+</script>
|
|
|
+
|
|
|
+<style lang="scss" scoped>
|
|
|
+.graphic-verification-code-container {
|
|
|
+ width: 100%;
|
|
|
+ box-sizing: border-box;
|
|
|
+
|
|
|
+ .rightValidate {
|
|
|
+ width: 280px;
|
|
|
+ margin: 0px auto;
|
|
|
+ position: relative;
|
|
|
+ line-height: 30px;
|
|
|
+ height: 30px;
|
|
|
+ text-align: center;
|
|
|
+ z-index: 99;
|
|
|
+ margin-top: 60px;
|
|
|
+ }
|
|
|
+
|
|
|
+ .v_rightBtn {
|
|
|
+ position: absolute;
|
|
|
+ left: 0;
|
|
|
+ top: 0;
|
|
|
+ height: 33px;
|
|
|
+ width: 40px;
|
|
|
+ background: #ddd;
|
|
|
+ cursor: pointer;
|
|
|
+ }
|
|
|
+
|
|
|
+ .imgBtn {
|
|
|
+ width: 44px;
|
|
|
+ height: 171px;
|
|
|
+ position: absolute;
|
|
|
+ left: 0;
|
|
|
+ }
|
|
|
+
|
|
|
+ .imgBtn img {
|
|
|
+ z-index: 99;
|
|
|
+ align: center;
|
|
|
+ }
|
|
|
+
|
|
|
+ .imgBg {
|
|
|
+ position: absolute;
|
|
|
+ bottom: 35px;
|
|
|
+ width: 280px;
|
|
|
+ height: 171px;
|
|
|
+ display: block;
|
|
|
+ z-index: 9;
|
|
|
+ }
|
|
|
+
|
|
|
+ .imgBg_2 {
|
|
|
+ position: absolute;
|
|
|
+ bottom: -80px;
|
|
|
+ bottom: 35px;
|
|
|
+ width: 150px;
|
|
|
+ height: 60px;
|
|
|
+ display: block;
|
|
|
+ z-index: 9;
|
|
|
+ }
|
|
|
+
|
|
|
+ .imgBg_3 {
|
|
|
+ position: absolute;
|
|
|
+ bottom: -80px;
|
|
|
+ bottom: 35px;
|
|
|
+ width: 170px;
|
|
|
+ height: 60px;
|
|
|
+ display: block;
|
|
|
+ z-index: 9;
|
|
|
+ }
|
|
|
+
|
|
|
+ .hkinnerWrap {
|
|
|
+ border: 1px solid #eee;
|
|
|
+ z-index: 9999
|
|
|
+ }
|
|
|
+
|
|
|
+ .green {
|
|
|
+ border-color: #34C6C2 !important;
|
|
|
+ }
|
|
|
+
|
|
|
+ .green .v_rightBtn {
|
|
|
+ background: #34C6C2;
|
|
|
+ color: #fff;
|
|
|
+ }
|
|
|
+
|
|
|
+ .red {
|
|
|
+ border-color: red !important;
|
|
|
+ }
|
|
|
+
|
|
|
+ .red .v_rightBtn {
|
|
|
+ background: red;
|
|
|
+ color: #fff;
|
|
|
+ }
|
|
|
+
|
|
|
+ .refresh {
|
|
|
+ position: absolute;
|
|
|
+ width: 30px;
|
|
|
+ height: 30px;
|
|
|
+ right: 4px;
|
|
|
+ top: 4px;
|
|
|
+ font-size: 12px;
|
|
|
+ color: #fff;
|
|
|
+ text-shadow: 0px 0px 9px #333;
|
|
|
+ cursor: pointer;
|
|
|
+ display: none;
|
|
|
+ }
|
|
|
+
|
|
|
+ .notSel {
|
|
|
+ user-select: none;
|
|
|
+ -webkit-user-select: none;
|
|
|
+ -moz-user-select: none;
|
|
|
+ -ms-user-select: none;
|
|
|
+ -webkit-touch-callout: none;
|
|
|
+ }
|
|
|
+}
|
|
|
+</style>
|