微信小程序中封装天爱验证码

前些时候,开发一码游小程序,产品提出一个需求,需要在小程序中使用图片滑块校验,需要真实的,配合后端请求的,不要前端写的假验证码。恰巧一码游后台登陆就采用的是天爱的图形验证码。后端接口可以直接使用,可前端验证码组件需要如何实现呢?

由于开发时间有限,我没有从头一点点码代码,而是从uniapp的插件市场(点击打开)搜索到一款他人封装好的插件(x-verify-code),此插件就是基于tianai-captcha 封装,经过部分改动,结合我们项目的特点,成功将天爱验证码小程序uniapp版本插件搞定。

有需要的可以直接拿来用。

验证码组件

ymy-mp/src/components/tianai-verify-code/index.vue

html 复制代码
<template>
	<view class="verify-wrap" v-if="verifyShow">
		<view class="verify-code">
			<!-- <view class="verify-title">安全验证</view> -->
			<view class="verify-tip">拖动下方滑块完成拼图</view>
			<view class="verify-content">
				<view class="verify-body">
					<view class="verify-bg">
						<image id="bg" :src="captchaData.background_image" mode="heightFix">
						</image>
					</view>
					<view class="verify-slider">
						<image id="slider-img" :style="{ left: leftDistance + 'px' }" :src="captchaData.slider_image"
							mode="heightFix"></image>
					</view>
					<view v-if="isSuccess" class="check-status check-success">
						<text>验证成功</text>
					</view>
					<view v-if="isErr" class="check-status check-error">
						<text>{{ errorText }}</text>
					</view>
				</view>
				<view @touchstart="touchstart" @touchmove="touchmove" @touchend="touchend" v-if="isActive"
					class="move-area">
					<movable-area class="move-block" :animation="true">
						<view class="color-change" :style="{ width: colorWidth + 'px' }"></view>
						<view class="move-shadow"></view>
						<movable-view class="block-button" :x='x' :animation="true" direction="horizontal"
							@change="StartMove">
							<u-icon name="arrow-rightward" color="#fff" size="20" class="icon-drag"></u-icon>
						</movable-view>
					</movable-area>
				</view>
			</view>
			<view class="verify-opts">
				<u-icon name="reload" color="#686868" size="30" @click="refresh"></u-icon>
				<view class="divide"></view>
				<u-icon name="close-circle" color="#686868" size="26" @click="verifyShow = false"></u-icon>
			</view>
		</view>
	</view>

</template>
<script>
export default {
	name: "tianai-verify-code",
	data() {
		return {
			verifyShow: false,
			isActive: true, //刷新滑块
			colorWidth: uni.upx2px(80),
			leftDefault: 0,
			leftDistance: 0, //默认卡片位置固定为靠左
			isSuccess: false, //验证成功
			isErr: false, //验证失败
			errorText: "验证失败,请重新尝试!",
			x: 0, //滑块的X距离
			xpos: 0, //读取X滑动的距离
			bgImg: { //当前环境滑块背景尺寸
				width: 0,
				height: 0
			},
			sliderImg: { //当前环境滑块尺寸
				width: 0,
				height: 0
			},
			captchaData: {
				id: null, //当前生成的滑块ID, 后端生成
				background_image: '', //滑块背景图
				slider_image: '', //滑块图片
				startTime: new Date(), //起始时间
				trackArr: [], //滑动轨迹
				movePercent: 0, //滑动距离与背景图百分比
				background_image_width: 0, //当前环境滑块背景宽
				background_image_height: 0, //当前环境滑块背景高
				slider_image_width: 0, //当前环境滑块宽
				slider_image_height: 0, //当前环境滑块高
				end: 206 //滑块滑动界限值
			}
		};
	},
	methods: {
		open() {
			uni.$u.debounce(this.getVerifyData, 80)
			this.verifyShow = true
		},
		close() {
			this.verifyShow = false
			this.$nextTick(() => {
				this.captchaData = {
					id: null, //当前生成的滑块ID, 后端生成
					background_image: '', //滑块背景图
					slider_image: '', //滑块图片
					startTime: new Date(), //起始时间
					trackArr: [], //滑动轨迹
					movePercent: 0, //滑动距离与背景图百分比
					background_image_width: 0, //当前环境滑块背景宽
					background_image_height: 0, //当前环境滑块背景高
					slider_image_width: 0, //当前环境滑块宽
					slider_image_height: 0, //当前环境滑块高
					end: 206 //滑块滑动界限值
				}
				this.isActive = false
				this.$nextTick(() => {
					this.isActive = true
				})
				this.colorWidth = 0
				this.x = this.xpos
				this.$nextTick(function () {
					this.x = 0
					this.colorWidth = uni.upx2px(80)
				})
				this.isErr = false
				this.isSuccess = false
				this.leftDistance = 0
			})

		},
		// 日志打印
		printLog(...params) {
			console.log(JSON.stringify(params));
		},
		// 滑块初始位置
		touchstart(e) {
			// console.log("---touchstart---", e)
			let startX = e.changedTouches[0].pageX;
			let startY = e.changedTouches[0].pageY;
			this.captchaData.startX = startX;
			this.captchaData.startY = startY;
			const pageX = this.captchaData.startX;
			const pageY = this.captchaData.startY;
			const startTime = this.captchaData.startTime;
			const trackArr = this.captchaData.trackArr;
			trackArr.push({
				x: pageX - startX,
				y: pageY - startY,
				type: "down",
				t: (new Date().getTime() - startTime.getTime())
			})
			this.printLog('start', startX, startY)
		},
		// 滑块滑动中位置
		touchmove(e) {
			let pageX = Math.round(e.changedTouches[0].pageX);
			let pageY = Math.round(e.changedTouches[0].pageY);
			const startX = this.captchaData.startX;
			const startY = this.captchaData.startY;
			const startTime = this.captchaData.startTime;
			const end = this.captchaData.end;
			const bgImageWidth = this.captchaData.background_image_width;
			const trackArr = this.captchaData.trackArr;
			let moveX = pageX - startX;
			const track = {
				x: pageX - startX,
				y: pageY - startY,
				type: "move",
				t: (new Date().getTime() - startTime.getTime())
			};
			trackArr.push(track);
			// console.log("moveX", moveX)
			if (moveX < 0) {
				moveX = 0;
			} else if (moveX > end) {
				moveX = end;
			}
			this.captchaData.moveX = moveX;
			this.captchaData.movePercent = moveX / bgImageWidth;
			this.printLog("move", track)
		},
		// 滑块停止滑动
		touchend(e) {
			this.captchaData.stopTime = new Date()
			let pageX = Math.round(e.changedTouches[0].pageX);
			let pageY = Math.round(e.changedTouches[0].pageY);
			const startX = this.captchaData.startX;
			const startY = this.captchaData.startY;
			const startTime = this.captchaData.startTime;
			const trackArr = this.captchaData.trackArr;
			const track = {
				x: pageX - startX,
				y: pageY - startY,
				type: "up",
				t: (new Date().getTime() - startTime.getTime())
			}
			trackArr.push(track);
			this.printLog("up", track)
			this.setVertifyData()
		},
		// 滑块绑定卡片移动距离
		StartMove(e) {
			this.xpos = e.detail.x
			// console.log("xpos", this.xpos)
			this.$nextTick(() => {
				this.colorWidth = this.xpos + uni.upx2px(80)
				this.leftDistance = this.xpos
			})
		},
		// 获取背景图和滑块图片
		getVerifyData() {
			// console.log('---getVerifyData---')
			let that = this
			// 需要更改为自己的请求
			this.$http.post(`${process.env.VUE_APP_ADMIN_API}/captcha/gen`, {}, {
				custom: {
					noLoading: true
				}
			}).then(data => {
				// console.log("---/captcha/gen---", data)

				this.captchaData.id = data.id
				this.captchaData.background_image = data.captcha.backgroundImage
				this.captchaData.slider_image = data.captcha.templateImage
				// this.captchaData.end = data.captcha.backgroundImageWidth - data.captcha
				// 	.templateImageWidth
				this.$nextTick(() => {
					// 获取当前设备上渲染的背景图和滑块尺寸
					let query = uni.createSelectorQuery().in(this)
					query.select('#bg').boundingClientRect(res => {
						this.bgImg.width = res.width
						this.bgImg.height = res.height
					}).exec();
					query.select('#slider-img').boundingClientRect(res => {
						this.sliderImg.width = res.width
						this.sliderImg.height = res.height
					}).exec();
					setTimeout(() => {
						this.captchaData.end = this.bgImg.width - uni.upx2px(40)
						this.initConfig(this.bgImg.width, this.bgImg.height, this.sliderImg
							.width, this
								.sliderImg.height, this.captchaData.end)
					}, 500)
				})

			}).catch(err => {
				console.log(err);
			})
		},
		// 初始化配置
		initConfig(bgImageWidth, bgImageHeight, sliderImageWidth, sliderImageHeight, end) {
			this.captchaData.background_image_width = bgImageWidth
			this.captchaData.background_image_height = bgImageHeight
			this.captchaData.slider_image_width = sliderImageWidth
			this.captchaData.slider_image_height = sliderImageHeight
			this.captchaData.end = end
		},
		// 校验是否成功
		setVertifyData() {
			const dataForm = {
				id: this.captchaData.id,
				data: {
					bgImageWidth: this.captchaData.background_image_width,
					bgImageHeight: this.captchaData.background_image_height,
					// sliderImageWidth: this.captchaData.slider_image_width,
					// sliderImageHeight: this.captchaData.slider_image_height,
					// startSlidingTime: this.captchaData.startTime,
					// endSlidingTime: this.captchaData.stopTime,
					startTime: this.captchaData.startTime,
					stopTime: this.captchaData.stopTime,
					trackList: this.captchaData.trackArr
				}
			}
			this.printLog("dataForm", dataForm)
			// 更改为自己的请求
			this.$http.post(`${process.env.VUE_APP_ADMIN_API}/captcha/check`, dataForm, {
				custom: {
					noLoading: true
				}
			}).then((res) => {
				// console.log("---/captcha/check---", res)
				if (res.code == 200) {
					if (res.success) {
						this.isSuccess = true
						let tm = setTimeout(() => {
							clearTimeout(tm)
							this.close()
							this.$emit('success', res.data.id)
						}, 1000)
					}
				} else {
					if (res.code == 4001) {
						this.isErr = true
						this.errorText = "验证失败,请重新尝试!"
					}
					else if (res.code == 4000) {
						this.isErr = true
						this.errorText = "验证码被黑洞吸走了!"
					}
					else {
						uni.showToast({
							title: res.msg,
							icon: 'none'
						})
					}
					let tm = setTimeout(() => {
						clearTimeout(tm)
						this.refresh()
					}, 1000)
				}
			}).catch(err => {
				console.log(err);
				this.refresh()
			})
		},
		// 刷新滑块
		refresh() {
			this.isActive = false
			this.$nextTick(() => {
				this.isActive = true
			})
			this.colorWidth = 0
			this.x = this.xpos
			this.$nextTick(function () {
				this.x = 0
				this.colorWidth = uni.upx2px(80)
			})

			this.isErr = false
			this.isSuccess = false
			this.leftDistance = 0
			this.captchaData.trackArr = []
			this.getVerifyData()
		}
	}
}
</script>

<style scoped lang="scss">
.verify-wrap {
	position: fixed;
	top: 0;
	left: 0;
	width: 100%;
	height: 100%;
	background-color: rgba(0, 0, 0, 0.5);
	z-index: 999;
	overflow: hidden;

	.verify-code {
		position: absolute;
		left: 50%;
		top: 50%;
		transform: translate(-50%, -50%);
		width: 640rpx;
		max-height: 740rpx;
		background-color: #FFFFFF;
		padding: 40rpx 0 20rpx;
		z-index: 999;
		box-shadow: 0 0 10rpx rgba(227, 227, 227, 0.7);
		border-radius: 10px;
		overflow: hidden;

		// .verify-title {
		// 	width: 100%;
		// 	text-align: center;
		// 	line-height: 1;
		// 	font-size: 40rpx;
		// 	font-weight: 800;
		// 	margin: 20rpx 0;
		// }

		.verify-tip {
			font-size: 32rpx;
			font-weight: bold;
			color: #686868;
			padding: 0 20rpx;
		}


		.verify-content {
			width: 100%;
			padding: 20rpx 20rpx;
			background-color: #ffffff;
			box-sizing: border-box;
			overflow: hidden;

			.verify-body {
				width: 100%;
				height: 360rpx;
				border-radius: 6px;
				position: relative;
				overflow: hidden;

				.verify-bg {
					width: 100%;
					height: 100%;
					position: absolute;

					image {
						width: 100%;
						height: 100%;
					}
				}

				.verify-slider {
					height: 100%;
					position: absolute;
					left: 0;
					top: 0;

					image {
						overflow: hidden;
						width: 55px;
						height: 100%;
						position: relative;
					}
				}
			}

			.move-area {
				overflow: hidden;
				width: 100%;
				height: 80rpx;
				margin-top: 20rpx;
			}

			.move-block {
				width: 100%;
				height: 100%;
				background-color: #f0f0f0;
				border-radius: 100rpx;
				position: relative;
				overflow: hidden;

				.move-shadow {
					height: 100%;
					width: 4px;
					background-color: rgba(255, 255, 255, 0.5);
					position: absolute;
					top: 0;
					left: 0;
					box-shadow: 1px 1px 1px #fff;
					border-radius: 50%;
					animation: moveAnimate 2s linear infinite;
				}

				@keyframes moveAnimate {
					0% {
						left: 0;
						opacity: 0.5;
					}

					50% {
						left: 50%;
						opacity: 1;
					}

					100% {
						left: 100%;
						opacity: 0.5;
					}
				}

				.color-change {
					height: 80rpx;
					border-radius: 100rpx;
					background-color: #c6a876;
					z-index: 2;
				}

				.block-button {
					// transform-origin: left center !important;
					border-radius: 100rpx;
					background-color: #b48d4d;
					;
					height: 80rpx;
					width: 80rpx;
					margin-top: -10rpx;
					touch-action: none;
					display: flex;
					flex-direction: row;
					align-items: center;
					justify-content: center;
					color: #fff;
				}

			}

			.check-status {
				position: absolute;
				left: 0;
				right: 0;
				bottom: -1px;
				height: 50rpx;
				line-height: 50rpx;
				width: 100%;
				text-align: center;
				font-size: 24rpx;
				color: #fff;

				&.check-success {
					background: #5ac725;
				}

				&.check-error {
					background: #f56c6c;
				}
			}

		}

		.verify-opts {
			display: flex;
			justify-content: flex-end;
			margin: 0 20rpx;

			.divide {
				height: 20px;
				width: 20rpx;
			}
		}
	}
}
</style>

具体应用

html 复制代码
<template> 
    <div class="rob-button" @click="handleRobButton">抢券</div> 
    <tianai-verify-code ref="verifyRef" @success="verifySuccess"></tianai-verify-code> 
</template> 

<script> 
import tianaiVerifyCode from '@/components/tianai-verify-code/index'; 
export default { 
    components:{ 
        tianaiVerifyCode, 
    }, 
    methods: { 
        handleRobButton(){ 
            this.$refs?.verifyRef?.open() 
        }, 
        verifySuccess(){ 
            this.toRobCoupon() 
        } 
    } 
} 
</script>
相关推荐
庸俗今天不摸鱼几秒前
【万字总结】前端全方位性能优化指南(十)——自适应优化系统、遗传算法调参、Service Worker智能降级方案
前端·性能优化·webassembly
黄毛火烧雪下7 分钟前
React Context API 用于在组件树中共享全局状态
前端·javascript·react.js
Apifox18 分钟前
如何在 Apifox 中通过 CLI 运行包含云端数据库连接配置的测试场景
前端·后端·程序员
一张假钞20 分钟前
Firefox默认在新标签页打开收藏栏链接
前端·firefox
高达可以过山车不行21 分钟前
Firefox账号同步书签不一致(火狐浏览器书签同步不一致)
前端·firefox
m0_5937581022 分钟前
firefox 136.0.4版本离线安装MarkDown插件
前端·firefox
掘金一周25 分钟前
金石焕新程 >> 瓜分万元现金大奖征文活动即将回归 | 掘金一周 4.3
前端·人工智能·后端
三翼鸟数字化技术团队43 分钟前
Vue自定义指令最佳实践教程
前端·vue.js
Jasmin Tin Wei1 小时前
蓝桥杯 web 学海无涯(axios、ecahrts)版本二
前端·蓝桥杯
圈圈编码1 小时前
Spring Task 定时任务
java·前端·spring