基于 Vue3.x + Vant UI 的多功能记账本(四)


文章目录

  • 基于 Vue3.x + Vant UI 的多功能记账本(四)
    • 项目演示
    • 1、登录注册页面
    • 2、图片验证码
    • 3、修改 axios
    • 4、写到最后(附源码)
系列内容 参考链接
基于 Vue3.x + Vant UI 的多功能记账本(一) 项目演示,涉及知识点
基于 Vue3.x + Vant UI 的多功能记账本(二) 搭建开发环境
基于 Vue3.x + Vant UI 的多功能记账本(三) 开发导航栏及公共部分

项目演示

Vue3 + Vant UI_多功能记账本

1、登录注册页面

页面设计,页面跳转

Login.vue

<template><!-- 根据页面显示相应头部 --><Header :title="type == 'login' ? '登录' : '注册'" /><div class="auth"><img class="logo" src="//s.yezgea02.com/1606836859539/onpeice.png" alt="" /><!-- 登录界面的表单 --><van-form class="form-wrap" @submit="onSubmit" v-if="type == 'login'"><div class="form"><!-- 账号输入框,clearable:清除图标,rules:表单校验规则 --><van-fieldclearablev-model="username"name="username"label="账号"placeholder="请输入账号":rules="[{ required: true, message: '请填写账户' }]"/><!-- 密码输入框 --><van-fieldclearablev-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="chanegType('register')" class="change-btn">没有账号,前往注册</p></div></van-form><!-- 注册页面的表单 --><van-form class="form-wrap" @submit="onSubmit" v-if="type == 'register'"><div class="form"><van-fieldclearablev-model="username"name="username"label="账号"placeholder="请输入账号":rules="[{ required: true, message: '请填写账号' }]"/><van-fieldclearablev-model="password"type="password"name="password"label="密码"placeholder="请输入密码":rules="[{ required: true, message: '请填写密码' }]"/><!-- 验证码输入框 --><van-fieldcenterclearablelabel="验证码"placeholder="输入验证码"v-model="verify"><!-- 点击刷新验证码 --><template #button><!-- 生成验证码图片组件,ref 方便拿到组件内的实例属性 --><VueImgVerify ref="verifyRef" /></template></van-field></div><div style="margin: 16px 0"><van-button round block type="primary" native-type="submit">注册</van-button><p @click="chanegType('login')" class="change-btn">登录已有账号</p></div></van-form></div>
</template><script>
import { reactive, toRefs, ref, onMounted } from "vue";
// 生成验证码的组件
import VueImgVerify from "../components/VueImageVerify.vue";
import Header from "../components/Header.vue";
import axios from "../utils/axios";
// 轻提示(成功/失败...)
import { Toast } from "vant";
import router from "../router";
export default {name: "Login",components: {VueImgVerify, // 验证码组件Header, //公共头组件},setup() {// 便于拿到 verifyRef 组件内的实例属性const verifyRef = ref(null);// 注册登录的相关内容const state = reactive({username: "",password: "",type: "login", // 登录注册模式切换参数verify: "", // 验证码输入框输入的内容imgCode: "", // 生成的验证图片内的文字});console.log("verifyRef", verifyRef);// 提交登录 or 注册表单const onSubmit = async (values) => {// 登录功能if (state.type == "login") {const { data } = await axios.post("/user/login", {username: state.username,password: state.password,});// 添加 token 到本地存储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);Toast.fail("验证码错误");return;}// 验证码匹配成功,注册=>注册成功await axios.post("/user/register", {username: state.username,password: state.password,});Toast.success("注册成功");}};// 切换登录和注册两种模式const chanegType = (type) => {state.type = type;};return {...toRefs(state),onSubmit,chanegType,verifyRef,};},
};
</script><style lang='less' scoped>
@import url("../config/custom.less");
.auth {height: calc(~"(100% - 46px)");padding: 30px 20px 0 20px;background: @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-top: 20px;}.van-cell:last-child {padding-bottom: 20px;}}}.change-btn {text-align: center;margin: 10px 0;color: @link-color;font-size: 14px;}
}
</style>

在 custom.less 下补充 link-color 变量的定义,在写样式的时候,以 color: @link-color; 这样的形式引用它

custom.less

@primary: #39be77; // 主题色
@danger: #fc3c0c;
@primary-bg: #f5f5f5;
@link-color: #597fe7;

当前页面的外层是 #app、body,作为父级,它们需要先把高度撑开

index.css

body,
html,
p {height: 100%;margin: 0;padding: 0;
}* {box-sizing: border-box;
}#app {height: 100%;
}

此时,yarn dev,打开浏览器可以看到…

2、图片验证码

注:验证码基本上都是由服务端接口提供,然后上报之后由服务端验证是否正确,所以此部分内容可以自行选择是否去做。

<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>

此时,yarn dev,打开浏览器可以看到…

3、修改 axios

为避免在页面内请求接口的时候,每次都通过 code 码去判断接口请求是否成功,我们可以这样修改 axios.js 文件

axios.js

import axios from 'axios'
// 轻提示插件(Vant UI)
import { Toast } from 'vant'
import router from '../router'// 根据环境变量切换本地和线上的请求地址
axios.defaults.baseURL = process.env.NODE_ENV == 'development' ? '/api' : '//47.99.134.126:7008/api'
// 允许跨域
axios.defaults.withCredentials = true
axios.defaults.headers['X-Requested-With'] = 'XMLHttpRequest'
// token的用户鉴权方式,在请求头的 headers 内添加 token,每次请求都会验证用户信息
axios.defaults.headers['Authorization'] = `${localStorage.getItem('token') || null}`
axios.defaults.headers.post['Content-Type'] = 'application/json'axios.interceptors.response.use(res => {// 返回数据的类型不是对象,则报异常if (typeof res.data !== 'object') {Toast.fail('服务端异常!')return Promise.reject(res)}// code 状态码不是200,则报异常if (res.data.code != 200) {if (res.data.msg) Toast.fail(res.data.msg)// code 状态码为 401 代表接口需要登录,继而跳转到登录页面if (res.data.code == 401) {router.push({ path: '/login' })}// 返回失败的实例return Promise.reject(res.data)}// code 为 200 时,请求成功,返回数据return res.data
})export default axios

4、写到最后(附源码)

看到这么好的项目,是不是有种想自己做出来的冲动?

如果有,那么说明你非常的想提升自己,想检验自己这段时间的学习成果,这个项目绝对是你的 不二选择

心动不如行动

那么接下来,一起从0搭建,开始我们基于 Vue3.x + Vant UI 的项目之旅吧~

源码在下方 ↓【回复:记账本】即可

Vue项目实战——【基于 Vue3.x + Vant UI】实现一个多功能记账本(登录注册页面,验证码)相关推荐

  1. Vue项目实战——【基于 Vue3.x + Vant UI】实现一个多功能记账本(项目演示、涉及知识点、源码分享)

    基于 Vue3.x + Vant UI 的多功能记账本(一) 文章目录 基于 Vue3.x + Vant UI 的多功能记账本(一) Vue3.x 实现多功能记账本 1.前言 2.项目演示 3.涉及知 ...

  2. 【Vue项目实战】Vue3动画神操作!教你如何实现PPT一样的动画效果!

    文章目录 前言 一.Animate.css是什么? 二.安装和使用 1.安装 2.基本用法 3.JavaScript用法 三.动画制作 1.弹入动画 总结 前言 最近写界面的时候,发现一个前端组件很好 ...

  3. 自己使用vue写的一个还觉得不错的登录注册页面

    登录页面效果 登录页面代码 <template><div class="content"> <div class="content-box& ...

  4. 【Vue3 造轮子项目 ------ kaite-ui】基于vue3.0 + vite + TypeScript 实现一个UI框架 - kaiteUI

    基于vue3.0 + vite + TypeScript 实现一个UI框架 - kaiteUI 前言 前段时间笔者一直忙于学习Vue3方面新知识,比如如何从vue2.0版本过渡到vue3.0,如何理解 ...

  5. Vue项目实战-vue2(移动端)

    Vue项目实战(移动端)# 相关资料 (一) 创建项目 (二) 禁用Eslint (三) devtool (四) 添加less支持 (五) vue路由配置(背诵) (六) 父子组件通信(背诵) (七) ...

  6. vue 动态添加class_前端开发:Vue项目实战-Music

    大家好,我来了,本期为大家带来的前端开发知识是"前端开发:Vue项目实战-Music",有兴趣做前端的朋友,和我一起来看看吧! 主要内容 项目环境搭建 路由导航实现 ListVie ...

  7. vue 将字符串最后一个字符给替换_前端开发:Vue项目实战-Music

    大家好,我来了,本期为大家带来的前端开发知识是"前端开发:Vue项目实战-Music",有兴趣做前端的朋友,和我一起来看看吧! 主要内容 项目环境搭建 路由导航实现 ListVie ...

  8. Vue项目实战 —— 哔哩哔哩移动端开发—— 第一篇

    目录 前言完 效果图 : 登录含签权 注册带正则 个人中心 下拉加载更多主页 修改个人中心 视频播放加关注+收藏 评论盖楼A回复B B回复C C回复A类似 项目开始 封装登录.注册 封装登录 从零到一 ...

  9. Vue项目实战 —— 后台管理系统( pc端 ) 第一篇

    前期回顾     我只写注释 -- 让Ai写代码_0.活在风浪里的博客-CSDN博客前期回顾 Vue项目实战 -- 哔哩哔哩移动端开发-- 第二篇_0.活在风浪里的博客-CSDN博客https://b ...

最新文章

  1. 蓝桥杯java第八届第四题--魔方状态
  2. yum-utils离线安装包下载地址
  3. PyTorch代码出现“interrupted by signal 9: SIGKILL”
  4. 谈谈struts2的工作原理
  5. @ResponseBody与@RestController的作用与区别
  6. SpringBoot整合jsp的使用
  7. 是德科技发力5G测试测量 欲攻占市场领导地位
  8. 今天第一次来这里开博,大家多多指教
  9. CMD如何直接运行文件
  10. vue路由守卫和重定向
  11. jzy3D从入门到弃坑_2使用jzy3D0.9画2D散点图
  12. 项目延期,不重视过程项目团队将自食其果
  13. android验证数字字母和特殊字符,正则表达式必须包含字母数字和特殊字符
  14. Android小心心动画
  15. 解决AE在mac big sur系统上安装闪退打不开问题,最新After Effects 2021 for Mac中文直装(ae 2021 mac)详细安装教程
  16. SpringBoot 自定义全局异常处理器
  17. MPLAB X IDE开发环境
  18. 《裸辞的程序猿漂流记十三》——奔跑在逆袭的路上
  19. HADOOP集群大数据词频统计及设计比较(完整教程)
  20. Java对象的持久化,将对象入库保存以及取出转换成对象实例

热门文章

  1. 资源|28本必读的经典机器学习/数据挖掘书籍(免费下载)
  2. Netty实战(一)
  3. 网上学什么副业赚钱?分享多个赚钱的副业,总有一个适合你
  4. 微信授权错误码 48001
  5. 四阶行列式直接展开_4行列式按行(列)展开课程.ppt
  6. QingScan 支持墨菲安全项目 murphysec
  7. jQuery停止动画效果
  8. priPrinter Pro v6.6.0.2524 虚拟打印机软件
  9. 浪潮集团入职技术培训(一)
  10. 【python代码实现】决策树分类算法