380 lines
13 KiB
Vue
380 lines
13 KiB
Vue
<template>
|
|
<div class="wg-cap-btn" :style="style">
|
|
<div class="wg-cap-btn__inner" :class="activeClass">
|
|
<!-- wg-cap-active__default wg-cap-active__error wg-cap-active__over wg-cap-active__success -->
|
|
<template>
|
|
<div @click="handleBtnEvent" class="wg-cap-state__default">
|
|
<!-- 初始状态 -->
|
|
<div class="wg-cap-state__inner">
|
|
<div class="wg-cap-btn__ico wg-cap-btn__verify">
|
|
<img
|
|
src="" />
|
|
</div>
|
|
<span class="wg-cap-btn__text">点击按键进行人机验证</span>
|
|
</div>
|
|
</div>
|
|
<div @click="() => false" class="wg-cap-state__check">
|
|
<!-- 验证状态 -->
|
|
<div class="wg-cap-state__inner">
|
|
<div class="wg-cap-btn__ico">
|
|
<img
|
|
src=""
|
|
alt="" />
|
|
</div>
|
|
<span class="wg-cap-btn__text">正在进行人机验证...</span>
|
|
</div>
|
|
</div>
|
|
<div @click="handleBtnEvent" class="wg-cap-state__error">
|
|
<!-- 验证失败状态 -->
|
|
<div class="wg-cap-state__inner">
|
|
<div class="wg-cap-btn__ico">
|
|
<img
|
|
src=""
|
|
alt="失败" />
|
|
</div>
|
|
<span>
|
|
人机验证失败
|
|
<em>点击重试</em>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<div @click="handleBtnEvent" class="wg-cap-state__over">
|
|
<!-- 验证次数过多状态 -->
|
|
<div class="wg-cap-state__inner">
|
|
<div class="wg-cap-btn__ico">
|
|
<img
|
|
src=""
|
|
alt="失败" />
|
|
</div>
|
|
<span>
|
|
点击次数过多
|
|
<em>点击重试</em>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<div @click="() => false" class="wg-cap-state__success">
|
|
<!-- 验证成功状态 -->
|
|
<div class="wg-cap-state__inner">
|
|
<div class="wg-cap-btn__ico">
|
|
<img
|
|
src=""
|
|
alt="成功" />
|
|
</div>
|
|
<span>人机验证已通过</span>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<el-dialog v-model:visible="popoverVisible" :close-on-click-modal="false" append-to-body :center="true" title="人机校验"
|
|
:show-close="false" z-index="999999" width="360px">
|
|
<CommonCaptcha v-model="popoverVisible" width="300px" height="240px" :max-dot="maxDot" :image-base64="imageBase64"
|
|
:thumb-base64="thumbBase64" @close="handleCloseEvent" @refresh="handleRefreshEvent"
|
|
@confirm="handleConfirmEvent" />
|
|
</el-dialog>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script>
|
|
import CommonCaptcha from './captcha.vue'
|
|
import { defineComponent, watch, toRefs, computed, ref, reactive } from 'vue'
|
|
export default defineComponent({
|
|
name: 'CommonCaptchaDialog',
|
|
components: { CommonCaptcha },
|
|
props: {
|
|
value: {
|
|
type: String,
|
|
default: 'default',
|
|
validator: value =>
|
|
['default', 'check', 'error', 'over', 'success'].indexOf(value) > -1,
|
|
},
|
|
width: String,
|
|
height: String,
|
|
maxDot: {
|
|
type: Number,
|
|
default: 5,
|
|
},
|
|
imageBase64: String,
|
|
thumbBase64: String,
|
|
},
|
|
setup (props, { emit }) {
|
|
const state = reactive({
|
|
popoverVisible: ref(false),
|
|
captStatus: ref('default'),
|
|
activeClass: computed(() => {
|
|
return `wg-cap-active__${state.captStatus}`
|
|
}),
|
|
style: computed(() => {
|
|
return `width:${props.width}; height:${props.height};`
|
|
}),
|
|
handleBtnEvent () {
|
|
setTimeout(() => {
|
|
state.popoverVisible = true
|
|
}, 0)
|
|
},
|
|
handleRefreshEvent () {
|
|
state.captStatus = 'check'
|
|
emit('refresh')
|
|
},
|
|
handleConfirmEvent (data) {
|
|
emit('confirm', data)
|
|
},
|
|
handleCloseEvent () {
|
|
state.popoverVisible = false
|
|
},
|
|
})
|
|
|
|
watch(
|
|
() => state.popoverVisible,
|
|
v => {
|
|
if (v) {
|
|
state.captStatus = 'check'
|
|
emit('refresh')
|
|
} else if (state.captStatus === 'check') {
|
|
state.captStatus = props.value
|
|
}
|
|
}
|
|
)
|
|
watch(
|
|
() => props.value,
|
|
val => {
|
|
if (state.captStatus !== 'check') {
|
|
state.captStatus = val
|
|
}
|
|
if (val === 'over' || val === 'success') {
|
|
setTimeout(() => {
|
|
state.popoverVisible = false
|
|
}, 0)
|
|
}
|
|
}
|
|
)
|
|
watch(
|
|
() => state.captStatus,
|
|
val => {
|
|
if (val !== 'check' && props.value !== val) {
|
|
emit('input', val)
|
|
}
|
|
}
|
|
)
|
|
|
|
return {
|
|
...toRefs(state),
|
|
}
|
|
},
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.wg-cap-btn {
|
|
width: 100%;
|
|
height: 48px;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-btn__inner {
|
|
width: 100%;
|
|
height: 48px;
|
|
position: relative;
|
|
letter-spacing: 1px;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__default,
|
|
.wg-cap-btn .wg-cap-state__check,
|
|
.wg-cap-btn .wg-cap-state__error,
|
|
.wg-cap-btn .wg-cap-state__success,
|
|
.wg-cap-btn .wg-cap-state__over {
|
|
position: absolute;
|
|
width: 100%;
|
|
height: 48px;
|
|
font-size: 13px;
|
|
-webkit-border-radius: 5px;
|
|
-moz-border-radius: 5px;
|
|
border-radius: 5px;
|
|
display: inline-block;
|
|
line-height: 1;
|
|
white-space: nowrap;
|
|
cursor: pointer;
|
|
-webkit-appearance: none;
|
|
appearance: none;
|
|
box-sizing: border-box;
|
|
outline: none;
|
|
margin: 0;
|
|
transition: 0.1s;
|
|
font-weight: 500;
|
|
-moz-user-select: none;
|
|
-webkit-user-select: none;
|
|
user-select: none;
|
|
|
|
display: -webkit-box;
|
|
display: -webkit-flex;
|
|
display: -ms-flexbox;
|
|
display: flex;
|
|
-webkit-box-align: center;
|
|
-webkit-align-items: center;
|
|
-ms-flex-align: center;
|
|
align-items: center;
|
|
justify-content: center;
|
|
justify-items: center;
|
|
|
|
visibility: hidden;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__default {
|
|
color: #3e7cff;
|
|
border: 1px solid #50a1ff;
|
|
background: #ecf5ff;
|
|
box-shadow: 0 0 20px rgba(62, 124, 255, 0.1);
|
|
-webkit-box-shadow: 0 0 20px rgba(62, 124, 255, 0.1);
|
|
-moz-box-shadow: 0 0 20px rgba(62, 124, 255, 0.1);
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__check {
|
|
cursor: default;
|
|
color: #ffa000;
|
|
background: #fdf6ec;
|
|
border: 1px solid #ffbe09;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__error {
|
|
color: #ed4630;
|
|
background: #fef0f0;
|
|
border: 1px solid #ff5a34;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__over {
|
|
color: #ed4630;
|
|
background: #fef0f0;
|
|
border: 1px solid #ff5a34;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__success {
|
|
color: #5eaa2f;
|
|
background: #f0f9eb;
|
|
border: 1px solid #8bc640;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-active__default .wg-cap-state__default,
|
|
.wg-cap-btn .wg-cap-active__error .wg-cap-state__error,
|
|
.wg-cap-btn .wg-cap-active__over .wg-cap-state__over,
|
|
.wg-cap-btn .wg-cap-active__success .wg-cap-state__success,
|
|
.wg-cap-btn .wg-cap-active__check .wg-cap-state__check {
|
|
visibility: visible;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__inner {
|
|
display: -webkit-box;
|
|
display: -webkit-flex;
|
|
display: -ms-flexbox;
|
|
display: flex;
|
|
-webkit-box-align: center;
|
|
-webkit-align-items: center;
|
|
-ms-flex-align: center;
|
|
align-items: center;
|
|
justify-content: center;
|
|
justify-items: center;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-state__inner em {
|
|
padding-left: 5px;
|
|
color: #3e7cff;
|
|
font-style: normal;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-btn__inner .wg-cap-btn__ico {
|
|
position: relative;
|
|
width: 24px;
|
|
height: 24px;
|
|
margin-right: 12px;
|
|
font-size: 14px;
|
|
display: inline-block;
|
|
/*float: left;*/
|
|
flex: 0;
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-btn__inner .wg-cap-btn__ico img {
|
|
width: 24px;
|
|
height: 24px;
|
|
float: left;
|
|
position: relative;
|
|
z-index: 10;
|
|
}
|
|
|
|
@keyframes ripple {
|
|
0% {
|
|
opacity: 0;
|
|
}
|
|
|
|
5% {
|
|
opacity: 0.05;
|
|
}
|
|
|
|
20% {
|
|
opacity: 0.35;
|
|
}
|
|
|
|
65% {
|
|
opacity: 0.01;
|
|
}
|
|
|
|
100% {
|
|
transform: scaleX(2) scaleY(2);
|
|
opacity: 0;
|
|
}
|
|
}
|
|
|
|
@-webkit-keyframes ripple {
|
|
0% {
|
|
opacity: 0;
|
|
}
|
|
|
|
5% {
|
|
opacity: 0.05;
|
|
}
|
|
|
|
20% {
|
|
opacity: 0.35;
|
|
}
|
|
|
|
65% {
|
|
opacity: 0.01;
|
|
}
|
|
|
|
100% {
|
|
transform: scaleX(2) scaleY(2);
|
|
opacity: 0;
|
|
}
|
|
}
|
|
|
|
.wg-cap-btn .wg-cap-btn__inner .wg-cap-btn__verify::after {
|
|
background: #409eff;
|
|
-webkit-border-radius: 50px;
|
|
-moz-border-radius: 50px;
|
|
border-radius: 50px;
|
|
content: '';
|
|
display: block;
|
|
width: 24px;
|
|
height: 24px;
|
|
opacity: 0;
|
|
position: absolute;
|
|
top: 0;
|
|
left: 0;
|
|
z-index: 9;
|
|
|
|
animation: ripple 1.3s infinite;
|
|
-moz-animation: ripple 1.3s infinite;
|
|
-webkit-animation: ripple 1.3s infinite;
|
|
animation-delay: 2s;
|
|
-moz-animation-delay: 2s;
|
|
-webkit-animation-delay: 2s;
|
|
}
|
|
|
|
.wg-cap-tip {
|
|
padding: 50px 20px 100px;
|
|
font-size: 13px;
|
|
color: #76839b;
|
|
text-align: center;
|
|
line-height: 180%;
|
|
width: 100%;
|
|
max-width: 680px;
|
|
}
|
|
</style>
|