記錄--使用Canvas繪製一個驗證碼組件

来源:https://www.cnblogs.com/smileZAZ/archive/2023/09/22/17723036.html
-Advertisement-
Play Games

這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助 使用Canvas繪製一個驗證碼組件 前言 驗證碼,這一日常伴隨我們的要素,是我們線上交互的重要安全保障。你的手機簡訊里是否被它占據半壁江山,今天我們就來聊聊如何在網頁上實現一個簡單的驗證碼組件。大家在登錄網站時為了防止被惡意攻擊或者多次點 ...


這裡給大家分享我在網上總結出來的一些知識,希望對大家有所幫助

使用Canvas繪製一個驗證碼組件

前言

驗證碼,這一日常伴隨我們的要素,是我們線上交互的重要安全保障。你的手機簡訊里是否被它占據半壁江山,今天我們就來聊聊如何在網頁上實現一個簡單的驗證碼組件。大家在登錄網站時為了防止被惡意攻擊或者多次點擊操作,使用驗證碼是最常用的實現方式。在學習完Canvas後,通過Canvas實現簡單的驗證碼就比較簡單了,該組件已開源上傳npm,可以直接安裝使用,源碼已上傳Git,地址在文尾。

聊聊Canvas

1、在開始繪製Canvas時,我們需要先創建一個html節點,併為其設置寬高

<canvas id="canvas" width="200" height="150""></canvas>

2、可以使用 cnv.width 和 cnv.height 分別獲取 Canvas 的寬度和高度,可以使用 cnv.getContext("2d")來獲取 canvas 2D 上下文環境對象。

如何繪製一條直線

使用canvas的 moveTo()和 lineTo()方法,比如將畫筆moveTo(0,0),移動畫筆到極點坐標,使用lineTo(100,100),這樣執行完兩個步驟將能夠繪製一條從坐標(0,0)到坐標(100,100)的一條直線,不過此時canvas上並不會展示出一條直線,我們還需要調用stroke()方法結束繪畫,這樣就能在頁面上看到一條直線。

隨機畫一條直線:

    context.moveTo(Math.random() * Canvas_DOM.width, Math.random() * Canvas_DOM.height);
    context.lineTo(Math.random() * Canvas_DOM.width, Math.random() * Canvas_DOM.height);

如何繪製一個點

從上面我們知道瞭如何繪製一條直線,那麼繪製一個點其實就是繪製一個長度為1的直線。

繪製一個隨機點:

    var x = Math.random() * Canvas_DOM.width;
    var y = Math.random() * Canvas_DOM.height;
    context.moveTo(x, y);
    context.lineTo(x + 1, y + 1);

怎麼繪製文字

通過fillText(txt, 0, 0)就能在canvas里添加一個文字,其中txt為需要繪製的內容,後面兩個參數為文字在canvas中的位置,當然也可以通過rotate()和translate()來改變文字的方向和位置。

    context.fillText(txt, 0, 0);
    context.rotate(-deg);
    context.translate(-x, -y);

使用fillStyle可以修改文字、線條的顏色

繪製驗證碼組件

從上面我們知道瞭如何繪製直線、點、以及文字,那麼一個驗證碼也就由這些組成,我們只需要把他們結合起來,在點擊驗證碼時,重新繪製一個隨機的canvas,就能得到一個完整的驗證碼組件:

<template>
    <canvas class="canvas" ref="cav" @click="Refresh"></canvas>
</template>
<script lang="ts" setup>
import { ref, onMounted } from 'vue';
const show_num = ref<any>([])
const cav = ref<any>(null)
onMounted(() => {
    draw();
});
const draw = () => {
    var Canvas_DOM = cav.value;
    var cav_width = Canvas_DOM.clientWidth;
    var cav_height = Canvas_DOM.clientHeight;
    var context = Canvas_DOM.getContext('2d');
    Canvas_DOM.width = cav_width;
    Canvas_DOM.height = cav_height;
    var sCode = "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,1,2,3,4,5,6,7,8,9,0";
    var aCode = sCode.split(',');
    var aLength = aCode.length; //獲取到數組的長度

    for (var i = 0; i <= 3; i++) {
        var j = Math.floor(Math.random() * aLength); //獲取到隨機的索引值
        var deg = (Math.random() * 30 * Math.PI) / 180; //產生0~30之間的隨機弧度
        var txt = aCode[j]; //得到隨機的一個內容
        show_num.value[i] = txt.toLowerCase();
        var x = 50 + i * 20; //文字在canvas上的x坐標
        var y = 20 + Math.random() * 8; //文字在canvas上的y坐標
        context.font = 'bold 23px 微軟雅黑';

        context.translate(x, y);
        context.rotate(deg);

        context.fillStyle = _randomColor();
        context.fillText(txt, 0, 0);
        context.rotate(-deg);
        context.translate(-x, -y);
    }
    for (var i = 0; i <= 5; i++) {
        //驗證碼上顯示隨機線條
        context.strokeStyle = _randomColor();
        context.beginPath();
        context.moveTo(Math.random() * Canvas_DOM.width, Math.random() * Canvas_DOM.height);
        context.lineTo(Math.random() * Canvas_DOM.width, Math.random() * Canvas_DOM.height);
        context.stroke();
    }
    for (var i = 0; i <= 30; i++) {
        //驗證碼上顯示隨機小點
        context.strokeStyle = _randomColor();
        context.beginPath();
        var x = Math.random() * Canvas_DOM.width;
        var y = Math.random() * Canvas_DOM.height;
        context.moveTo(x, y);
        context.lineTo(x + 1, y + 1);
        context.stroke();
    }
}
const _randomColor = () => {
    //得到隨機的顏色值
    var r = Math.floor(Math.random() * 256);
    var g = Math.floor(Math.random() * 256);
    var b = Math.floor(Math.random() * 256);
    return 'rgb(' + r + ',' + g + ',' + b + ')';
}
const Refresh = () => {
    draw();
}
</script>
<style  scoped>
.canvas {
    width: 200px;
    height: 40px;
    display: inline-block;
    margin-left: 12px;
    cursor: pointer;
}
</style>

使用示例

npm install @fcli/vue-vercode --save-dev 來安裝

在項目中使用
import VerCode from '@fcli/vue-vercode';
const app=createApp(App)
app.use(VerCode);
示例:
<div class="content">
    <ver-code ref=verCode></ver-code>
</div>

獲取驗證碼組件的數據

let code = verCode.value.show_num.join('');

本文轉載於:

https://juejin.cn/post/7280786332712173626

如果對您有所幫助,歡迎您點個關註,我會定時更新技術文檔,大家一起討論學習,一起進步。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1、首先你需要提前準備好jar包或者war包,並想辦法放入Linux環境(或虛擬機)中; 2、java項目的部署需要用到Tomcat或者Jetty,docker可以直接拉取他倆的鏡像,這裡以Tomcat為例: # : 後面需要加上war或者jar對應的Tomcat版本,最好加上, # 否則預設最新的 ...
  • 一、文檔的查看指令 1、tail指令 作用:查看一個文件的末n行 語法:#tail -n 文件的路徑 說明:-n可以不寫,不寫,預設表示10行。 案例:新建一個1.txt文檔,使用tail指令查看root/1.txt文件的末5行和末10行 tail -5 /root/1.txt tail /root ...
  • 1. 線上備份 2. 離線備份 2.1. 關閉MySQL做備份是最簡單、最安全的 2.2. 所有獲取一致性副本的方法中最好的 2.3. 損壞或不一致的風險最小 2.4. 根本不用關心InnoDB緩衝池中的臟頁或其他緩存 2.5. 不需要擔心數據在嘗試備份的過程中被修改 2.5.1. 伺服器不對應用提 ...
  • MySQL 主從複製與讀寫分離 1、什麼是讀寫分離? 讀寫分離,基本的原理是讓主資料庫處理事務性增、改、刪操作(INSERT、UPDATE、DELETE),而從資料庫處理SELECT查詢操作。資料庫複製被用來把事務性操作導致的變更同步到集群中的從資料庫。 2、為什麼要讀寫分離呢? 因為資料庫的“寫” ...
  • 最近,某白酒品牌頻頻吸引大眾眼球,白酒與咖啡、巧克力等聯名衍生品一經推出便掀起熱潮。某商品由於太過火爆,甚至一度售罄下架。 不得不說,我國擁有超大規模內需市場,消費潛力巨大。 當前,創新消費場景加上數字化融合轉型,成為酒企品牌開疆擴土、逆勢增長的重要途徑。 如今越來越多的酒企開始擁抱數字化,建立涵蓋 ...
  • 1. 每個人都知道需要備份,但並不是每個人都能意識到需要的是可恢復的備份 1.1. 如果你沒有提前做好備份規劃,也許以後會發現已經錯失了一些最佳的選擇 1.2. 在伺服器已經配置好以後,才想起應該使用LVM,以便獲取文件系統的快照——但這時已經太遲了 1.3. 如果你沒有計劃做定期的恢復演練,當真的 ...
  • 在日常開發中,很多時候需要對數組進行分組,每次都要手寫一個分組函數,或者使用lodash的groupBy函數。好消息是,JavaScript 現在正在引入全新的分組方法:Object.groupBy和Map.groupBy,以後再也不需要手寫分組函數了,目前最新版本的 Chrome(117)已經支持 ...
  • 前言 Vue 3是一個功能強大的前端框架,它引入了一些令人興奮的新特性,其中最引人註目的是ref和reactive。這兩個API是Vue 3中響應式編程的核心,本文將深入探討它們的用法和差異。 什麼是響應式編程? 在Vue中,響應式編程是一種使數據與UI保持同步的方式。當數據變化時,UI會自動更新, ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...