zdppy_api+vue3实现前后端分离的登录功能

实现思路

1、准备zdppy的开发环境
2、使用amauth提供的低代码接口,直接生成login登录接口
3、使用之前开发的登录模板渲染登录界面
4、给登录按钮绑定点击事件
5、给用户名和密码的输入框双向绑定数据
6、使用axios在登录按钮点击的时候,携带用户数据发送POST登录请求
7、处理登录接口的响应

完整代码

后端代码

.env
ZDPPY_MCRUD_HOST=127.0.0.1
ZDPPY_MCRUD_PORT=3306
ZDPPY_MCRUD_USERNAME=root
ZDPPY_MCRUD_PASSWORD=zhangdapeng520
ZDPPY_MCRUD_DATABASE=zdppy_demo
main.py
import contextlibimport api
import mcrud
import amauth
import envenv.load(".env")
db = mcrud.new_env()amauth.data.init(db, is_init_role=True, is_init_auth=True)@contextlib.asynccontextmanager
async def lifespan(app):yield {"db": db}app = api.Api(routes=[api.resp.post("/login", amauth.user.login),],middleware=[api.middleware.cors(),],lifespan=lifespan,
)if __name__ == '__main__':app.run()

前端代码

package.json
{"name": "tailwindcss_demo","private": true,"version": "0.0.0","type": "module","scripts": {"dev": "vite","build": "vite build","preview": "vite preview"},"dependencies": {"@ant-design/icons-vue": "^7.0.1","ant-design-vue": "^4.2.3","axios": "^1.7.2","dayjs": "^1.11.11","vue": "^3.4.29","vue-router": "^4.4.0"},"devDependencies": {"@vitejs/plugin-vue": "^5.0.5","autoprefixer": "^10.4.19","postcss": "^8.4.38","tailwindcss": "^3.4.4","vite": "^5.3.1"}
}
vite.config.js
import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue'// https://vitejs.dev/config/
export default defineConfig({plugins: [vue()],
})
tailwind.config.js
/** @type {import('tailwindcss').Config} */
export default {content: ["./src/**/*.{html,js,vue}"],theme: {extend: {},},plugins: [],
}
postcss.config.js
export default {plugins: {tailwindcss: {},autoprefixer: {},}}
index.html
<!doctype html>
<html lang="en"><head><meta charset="UTF-8" /><link rel="icon" type="image/svg+xml" href="/vite.svg" /><meta name="viewport" content="width=device-width, initial-scale=1.0" /><title>Vite + Vue</title></head><body><div id="app"></div><script type="module" src="/src/main.js"></script></body>
</html>
src/main.js
import {createApp} from 'vue'
import './style.css'
import 'ant-design-vue/dist/reset.css';import Antd from 'ant-design-vue';
import App from './App.vue';const app = createApp(App)
app.use(Antd)
app.mount('#app')
src/style.css
@tailwind base;
@tailwind components;
@tailwind utilities;
src/App.vue
<script setup>
import {message} from "ant-design-vue";
import {ref} from "vue";
import axios from "axios";const username = ref("")
const password = ref("")const onLoginButtonClick = () => {axios.post('http://127.0.0.1:8888/login', {username: username.value,password: password.value}).then(function (response) {message.success("login success")console.log("response=", response)const data = response.data.dataconsole.log("data=", data)console.log("token=", data.token)}).catch(function (error) {console.log(error);message.error(error)});
}
</script>
<template><section><span v-for="i in 297" :key="i"></span><div class="signin"><div class="content"><h2>用户登录</h2><div class="form"><div class="inputBox"><input type="text" v-model="username"> <i>账号</i></div><div class="inputBox"><input type="password" v-model="password"> <i>密码</i></div><div class="links"><a href="#">忘记密码</a><a href="#">注册</a></div><div class="inputBox"><input type="submit" value="立即登录" @click.prevent="onLoginButtonClick"></div></div></div></div></section>
</template>
<style scoped>
* {margin: 0;padding: 0;box-sizing: border-box;font-family: 'Quicksand', sans-serif;
}body {display: flex;justify-content: center;align-items: center;min-height: 100vh;background: #000;
}section {position: absolute;width: 100vw;height: 100vh;display: flex;justify-content: center;align-items: center;gap: 2px;flex-wrap: wrap;overflow: hidden;
}section::before {content: '';position: absolute;width: 100%;height: 100%;background: linear-gradient(#000, #0f0, #000);animation: animate 5s linear infinite;
}@keyframes animate {0% {transform: translateY(-100%);}100% {transform: translateY(100%);}
}section span {position: relative;display: block;width: calc(6.25vw - 2px);height: calc(6.25vw - 2px);background: #181818;z-index: 2;transition: 1.5s;
}section span:hover {background: #0f0;transition: 0s;
}section .signin {position: absolute;width: 400px;background: #222;z-index: 1000;display: flex;justify-content: center;align-items: center;padding: 40px;border-radius: 4px;box-shadow: 0 15px 35px rgba(0, 0, 0, 9);
}section .signin .content {position: relative;width: 100%;display: flex;justify-content: center;align-items: center;flex-direction: column;gap: 40px;
}section .signin .content h2 {font-size: 2em;color: #0f0;text-transform: uppercase;
}section .signin .content .form {width: 100%;display: flex;flex-direction: column;gap: 25px;
}section .signin .content .form .inputBox {position: relative;width: 100%;
}section .signin .content .form .inputBox input {position: relative;width: 100%;background: #333;border: none;outline: none;padding: 25px 10px 7.5px;border-radius: 4px;color: #fff;font-weight: 500;font-size: 1em;
}section .signin .content .form .inputBox i {position: absolute;left: 0;padding: 15px 10px;font-style: normal;color: #aaa;transition: 0.5s;pointer-events: none;
}.signin .content .form .inputBox input:focus ~ i,
.signin .content .form .inputBox input:valid ~ i {transform: translateY(-7.5px);font-size: 0.8em;color: #fff;
}.signin .content .form .links {position: relative;width: 100%;display: flex;justify-content: space-between;
}.signin .content .form .links a {color: #fff;text-decoration: none;
}.signin .content .form .links a:nth-child(2) {color: #0f0;font-weight: 600;
}.signin .content .form .inputBox input[type="submit"] {padding: 10px;background: #0f0;color: #000;font-weight: 600;font-size: 1.35em;letter-spacing: 0.05em;cursor: pointer;
}input[type="submit"]:active {opacity: 0.6;
}@media (max-width: 900px) {section span {width: calc(10vw - 2px);height: calc(10vw - 2px);}
}@media (max-width: 600px) {section span {width: calc(20vw - 2px);height: calc(20vw - 2px);}
}
</style>
<script setup lang="ts">
</script>

效果预览

在这里插入图片描述

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

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

相关文章

C++ | Leetcode C++题解之第207题课程表

题目&#xff1a; 题解&#xff1a; class Solution { private:vector<vector<int>> edges;vector<int> indeg;public:bool canFinish(int numCourses, vector<vector<int>>& prerequisites) {edges.resize(numCourses);indeg.resize(numCo…

昇思25天学习打卡营第13天|MindNLP ChatGLM-6B StreamChat

学AI还能赢奖品&#xff1f;每天30分钟&#xff0c;25天打通AI任督二脉 (qq.com) MindNLP ChatGLM-6B StreamChat 本案例基于MindNLP和ChatGLM-6B实现一个聊天应用。 1 环境配置 %%capture captured_output # 实验环境已经预装了mindspore2.2.14&#xff0c;如需更换mindspo…

[知识点篇]《计算机组成原理》之数据信息的表示

1、数据表示的作用 &#xff08;1&#xff09;定义&#xff1a;将数据按照某种方式组织&#xff0c;以便机器硬件能直接识别和使用。现代计算机采用二进制进行数据表示。 &#xff08;2&#xff09;数据表示考虑因素&#xff1a; 数据的类型&#xff1a; 数值/非数值、小数、…

读AI新生:破解人机共存密码笔记17不确定性和概率

1. 前向搜索 1.1. 通过前向搜索&#xff0c;通过考虑各种可能的动作序列的结果&#xff0c;来选择动作&#xff0c;是智能系统的基本能力 1.2. 如果一家卡车运输公司想要优化其100辆卡车在美国的运输&#xff0c;那么该公司可能需要考虑的状态数量将是10^700个 1.3. 几乎所有…

解决Install/Remove of the Service Denied报错

1、问题概述&#xff1f; 在Windows系统中安装MySQL5.7.43的时候&#xff0c;运行mysqld install命令提示报错&#xff1a;Install/Remove of the Service Denied 意思是&#xff1a;安装/删除服务被拒绝 问题原因所在&#xff1a;就是你当前的权限不够&#xff0c;以管理员…

Linux【环境 CenOS7】部分软件安装链接整理

优质博文&#xff1a;IT-BLOG-CN 一、开启网络 【问题】&#xff1a; 刚安装完CentOS&#xff0c;当ping www.baidu.com时&#xff0c;ping不通&#xff1b; 【解决】&#xff1a; 进入cd /etc/sysconfig/network-scripts/我这里修改的是ifcfg-ens33文件&#xff0c;将ONBOOT…

p2p、分布式,区块链笔记:试用ZeroTier组网

ZeroTier 是一种用于创建和管理虚拟局域网&#xff08;Virtual Local Area Network&#xff0c;VLAN&#xff09;的软件定义网络&#xff08;SDN&#xff09;解决方案。它可以通过互联网将多个设备安全地连接在一起&#xff0c;就像它们在同一个本地网络上一样。主要开发语言为…

【Python从入门到进阶】59、Pandas库中Series对象的操作(二)

接上篇《58、Pandas库中Series对象的操作(一)》 上一篇我们讲解了Pandas库中Series对象的基本概念、对象创建和操作&#xff0c;本篇我们来继续学习Series对象的运算、函数应用、时间序列操作&#xff0c;以及Series的案例实践。 一、Series对象的运算 1. 数值型数据的算术运…

1、音视频解封装流程---解复用

对于一个视频文件(mp4格式/flv格式)&#xff0c;audio_pkt或者video_pkt是其最基本的数据单元&#xff0c;即视频文件是由独立的视频编码包或者音频编码包组成的。 解复用就是从视频文件中把视频包/音频包单独读取出来保存成独立文件&#xff0c;那么如何得知packet是视频包还是…

【一篇搞懂】操作系统期末大题:进程同步与互斥 PV操作

文章目录 一、前言&#x1f680;&#x1f680;&#x1f680;二、正文&#xff1a;☀️☀️☀️题型一&#xff1a;利用信号量实现前驱关系题型二&#xff1a;利用信号量实现资源同步与互斥 一、前言&#x1f680;&#x1f680;&#x1f680; 本文简介&#xff1a;这是一篇基于b…

无人机远程控制:北斗短报文技术详解

无人机&#xff08;UAV&#xff09;技术的快速发展和应用&#xff0c;使得远程控制成为了一项关键技术。无人机远程控制涉及无线通信、数据处理等多个方面&#xff0c;其中北斗短报文技术以其独特的优势&#xff0c;在无人机远程控制领域发挥着重要作用。本文将详细解析无人机远…

2024-06-26 base SAS programming 学习笔记6(proc report)

proc report可以生成报表&#xff0c;基本格式&#xff1a; proc report data options; (options 可以是windows/WD表示将结果输出至单独的报表窗口&#xff0c;或者nowindows/nowd将结果输出至HTML结果窗口) column variables ;(筛选待输出的变量&#xff0c;变量名与变量名之…

09_计算机网络模型

目录 OSI/RM七层模型 OSI/RM七层模型 各层介绍及硬件设备 传输介质 TCP/IP协议簇 网络层协议 传输层协议 应用层协议 完整URL的组成 IP地址表示与计算 分类地址格式 子网划分和超网聚合 无分类编址 特殊含义的IP地址 IPv6协议 过渡技术 OSI/RM七层模型 OSI/RM七…

区间动态规划——最长回文子序列长度(C++)

把夜熬成粥&#xff0c;然后喝了它。 ——2024年7月1日 书接上回&#xff1a;区间动态规划——最长回文子串&#xff08;C&#xff09;-CSDN博客&#xff0c;大家有想到解决办法吗&#xff1f; 题目描述 给定一个字符串s&#xff08;s仅由数字和英文大小写字母组成&#xff0…

微积分-导数3(微分法则)

常见函数的导数 常量函数的导数 d d x ( c ) 0 \frac{d}{dx}(c) 0 dxd​(c)0 常量函数的图像是一条水平线 y c y c yc&#xff0c;它的斜率为0&#xff0c;所以我们必须有 f ′ ( x ) 0 f(x) 0 f′(x)0。从导数的定义来看&#xff0c;证明也很简单&#xff1a; f ′ …

QT拖放事件之三:自定义拖放操作-利用QDrag来拖动完成数据的传输

1、运行效果 1)Qt::MoveAction 2)Qt::CopyAction 2、源码 #include "Widget.h" #include "ui_Widget.h" #include "common.h"

二级建造师(建筑工程专业)考试题库,高效备考!!!

16.在施工合同履行期间发生的变更事项中&#xff0c;属于工程变更的是&#xff08;&#xff09;。 A.质量要求变更 B.分包单位变更 C.合同价款变更 D.相关法规变更 答案&#xff1a;A 解析&#xff1a;工程变更一般是指在工程施工过程中&#xff0c;根据合同约定对施工的…

恭贺甘露海首届道教南宗养生论坛暨天台山第十届道医大会圆满成功

6月13日&#xff0c;首届中国道教南宗养生论坛暨天台山第十届道医学术交流大会在浙江新昌重阳宫千人会场隆重开幕。 本次大会主办单位&#xff1a;天台山桐柏宫 中国民间中医医药研究开发协会道医学分会&#xff0c; 承办单位&#xff1a;新昌县重阳宫 &#xff0c;协办单位&…

网络基础:静态路由

静态路由是一种由网络管理员手动配置的路由方式&#xff0c;用于在网络设备&#xff08;如路由器或交换机&#xff09;之间传递数据包。与动态路由不同&#xff0c;静态路由不会根据网络状态的变化自动调整。 不同厂商的网络设备在静态路由的配置上有些许差异&#xff1b;下面…

网络构建关键技术_2.IPv4与IPv6融合组网技术

互联网数字分配机构&#xff08;IANA&#xff09;在2016年已向国际互联网工程任务组&#xff08;IETF&#xff09;提出建议&#xff0c;要求新制定的国际互联网标准只支持IPv6&#xff0c;不再兼容IPv4。目前&#xff0c;IPv6已经成为唯一公认的下一代互联网商用解决方案&#…