前言:
最近小编闲暇之余,想到ruoyi后台的计算验证有点单一,网上很多的网站都有像滑块图片验证或文字点选验证的机器验证,所以就想集成一下,可以是百度半天网上终究是没有特别详细的教程,所以小编通过参考网络教程加上自身实践,实现了这几种验证,效果还不错。
效果展示:
小编将这些验证部分统一放入一个组件文件夹里面,目录如图:
在components里面新建Veriftion文件夹用于存放这些验证的组件。其中SilderVerify.vue是简单滑动验证组件,Verify用于统一管理写好的滑动图片和文字点选组件。
ruoyi-ui里面如果没有crypto-js则需要先安装一下:npm install crypto-js --save-dev
流程简述:让用户进行滑动验证后,展示图片验证或文字点选验证,验证通过后才进行登录。
SilderVerify.vue组件:
<template> <div class="drag" :style="style" > <div class="background"/> <div class="text shadow" onselectstart="return false" :style="{ color: textColor }"> <slot name="content"> {{ content }} </slot> </div> <div class="slider" :style="{height,width:sliderWidth}"> <slot v-if="icon" name="icon"> <i :class="icon"></i> </slot> <slot v-else name="icon"> >> </slot> </div> </div> </template> <script> const debounce = (function () { let timer = 0 return function (callback, ms) { clearTimeout(timer) timer = setTimeout(callback, ms) } })(); export default { name: 'slider-verify-code', model: { event: 'change', prop: 'isLock' }, props: { isLock: { //解锁状态 type: [String, Boolean, Number, Object], required: true, default: false }, icon: { //滑块图标 type: [String], default: "el-icon-d-arrow-right" }, activeValue: { //滑块解锁后的值 type: [String, Boolean, Number, Object], default: true }, inactiveValue: { //滑块解锁前的值 type: [String, Boolean, Number, Object], default: false }, content: { //滑块的文字 type: [String], default: "请向右拖动滑块" }, height: { //高度 type: [String], default: "40px" }, sliderWidth: { //滑块宽度 type: [String], default: "40px" }, background: { //高度 type: [String], default: "#e8e8e8" }, textColor: { //滑块的文字颜色 type: [String], default: "#777" } }, watch: { isLock(data) { //重置样式 !data && this.init(); }, }, computed: { style() { const {height, background} = this; return {height, 'line-height': height, background}; }, resize() { return document.body.clientWidth; }, }, mounted() { this.init(); window.onresize = () => { debounce(() => { this.init(); }, 120); }; }, methods: { /** * 定义一个获取DOM元素的方法-选择器 */ selector(selector) { return document.querySelector(selector); }, /** * 初始化 */ init() { const box = this.selector('.drag'); //容器 const background = this.selector('.background'); //背景 const text = this.selector('.text'); //文字 const slider = this.selector('.slider');//滑块 const distance = box.offsetWidth - slider.offsetWidth;//滑动成功的宽度(距离) let success = this.inactiveValue;//是否通过验证的标志 // 初始化的时候 清除所有属性 slider.style.transition = null; background.style.transition = null; slider.style.left = 0 + 'px'; background.style.width = 0 + 'px'; text.innerHTML = this.content; slider.innerHTML = '<i class="el-icon-d-arrow-right"></i>'; slider.style.color = '#3fcd26'; //二、给滑块注册鼠标按下事件 slider.onmousedown = (event) => { //1.鼠标按下之前必须清除掉后面设置的过渡属性 slider.style.transition = null; background.style.transition = null; //说明:clientX 事件属性会返回当事件被触发时,鼠标指针向对于浏览器页面(或客户区)的水平坐标。 //2.当滑块位于初始位置时,得到鼠标按下时的水平位置 const ev = event || window.event; const downX = ev.clientX; //三、给文档注册鼠标移动事件 document.onmousemove = (e) => { const evt = e || window.event;//是为了更好的兼容IE浏览器和非ie浏览器。在ie浏览器中,window.event是全局变量,在非ie中,就需要自己传入一个参数来获取event啦,所以就有了var e = e||window.event //1.获取鼠标移动后的水平位置 const moveX = evt.clientX; //2.得到鼠标水平位置的偏移量(鼠标移动时的位置 - 鼠标按下时的位置) let offsetX = moveX - downX; //3.在这里判断一下:鼠标水平移动的距离 与 滑动成功的距离 之间的关系 if (offsetX > distance) { offsetX = distance;//如果滑过了终点,就将它停留在终点位置 } else if (offsetX < 0) { offsetX = 0;//如果滑到了起点的左侧,就将它重置为起点位置 } //4.根据鼠标移动的距离来动态设置滑块的偏移量和背景颜色的宽度 slider.style.left = offsetX + 'px'; background.style.width = offsetX + 'px'; //如果鼠标的水平移动距离 = 滑动成功的宽度 if (offsetX == distance) { //1.设置滑动成功后的样式 text.innerHTML = '验证成功'; text.style.color = '#53C300'; slider.innerHTML = '<i class="el-icon-success"></i>'; slider.style.color = '#53C300'; //2.设置滑动成功后的状态 success = this.activeValue; //成功后,清除掉鼠标按下事件和移动事件(因为移动时并不会涉及到鼠标松开事件) slider.onmousedown = null; document.onmousemove = null; //3.成功解锁后的回调函数 setTimeout(() => { this.$emit('change', this.activeValue); }, 100); } }; //四、给文档注册鼠标松开事件 document.onmouseup = () => { //如果鼠标松开时,滑到了终点,则验证通过 if (success == this.activeValue) return true; //反之,则将滑块复位(设置了1s的属性过渡效果) slider.style.left = 0; background.style.width = 0; slider.style.transition = 'left 1s ease'; background.style.transition = 'width 1s ease'; //只要鼠标松开了,说明此时不需要拖动滑块了,那么就清除鼠标移动和松开事件。 document.onmousemove = null; document.onmouseup = null; }; }; /* 移动端 */ //二、给滑块注册鼠标按下事件 slider.ontouchstart = (event) => { const touch = event.changedTouches[0]; //1.鼠标按下之前必须清除掉后面设置的过渡属性 slider.style.transition = null; background.style.transition = null; //说明:clientX 事件属性会返回当事件被触发时,鼠标指针向对于浏览器页面(或客户区)的水平坐标。 //2.当滑块位于初始位置时,得到鼠标按下时的水平位置 const downX = touch.pageX; //三、给文档注册鼠标移动事件 document.ontouchmove = (e) => { const tev = e.changedTouches[0]; //1.获取鼠标移动后的水平位置 const moveX = tev.pageX; //2.得到鼠标水平位置的偏移量(鼠标移动时的位置 - 鼠标按下时的位置) let offsetX = moveX - downX; //3.在这里判断一下:鼠标水平移动的距离 与 滑动成功的距离 之间的关系 if (offsetX > distance) { offsetX = distance;//如果滑过了终点,就将它停留在终点位置 } else if (offsetX < 0) { offsetX = 0;//如果滑到了起点的左侧,就将它重置为起点位置 } //4.根据鼠标移动的距离来动态设置滑块的偏移量和背景颜色的宽度 slider.style.left = offsetX + 'px'; background.style.width = offsetX + 'px'; //如果鼠标的水平移动距离 = 滑动成功的宽度 if (offsetX == distance) { //1.设置滑动成功后的样式 text.innerHTML = '验证成功'; text.style.color = '#53C300'; slider.innerHTML = '√'; slider.style.color = '#53C300'; //2.设置滑动成功后的状态 success = this.activeValue; //成功后,清除掉鼠标按下事件和移动事件(因为移动时并不会涉及到鼠标松开事件) slider.ontouchstart = null; document.ontouchmove = null; //3.成功解锁后的回调函数 setTimeout(() => { this.$emit('change', this.activeValue); // console.log('解锁成功'); }, 100); } }; //四、给文档注册鼠标松开事件 document.ontouchend = () => { //如果鼠标松开时,滑到了终点,则验证通过 if (success == this.activeValue) return true; //反之,则将滑块复位(设置了1s的属性过渡效果) slider.style.left = 0; background.style.width = 0; slider.style.transition = 'left 1s ease'; background.style.transition = 'width 1s ease'; //只要鼠标松开了,说明此时不需要拖动滑块了,那么就清除鼠标移动和松开事件。 document.ontouchmove = null; document.ontouchend = null; }; }; } } }; </script> <style scoped lang="scss"> * { margin: 0px; padding: 0px; font-family: "微软雅黑"; box-sizing: border-box; } .drag { height: 2.5rem; line-height: 2.5rem; background-color: #e8e8e8; position: relative; margin: 0 auto; border-radius: 3px; } .background { width: 2.5rem; height: 100%; position: absolute; background-color: #53C300; border-radius: 3px 0 0 3px; } .text { position: absolute; width: 100%; height: 100%; text-align: center; user-select: none; } .slider { width: 2.5rem; height: 2.375rem; position: absolute; border: 1px solid #ccc; cursor: move; font-family: "宋体"; text-align: center; background-color: #fff; user-select: none; color: #666; } .shadow { text-align: center; background: -webkit-gradient(linear, left top, right top, color-stop(0, #4d4d4d), color-stop(.2, #5d5d5d), color-stop(.4, #6d6d6d), color-stop(.5, white), color-stop(.6, #6d6d6d), color-stop(.8, #5d5d5d), color-stop(1, #4d4d4d)); -webkit-background-clip: text; -webkit-text-fill-color: transparent; -webkit-animation: animate 3s infinite; } @-webkit-keyframes animate { from { background-position: -80px; } to { background-position: 80px; } } @keyframes animate { from { background-position: -80px; } to { background-position: 80px; } } </style>
复制
Login.vue使用:
多余的代码小编这里就不贴了,仅贴一下需要使用的
<el-form-item v-if="captchaOnOff&&(captcha_type=='captcha_type_imgfill'||captcha_type=='captcha_type_clickword')" prop="isLock"> <SilderVerify ref="silderverify" v-model="loginForm.isLock" @change="handlerLock"></SilderVerify> </el-form-item> <el-form-item v-if="captchaOnOff&&(captcha_type=='captcha_type_imgfill'||captcha_type=='captcha_type_clickword')" prop="code"> <Verify @success="capctchaCheckSuccess" :mode="'pop'" :captchaType="captchaType" :imgSize="{ width: '330px', height: '155px' }" ref="verify" ></Verify> </el-form-item>
复制
<script> import Verify from "@/components/wujiweijunyi/Verifition/Verify"; import SilderVerify from "@/components/wujiweijunyi/Verifition/SilderVerify"; export default { components: { Verify,SilderVerify }, name: "Login", data() { const checkStatus = (rule, value, callback) => { if (!value) { return callback(new Error("请拖动滑块完成验证")); } else { if (this.loginForm.username == '' || this.loginForm.password == '' || !this.loginForm.username || !this.loginForm.password) { setTimeout(() => { this.loginForm.isLock = false; this.$refs.loginForm.validateField('username'); this.$refs.loginForm.validateField('password'); return callback(new Error("验证未通过")); }, 1); } callback(); } }; return { loginForm: { id: "", //默认用户名 pwd: "", //默认密码 rememberMe: false, //是否记住密码 code: "", //验证码的值 uuid: "" ,//验证码的uuid isLock:false, }, loginRules: { id: [{ required: true, trigger: "blur", message: "请输入您的账号" }], pwd: [{ required: true, trigger: "blur", message: "请输入您的密码" }], code: [{ required: true, trigger: "change", message: "请进行验证码验证" }], isLock: [ {validator: checkStatus, trigger: 'blur'}, ], }, captchaOnOff: true,// 验证码开关 captcha_type:"",//存放接口返回的验证类型 captchaType:"",//存放需要执行的验证方式 }, //#省略生命周期等代码# methods: { getCode() { let that = this;that.showCaptchatImg=false; //#省略请求接口代码,接口用于查询需要实现哪种验证方式# if(res.data.captcha_type=="captcha_type_imgfill"){ that.captchaType="blockPuzzle"; } if(res.data.captcha_type=="captcha_type_clickword"){ that.captchaType="clickWord"; } }, //滑块验证成功的回调 handlerLock(data){ let that=this; //验证成功后 data值为true if(data){ setTimeout(function() { that.$refs.verify.show(); }, 1000); } }, //图片验证或文字点选验证成功后的回调,有兴趣的用户后续可以研究一下根据返回的 params.captchaVerification 进行二次验证 capctchaCheckSuccess(params) { this.loginForm.code = params.captchaVerification; }, } } } <script>
复制
Login.vue使用组件到此差不多就结束了,下面贴一下其他组件的详细代码。
Verifition/api/index.js:
import request from '@/utils/request' //获取验证图片 export function reqGet(data) { return request({ url: '/base/base_anji/get', method: 'post', data }) } //滑动或者点选验证 export function reqCheck(data) { return request({ url: '/base/base_anji/check', method: 'post', data }) }
复制
Verifition/utils/ase.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(); }
复制
Verifition/utils/util.js:
export function resetSize(vm) { var img_width, img_height, bar_width, bar_height; //图片的宽度、高度,移动条的宽度、高度 var parentWidth = vm.$el.parentNode.offsetWidth || window.offsetWidth var parentHeight = vm.$el.parentNode.offsetHeight || window.offsetHeight if (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']
复制
Verifition/Verify/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=" pointBackImgBase ? 'data:image/png;base64,' + pointBackImgBase : defaultImg " ref="canvas" alt="" style="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,固定fixed mode: { 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", }; }, }, defaultImg: { type: String, default: "", }, }, 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((res) => { 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, clientUid: localStorage.getItem("point"), ts: Date.now(), // 现在的时间戳 }; reqGet(data).then((res) => { 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; } // 判断接口请求次数是否失效 if (res.repCode == "6201") { this.pointBackImgBase = null; } }); }, //坐标转换函数 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 }; }); return newPointArr; }, }, watch: { // type变化则全面刷新 type: { immediate: true, handler() { this.init(); }, }, }, mounted() { // 禁止拖拽 this.$el.onselectstart = function () { return false; }; }, }; </script>
复制
Verifition/Verify/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=" backImgBase ? 'data:image/png;base64,' + backImgBase : defaultImg " 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" alt="" style="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,固定fixed mode: { 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", }; }, }, defaultImg: { type: String, default: "", }, }, 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((res) => { if (res.repCode == "0000") { this.moveBlockBackgroundColor = "#5cb85c"; this.leftBarBorderColor = "#5cb85c"; this.iconColor = "#fff"; this.iconClass = "icon-check"; this.showRefresh = false; this.isEnd = true; 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, clientUid: localStorage.getItem("slider"), ts: Date.now(), };// 现在的时间戳 console.log("请求背景图片和验证图片 请求数据:"+JSON.stringify(data)); reqGet(data).then((res) => { console.log("滑块验证码请求返回:"+JSON.stringify(res)); 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; } // 判断接口请求次数是否失效 if (res.repCode == "6201") { this.backImgBase = null; this.blockBackImgBase = null; } }); }, }, watch: { // type变化则全面刷新 type: { immediate: true, handler() { this.init(); }, }, }, mounted() { // 禁止拖拽 this.$el.onselectstart = function () { return false; }; }, }; </script>
复制
Verifition/Verify.vue:
<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" :defaultImg="defaultImg" 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, // 默认图片 defaultImg: require("@/assets/images/default.jpg"), }; }, mounted() { this.uuid(); }, methods: { // 生成 uuid uuid() { var s = []; var hexDigits = "0123456789abcdef"; for (var i = 0; i < 36; i++) { s[i] = hexDigits.substr(Math.floor(Math.random() * 0x10), 1); } s[14] = "4"; // bits 12-15 of the time_hi_and_version field to 0010 s[19] = hexDigits.substr((s[19] & 0x3) | 0x8, 1); // bits 6-7 of the clock_seq_hi_and_reserved to 01 s[8] = s[13] = s[18] = s[23] = "-"; var slider = "slider" + "-" + s.join(""); var point = "point" + "-" + s.join(""); // 判断下是否存在 slider if (!localStorage.getItem("slider")) { localStorage.setItem("slider", slider); } if (!localStorage.getItem("point")) { localStorage.setItem("point", point); } }, /** * 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: 2px; border: 1px solid #e4e7eb; background-color: #fff; box-shadow: 0 0 10px rgba(0, 0, 0, 0.3); left: 50%; top: 50%; transform: translate(-50%, -50%); } .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; background: rgba(0, 0, 0, 0.3); /* 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>
复制
此组件在验证失败时候需要一张图片,小编放置如下:
前端部分到此基本就结束了,下面开始Java Spring Boot部分.
引入依赖:
<dependency> <groupId>com.github.anji-plus</groupId> <artifactId>captcha-spring-boot-starter</artifactId> <version>1.2.7</version> </dependency>
复制
主要目录如下:
CaptchaRedisServiceImpl复制
package com.wjwjy.web.BaseAnji.service.impl; import com.anji.captcha.service.CaptchaCacheService; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.stereotype.Service; import java.util.concurrent.TimeUnit; @Service public class CaptchaRedisServiceImpl implements CaptchaCacheService { @Autowired private StringRedisTemplate stringRedisTemplate; @Override public void set(String s, String s1, long l) { stringRedisTemplate.opsForValue().set(s, s1, l, TimeUnit.SECONDS); } @Override public boolean exists(String s) { return stringRedisTemplate.hasKey(s); } @Override public void delete(String s) { stringRedisTemplate.delete(s); } @Override public String get(String s) { return stringRedisTemplate.opsForValue().get(s); } @Override public String type() { return "redis"; } }
复制
在resources/META-INF/ 下新建services目录,然后在services下新建com.anji.captcha.service.CaptchaCacheService,这里需要注意,新建的是文件。
com.anji.captcha.service.CaptchaCacheService代码如下:
com.wjwjy.web.BaseAnji.service.impl.CaptchaRedisServiceImpl
复制
这里需要注意,com.anji.captcha.service.CaptchaCacheService的内容必须是CaptchaRedisServiceImpl的路径。
BaseAnjiController:复制
package com.wjwjy.web.BaseAnji.controller; import com.anji.captcha.model.common.ResponseModel; import com.anji.captcha.model.vo.CaptchaVO; import com.anji.captcha.service.CaptchaService; import lombok.extern.slf4j.Slf4j; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.web.bind.annotation.*; @RestController @CrossOrigin @Slf4j @RequestMapping(value="/api/base/base_anji") public class BaseAnjiController { @Autowired private CaptchaService captchaService; @PostMapping("/get") public ResponseModel get(@RequestBody CaptchaVO captchaVO) { return captchaService.get(captchaVO); } @PostMapping("/check") public ResponseModel check(@RequestBody CaptchaVO captchaVO) { return captchaService.check(captchaVO); } @PostMapping("/verify") public ResponseModel verify(@RequestBody CaptchaVO captchaVO) { return captchaService.verification(captchaVO); } }
复制
Java的请求代码到这里基本结束,但还需配置一下aj-captcha
新建Config类:
package com.wjwjy.trendscommon.config; import com.anji.captcha.model.common.Const; import com.anji.captcha.service.CaptchaService; import com.anji.captcha.service.impl.CaptchaServiceFactory; import com.anji.captcha.util.Base64Utils; import com.anji.captcha.util.FileCopyUtils; import com.anji.captcha.util.ImageUtils; import com.anji.captcha.util.StringUtils; import lombok.extern.slf4j.Slf4j; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.context.annotation.DependsOn; import org.springframework.core.io.Resource; import org.springframework.core.io.support.PathMatchingResourcePatternResolver; import org.springframework.core.io.support.ResourcePatternResolver; import java.util.HashMap; import java.util.Map; import java.util.Properties; @Slf4j @Configuration public class Config { @Bean @DependsOn("AjCaptchaCacheService") public CaptchaService captchaService() { Properties config = new Properties(); // try { // try (InputStream input = CaptchaConfig.class.getClassLoader() // .getResourceAsStream("application.properties")) { // config.load(input); // } // }catch (Exception ex){ // ex.printStackTrace(); // } //各种参数设置.... //缓存类型redis/local/.... config.put(Const.CAPTCHA_CACHETYPE, "local");//缓存类型 config.put(Const.CAPTCHA_WATER_MARK, "吾疾唯君医Java");//水印文字 config.put(Const.CAPTCHA_FONT_TYPE, "宋体");//字体 config.put(Const.CAPTCHA_TYPE, "default");//blockPuzzle 图片滑块 clickWord 文字点选 default默认两者都实例化 config.put(Const.CAPTCHA_INTERFERENCE_OPTIONS, "0");//滑动干扰项(0/1/2) config.put(Const.ORIGINAL_PATH_JIGSAW, ""); config.put(Const.ORIGINAL_PATH_PIC_CLICK, ""); config.put(Const.CAPTCHA_SLIP_OFFSET, 5);//校验滑动拼图允许误差偏移量(默认5像素) config.put(Const.CAPTCHA_AES_STATUS, "true"); config.put(Const.CAPTCHA_WATER_FONT, "宋体");//aes加密坐标开启或者禁用(true|false) config.put(Const.CAPTCHA_CACAHE_MAX_NUMBER, "1000"); config.put(Const.CAPTCHA_TIMING_CLEAR_SECOND, "180"); //更多自定义参数,请参考service/springboot/../resources/application.properties if ((StringUtils.isNotBlank(config.getProperty(Const.ORIGINAL_PATH_JIGSAW)) && config.getProperty(Const.ORIGINAL_PATH_JIGSAW).startsWith("classpath:")) || (StringUtils.isNotBlank(config.getProperty(Const.ORIGINAL_PATH_PIC_CLICK)) && config.getProperty(Const.ORIGINAL_PATH_PIC_CLICK).startsWith("classpath:"))) { //自定义resources目录下初始化底图 config.put(Const.CAPTCHA_INIT_ORIGINAL, "true"); initializeBaseMap(config.getProperty(Const.ORIGINAL_PATH_JIGSAW), config.getProperty(Const.ORIGINAL_PATH_PIC_CLICK)); } CaptchaService s = CaptchaServiceFactory.getInstance(config); return s; } private static void initializeBaseMap(String jigsaw, String picClick) { ImageUtils.cacheBootImage(getResourcesImagesFile(jigsaw + "/original/*.png"), getResourcesImagesFile(jigsaw + "/slidingBlock/*.png"), getResourcesImagesFile(picClick + "/*.png")); } public static Map<String, String> getResourcesImagesFile(String path) { Map<String, String> imgMap = new HashMap<>(); ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver(); try { Resource[] resources = resolver.getResources(path); for (Resource resource : resources) { byte[] bytes = FileCopyUtils.copyToByteArray(resource.getInputStream()); String string = Base64Utils.encodeToString(bytes); String filename = resource.getFilename(); imgMap.put(filename, string); } } catch (Exception e) { e.printStackTrace(); } return imgMap; } }
复制
到这里,验证实现的代码就基本结束了,如有不足之处欢迎大家指正。还希望大家点点关注,多多支持,谢谢~