You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
408 lines
10 KiB
408 lines
10 KiB
<template> |
|
<!-- #ifdef VUE2 --> |
|
<view class="xt__verify-code"> |
|
<!-- 输入框 --> |
|
<input id="xt__input" :value="code" class="xt__input" :focus="isFocus" :type="inputType" :maxlength="itemSize" @input="onInput" @focus="inputFocus" @blur="inputBlur" /> |
|
|
|
<!-- 光标 --> |
|
<view |
|
id="xt__cursor" |
|
v-if="cursorVisible && type !== 'middle'" |
|
class="xt__cursor" |
|
:style="{ left: codeCursorLeft[code.length] + 'px', height: cursorHeight + 'px', backgroundColor: cursorColor }" |
|
></view> |
|
|
|
<!-- 输入框 - 组 --> |
|
<view id="xt__input-ground" class="xt__input-ground"> |
|
<view |
|
v-for="(item, index) in itemSize" |
|
:key="index" |
|
:style="{ borderColor: code.length === index && cursorVisible ? boxActiveColor : boxNormalColor }" |
|
:class="['xt__box', `xt__box-${type + ''}`, `xt__box::after`]" |
|
> |
|
<view :style="{ borderColor: boxActiveColor }" class="xt__middle-line" v-if="type === 'middle' && !code[index]"></view> |
|
|
|
<text :style="{ color: color }" class="xt__code-text">{{ code[index] | codeFormat(isPassword) }}</text> |
|
</view> |
|
</view> |
|
</view> |
|
<!-- #endif --> |
|
|
|
<!-- #ifdef VUE3 --> |
|
<view class="xt__verify-code"> |
|
<!-- 输入框 --> |
|
<input |
|
id="xt__input" |
|
:value="code" |
|
class="xt__input" |
|
:focus="props.isFocus" |
|
:type="props.inputType" |
|
:maxlength="itemSize" |
|
@input="onInput" |
|
@focus="inputFocus" |
|
@blur="inputBlur" |
|
/> |
|
|
|
<!-- 光标 --> |
|
<view |
|
id="xt__cursor" |
|
v-if="cursorVisible && props.type !== 'middle'" |
|
class="xt__cursor" |
|
:style="{ left: codeCursorLeft[code.length] + 'px', height: cursorHeight + 'px', backgroundColor: props.cursorColor }" |
|
></view> |
|
|
|
<!-- 输入框 - 组 --> |
|
<view id="xt__input-ground" class="xt__input-ground"> |
|
<view |
|
v-for="(item, index) in itemSize" |
|
:key="index" |
|
:style="{ borderColor: code.length === index && cursorVisible ? props.boxActiveColor : props.boxNormalColor }" |
|
:class="['xt__box', `xt__box-${props.type + ''}`, `xt__box::after`]" |
|
> |
|
<view :style="{ borderColor: props.boxActiveColor }" class="xt__middle-line" v-if="props.type === 'middle' && !code[index]"></view> |
|
<text :style="{ color: props.color }" class="xt__code-text">{{ codeFormat(code[index], props.isPassword) }}</text> |
|
</view> |
|
</view> |
|
</view> |
|
<!-- #endif --> |
|
</template> |
|
|
|
<!-- #ifdef VUE3 --> |
|
<script setup> |
|
/** |
|
* @description 输入验证码组件 |
|
* @property {string} type = [box|middle|bottom] - 显示类型 默认:box -eg:bottom |
|
* @property {string} inputType = [text|number] - 输入框类型 默认:number -eg:number |
|
* @property {number} size - 验证码输入框数量 默认:6 -eg:6 |
|
* @property {boolean} isFocus - 是否立即聚焦 默认:true |
|
* @property {boolean} isPassword - 是否以密码形式显示 默认false -eg: false |
|
* @property {string} cursorColor - 光标颜色 默认:#cccccc |
|
* @property {string} boxNormalColor - 光标未聚焦到的框的颜色 默认:#cccccc |
|
* @property {string} boxActiveColor - 光标聚焦到的框的颜色 默认:#000000 |
|
* @property {string} color - 光标聚焦到的框的颜色 默认:#333333 |
|
* @event {Function(data)} confirm - 输入完成回调函数 |
|
*/ |
|
import { defineProps, onMounted, ref, getCurrentInstance, watch } from 'vue'; |
|
import { propsMap } from './config.js'; |
|
import { getElementRect } from './util.js'; |
|
|
|
const props = defineProps(propsMap); |
|
const emits = defineEmits(['update:modelValue', 'confirm']); |
|
|
|
// 获取当前组件的实例 |
|
const currentInstance = getCurrentInstance(); |
|
|
|
const getElement = getElementRect(currentInstance); |
|
|
|
const cursorVisible = ref(false); |
|
const cursorHeight = ref(35); |
|
const code = ref(''); // 输入的验证码 |
|
const codeCursorLeft = ref([]); // 向左移动的距离数组, |
|
const itemSize = ref(6); |
|
const isPatch = ref(false); |
|
|
|
/** |
|
* 设置验证码框数量 |
|
*/ |
|
function validatorSize() { |
|
if (props.size > 0) { |
|
itemSize.value = Math.floor(props.size); |
|
} else { |
|
throw "methods of 'size' is integer"; |
|
} |
|
} |
|
/** |
|
* @description 初始化 |
|
*/ |
|
function init() { |
|
getCodeCursorLeft(); |
|
setCursorHeight(); |
|
} |
|
|
|
/** |
|
* @description 计算光标的高度 |
|
*/ |
|
function setCursorHeight() { |
|
getElement('.xt__box', 'single', boxElm => { |
|
cursorHeight.value = boxElm.height * 0.6; |
|
}); |
|
} |
|
/** |
|
* @description 获取光标在每一个box的left位置 |
|
*/ |
|
function getCodeCursorLeft() { |
|
// 获取父级框的位置信息 |
|
getElement('#xt__input-ground', 'single', parentElm => { |
|
const parentLeft = parentElm.left; |
|
// 获取各个box信息 |
|
getElement('.xt__box', 'array', elms => { |
|
codeCursorLeft.value = []; |
|
elms.forEach(elm => { |
|
codeCursorLeft.value.push(elm.left - parentLeft + elm.width / 2); |
|
}); |
|
}); |
|
}); |
|
} |
|
|
|
// 输入框输入变化的回调 |
|
function onInput(e) { |
|
let { value, keyCode } = e.detail; |
|
cursorVisible.value = value.length < itemSize.value; |
|
code.value = value; |
|
emits('update:modelValue', value); |
|
inputSuccess(value); |
|
} |
|
|
|
// 输入完成回调 |
|
function inputSuccess(value) { |
|
if (value.length === itemSize.value && !isPatch.value) { |
|
isPatch.value = true; |
|
emits('confirm', value); |
|
} else { |
|
isPatch.value = false; |
|
} |
|
} |
|
// 输入聚焦 |
|
function inputFocus() { |
|
cursorVisible.value = code.value.length < itemSize.value; |
|
} |
|
// 输入失去焦点 |
|
function inputBlur() { |
|
cursorVisible.value = false; |
|
} |
|
|
|
// 判断文本如何显示 |
|
function codeFormat(val, isPassword) { |
|
return val ? (isPassword ? '*' : val) : ''; |
|
} |
|
|
|
onMounted(() => { |
|
cursorVisible.value = props.isFocus; |
|
validatorSize(); |
|
init(); |
|
}); |
|
|
|
watch( |
|
() => props.modelValue, |
|
val => { |
|
if (val !== code.value) { |
|
code.value = val; |
|
} |
|
} |
|
); |
|
</script> |
|
<!-- #endif --> |
|
|
|
<!-- #ifdef VUE2 --> |
|
<script> |
|
/** |
|
* @description 输入验证码组件 |
|
* @property {string} type = [box|middle|bottom] - 显示类型 默认:box -eg:bottom |
|
* @property {string} inputType = [text|number] - 输入框类型 默认:number -eg:number |
|
* @property {number} size - 验证码输入框数量 默认:6 -eg:6 |
|
* @property {boolean} isFocus - 是否立即聚焦 默认:true |
|
* @property {boolean} isPassword - 是否以密码形式显示 默认false -eg: false |
|
* @property {string} cursorColor - 光标颜色 默认:#cccccc |
|
* @property {string} boxNormalColor - 光标未聚焦到的框的颜色 默认:#cccccc |
|
* @property {string} boxActiveColor - 光标聚焦到的框的颜色 默认:#000000 |
|
* @property {string} color - 光标聚焦到的框的颜色 默认:#333333 |
|
* @event {Function(data)} confirm - 输入完成回调函数 |
|
*/ |
|
import { propsMap } from './config.js'; |
|
import { getElementRect } from './util.js'; |
|
export default { |
|
name: 'xt-verify-code', |
|
props: propsMap, |
|
model: { |
|
prop: 'value', |
|
event: 'input' |
|
}, |
|
data() { |
|
return { |
|
cursorVisible: false, |
|
cursorHeight: 35, |
|
code: '', // 输入的验证码 |
|
codeCursorLeft: [], // 向左移动的距离数组, |
|
itemSize: 6, |
|
getElement: getElementRect(this), |
|
isPatch: false |
|
}; |
|
}, |
|
created() { |
|
this.cursorVisible = this.isFocus; |
|
this.validatorSize(); |
|
}, |
|
mounted() { |
|
this.init(); |
|
}, |
|
methods: { |
|
/** |
|
* 设置验证码框数量 |
|
*/ |
|
validatorSize() { |
|
if (this.size > 0) { |
|
this.itemSize = Math.floor(this.size); |
|
} else { |
|
throw "methods of 'size' is integer"; |
|
} |
|
}, |
|
/** |
|
* @description 初始化 |
|
*/ |
|
init() { |
|
this.getCodeCursorLeft(); |
|
this.setCursorHeight(); |
|
}, |
|
/** |
|
* @description 计算光标的高度 |
|
*/ |
|
setCursorHeight() { |
|
this.getElement('.xt__box', 'single', boxElm => { |
|
this.cursorHeight = boxElm.height * 0.6; |
|
}); |
|
}, |
|
/** |
|
* @description 获取光标在每一个box的left位置 |
|
*/ |
|
getCodeCursorLeft() { |
|
// 获取父级框的位置信息 |
|
this.getElement('#xt__input-ground', 'single', parentElm => { |
|
const parentLeft = parentElm.left; |
|
// 获取各个box信息 |
|
this.getElement('.xt__box', 'array', elms => { |
|
this.codeCursorLeft = []; |
|
elms.forEach(elm => { |
|
this.codeCursorLeft.push(elm.left - parentLeft + elm.width / 2); |
|
}); |
|
}); |
|
}); |
|
}, |
|
|
|
// 输入框输入变化的回调 |
|
onInput(e) { |
|
let { value, keyCode } = e.detail; |
|
this.cursorVisible = value.length < this.itemSize; |
|
this.code = value; |
|
this.$emit('input', value); |
|
this.inputSuccess(value); |
|
}, |
|
|
|
// 输入完成回调 |
|
inputSuccess(value) { |
|
if (value.length === this.itemSize && !this.isPatch) { |
|
this.$emit('confirm', value); |
|
} else { |
|
this.isPatch = false; |
|
} |
|
}, |
|
// 输入聚焦 |
|
inputFocus() { |
|
this.cursorVisible = this.code.length < this.itemSize; |
|
}, |
|
// 输入失去焦点 |
|
inputBlur() { |
|
this.cursorVisible = false; |
|
} |
|
}, |
|
watch: { |
|
value(val) { |
|
if (val !== this.code) { |
|
this.code = val; |
|
} |
|
} |
|
}, |
|
filters: { |
|
codeFormat(val, isPassword) { |
|
return val ? (isPassword ? '*' : val) : ''; |
|
} |
|
} |
|
}; |
|
</script> |
|
<!-- #endif --> |
|
|
|
<style > |
|
.xt__verify-code { |
|
position: relative; |
|
width: 100%; |
|
box-sizing: border-box; |
|
} |
|
|
|
.xt__verify-code .xt__input { |
|
height: 100%; |
|
width: 200vw; |
|
position: absolute; |
|
left: -100vw; |
|
z-index: 1; |
|
color: transparent; |
|
caret-color: transparent; |
|
background-color: rgba(0, 0, 0, 0); |
|
} |
|
.xt__verify-code .xt__cursor { |
|
position: absolute; |
|
top: 50%; |
|
transform: translateY(-50%); |
|
display: inline-block; |
|
width: 2px; |
|
animation-name: xt__cursor; |
|
animation-duration: 0.8s; |
|
animation-iteration-count: infinite; |
|
z-index: 1; |
|
} |
|
|
|
.xt__verify-code .xt__input-ground { |
|
display: flex; |
|
justify-content: space-between; |
|
align-items: center; |
|
width: 100%; |
|
box-sizing: border-box; |
|
} |
|
|
|
.xt__verify-code .xt__input-ground .xt__box { |
|
position: relative; |
|
display: inline-block; |
|
width: 100rpx; |
|
height: 140rpx; |
|
} |
|
|
|
.xt__verify-code .xt__input-ground .xt__box-bottom { |
|
border-bottom-width: 2px; |
|
border-bottom-style: solid; |
|
} |
|
|
|
.xt__verify-code .xt__input-ground .xt__box-box { |
|
border-width: 2px; |
|
border-style: solid; |
|
} |
|
|
|
.xt__verify-code .xt__input-ground .xt__box-middle { |
|
border: none; |
|
} |
|
|
|
.xt__input-ground .xt__box .xt__middle-line { |
|
position: absolute; |
|
top: 50%; |
|
left: 50%; |
|
width: 50%; |
|
transform: translate(-50%, -50%); |
|
border-bottom-width: 2px; |
|
border-bottom-style: solid; |
|
} |
|
|
|
.xt__input-ground .xt__box .xt__code-text { |
|
position: absolute; |
|
top: 50%; |
|
left: 50%; |
|
font-size: 80rpx; |
|
transform: translate(-50%, -50%); |
|
} |
|
|
|
@keyframes xt__cursor { |
|
0% { |
|
opacity: 1; |
|
} |
|
|
|
100% { |
|
opacity: 0; |
|
} |
|
} |
|
</style>
|
|
|