vue中验证码的实现方式

news2024/12/23 4:04:21

 在写登录页的时候有的系统会让你也进行一下验证码绘制,那么验证码如何实现的呢?我在写登录页的时候通过将登录框,验证码分开页面来写,最后将它们变成标签来导入到我的样式页面中,这样写不仅方便,更容易修改代码,很便捷。

也就是说我登录页的样式,输入框,验证码是分别写了三个页面。 

然后将验证码绘制的图形,通过组件的形式放入输入框的页面中。

import Identify from '../component/code.vue'; //引用验证码组件
<!-- 验证码组件 -->
			<el-col :span="8">
				<div class="change-email">
					<Identify :identifyCode="identifyCode" :changeCode="changeCode"></Identify>
					<!-- 引用验证码组件 -->
				</div>
			</el-col>

然后下面这里是验证码的逻辑,比如验证码的初始值或者它的生成范围,可以只生成数字或者字母

// 验证码逻辑
const identifyCode = ref('1234'); // 验证码的初始值
const identifyCodes = '1234567890abcdefghijklmnopqrstuvwxyz'; // 验证码生成范围
// 生成验证码的方法
const changeCode = () => {
	identifyCode.value = ''; // 清空当前验证码
	makeCode(identifyCodes, 4); // 重新生成新的验证码
};
// 生成随机数字
const randomNum = (min: any, max: any) => {
	max = max + 1; // 最大值加1,使得能够取到最大值
	return Math.floor(Math.random() * (max - min) + min); // 返回一个在[min, max]范围内的随机数
};
// 根据给定的字符集和长度生成验证码
const makeCode = (data: any, len: any) => {
	for (let i = 0; i < len; i++) {
		identifyCode.value += data[randomNum(0, data.length - 1)]; // 从字符集中随机取字符拼接成验证码
	}
};
onMounted(() => {
	changeCode(); // 组件挂载时,生成验证码
});

然后changeCode就是每次触发验证码的事件,它都会重新生成一个新的验证码,当然如果在输入验证码错误的时候就需要调用这个验证码重新生成。

最主要的就是这一生成验证码的部分,直接复制就可以,它是通过canvas来进行绘制的

<template>
    <!-- 二维码识别组件容器,包含一个 canvas 元素用于绘制验证码 -->
    <div class="s-canvas">
        <!-- canvas 元素,用于绘制验证码图案 -->
        <canvas @click="changeCode" class="s-canvas" :width="contentWidth" :height="contentHeight"></canvas>
    </div>
</template>
<script>
export default {
    name: 'Identify',  // 组件名称
    props: {
        identifyCode: {  // 验证码内容
            type: String,
            default: '1234'
        },
        fontSizeMin: {  // 最小字体大小
            type: Number,
            default: 20
        },
        fontSizeMax: {  // 最大字体大小
            type: Number,
            default: 35
        },
        backgroundColorMin: {  // 背景颜色最小值(用于生成干扰背景颜色)
            type: Number,
            default: 180
        },
        backgroundColorMax: {  // 背景颜色最大值
            type: Number,
            default: 240
        },
        colorMin: {  // 字体颜色最小值(用于生成干扰字体颜色)
            type: Number,
            default: 50
        },
        colorMax: {  // 字体颜色最大值
            type: Number,
            default: 160
        },
        lineColorMin: {  // 干扰线颜色最小值
            type: Number,
            default: 40
        },
        lineColorMax: {  // 干扰线颜色最大值
            type: Number,
            default: 180
        },
        dotColorMin: {  // 干扰点颜色最小值
            type: Number,
            default: 0
        },
        dotColorMax: {  // 干扰点颜色最大值
            type: Number,
            default: 255
        },
        contentWidth: {  // 验证码画布宽度
            type: Number,
            default: 100
        },
        contentHeight: {  // 验证码画布高度
            type: Number,
            default: 40
        },
        changeCode: {  // 切换验证码的事件函数
            type: Function
        }
    },
    methods: {
        // 生成一个指定范围内的随机数
        randomNum(min, max) {
            return Math.floor(Math.random() * (max - min) + min)
        },
        // 生成一个随机的颜色
        randomColor(min, max) {
            let r = this.randomNum(min, max)
            let g = this.randomNum(min, max)
            let b = this.randomNum(min, max)
            return 'rgb(' + r + ',' + g + ',' + b + ')'
        },
        // 绘制验证码图案
        drawPic() {
            let canvas = document.getElementsByClassName('s-canvas');  // 获取所有 canvas 元素
            let ctx = canvas[1].getContext('2d');  // 获取第一个 canvas 的上下文
            ctx.textBaseline = 'bottom'  // 设置文字的基准线为底部
            // 绘制背景
            ctx.fillStyle = this.randomColor(this.backgroundColorMin, this.backgroundColorMax)
            ctx.fillRect(0, 0, this.contentWidth, this.contentHeight)  // 绘制背景矩形
            // 绘制验证码文字
            for (let i = 0; i < this.identifyCode.length; i++) {
                this.drawText(ctx, this.identifyCode[i], i)  // 绘制每个字符
            }
            // 绘制干扰线
            this.drawLine(ctx)
            // 绘制干扰点
            this.drawDot(ctx)
            // 为其他 canvas 元素绘制同样的验证码图案(如果存在的话)
            if (canvas[3]) {
                let ctx1 = canvas[3].getContext('2d');
                ctx1.textBaseline = 'bottom'
                ctx1.fillStyle = this.randomColor(this.backgroundColorMin, this.backgroundColorMax)
                ctx1.fillRect(0, 0, this.contentWidth, this.contentHeight)
                for (let i = 0; i < this.identifyCode.length; i++) {
                    this.drawText(ctx1, this.identifyCode[i], i)
                }
                this.drawLine(ctx1)
                this.drawDot(ctx1)
            }
            if (canvas[5]) {
                let ctx2 = canvas[5].getContext('2d');
                ctx2.textBaseline = 'bottom'
                ctx2.fillStyle = this.randomColor(this.backgroundColorMin, this.backgroundColorMax)
                ctx2.fillRect(0, 0, this.contentWidth, this.contentHeight)
                for (let i = 0; i < this.identifyCode.length; i++) {
                    this.drawText(ctx2, this.identifyCode[i], i)
                }
                this.drawLine(ctx2)
                this.drawDot(ctx2)
            }
        },
        // 绘制文字
        drawText(ctx, txt, i) {
            ctx.fillStyle = this.randomColor(this.colorMin, this.colorMax)  // 随机字体颜色
            ctx.font = this.randomNum(this.fontSizeMin, this.fontSizeMax) + 'px SimHei'  // 随机字体大小
            let x = (i + 1) * (this.contentWidth / (this.identifyCode.length + 1))  // 计算字符的 x 坐标
            let y = this.randomNum(this.fontSizeMax, this.contentHeight - 5)  // 计算字符的 y 坐标
            var deg = this.randomNum(-45, 45)  // 随机旋转角度
            // 修改坐标原点和旋转角度
            ctx.translate(x, y)
            ctx.rotate(deg * Math.PI / 180)
            ctx.fillText(txt, 0, 0)  // 绘制文字
            // 恢复坐标原点和旋转角度
            ctx.rotate(-deg * Math.PI / 180)
            ctx.translate(-x, -y)
        },
        // 绘制干扰线
        drawLine(ctx) {
            for (let i = 0; i < 3; i++) {
                ctx.strokeStyle = this.randomColor(this.lineColorMin, this.lineColorMax)  // 随机线条颜色
                ctx.beginPath()
                ctx.moveTo(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight))  // 起点
                ctx.lineTo(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight))  // 终点
                ctx.stroke()  // 绘制线条
            }
        },
        // 绘制干扰点
        drawDot(ctx) {
            for (let i = 0; i < 30; i++) {
                ctx.fillStyle = this.randomColor(0, 255)  // 随机点的颜色
                ctx.beginPath()
                ctx.arc(this.randomNum(0, this.contentWidth), this.randomNum(0, this.contentHeight), 1, 0, 2 * Math.PI)  // 绘制圆形
                ctx.fill()  // 填充圆形
            }
        }
    },
    watch: {
        // 监视验证码内容变化时,重新绘制验证码图案
        identifyCode() {
            this.drawPic()
        }
    },
    mounted() {
        // 页面加载时绘制验证码图案
        this.drawPic()
    }
}
</script>
<style scoped>
/* canvas 容器样式 */
.s-canvas {
    margin-left: 10px;  /* 设置左边距 */
    display: flex;  /* 使用 flex 布局 */
    align-items: center;  /* 垂直居中对齐 */
    cursor: pointer;  /* 鼠标悬浮时显示为可点击 */
}
/* 可选的验证码提示文字样式 */
.s-canvas-tip {
    color: #5C7099;  /* 设置提示文字的颜色 */
    font-size: 14px;  /* 设置字体大小 */
    margin-left: 14px;  /* 设置左边距 */
    cursor: pointer;  /* 鼠标悬浮时显示为可点击 */
}
</style>

但是生成验证码的书写格式并不是vue3中的,因为我不是通过setup直接写,是通过export来进行导出。

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

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

相关文章

Spring(三)-SpringWeb-概述、特点、搭建、运行流程、组件、接受请求、获取请求数据、特殊处理、拦截器

文章目录 一、SpringWeb概述 二、SpringWeb特点 三、搭建SpringWeb&#xff08;在web项目中&#xff09; 1、导包 2、在web.xml文件中配置统一拦截分发器 DispatcherServlet 3、开启 SpringWEB 注解 4、处理器搭建 四、SpringWeb运行流程 五、SpringWeb组件 1、前端控…

构建MacOS应用小白教程(打包 签名 公证 上架)

打包 在package.json中&#xff0c;dependencies会被打进 Electron 应用的包里&#xff0c;而devDependencies则不会&#xff0c;所以必要的依赖需要放到dependencies中。files中定义自己需要被打进 Electron 包里的文件。以下是一个完整的 mac electron-builder的配置文件。 …

2.4 网络概念(分层、TCP)

网络层与传输层概述 网络层&#xff1a; 抽象概念&#xff1a;网络层是基于 IP 的抽象概念&#xff0c;与数据链路层用 MAC 地址标记设备不同。MAC 地址是一种具体化的概念&#xff0c;绑定于所在的物理网络&#xff0c;而 IP 地址可以是固定的&#xff0c;也可以通过路由动态…

【JetPack】Room数据库笔记

Room数据库笔记 ORM框架&#xff1a;对齐数据库数据结构与面向对象数据结构之间的关系&#xff0c;使开发编程只考虑面向对象不需要考虑数据库的结构 Entity : 数据实体&#xff0c;对应数据库中的表 <完成面向对象与数据库表结构的映射> 注解&#xff1a; 类添加注解…

基于前端技术UniApp和后端技术Node.js的电影购票系统

文章目录 摘要Abstruct第一章 绪论1.1 研究背景与意义1.2 国内外研究现状 第二章 需求分析2.1 功能需求分析2.2 非功能性需求分析 第二章系统设计3.1 系统架构设计3.1.1 总体架构3.1.2 技术选型 3.2 功能架构 第四章 系统实现4.1 用户端系统实现4.1.1 用户认证模块实现4.1.2 电…

大模型微调---Lora微调实战

目录 一、前言二、LoRA实战2.1、下载模型到本地2.2、加载模型与数据集2.3、处理数据2.4、LoRA微调2.5、训练参数配置2.6、开始训练 三、模型评估四、完整训练代码 一、前言 LoRA是一种参数高效的微调技术&#xff0c;通过低秩转换对大型语言模型进行适应性更新&#xff0c;减少…

centos7下docker 容器实现redis主从同步

1.下载redis 镜像 docker pull bitnami/redis2. 文件夹授权 此文件夹是 你自己映射到宿主机上的挂载目录 chmod 777 /app/rd13.创建docker网络 docker network create mynet4.运行docker 镜像 安装redis的master -e 是设置环境变量值 docker run -d -p 6379:6379 \ -v /a…

SLAAC如何工作?

SLAAC如何工作&#xff1f; IPv6无状态地址自动配置(SLAAC)-常见问题 - 苍然满关中 - 博客园 https://support.huawei.com/enterprise/zh/doc/EDOC1100323788?sectionj00shttps://www.zhihu.com/question/6691553243/answer/57023796400 主机在启动或接口UP后&#xff0c;发…

2024.12.21辩论赛感受

背景 今天辩论赛的双方论点是&#xff1a; 正方&#xff1a;寒假留在研发中心的收获大 反方&#xff1a;寒假去做其他事情的收获 辩论赛&#xff0c;为了锻炼自己&#xff0c;选择了不想选择以及相对不好辩论的反方。出现的状况有一下几点&#xff1a; 1.发现自己脑子完全跟不…

【从零开始入门unity游戏开发之——C#篇21】C#面向对象的封装——`this`扩展方法、运算符重载、内部类、`partial` 定义分部类

文章目录 一、this扩展方法1、扩展方法的基本语法2、使用扩展方法3、扩展方法的注意事项5、扩展方法的限制6、总结 二、运算符重载1、C# 运算符重载2、运算符重载的基本语法3. 示例&#xff1a;重载加法运算符 ()4、使用重载的运算符5、支持重载的运算符6、不能重载的运算符7、…

C语言:文件IO

C语言&#xff1a;文件IO 文件操作 概述 什么是文件 文件是保存在外存储器&#xff08;一般代指磁盘&#xff0c;U盘&#xff0c;移动硬盘等&#xff09;的数据的集合 文件操作体现在哪几个方面 文件内容的读取文件内容的写入 数据的读取和写入可被视为针对文件进行输入&…

时间序列异常值处理方法

文章目录 一、删除法二、替换法三、插值法四、滑动窗口五、基于模型的替换 时间序列相关参考文章&#xff1a; 时间序列预测算法—ARIMA 时间序列预测算法—Prophet 时间序列分类任务—tsfresh python时间序列处理 有季节效应的非平稳序列分析 时间序列异常值检测方法 时间序列…

【报错】node:internal/modules/cjs/loader:936

报错问题&#xff1a; 当执行npm run dev后&#xff0c;出现下面错误 这个错误一般是由于Node.js无法找到所需的模块而引起的&#xff0c;解决此问题的一种方法就是重新安装所需的模块。 解决办法&#xff1a; 删除npm install 所下载在项目里的node_modules文件执行操作&…

故障诊断 | 一个小创新:特征提取+KAN分类

往期精彩内容&#xff1a; Python-凯斯西储大学&#xff08;CWRU&#xff09;轴承数据解读与分类处理 基于FFT CNN - BiGRU-Attention 时域、频域特征注意力融合的轴承故障识别模型-CSDN博客 基于FFT CNN - Transformer 时域、频域特征融合的轴承故障识别模型-CSDN博客 P…

javac 编译java文件源码 怎么生成 ast语法树 步骤详解

在 javac 中&#xff0c;编译源代码并生成抽象语法树&#xff08;AST&#xff09;是一个多步骤的过程&#xff0c;涉及从源码解析到最终生成字节码。以下是详细步骤&#xff0c;描述了如何使用 javac 编译源码并生成 AST。 1. 准备源文件 javac 首先需要源文件。这些源文件是…

人工智能入门是先看西瓜书还是先看花书?

在人工智能入门时&#xff0c;关于先看《机器学习》&#xff08;西瓜书&#xff09;还是先看《深度学习》&#xff08;花书&#xff09;的问题&#xff0c;实际上取决于个人的学习目标和背景。 《机器学习》&#xff08;西瓜书&#xff09;由周志华教授撰写&#xff0c;是一本…

Linux 安装Nginx 并配置启动 (已实测)

文章目录 一、安装Nginx二、配置 Nginx 为系统服务 一、安装Nginx 安装依赖&#xff0c;确保Nginx编译和运行正常&#xff0c;打开终端执行以下命令 yum install -y wget gcc-c pcre-devel zlib-devel openssl-devel下载Nginx # 例如&#xff0c;下载Nginx 1.24.0版本 wget …

Linux 下的 GPT 和 MBR 分区表详解

文章目录 Linux 下的 GPT 和 MBR 分区表详解一、分区表的作用二、MBR&#xff08;Master Boot Record&#xff09;1. **特点**2. **优点**3. **缺点**4. **适用场景** 三、GPT&#xff08;GUID Partition Table&#xff09;1. **特点**2. **优点**3. **缺点**4. **适用场景** 四…

10. 虚拟机VMware Workstation Pro下共享Ubuntu和Win11文件夹

本文记录当前最新版虚拟机VMware Workstation Pro&#xff08;2024.12&#xff09;如何在win11下共享文件&#xff0c;以实现Windows与Ubuntu互传文件的目的。 1. 创建共享文件夹 1.1 先关闭虚拟机的客户机&#xff0c;打开虚拟机设置 1.2 在虚拟机设置界面找到“选项”->“…

有哪些免费的 ERP 软件可供选择?哪些 ERP 软件使用体验较好?

想找个 “免费” 的 ERP 软件&#xff1f; 咱得知道&#xff0c;ERP 那可是涉及财务、人力、供应链、采购、销售等好多方面的重要企业软件。功能这么全&#xff0c;能免费才怪呢&#xff01;真要是有免费的&#xff0c;早就火遍大江南北&#xff0c;说不定把市场都垄断了&…