滑块验证码

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.mzph.cn/news/751301.shtml

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

相关文章

Docker出现容器名称重复如何解决

假如你的重复容器名称是mysql5 删除已存在的容器&#xff1a;如果你不再需要那个已经存在的名为“mysql5”的容器&#xff0c;你可以删除它。使用下面的命令&#xff1a; docker rm -f mysql5这条命令会强制删除正在运行的容器。一旦容器被删除&#xff0c;你就可以重新使用这个…

Java 面向对象(类与对象 成员方法 方法重载 可变参数 构造方法 / 构造器 this关键字 包 访问修饰符)

目录 一、类与对象1. 类与对象的定义2. 类和对象的内存分配机制 二、成员方法1. 成员方法的定义2. 方法的调用机制3. 成员方法传参机制 三、方法重载四、可变参数1. 基本概念2. 基本语法3. 应用 五、 构造方法 / 构造器1. 特点2. 使用案例3. 对象创建的流程 六、this关键字1. 运…

P6安装:安装P6提示1433端口无效

错误描述 尝试运行 Microsoft SQL Server 2005 的 Primavera P6 数据库时&#xff0c;遇到以下错误&#xff1a; SQLServerException: The TCP/IP connection to the host [name], port 1433 has failed. Error: “Connection refused: connect. Verify the connection prope…

MATLAB的多项式相加

多项式的加减在阶次相同的情况下可直接运算&#xff0c;若两个相加减的多项式阶次不同&#xff0c;则低阶多项式必须用零填补高阶项系数&#xff0c;使其与高阶多项式有相同的阶次。而且通常情况下&#xff0c;进行加减的两个多项 式的阶次不会相同&#xff0c;这时可以自定义一…

视频素材库大全免费无水印必备素材网站,整理分享抖音视频素材剪辑必备

想要做出容易上热门的抖音视频作品&#xff0c;你一定要找到合适的短视频剪辑素材&#xff0c;今天我就给你收集了视频素材库大全免费无水印必备素材网站&#xff0c;都给你提供了海量的短视频素材库大全免费无水印&#xff0c;从风景&#xff0c;美食&#xff0c;动漫&#xf…

从排序算法的艺术看C语言qsort函数的魅力:一场数据的时空穿越

欢迎来到白刘的领域 Miracle_86.-CSDN博客 系列专栏 C语言知识 先赞后看&#xff0c;已成习惯 创作不易&#xff0c;多多支持&#xff01; 目录 一 、回调函数 二、qsort函数 1.qsort函数排序整型数据 2.qsort函数排序结构数据 一 、回调函数 何为回调函数&#xff1…

基于matlab使用 fmincon 函数来进行有约束条件的最小化问题求解

一、一般步骤 生成带有噪声的正态分布数据&#xff1b;定义拟合模型。 model (params, x) normpdf(x, params(1), params(2)); 初始参数猜测 initial_guess [mu, sigma]; 设置约束条件 lb [0, 0]; % 参数的最小值 ub [10, 10]; % 参数的最大值 定义优化问题 opts …

deepseek-coder模型量化

1 简介 DeepSeek-Coder在多种编程语言和各种基准测试中取得了开源代码模型中最先进的性能。 为尝试在开发板进行部署&#xff0c;首先利用llama.cpp对其进行量化。 2 llama.cpp安装 git clone之后进入文件夹make即可&#xff0c;再将依赖补全pip install -r requirements.tx…

【吊打面试官系列】Java虚拟机JVM篇 - 关于双亲委派模型

大家好&#xff0c;我是锋哥。今天分享关于JVM双亲委派模型的JVM面试题&#xff0c;希望对大家有帮助&#xff1b; 什么是双亲委派模型&#xff1f; 双亲委派模型针对的是 Java 虚拟机中三个类加载器的&#xff0c;这三个类加载器分别是&#xff1a; 启动类加载器&#xff08;B…

node.js快速入门-day03

个人名片&#xff1a; &#x1f60a;作者简介&#xff1a;一名大二在校生 &#x1f921; 个人主页&#xff1a;坠入暮云间x &#x1f43c;座右铭&#xff1a;给自己一个梦想&#xff0c;给世界一个惊喜。 &#x1f385;**学习目标: 坚持每一次的学习打卡 文章目录 web服务器创建…

力扣hot100:34. 在排序数组中查找元素的第一个和最后一个位置(二分查找的理解)

我们知道使用二分查找能找到值所在的位置。假如我们在找到值后仍然不断的更新指针会发生什么&#xff1f;我们可以利用这一点来找到最左边的以及最右边的值。 如果当nums[mid]target时&#xff0c;使得 rightmid-1&#xff0c;那么最终会使得target在right的右边。 如果当nums[…

海外媒体宣发套餐推广攻略实现品牌全球化-华媒舍

如今&#xff0c;在全球经济一体化的浪潮下&#xff0c;品牌全球化已成为企业成功的重要因素之一。海外市场作为一个巨大而具有潜力的机会&#xff0c;吸引着越来越多的企业前往探索。而在海外市场的推广过程中&#xff0c;海外媒体宣发套餐成为了重要的推广方式之一。本文将为…

【S5PV210_视频编解码项目】裸机开发:实现按键的外部中断处理

加粗样式本文所作内容&#xff1a; 基于S5PV210芯片实现按键的外部中断处理程序&#xff0c;搭建中断处理流程框架 S5PV210对于中断处理的操作流程 1 外部中断得到触发&#xff1a; 1&#xff09;外部中断在初始化阶段得到使能 2&#xff09;外界达到了外部中断的触发条件 …

24考研数学最大教训❗️660/880过时了?

我没看错吧&#xff0c;说660题和880题过时了&#xff1f; 660题和880题好好用&#xff0c;这俩很经典不会过时。 660题是客观题训练必刷的一本题集&#xff0c;而880是强化阶段非常好的一本综合性题集。我本身在考研的时候使用的也是这两本题集&#xff0c;所以对这两本题集…

如何学习一个大型分布式Java项目

前言 很多同学在没有实习经验的时候看到一个多模块分布式项目总是有一种老虎吃天的无力感&#xff0c;就像我刚毕业去到公司接触项目的时候一样&#xff0c;模块多的夸张&#xff0c;想学都不知道从哪开始学&#xff0c;那么我们拿到一份代码后如何从头开始学习一个新项目呢。…

Oracle Primavera Analytics 是什么,与P6的关系?

前言 Oracle Primavera P6 Analytics 是与P6有关的一个相对较新的模块&#xff0c;Primavera 用户社区在很大程度上尚未对其进行探索。 那么它到底有什么作用呢&#xff1f; 通过了解得知它旨在通过深入了解组织的项目组合绩效&#xff0c;帮助高级管理层对其项目组合做出更好…

MySQL | 库的操作 | 表的操作

目录 1. 库的操作 1.1. 创建数据库 1.2. 字符集和校验规则 1.2.1. 查看系统默认字符集以及校验规则 1.2.2. 查看数据库支持的字符集 1.2.3. 查看数据库支持的字符校验规则 2. 操作数据库 2.1. 查看数据库 2.2. 显示创建语句 2.3. 修改数据库 2.4. 数据库的删除 2.4.…

维基百科推广秘诀13个方法助你成为行业领导者-华媒舍

维基百科&#xff08;Wikipedia&#xff09;作为全球最大、最权威的在线百科全书&#xff0c;拥有海量的知识内容&#xff0c;被广大用户广泛使用。对于任何一个领域的从业者来说&#xff0c;建立自己的维基百科页面&#xff0c;无疑是提升行业影响力的重要手段。本文将向您介绍…

Linux学习(4)——使用编辑器

1.gedit编辑器 简单易懂&#xff0c;依赖图形界面。可以使用ctrlc ctrlv等快捷键&#xff0c;ctrls进行保存&#xff0c;与windows系统中相类似。 2.vi/vim编辑器 vi/vim可以直接通过控制台的终端完成文本的编辑&#xff0c;不依赖图形界面&#xff0c;使用范围更广。它的编辑…

Redis数据结构对象之字符串对象

字符串对象 字符串对象的编码可以是int、raw或者embstr 如果一个字符串对象保存的是整数值&#xff0c;并且这个整数值可以用long类型来表示&#xff0c;那么字符串对象会将整数值保存在字符串对象结构的ptr属性里面(将void *转换成long)&#xff0c;并且将字符串对象的编码设…