滑块验证码

1.这里针对滑块验证给了一个封装的组件verifition,使用直接可以调用

2.组件目录

在这里插入图片描述

3.每个文件的内容

3.1 Api文件中只有一个index.js文件,用来存放获取滑块和校验滑块结果的api

import request from '@/router/axios'//获取验证图片
export function reqGet(data) {return request({url: '/code',method: 'get',data})
}//滑动或者点选验证
export function reqCheck(data) {return request({url: '/code/check',method: 'post',params: data})
}

3.2 utils文件夹中反了一些工具函数为校验提供支持,包含三个文件:ase.js、axios.js、util.js,不涉及任何组件的引用,可放心使用

//ase.js文件内容
//这里需要安装crypto-js进行加密
import CryptoJS from 'crypto-js'
/*** @word 要加密的内容* @keyWord String  服务器随机返回的关键字*  */
export function aesEncrypt(word,keyWord="XwKsGlMcdPMEhR1B"){var key = CryptoJS.enc.Utf8.parse(keyWord);var srcs = CryptoJS.enc.Utf8.parse(word);var encrypted = CryptoJS.AES.encrypt(srcs, key, {mode:CryptoJS.mode.ECB,padding: CryptoJS.pad.Pkcs7});return encrypted.toString();
}
//axios.js文件内容
import axios from 'axios';axios.defaults.baseURL = process.env.BASE_API;const service = axios.create({timeout: 40000,headers: {'X-Requested-With': 'XMLHttpRequest','Content-Type': 'application/json; charset=UTF-8'},
})
service.interceptors.request.use(config => {return config},error => {Promise.reject(error)}
)// response interceptor
service.interceptors.response.use(response => {const res = response.data;return res},error => {}
)
export default service
//util.js文件内容
export function resetSize(vm) {var img_width, img_height, bar_width, bar_height;	//图片的宽度、高度,移动条的宽度、高度var parentWidth = vm.$el.parentNode.offsetWidth || window.offsetWidthvar parentHeight = vm.$el.parentNode.offsetHeight || window.offsetHeightif (vm.imgSize.width.indexOf('%') != -1) {img_width = parseInt(this.imgSize.width) / 100 * parentWidth + 'px'} else {img_width = this.imgSize.width;}if (vm.imgSize.height.indexOf('%') != -1) {img_height = parseInt(this.imgSize.height) / 100 * parentHeight + 'px'} else {img_height = this.imgSize.height}if (vm.barSize.width.indexOf('%') != -1) {bar_width = parseInt(this.barSize.width) / 100 * parentWidth + 'px'} else {bar_width = this.barSize.width}if (vm.barSize.height.indexOf('%') != -1) {bar_height = parseInt(this.barSize.height) / 100 * parentHeight + 'px'} else {bar_height = this.barSize.height}return {imgWidth: img_width, imgHeight: img_height, barWidth: bar_width, barHeight: bar_height}
}export const _code_chars = [1, 2, 3, 4, 5, 6, 7, 8, 9, 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z']
export const _code_color1 = ['#fffff0', '#f0ffff', '#f0fff0', '#fff0f0']
export const _code_color2 = ['#FF0033', '#006699', '#993366', '#FF9900', '#66CC66', '#FF33CC']

3.3Verify文件夹中放的是关键的组件内容,包含两个文件:VerifyPoints.vue,VerifySlide.vue

//VerifyPoints.vue内容
<template><div style="position: relative"><div class="verify-img-out"><div class="verify-img-panel" :style="{width: setSize.imgWidth,height: setSize.imgHeight,'background-size': setSize.imgWidth + ' ' + setSize.imgHeight,'margin-bottom': vSpace + 'px',}"><div class="verify-refresh" style="z-index: 3" @click="refresh" v-show="showRefresh"><i class="iconfont icon-refresh"></i></div><img :src="'data:image/png;base64,' + pointBackImgBase" ref="canvas" altstyle="width: 100%; height: 100%; display: block" @click="bindingClick ? canvasClick($event) : undefined" /><div v-for="(tempPoint, index) in tempPoints" :key="index" class="point-area" :style="{'background-color': '#1abd6c',color: '#fff','z-index': 9999,width: '20px',height: '20px','text-align': 'center','line-height': '20px','border-radius': '50%',position: 'absolute',top: parseInt(tempPoint.y - 10) + 'px',left: parseInt(tempPoint.x - 10) + 'px',}">{{ index + 1 }}</div></div></div><!-- 'height': this.barSize.height, --><div class="verify-bar-area" :style="{width: setSize.imgWidth,color: this.barAreaColor,'border-color': this.barAreaBorderColor,'line-height': this.barSize.height,}"><span class="verify-msg">{{ text }}</span></div></div>
</template>
<script type="text/babel">
/*** VerifyPoints* @description 点选* */
import {resetSize,_code_chars,_code_color1,_code_color2,
} from "./../utils/util";
import { aesEncrypt } from "./../utils/ase";
import { reqGet, reqCheck } from "./../api/index";export default {name: "VerifyPoints",props: {//弹出式pop,固定fixedmode: {type: String,default: "fixed",},captchaType: {type: String,},//间隔vSpace: {type: Number,default: 5,},imgSize: {type: Object,default() {return {width: "310px",height: "155px",};},},barSize: {type: Object,default() {return {width: "310px",height: "40px",};},},},data() {return {secretKey: "", //后端返回的ase加密秘钥checkNum: 3, //默认需要点击的字数fontPos: [], //选中的坐标信息checkPosArr: [], //用户点击的坐标num: 1, //点击的记数pointBackImgBase: "", //后端获取到的背景图片poinTextList: [], //后端返回的点击字体顺序backToken: "", //后端返回的token值setSize: {imgHeight: 0,imgWidth: 0,barHeight: 0,barWidth: 0,},tempPoints: [],text: "",barAreaColor: undefined,barAreaBorderColor: undefined,showRefresh: true,bindingClick: true,};},computed: {resetSize() {return resetSize;},},methods: {init() {//加载页面this.fontPos.splice(0, this.fontPos.length);this.checkPosArr.splice(0, this.checkPosArr.length);this.num = 1;this.getPictrue();this.$nextTick(() => {this.setSize = this.resetSize(this); //重新设置宽度高度this.$parent.$emit("ready", this);});},canvasClick(e) {this.checkPosArr.push(this.getMousePos(this.$refs.canvas, e));if (this.num == this.checkNum) {this.num = this.createPoint(this.getMousePos(this.$refs.canvas, e));//按比例转换坐标值this.checkPosArr = this.pointTransfrom(this.checkPosArr, this.setSize);//等创建坐标执行完setTimeout(() => {// var flag = this.comparePos(this.fontPos, this.checkPosArr);//发送后端请求var captchaVerification = this.secretKey? aesEncrypt(this.backToken + "---" + JSON.stringify(this.checkPosArr),this.secretKey): this.backToken + "---" + JSON.stringify(this.checkPosArr);let data = {captchaType: this.captchaType,pointJson: this.secretKey? aesEncrypt(JSON.stringify(this.checkPosArr), this.secretKey): JSON.stringify(this.checkPosArr),token: this.backToken,};reqCheck(data).then((response) => {let res = response.data.data;if (res.repCode == "0000") {this.barAreaColor = "#4cae4c";this.barAreaBorderColor = "#5cb85c";this.text = "验证成功";this.bindingClick = false;if (this.mode == "pop") {setTimeout(() => {this.$parent.clickShow = false;this.refresh();}, 1500);}this.$parent.$emit("success", { captchaVerification });} else {this.$parent.$emit("error", this);this.barAreaColor = "#d9534f";this.barAreaBorderColor = "#d9534f";this.text = "验证失败";setTimeout(() => {this.refresh();}, 700);}});}, 400);}if (this.num < this.checkNum) {this.num = this.createPoint(this.getMousePos(this.$refs.canvas, e));}},//获取坐标getMousePos: function (obj, e) {var x = e.offsetX;var y = e.offsetY;return { x, y };},//创建坐标点createPoint: function (pos) {this.tempPoints.push(Object.assign({}, pos));return ++this.num;},refresh: function () {this.tempPoints.splice(0, this.tempPoints.length);this.barAreaColor = "#000";this.barAreaBorderColor = "#ddd";this.bindingClick = true;this.fontPos.splice(0, this.fontPos.length);this.checkPosArr.splice(0, this.checkPosArr.length);this.num = 1;this.getPictrue();this.text = "验证失败";this.showRefresh = true;},// 请求背景图片和验证图片getPictrue() {let data = {captchaType: this.captchaType,};reqGet(data).then((response) => {let res = response.data.data;if (res.repCode == "0000") {this.pointBackImgBase = res.repData.originalImageBase64;this.backToken = res.repData.token;this.secretKey = res.repData.secretKey;this.poinTextList = res.repData.wordList;this.text = "请依次点击【" + this.poinTextList.join(",") + "】";} else {this.text = res.repMsg;}});},//坐标转换函数pointTransfrom(pointArr, imgSize) {var newPointArr = pointArr.map((p) => {let x = Math.round((310 * p.x) / parseInt(imgSize.imgWidth));let y = Math.round((155 * p.y) / parseInt(imgSize.imgHeight));return { x, y };});// console.log(newPointArr,"newPointArr");return newPointArr;},},watch: {// type变化则全面刷新type: {immediate: true,handler() {this.init();},},},mounted() {// 禁止拖拽this.$el.onselectstart = function () {return false;};},
};
</script>
//VerifySlide.vue文件内容
<template><div style="position: relative"><div v-if="type === '2'" class="verify-img-out" :style="{ height: parseInt(setSize.imgHeight) + vSpace + 'px' }"><div class="verify-img-panel" :style="{ width: setSize.imgWidth, height: setSize.imgHeight }"><img :src="'data:image/png;base64,' + backImgBase" alt style="width: 100%; height: 100%; display: block" /><div class="verify-refresh" @click="refresh" v-show="showRefresh"><i class="iconfont icon-refresh"></i></div><transition name="tips"><span class="verify-tips" v-if="tipWords" :class="passFlag ? 'suc-bg' : 'err-bg'">{{ tipWords }}</span></transition></div></div><!-- 公共部分 --><div class="verify-bar-area" :style="{width: setSize.imgWidth,height: barSize.height,'line-height': barSize.height,}"><span class="verify-msg" v-text="text"></span><div class="verify-left-bar" :style="{width: leftBarWidth !== undefined ? leftBarWidth : barSize.height,height: barSize.height,'border-color': leftBarBorderColor,transaction: transitionWidth,}"><span class="verify-msg" v-text="finishText"></span><div class="verify-move-block" @touchstart="start" @mousedown="start" :style="{width: barSize.height,height: barSize.height,'background-color': moveBlockBackgroundColor,left: moveBlockLeft,transition: transitionLeft,}"><i :class="['verify-icon iconfont', iconClass]" :style="{ color: iconColor }"></i><div v-if="type === '2'" class="verify-sub-block" :style="{width: Math.floor((parseInt(setSize.imgWidth) * 47) / 310) + 'px',height: setSize.imgHeight,top: '-' + (parseInt(setSize.imgHeight) + vSpace) + 'px','background-size': setSize.imgWidth + ' ' + setSize.imgHeight,}"><img :src="'data:image/png;base64,' + blockBackImgBase" altstyle="width: 100%; height: 100%; display: block" /></div></div></div></div></div>
</template>
<script type="text/babel">
/*** VerifySlide* @description 滑块* */
import { aesEncrypt } from "./../utils/ase";
import { resetSize } from "./../utils/util";
import { reqGet, reqCheck } from "./../api/index";//  "captchaType":"blockPuzzle",
export default {name: "VerifySlide",props: {captchaType: {type: String,},type: {type: String,default: "1",},//弹出式pop,固定fixedmode: {type: String,default: "fixed",},vSpace: {type: Number,default: 5,},explain: {type: String,default: "向右滑动完成验证",},imgSize: {type: Object,default() {return {width: "310px",height: "155px",};},},blockSize: {type: Object,default() {return {width: "50px",height: "50px",};},},barSize: {type: Object,default() {return {width: "310px",height: "40px",};},},},data() {return {secretKey: "", //后端返回的加密秘钥 字段passFlag: "", //是否通过的标识backImgBase: "", //验证码背景图片blockBackImgBase: "", //验证滑块的背景图片backToken: "", //后端返回的唯一token值startMoveTime: "", //移动开始的时间endMovetime: "", //移动结束的时间tipsBackColor: "", //提示词的背景颜色tipWords: "",text: "",finishText: "",setSize: {imgHeight: 0,imgWidth: 0,barHeight: 0,barWidth: 0,},top: 0,left: 0,moveBlockLeft: undefined,leftBarWidth: undefined,// 移动中样式moveBlockBackgroundColor: undefined,leftBarBorderColor: "#ddd",iconColor: undefined,iconClass: "icon-right",status: false, //鼠标状态isEnd: false, //是够验证完成showRefresh: true,transitionLeft: "",transitionWidth: "",};},computed: {barArea() {return this.$el.querySelector(".verify-bar-area");},resetSize() {return resetSize;},},methods: {init() {this.text = this.explain;this.getPictrue();this.$nextTick(() => {let setSize = this.resetSize(this); //重新设置宽度高度for (let key in setSize) {this.$set(this.setSize, key, setSize[key]);}this.$parent.$emit("ready", this);});var _this = this;window.removeEventListener("touchmove", function (e) {_this.move(e);});window.removeEventListener("mousemove", function (e) {_this.move(e);});//鼠标松开window.removeEventListener("touchend", function () {_this.end();});window.removeEventListener("mouseup", function () {_this.end();});window.addEventListener("touchmove", function (e) {_this.move(e);});window.addEventListener("mousemove", function (e) {_this.move(e);});//鼠标松开window.addEventListener("touchend", function () {_this.end();});window.addEventListener("mouseup", function () {_this.end();});},//鼠标按下start: function (e) {e = e || window.event;if (!e.touches) {//兼容PC端var x = e.clientX;} else {//兼容移动端var x = e.touches[0].pageX;}this.startLeft = Math.floor(x - this.barArea.getBoundingClientRect().left);this.startMoveTime = +new Date(); //开始滑动的时间if (this.isEnd == false) {this.text = "";this.moveBlockBackgroundColor = "#337ab7";this.leftBarBorderColor = "#337AB7";this.iconColor = "#fff";e.stopPropagation();this.status = true;}},//鼠标移动move: function (e) {e = e || window.event;if (this.status && this.isEnd == false) {if (!e.touches) {//兼容PC端var x = e.clientX;} else {//兼容移动端var x = e.touches[0].pageX;}var bar_area_left = this.barArea.getBoundingClientRect().left;var move_block_left = x - bar_area_left; //小方块相对于父元素的left值if (move_block_left >=this.barArea.offsetWidth -parseInt(parseInt(this.blockSize.width) / 2) -2) {move_block_left =this.barArea.offsetWidth -parseInt(parseInt(this.blockSize.width) / 2) -2;}if (move_block_left <= 0) {move_block_left = parseInt(parseInt(this.blockSize.width) / 2);}//拖动后小方块的left值this.moveBlockLeft = move_block_left - this.startLeft + "px";this.leftBarWidth = move_block_left - this.startLeft + "px";}},//鼠标松开end: function () {this.endMovetime = +new Date();var _this = this;//判断是否重合if (this.status && this.isEnd == false) {var moveLeftDistance = parseInt((this.moveBlockLeft || "").replace("px", ""));moveLeftDistance =(moveLeftDistance * 310) / parseInt(this.setSize.imgWidth);let data = {captchaType: this.captchaType,pointJson: this.secretKey? aesEncrypt(JSON.stringify({ x: moveLeftDistance, y: 5.0 }),this.secretKey): JSON.stringify({ x: moveLeftDistance, y: 5.0 }),token: this.backToken,};reqCheck(data).then((response) => {let res = response.data.data;if (res.repCode == "0000") {this.moveBlockBackgroundColor = "#5cb85c";this.leftBarBorderColor = "#5cb85c";this.iconColor = "#fff";this.iconClass = "icon-check";this.showRefresh = false;this.isEnd = true;if (this.mode == "pop") {setTimeout(() => {this.$parent.clickShow = false;this.refresh();}, 1500);}this.passFlag = true;this.tipWords = `${((this.endMovetime - this.startMoveTime) /1000).toFixed(2)}s验证成功`;var captchaVerification = this.secretKey? aesEncrypt(this.backToken +"---" +JSON.stringify({ x: moveLeftDistance, y: 5.0 }),this.secretKey): this.backToken +"---" +JSON.stringify({ x: moveLeftDistance, y: 5.0 });setTimeout(() => {this.tipWords = "";this.$parent.closeBox();this.$parent.$emit("success", { captchaVerification });}, 1000);} else {this.moveBlockBackgroundColor = "#d9534f";this.leftBarBorderColor = "#d9534f";this.iconColor = "#fff";this.iconClass = "icon-close";this.passFlag = false;setTimeout(function () {_this.refresh();}, 1000);this.$parent.$emit("error", this);this.tipWords = "验证失败";setTimeout(() => {this.tipWords = "";}, 1000);}});this.status = false;}},refresh: function () {this.showRefresh = true;this.finishText = "";this.transitionLeft = "left .3s";this.moveBlockLeft = 0;this.leftBarWidth = undefined;this.transitionWidth = "width .3s";this.leftBarBorderColor = "#ddd";this.moveBlockBackgroundColor = "#fff";this.iconColor = "#000";this.iconClass = "icon-right";this.isEnd = false;this.getPictrue();setTimeout(() => {this.transitionWidth = "";this.transitionLeft = "";this.text = this.explain;}, 300);},// 请求背景图片和验证图片getPictrue() {let data = {captchaType: this.captchaType,};reqGet(data).then((response) => {let res = response.data.data;if (res.repCode == "0000") {this.backImgBase = res.repData.originalImageBase64;this.blockBackImgBase = res.repData.jigsawImageBase64;this.backToken = res.repData.token;this.secretKey = res.repData.secretKey;} else {this.tipWords = res.repMsg;}});},},watch: {// type变化则全面刷新type: {immediate: true,handler() {this.init();},},},mounted() {// 禁止拖拽this.$el.onselectstart = function () {return false;};},
};
</script>

3.4父组件内容,使用时调用show方法即可

<template><div :class="mode == 'pop' ? 'mask' : ''" v-show="showBox"><div :class="mode == 'pop' ? 'verifybox' : ''" :style="{ 'max-width': parseInt(imgSize.width) + 30 + 'px' }"><div class="verifybox-top" v-if="mode == 'pop'">请完成安全验证<span class="verifybox-close" @click="closeBox"><i class="iconfont icon-close"></i></span></div><div class="verifybox-bottom" :style="{ padding: mode == 'pop' ? '15px' : '0' }"><!-- 验证码容器 --><components v-if="componentType" :is="componentType" :captchaType="captchaType" :type="verifyType":figure="figure" :arith="arith" :mode="mode" :vSpace="vSpace" :explain="explain" :imgSize="imgSize":blockSize="blockSize" :barSize="barSize" ref="instance"></components></div></div></div>
</template>
<script type="text/babel">
/*** Verify 验证码组件* @description 分发验证码使用* */
import VerifySlide from "./Verify/VerifySlide";
import VerifyPoints from "./Verify/VerifyPoints";export default {name: "Vue2Verify",props: {// 双语化locale: {require: false,type: String,default() {// 默认语言不输入为浏览器语言if (navigator.language) {var language = navigator.language;} else {var language = navigator.browserLanguage;}return language;},},captchaType: {type: String,required: true,},figure: {type: Number,},arith: {type: Number,},mode: {type: String,default: "pop",},vSpace: {type: Number,},explain: {type: String,},imgSize: {type: Object,default() {return {width: "310px",height: "155px",};},},blockSize: {type: Object,},barSize: {type: Object,},},data() {return {// showBox:true,clickShow: false,// 内部类型verifyType: undefined,// 所用组件类型componentType: undefined,};},methods: {/*** i18n* @description 兼容vue-i18n 调用$t来转换ok* @param {String} text-被转换的目标* @return {String} i18n的结果* */i18n(text) {if (this.$t) {return this.$t(text);} else {// 兼容不存在的语言let i18n =this.$options.i18n.messages[this.locale] ||this.$options.i18n.messages["en-US"];return i18n[text];}},/*** refresh* @description 刷新* */refresh() {if (this.instance.refresh) {this.instance.refresh();}},closeBox() {this.clickShow = false;this.refresh();},show() {if (this.mode == "pop") {this.clickShow = true;}},},computed: {instance() {return this.$refs.instance || {};},showBox() {if (this.mode == "pop") {return this.clickShow;} else {return true;}},},watch: {captchaType: {immediate: true,handler(captchaType) {switch (captchaType.toString()) {case "blockPuzzle":this.verifyType = "2";this.componentType = "VerifySlide";break;case "clickWord":this.verifyType = "";this.componentType = "VerifyPoints";break;}},},},components: {VerifySlide,VerifyPoints,},
};
</script>
<style>
.verifybox {position: relative;box-sizing: border-box;border-radius: 18px;border: 1px solid #e4e7eb;background-color: #fff;box-shadow: 0 0 10px rgba(0, 0, 0, 0.3);left: 50%;top: 30%;transform: translate(-50%, -60%);
}.verifybox-top {padding: 0 15px;height: 50px;line-height: 50px;text-align: left;font-size: 16px;color: #45494c;border-bottom: 1px solid #e4e7eb;box-sizing: border-box;
}.verifybox-bottom {padding: 15px;box-sizing: border-box;
}.verifybox-close {position: absolute;top: 13px;right: 9px;width: 24px;height: 24px;text-align: center;cursor: pointer;
}.mask {position: fixed;top: 0;left: 0;z-index: 1001;width: 100%;height: 100vh;/* display: none; */transition: all 0.5s;
}.verify-tips {position: absolute;left: 0px;bottom: 0px;width: 100%;height: 30px;line-height: 30px;color: #fff;
}.suc-bg {background-color: rgba(92, 184, 92, 0.5);filter: progid:DXImageTransform.Microsoft.gradient(startcolorstr=#7f5CB85C, endcolorstr=#7f5CB85C);
}.err-bg {background-color: rgba(217, 83, 79, 0.5);filter: progid:DXImageTransform.Microsoft.gradient(startcolorstr=#7fD9534F, endcolorstr=#7fD9534F);
}.tips-enter,
.tips-leave-to {bottom: -30px;
}.tips-enter-active,
.tips-leave-active {transition: bottom 0.5s;
}/* ---------------------------- */
/*常规验证码*/
.verify-code {font-size: 20px;text-align: center;cursor: pointer;margin-bottom: 5px;border: 1px solid #ddd;
}.cerify-code-panel {height: 100%;overflow: hidden;
}.verify-code-area {float: left;
}.verify-input-area {float: left;width: 60%;padding-right: 10px;
}.verify-change-area {line-height: 30px;float: left;
}.varify-input-code {display: inline-block;width: 100%;height: 25px;
}.verify-change-code {color: #337ab7;cursor: pointer;
}.verify-btn {width: 200px;height: 30px;background-color: #337ab7;color: #ffffff;border: none;margin-top: 10px;
}/*滑动验证码*/
.verify-bar-area {position: relative;background: #ffffff;text-align: center;-webkit-box-sizing: content-box;-moz-box-sizing: content-box;box-sizing: content-box;border: 1px solid #ddd;-webkit-border-radius: 4px;
}.verify-bar-area .verify-move-block {position: absolute;top: 0px;left: 0;background: #fff;cursor: pointer;-webkit-box-sizing: content-box;-moz-box-sizing: content-box;box-sizing: content-box;box-shadow: 0 0 2px #888888;-webkit-border-radius: 1px;
}.verify-bar-area .verify-move-block:hover {background-color: #337ab7;color: #ffffff;
}.verify-bar-area .verify-left-bar {position: absolute;top: -1px;left: -1px;background: #f0fff0;cursor: pointer;-webkit-box-sizing: content-box;-moz-box-sizing: content-box;box-sizing: content-box;border: 1px solid #ddd;
}.verify-img-panel {margin: 0;-webkit-box-sizing: content-box;-moz-box-sizing: content-box;box-sizing: content-box;border-top: 1px solid #ddd;border-bottom: 1px solid #ddd;border-radius: 3px;position: relative;
}.verify-img-panel .verify-refresh {width: 25px;height: 25px;text-align: center;padding: 5px;cursor: pointer;position: absolute;top: 0;right: 0;z-index: 2;
}.verify-img-panel .icon-refresh {font-size: 20px;color: #fff;
}.verify-img-panel .verify-gap {background-color: #fff;position: relative;z-index: 2;border: 1px solid #fff;
}.verify-bar-area .verify-move-block .verify-sub-block {position: absolute;text-align: center;z-index: 3;/* border: 1px solid #fff; */
}.verify-bar-area .verify-move-block .verify-icon {font-size: 18px;
}.verify-bar-area .verify-msg {z-index: 3;
}/*字体图标的css*/
/*@font-face {font-family: "iconfont";*/
/*src: url('../fonts/iconfont.eot?t=1508229193188'); !* IE9*!*/
/*src: url('../fonts/iconfont.eot?t=1508229193188#iefix') format('embedded-opentype'), !* IE6-IE8 *!*/
/*url('data:application/x-font-woff;charset=utf-8;base64,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') format('woff'),*/
/*url('../fonts/iconfont.ttf?t=1508229193188') format('truetype'), !* chrome, firefox, opera, Safari, Android, iOS 4.2+*!*/
/*url('../fonts/iconfont.svg?t=1508229193188#iconfont') format('svg'); !* iOS 4.1- *!*/
/*}*/.iconfont {font-family: "iconfont" !important;font-size: 16px;font-style: normal;-webkit-font-smoothing: antialiased;-moz-osx-font-smoothing: grayscale;
}.icon-check:before {content: " ";display: block;width: 16px;height: 16px;position: absolute;margin: auto;left: 0;right: 0;top: 0;bottom: 0;z-index: 9999;background-image: url("");background-size: contain;
}.icon-close:before {content: " ";display: block;width: 16px;height: 16px;position: absolute;margin: auto;left: 0;right: 0;top: 0;bottom: 0;z-index: 9999;background-image: url("");background-size: contain;
}.icon-right:before {content: " ";display: block;width: 16px;height: 16px;position: absolute;margin: auto;left: 0;right: 0;top: 0;bottom: 0;background-size: cover;z-index: 9999;background-image: url("");background-size: contain;
}.icon-refresh:before {content: " ";display: block;width: 16px;height: 16px;position: absolute;margin: auto;left: 0;right: 0;top: 0;bottom: 0;z-index: 9999;background-image: url("");background-size: contain;
}
</style>

4.组件的使用

  <Verify @success="verifySuccess" :mode="'pop'" :captchaType="'blockPuzzle'":imgSize="{ width: '330px', height: '155px' }" ref="verify" />//使用时
this.$refs.verify.show() //展示滑块验证组件
//滑块验证成功后调取verifySuccess事件
verifySuccess(paramas){//其中 params.captchaVerification和登陆信息一起传给后端实现滑块验证完成
}

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/278440.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

centos7.9的GUI桌面样式不符合默认熟悉的操作习惯

一、问题描述&#xff1a; 原因&#xff1a;桌面样式选错了。 二、解决&#xff1a; 1.先登进去LogOut。 2.点击设置的工具图标中的GNOME Classic即可恢复成默认操作习惯的桌面样式。 3.恢复到默认熟悉的操作界面

(一)Neo4j下载安装以及初次使用

&#xff08;一&#xff09;下载 官网地址&#xff1a;Neo4j Graph Database & AnamConnect data as its stored with Neo4j. Perform powerful, complex queries at scale and speed with our graph data platform.https://neo4j.com/ &#xff08;二&#xff09;安装并配…

系统明天上线,PG表空间还不规划好疯狂给同事埋雷

&#x1f4e2;&#x1f4e2;&#x1f4e2;&#x1f4e3;&#x1f4e3;&#x1f4e3; 哈喽&#xff01;大家好&#xff0c;我是【IT邦德】&#xff0c;江湖人称jeames007&#xff0c;10余年DBA及大数据工作经验 一位上进心十足的【大数据领域博主】&#xff01;&#x1f61c;&am…

天眼销批量查询功能上线

天眼销是一款提供企业线索的产品&#xff0c;致力于帮助客户获取最新的企业联系方式、工商信息等关键数据。 数据库收录全国 3.3亿及以上企业(含个体)线索&#xff0c;涵盖企业名称、企业状态、注册时间、注册资本、经营范围、法人信息、联系方式等维度&#xff0c;为用户提供…

【python开发】并发编程(上)

并发编程&#xff08;上&#xff09; 一、进程和线程&#xff08;一&#xff09;多线程&#xff08;二&#xff09;多进程&#xff08;三&#xff09;GIL锁 二、多线程开发&#xff08;一&#xff09;t.start()&#xff08;二&#xff09;t.join()&#xff08;三&#xff09;t.…

数据的响应式:实现动态数据驱动的技巧

&#x1f90d; 前端开发工程师、技术日更博主、已过CET6 &#x1f368; 阿珊和她的猫_CSDN博客专家、23年度博客之星前端领域TOP1 &#x1f560; 牛客高级专题作者、打造专栏《前端面试必备》 、《2024面试高频手撕题》 &#x1f35a; 蓝桥云课签约作者、上架课程《Vue.js 和 E…

鸿蒙Harmony应用开发—ArkTS声明式开发(基础手势:StepperItem)

用作Stepper组件的页面子组件。 说明&#xff1a; 该组件从API Version 8开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 子组件 支持单个子组件。 接口 StepperItem() 属性 参数名参数类型参数描述prevLabelstring设置左侧文本按钮内…

upload文件上传漏洞复现

什么是文件上传漏洞&#xff1a; 文件上传漏洞是指由于程序员在对用户文件上传部分的控制不足或者处理缺陷&#xff0c;而导致的用户可以越过其本身权限向服务器上上传可执行的动态脚本文件。这里上传的文件可以是木马&#xff0c;病毒&#xff0c;恶意脚本或者WebShell等。“…

LeetCode108题:将有序数组转换为二叉搜索树(python3)

一个容易想到的思路&#xff1a;使用 nums 中最靠近中心的位置作为整棵 BST 的根节点&#xff0c;确保左右子树节点数量平衡。随后递归构造 nums 中下标范围为 [0,mid−1]作为左子树&#xff0c;递归构造 nums 中下标范围为 [mid1,n−1]作为右子树。 # Definition for a binar…

理论学习:with torch.no_grad()

如果不加上“with torch.no_grad():”&#xff0c;模型参数会发生改变吗&#xff1f; 如果不使用with torch.no_grad():&#xff0c;在进行模型推理&#xff08;即计算outputs_cls net(inputs[batch_size//2:])这一步&#xff09;时&#xff0c;模型参数不会发生改变&#xf…

鸿蒙Harmony应用开发—ArkTS声明式开发(基础手势:Web)中篇

onBeforeUnload onBeforeUnload(callback: (event?: { url: string; message: string; result: JsResult }) > boolean) 刷新或关闭场景下&#xff0c;在即将离开当前页面时触发此回调。刷新或关闭当前页面应先通过点击等方式获取焦点&#xff0c;才会触发此回调。 参数…

开发验证一切正常,而测试人员在性能测试时偶发报错,如何解决?

在业务系统逻辑实现中&#xff0c;经常涉及异步执行、异步更新场景的开发和使用。但在性能测试中&#xff0c;经常会出现因为异步逻辑设计不合理引发的不可预知问题&#xff0c;比如在开发验证时一切正常&#xff0c;测试人员在性能测试时偶发报错。 本文从Spring事务、业务逻辑…

前端三件套 | 综合练习:模拟抽奖活动,实现一个简单的随机抽取并显示三名获胜者

随机运行结果如下&#xff1a; 参考代码如下&#xff1a; <!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><tit…

P2934 [USACO09JAN] Safe Travel G 题解

题意 给定一张 n n n 个点 m m m 条边的无向图&#xff0c;对于每个除 1 1 1 以外的点 u u u&#xff0c;求在不允许经过原来从 1 1 1 到 u u u 的最短路径的最后一条边时&#xff0c; 1 1 1 到 u u u 的最短路。 保证 1 1 1 到其他点的最短路唯一&#xff0c;无解输出…

git撤回代码提交commit或者修改commit提交注释

执行commit后&#xff0c;还没执行push时&#xff0c;想要撤销之前的提交commit 撤销提交 使用命令&#xff1a; git reset --soft HEAD^命令详解&#xff1a; HEAD^ 表示上一个版本&#xff0c;即上一次的commit&#xff0c;也可以写成HEAD~1 如果进行两次的commit&#xf…

鸿蒙Socket通信示例(TCP通信)

前言 DevEco Studio版本&#xff1a;4.0.0.600 参考链接&#xff1a;OpenHarmony Socket 效果 TCPSocket 1、bind绑定本地IP地址 private bindTcpSocket() {let localAddress resolveIP(wifi.getIpInfo().ipAddress)console.info("111111111 localAddress: " …

配置阿里云加速器

国内镜像中心常用阿里云或者网易云。在本地docker中指定要使用国内加速器的地址后&#xff0c;就可以直接从阿里云镜像中心下载镜像。 2024阿里云-上云采购季-阿里云 根据操作系统来选择。

RequestResponse使用

文章目录 一、Request&Response介绍二、Request 继承体系三、Request 获取请求数据1、获取请求数据方法&#xff08;1&#xff09;、请求行&#xff08;2&#xff09;、请求头&#xff08;3&#xff09;、请求体 2、通过方式获取请求参数3、IDEA模板创建Servlet4、请求参数…

【AUTOSAR】【通信栈】Fls

AUTOSAR专栏——总目录-CSDN博客文章浏览阅读592次。本文主要汇总该专栏文章,以方便各位读者阅读。https://blog.csdn.net/qq_42357877/article/details/132072415?csdn_share_tail=%7B%22type%22%3A%22blog%22%2C%22rType%22%3A%22article%22%2C%22rId%22%3A%22132072415%22…

金融知识分享系列之:MACD指标精讲

金融知识分享系列之&#xff1a;MACD指标精讲 一、MACD指标二、指标原理三、MACD指标参考用法四、MACD计算步骤五、MACD分析要素六、根据快线DIF位置判断趋势七、金叉死叉作为多空信号八、快线位置交叉信号九、指标背离判断行情反转十、差离值的正负十一、差离值的变化十二、指…