首页 > 其他分享 >Vue验证码模块

Vue验证码模块

时间:2023-10-21 17:24:00浏览次数:33  
标签:canvas code ctx 验证码 randomNum Vue let 模块 options

一个普通图形验证码的模块,使用方式如下:

 

导入模块

import random from '@/util/VerificationCode'

使用方式

this.randomObj = random({
  id: 'random',
  api: {
      url: '/api/getCode',
      headers: {
          'Content-Type': 'application/json'
      }
  }
})

源码

!(function (window, document) {
    const charArr = 'abcdefghjkmnpqrstuvwxyz123456789ABCDEFGHJKMNPQRSTUVWXYZ'.split('');

    function GVerify(options) {
        this.options = {
            id: options.id,
            api: options.api,
            length: options.length ? options.length : 4,
            canvasId: 'VerifyCanvas_' + options.id,
            type: 'blend'
        };

        this._init();
        this.getCode();
    }

    function randomNum(min, max) {
        return Math.floor(Math.random() * (max - min) + min)
    }

    function randomColor(min, max) {
        let r = randomNum(min, max);
        let g = randomNum(min, max);
        let b = randomNum(min, max);
        return 'rgb(' + r + ',' + g + ',' + b + ')'
    }

    function generateCode(len) {
        let code = '';
        for (let i=0; i<len; i++) {
            code += charArr[randomNum(0, charArr.length)];
        }
        console.log(code);
        return code;
    }

    GVerify.prototype = {
        version: '1.0.0',
        _init() {
            let con = document.getElementById(this.options.id);
            let canvas = document.createElement('canvas');
            this.options.width = con.offsetWidth > 0 ? con.offsetWidth : '100';
            this.options.height = con.offsetHeight > 0 ? con.offsetHeight : '30';
            canvas.id = this.options.canvasId;
            canvas.width = this.options.width;
            canvas.height = this.options.height;
            canvas.style.cursor = 'pointer';
            canvas.innerHTML = '您的浏览器版本不支持canvas';
            con.appendChild(canvas);
            canvas.addEventListener('click', () => {
                this.getCode()
            })
        },
        // 刷新
        refresh() {
            let canvas = document.getElementById(this.options.canvasId);
            if (!canvas.getContext) return;
            let ctx = canvas.getContext('2d');
            ctx.textBaseline = 'middle';
            ctx.fillStyle = randomColor(180, 240);
            ctx.fillRect(0, 0, this.options.width, this.options.height);
            for (let i = 0; i < this.options.code.length; i++) {
                ctx.font = randomNum(this.options.height / 2, this.options.height) + 'px SimHei';
                ctx.fillStyle = randomColor(50, 160);
                ctx.shadowOffsetX = randomNum(-3, 3);
                ctx.shadowOffsetY = randomNum(-3, 3);
                ctx.shadowBlur = randomNum(-3, 3);
                ctx.shadowColor = 'rgba(0, 0, 0, 0.3)';
                let x = this.options.width / this.options.code.length * i;
                x = x + (this.options.width / this.options.code.length / 2) - 10; // 居中
                let y = this.options.height / 2 - 1;
                let deg = randomNum(-30, 30);
                ctx.translate(x, y);
                ctx.rotate(deg * Math.PI / 180);
                ctx.fillText(this.options.code.substring(i, i + 1), 0, 0);
                ctx.rotate(-deg * Math.PI / 180);
                ctx.translate(-x, -y);
            }
            for (let i = 0; i < 4; i++) {
                ctx.strokeStyle = randomColor(40, 180);
                ctx.beginPath();
                ctx.moveTo(randomNum(0, this.options.width), randomNum(0, this.options.height));
                ctx.lineTo(randomNum(0, this.options.width), randomNum(0, this.options.height));
                ctx.stroke();
            }
            for (let i = 0; i < this.options.width / 4; i++) {
                ctx.fillStyle = randomColor(0, 255);
                ctx.beginPath();
                ctx.arc(randomNum(0, this.options.width), randomNum(0, this.options.height), 1, 0, 2 * Math.PI);
                ctx.fill();
            }
            this.ctx = canvas;
        },
        // 重载大小
        reload() {},
        // 获取验证码
        getCode() {
            if (this.options.api && this.options.api.url) { // 从api获取
                let xhr = new XMLHttpRequest();
                xhr.onreadystatechange = () => {
                    if (xhr.readyState == 4 && xhr.status == 200) {
                        this.options.code = xhr.responseText;
                        this.refresh();
                    }
                };
                xhr.open(this.options.api.method ? this.options.api.method : 'POST', this.options.api.url);
                for (const key in this.options.api.headers) {
                    xhr.setRequestHeader(key, this.options.api.headers[key]);
                }
                xhr.send();
            } else { // 本地生成
                let code = generateCode(this.options.length);
                this.options.code = code;
                this.refresh();
            }
        },
        // 校验输入
        validate(code) {
            code = code.toLowerCase();
            let v_code = this.options.code.toLowerCase();
            return code == v_code;
        }
    };

    window.GVerify = GVerify;
})(window, document);

export default conf => {
    if (!conf.id) return false;
    return new GVerify(conf);
}

 

标签:canvas,code,ctx,验证码,randomNum,Vue,let,模块,options
From: https://www.cnblogs.com/sanrenblog/p/17779238.html

相关文章

  • 手撕Vue-实现事件相关指令
    经过上一篇文章的学习,实现了界面驱动数据更新,接下来实现一下其它相关的指令,比如事件相关的指令,v-on这个指令的使用频率还是很高的,所以我们先来实现这个指令。v-on的作用是什么,是不是可以给某一个元素绑定一个事件。紧接着了解了v-on的作用之后,我在example.html的结构代码......
  • uniCloud cms 自媒体资讯新闻文章应用系统 uniapp+uniCloud+AntDesignVue Life cms
    介绍LifeCMS是uniCloud+uni-app云端一体全套CMS/自媒体/资讯/新闻/文章应用系统,前台包含注册、登录(账号密码登录、短信登录、微信手机号快捷登录、微信一键登录、App手机一键登录、Apple登录)、文章列表、文章详情、搜索、广告、分享、评论、回复、点赞、收藏、用户中心、意见......
  • Vue进阶(幺玖玖):vue 输入框中按enter键实现搜索或表单提交
    在前端项目开发过程中,为优化用户体验,可考虑在用户输入查询条件后按回车键实现搜索效果。实现方法如下:el-input监听键盘按下状态得用@keyup.enter.native,如果是非el-input组件,可以直接用@keyup.enter。<[email protected]="search"v-model='form.searchAttr'></el......
  • vue实现点击复制功能(无需安装库)
     1.标签<buttonv-copy="text">复制文本</button>text是要复制的内容,在data函数中 2.在main.js中注册copy指令Vue.directive('copy',{bind:function(el,binding){el.$copy=function(){consttextarea=document.createElement('text......
  • go mod tidy总是安装最新依赖,如何查找哪个模块导致某个包安装最新依赖,提供一个小工具
    安装:goinstallgithub.com/jan-bar/interesting/findModVer@latest执行:findModVerd:\myproject结果如下图所示:根据结果可以找到哪个依赖导致google.golang.org/grpcv1.45.0使用了这个版本,这样每次执行gomodtidy会自动修改该模块到v1.45.0版本。我看了下github.com/spf1......
  • VUE
    在body标签中编写视图       ......
  • [Vue]键盘事件
    1.Vue中常用的按键别名:   回车=>enter   删除=>delete(捕获“删除”和“退格”键)   退出=>esc   空格=>space   换行=>tab(特殊,必须配合keydown使用)   上=>up   下=>down   左=>left  右=>right2.Vue未提供别名的按键,可以使用按键原始的key值去......
  • 短说PC端V3.1.0测试版发布|全新发布页、草稿箱、DIY门户首页新增页脚模块
    Hi大家好,我是给你们带来惊喜的运营小番茄。本期更新为短说PC端3.1.0测试版。此次V3.1.0版本带来的功能更新:①发布页全新设计;②草稿箱功能上线;③门户首页新增页脚模块。此外,优化了V3.0.1中的一些功能。一、新增功能01.PC端发布页全新设计保留原版动态类型内容支持快速发布的特点上,......
  • python tarfile模块
    一、简介tarfile 模块提供了创建、打开、读取和写入tar文件的函数和类。以下是该模块中常用的一些函数和类:tarfile.open(name,mode='r',fileobj=None,**kwargs):打开一个tar文件,返回一个TarFile对象。TarFile.add(name,arcname=None,recursive=True,filter=None):......
  • thinkPHP 项目只需要单个项目模块,比如去掉url中的admin
    thinkPHP项目只需要单个项目模块,比如去掉url中的admin这里以thinkPHP为例这里以thinkPHP为例1、找到入口文件的index.php,加入下面的define(‘BIND_MODULE’,‘admin’);<?php//定义应用目录define('APP_PATH',__DIR__.'/application/');//路由改写define('BIND_MODUL......