【记账本实战】07 记账本实战之登录注册页面(2)

记账本实战之登录注册页面(2)

目录

  • 记账本实战之登录注册页面(2)
    • 前言
    • 登录注册页面表单组件的完善
    • 图形验证码组件制作
    • 修改之前的错误的代码
    • axios 容错处理
    • 总结

前言

在本篇教程中,我们将会来完善登录和注册页面,并通过 canvas 手写一个验证码组件,然后再规整数据请求入口的容错处理。编写文章不易,如果你觉得这篇文章对你有帮助,请给博主点赞收藏评论加关注,你们的关注,将是博主继续编写文章的动力。另外,我也将我的文章教程制作成了 pdf 版,如果大家想的话,可以关注私信博主 免费获取 pdf 版的教程,也可以私信博主获取该项目需要用到的素材,在后面的教程中,我也会给出项目的源代码(敬请期待),下面的图片是 pdf 版教程的截图:

在这里插入图片描述

当然,博主也是一个正在学习前端的一员,能力有限,编写的文章也难免会存在错误之处,如果你发现错误,敬请指正!好了,下面进入正文部分。

登录注册页面表单组件的完善

在上篇文章记账本实战之登录注册页面(1)中,我们完成了登录注册页面的基本制作,如下图所示:

在这里插入图片描述

从效果图中可看出,表单组件里边界比较远,我们修改一下,打开 Login.vue,原来的代码如下图:

在这里插入图片描述

我们修改上图红框的代码,修改之后的代码如下:

<template><Header :title="type === 'login' ? '登录' : '注册'"></Header><div class="auth"><img class="logo" src="../assets/img/onpeice.png" alt="logo" /><van-form class="form-wrap" @submit="onSubmit" v-if="type === 'login'"><div class="form"><van-fieldv-model="username"name="username"label="账号"placeholder="请输入账号":rules="[{ required: true, message: '请填写账号' }]"/><van-fieldv-model="password"type="password"name="password"label="密码"placeholder="请输入密码":rules="[{ required: true, message: '请填写密码' }]"/></div><div style="margin: 16px 0"><van-button round block type="primary" native-type="submit">登录</van-button><p @click="changeType('register')" class="change-btn">没有账号,前往注册</p></div></van-form><van-form class="form-wrap" @submit="onSubmit" v-if="type === 'register'"><div class="form"><van-fieldv-model="username"name="username"label="账号"placeholder="请输入账号":rules="[{ required: true, message: '请填写账号' }]"/><van-fieldv-model="password"type="password"name="password"label="密码"placeholder="请输入密码":rules="[{ required: true, message: '请填写密码' }]"/><van-fieldcenterclearablelabel="验证码"placeholder="输入验证码"v-model="verify"><template #button><VueImgVerify ref="verifyRef" /></template></van-field></div><div style="margin: 16px 0"><van-button round block type="primary" native-type="submit">注册</van-button></div></van-form></div>
</template>// 省略其他代码...

然后我们在为 form 类添加样式,代码如下:

<style lang="less" scoped>
@import url("../style/custom.less");
.auth {// 省略其他代码....form-wrap {.form {border-radius: 10px;overflow: hidden;.van-cell:first-child {padding: 20px;}.van-cell:last-child {padding-bottom: 20px;}}// 省略其他代码...
}
</style>

然后我们运行 npm run dev 启动项目,打开浏览器查看效果如下:

在这里插入图片描述

可以看到,我们的表单组件变宽了,个人感觉会比较好看些。

图形验证码组件制作

在正常的注册流程中,验证码通常是由服务端接口提供的。当用户进行注册时,客户端会向服务端发送请求,服务端会生成一个验证码,并将其发送到用户的手机或电子邮件中。用户在注册时需要输入这个验证码以验证其身份。

服务端在接收到用户输入的验证码后,会将其与之前发送的验证码进行比对,以确认用户输入的验证码是否正确。如果验证码正确,服务端会认为用户是合法的,并允许其完成注册流程。如果验证码错误,服务端会拒绝用户的注册请求,并提示用户重新获取验证码。

这种验证码机制可以帮助防止自动化机器人恶意注册账号,保护用户账号的安全性。同时,它也可以防止已经存在的账号被未经授权的人员篡改或重置密码。因此,在注册流程中采用验证码机制是非常重要的安全措施之一。

前端手写图形验证码确实是一种有效的验证码形式,它要求用户在提供的画布上绘制特定的图案或字符,以验证他们不是机器人。这种验证码的前端实现通常涉及以下步骤:

  1. 生成验证码:在前端,使用某种随机数生成算法生成一个随机的图形或字符。
  2. 显示验证码:将生成的图形或字符显示给用户。
  3. 用户输入:用户根据显示的图形或字符进行绘制。
  4. 验证输入:当用户提交他们的输入时,前端代码将用户的输入与原始图形或字符进行比较。如果两者匹配,则认为验证码正确。

然而,需要注意的是,尽管前端验证可以提供用户体验和流畅性,但它不能作为安全的唯一保障。因为恶意用户可以通过查看或修改前端代码来绕过前端验证。因此,后端验证仍然是必需的,以确保整个注册流程的安全性。

总结来说,对于验证码的验证,最佳做法是结合前端和后端验证,以确保用户输入的验证码既正确又安全。

在本文中,为了让大家能尽可能多的掌握 Web 知识,我们采用前端验证的方式,通过手写图形验证码的形式,来验证用户是否输入正确的验证码。

首先我们在 components 新建验证码组件 VueImageVerify.vue,代码如下所示:

<template><div class="img-verify"><canvasref="verify":width="width":height="height"@click="handleDraw"></canvas></div>
</template>
<script type="text/ecmascript-6">
import { reactive, onMounted, ref, toRefs } from "vue";
export default {setup() {const verify = ref(null);const state = reactive({pool: 'ABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890', // 字符串width: 120,height: 40,imgCode: '',});onMounted(() => {// 初始化绘制图片验证码state.imgCode = draw();});// 点击图片重新绘制const handleDraw = () => {state.imgCode = draw();};// 随机数const randomNum = (min, max) => {return parseInt(Math.random() * (max - min) + min);};// 随机颜色const randomColor = (min, max) => {const r = randomNum(min, max);const g = randomNum(min, max);const b = randomNum(min, max);return `rgb(${r},${g},${b})`;};// 绘制图片const draw = () => {// 3.填充背景颜色,背景颜色要浅一点const ctx = verify.value.getContext('2d');// 填充颜色ctx.fillStyle = randomColor(180, 230);// 填充的位置ctx.fillRect(0, 0, state.width, state.height);// 定义paramTextlet imgCode = '';// 4.随机产生字符串,并且随机旋转for (let i = 0; i < 4; i++) {// 随机的四个字const text = state.pool[randomNum(0, state.pool.length)];imgCode += text;// 随机的字体大小const fontSize = randomNum(18, 40);// 字体随机的旋转角度const deg = randomNum(-30, 30);/** 绘制文字并让四个文字在不同的位置显示的思路 :* 1、定义字体* 2、定义对齐方式* 3、填充不同的颜色* 4、保存当前的状态(以防止以上的状态受影响)* 5、平移translate()* 6、旋转 rotate()* 7、填充文字* 8、restore出栈* */ctx.font = fontSize + 'px Simhei';ctx.textBaseline = 'top';ctx.fillStyle = randomColor(80, 150);/** save() 方法把当前状态的一份拷贝压入到一个保存图像状态的栈中。* 这就允许您临时地改变图像状态,* 然后,通过调用 restore() 来恢复以前的值。* save是入栈,restore是出栈。* 用来保存Canvas的状态。save之后,可以调用Canvas的平移、放缩、旋转、错切、裁剪等操作。 restore:用来恢复Canvas之前保存的状态。防止save后对Canvas执行的操作对后续的绘制有影响。** */ctx.save();ctx.translate(30 * i + 15, 15);ctx.rotate((deg * Math.PI) / 180);// fillText() 方法在画布上绘制填色的文本。文本的默认颜色是黑色。// 请使用 font 属性来定义字体和字号,并使用 fillStyle 属性以另一种颜色/渐变来渲染文本。// context.fillText(text,x,y,maxWidth);ctx.fillText(text, -15 + 5, -15);ctx.restore();}// 5.随机产生5条干扰线,干扰线的颜色要浅一点for (let i = 0; i < 5; i++) {ctx.beginPath();ctx.moveTo(randomNum(0, state.width), randomNum(0, state.height));ctx.lineTo(randomNum(0, state.width), randomNum(0, state.height));ctx.strokeStyle = randomColor(180, 230);ctx.closePath();ctx.stroke();}// 6.随机产生40个干扰的小点for (let i = 0; i < 40; i++) {ctx.beginPath();ctx.arc(randomNum(0, state.width), randomNum(0, state.height), 1, 0, 2 * Math.PI);ctx.closePath();ctx.fillStyle = randomColor(150, 200);ctx.fill();}return imgCode;};return {...toRefs(state),verify,handleDraw,};},
};
</script><style type="text/css">
.img-verify canvas {cursor: pointer;
}
</style>

代码的关键点就在 draw 方法,该方法内通过 verify.value.getContext('2d') 方法返回一个用于在画布上绘图的环境,赋值给 ctx 后,我们可以通过 ctxcanvas 画布上做文章。

绘制图形内的文字方法,我已经在上述代码中一一做了注释。这里再讲一个关键点,点击验证码的时候,注意要重新初始化 draw 方法,并将生成的值返回给 imgCode,后续调用组件的时候,我们可以从外部通过 ref,拿到组件内的 imgCode 变量,然后再于用户输入的值进行比较,这里若是不重新赋值,imgCode 会失去时效性。

下面我们需要在 Login.vue 引入验证码组件,关键代码如下所示:

<!-- 这里为了控制篇幅,不全部展示,下面代码接在注册表单的密码输入框后面 -->
...
<van-fieldcenterclearablelabel="验证码"placeholder="输入验证码"v-model="verify"
><template #button><VueImgVerify ref="verifyRef" /></template>
</van-field>
...

注册组件:

import VueImgVerify from "../components/VueImageVerify.vue";
components: {VueImgVerify;
}

我们需要给 VueImgVerify 组件添加 ref,以便拿到组件内的实例属性,如下所示:

setup() {const verifyRef = ref(null)...return {verifyRef}
}

到这里,验证码组件引入就完成了。

修改之前的错误的代码

在上一篇文章中,我们使用 Vant 组件库的 Toast 轻提示组件,当时我们使用的是 Vant3 的语法,但是在 Vant4 中原来的写法已经不适用了,所以我们要对其进行修改,首先要引入函数组件的样式,Vant 中有个别组件是以函数的形式提供的,包括 ToastDialogNotifyImagePreview 组件。在使用函数组件时,unplugin-vue-components 无法解析自动注册组件,导致 @vant/auto-import-resolver 无法解析样式,因此需要手动引入样式。

// Toast
import { showToast } from 'vant';
import 'vant/es/toast/style';// Dialog
import { showDialog } from 'vant';
import 'vant/es/dialog/style';// Notify
import { showNotify } from 'vant';
import 'vant/es/notify/style';// ImagePreview
import { showImagePreview } from 'vant';
import 'vant/es/image-preview/style';

这里我们主要用到 Toast 轻提示组件,所以我对之前的代码进行修改:

// 上篇文章写的代码
import { Toast } from "vant";Toast.fail("验证码错误");
Toast.success("注册成功");// Vant4
import { showFailToast, showSuccessToast } from "vant";
import "vant/es/toast/style";showFailToast("验证码错误");
showSuccessToast("注册成功");

在这里给出 Login.vue 修改之后的最终代码:

<template><Header :title="type === 'login' ? '登录' : '注册'"></Header><div class="auth"><img class="logo" src="../assets/img/onpeice.png" alt="logo" /><van-form class="form-wrap" @submit="onSubmit" v-if="type === 'login'"><div class="form"><van-fieldv-model="username"name="username"label="账号"placeholder="请输入账号":rules="[{ required: true, message: '请填写账号' }]"/><van-fieldv-model="password"type="password"name="password"label="密码"placeholder="请输入密码":rules="[{ required: true, message: '请填写密码' }]"/></div><div style="margin: 16px 0"><van-button round block type="primary" native-type="submit">登录</van-button><p @click="changeType('register')" class="change-btn">没有账号,前往注册</p></div></van-form><van-form class="form-wrap" @submit="onSubmit" v-if="type === 'register'"><div class="form"><van-fieldv-model="username"name="username"label="账号"placeholder="请输入账号":rules="[{ required: true, message: '请填写账号' }]"/><van-fieldv-model="password"type="password"name="password"label="密码"placeholder="请输入密码":rules="[{ required: true, message: '请填写密码' }]"/><van-fieldcenterclearablelabel="验证码"placeholder="输入验证码"v-model="verify"><template #button><VueImgVerify ref="verifyRef" /></template></van-field></div><div style="margin: 16px 0"><van-button round block type="primary" native-type="submit">注册</van-button></div></van-form></div>
</template><script>
import { reactive, toRefs } from "vue";
import Header from "../components/Header.vue";
import VueImgVerify from "../components/VueImageVerify.vue";
import axios from "../api/api";
import { showFailToast, showSuccessToast } from "vant";
import "vant/es/toast/style";
import { ref } from "vue";
export default {name: "LoginViews",components: {Header,VueImgVerify,},setup() {const verifyRef = ref(null);const state = reactive({username: "",password: "",type: "login", // 登录注册模式切换参数loading: false, // 点击注册时,让按钮处于加载状态verify: "", // 验证码输入框输入的内容imgCode: "", // 生成的验证图片内的文字});// 提交登录 or 注册表单const onSubmit = async (values) => {try {if (state.type === "login") {const { data } = await axios.post("/user/login", {username: state.username,password: state.password,});localStorage.setItem("token", data.token);window.location.href = "/";} else {state.imgCode = verifyRef.value.imgCode || "";if (verifyRef.value.imgCode.toLowerCase() != state.verify.toLowerCase()) {console.log("verifyRef.value.imgCode", verifyRef.value.imgCode);showFailToast("验证码错误");return;}state.loading = true;const { data } = await axios.post("/user/register", {username: state.username,password: state.password,});showSuccessToast("注册成功");state.type = "login";state.loading = false;}} catch (error) {state.loading = false;}};// 切换登录和注册两种模式const changeType = (type) => {state.type = type;};return {...toRefs(state),onSubmit,changeType,verifyRef,};},
};
</script><style lang="less" scoped>
@import url("../style/custom.less");
.auth {height: calc(~"(100% - 46px)");padding: 30px 20px 0 20px;background-color: @primary-bg;.logo {width: 150px;display: block;margin: 0 auto;margin-bottom: 30px;}.form-wrap {.form {border-radius: 10px;overflow: hidden;.van-cell:first-child {padding: 20px;}.van-cell:last-child {padding-bottom: 20px;}}.change-btn {text-align: center;margin: 10px 0;color: @link-color;font-size: 14px;}}
}
</style>

然后我们启动项目,打开浏览器查看效果如下:

在这里插入图片描述

axios 容错处理

我们在引入 Axios 网络框架 这篇文章的时候,对 Axios 做了二次封装,当时我们只做了简单的说明,在这里我们详细的讲一下 axios 的容错处理,接下来我们来分析一下 src/api/api.js

import axios from "axios";
import { Toast } from "vant";
import { useRouter } from "vue-router";axios.defaults.baseURL =process.env.NODE_ENV == "development"? "http://localhost:5173": "使用线上的域名或者IP"; // 根据环境变量切换本地和线上的请求地址
axios.defaults.withCredentials = true; // 允许跨域
axios.defaults.headers["X-Requested-With"] = "XMLHttpRequest";
axios.defaults.headers["token"] = localStorage.getItem("token") || ""; // 本项目采用 token 的用户鉴权方式,在请求头的 headers 内添加 token,每次请求都会验证用户信息
axios.defaults.headers.post["Content-Type"] = "application/json";// 响应拦截器
axios.interceptors.response.use((res) => {const router = useRouter(); // vue-router 4.x 的实例if (typeof res.data !== "object") {Toast.fail("服务端异常!");return Promise.reject(res);}// code 非 200 的情况下为异常情况// 代码 1if (res.data.code != 200) {// 代码 2if (res.data.msg) Toast.fail(res.data.msg);// 代码 3if (res.data.code == 401) {router.push({ path: "/login" });}return Promise.reject(res.data);}// 其他情况直接返回 data 数据// 代码 4return res.data;
});export default axios;

代码 1:当返回的 code 码为非 200 时,进入判断语句内。

代码 2:将错误提示全局展示。

代码 3:返回 401 代表接口需要登录,继而跳转到登录页面。

代码 4:code 码为 200 的时候,返回整个结构体。

后续随着项目的深入,此处的代码还会进一步的优化,有疑问的小伙伴可以在评论区提问喔。

总结

登录注册是此次实战之旅的“敲门砖”,登录之后,可以根据用户权限为当前用户创建账单、获取账单、图表数据等等。所以希望同学们能好好的吃透本实验的内容,这将会成为你们今后,独立扛项目的铺垫。

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

推荐几个Github高星GoLang管理系统

在Web开发领域&#xff0c;Go语言&#xff08;Golang&#xff09;以其高效、简洁、高并发等特性逐渐成为许多开发者的首选语言。有许多优秀的Go语言Web后台管理系统&#xff0c;这些项目星星众多&#xff0c;提供了丰富的功能和良好的代码质量。本文将介绍一些GitHub高星的GoLa…

Linux第33步_TF-A移植的第1步_创建新的设备树

TF-A移植第1步就是创建新的设备树&#xff0c;并命名为“stm32mp157d-atk”。 和“TF-A移植”有关的知识点&#xff1a; 1)设备树英文名字叫做Device tree&#xff0c;用来描述板子硬件信息的&#xff0c;比如开发板上的 CPU有几个核 、每个CPU核主频是多少&#xff0c;IIC、…

JVM:垃圾收集器(7种)

垃圾收集器关系图&#xff1a; 如果两个收集器之间存在连线&#xff0c;就说明它们可以搭配使用。它们说在的区域则表示这个收集器属于新生代收集器还是老年代收集器。其中Serial&#xff08;串行&#xff09;、Parallel&#xff08;并行&#xff09; 1、Serial收集器 Serial收…

【力扣·每日一题】2182.构造限制重复的字符串(模拟 贪心 优先队列 C++ Go)

题目链接 题意 给你一个字符串 s 和一个整数 repeatLimit &#xff0c;用 s 中的字符构造一个新字符串 repeatLimitedString &#xff0c;使任何字母 连续 出现的次数都不超过 repeatLimit 次。你不必使用 s 中的全部字符。 返回 字典序最大的 repeatLimitedString 。 如果…

AutoDL——终端训练神经网络模型(忽略本地问题)

前言&#xff1a; 本人之前分享过一篇文章&#xff1a;使用pycharm连接远程GPU训练神经网络模型&#xff08;超详细&#xff01;&#xff09;&#xff0c;其中详细介绍了如何利用pycharm连接AutoDL算力云平台租用的GPU服务器训练神经模型。但有些小伙伴可能会因为一些原因而导…

unity-声音与声效OLD

声音与声效 基本概念audio clipaudio listeneraudio source 基本操作如何创建音频源&#xff08;背景音乐&#xff09;如何在测试的时候关闭声音 常用代码一般流程如何在一个物体上播放多个音效如何在代码中延时播放多个声音如何在代码中停止音频的播放如何判断当前是否在播放音…

大模型 RAG 面试篇

1.LLMs 存在模型幻觉问题&#xff0c;请问如何处理&#xff1f; 检索LLM。 先用问题在领域数据库里检索到候选答案&#xff0c;再用LLM对答案进行加工。 2.基于LLM向量库的文档对话 思路是怎么样&#xff1f; 加载文件读取文本文本分割文本向量化问句向量化在文本向量中匹配…

基于深度学习的车牌识别(YOLOv5和CNN)

基于深度学习的车牌识别(YOLOv5和CNN&#xff09; 目录 一、综述 二、车牌检测 一、综述 本篇文章是面向的是小白&#xff0c;想要学习深度学习上的应用&#xff0c;本文中目前应用了YOLO v5和CNN来对车牌进行处理&#xff0c;最终形成一个完整的车牌信息记录&#xff0c;…

路飞项目--02

补充&#xff1a;axios封装 # 普通使用&#xff1a;安装 &#xff0c;导入使用 const filmListreactive({result:[]}) axios.get().then() async function load(){let responseawait axios.get()filmList.resultresponse.data.results } # 封装示例&#xff1a;请求发出去之前…

AP5101C 高压线性 LED恒流驱动器 DFN2*2 LED灯汽车雾灯转向灯

产品描述 AP5101C 是一款高压线性 LED 恒流芯片 &#xff0c; 简单 、 内置功率管 &#xff0c; 适用于6- 100V 输入的高精度降压 LED 恒流驱动芯片。电流2.0A。AP5101C 可实现内置MOS 做 2.0A,外置 MOS 可做 3.0A 的。AP5101C 内置温度保护功能 &#xff0c;温度保护点为 130 …

linux编译源码,安装valgrind

目录 1 下载源码 2 在虚拟机上解压 3 进入解压的目录&#xff0c;执行make 4 安装 5 检查安装是否成功 本文参考了内存检查工具valgrind介绍、安装与使用-CSDN博客 1 下载源码 我到Valgrind: Current Releases 下载了valgrind 3.22.0源码 2 在虚拟机上解压 我使用的虚…

Leetcode 用队列实现栈

题目&#xff1a; 请你仅使用两个队列实现一个后入先出&#xff08;LIFO&#xff09;的栈&#xff0c;并支持普通栈的全部四种操作&#xff08;push、top、pop 和 empty&#xff09;。 实现 MyStack 类&#xff1a; void push(int x) 将元素 x 压入栈顶。 int pop() 移除并…